// Copyright 2009 The Go Authors. All rights reserved.
|
// Use of this source code is governed by a BSD-style
|
// license that can be found in the LICENSE file.
|
|
#include "libcgo.h"
|
|
/* Stub for creating a new thread */
|
void
|
x_cgo_thread_start(ThreadStart *arg)
|
{
|
ThreadStart *ts;
|
|
/* Make our own copy that can persist after we return. */
|
_cgo_tsan_acquire();
|
ts = malloc(sizeof *ts);
|
_cgo_tsan_release();
|
if(ts == nil) {
|
fprintf(stderr, "runtime/cgo: out of memory in thread_start\n");
|
abort();
|
}
|
*ts = *arg;
|
|
_cgo_sys_thread_start(ts); /* OS-dependent half */
|
}
|
|
#ifndef CGO_TSAN
|
void(* const _cgo_yield)() = NULL;
|
#else
|
|
#include <string.h>
|
|
char x_cgo_yield_strncpy_src = 0;
|
char x_cgo_yield_strncpy_dst = 0;
|
size_t x_cgo_yield_strncpy_n = 0;
|
|
/*
|
Stub for allowing libc interceptors to execute.
|
|
_cgo_yield is set to NULL if we do not expect libc interceptors to exist.
|
*/
|
static void
|
x_cgo_yield()
|
{
|
/*
|
The libc function(s) we call here must form a no-op and include at least one
|
call that triggers TSAN to process pending asynchronous signals.
|
|
sleep(0) would be fine, but it's not portable C (so it would need more header
|
guards).
|
free(NULL) has a fast-path special case in TSAN, so it doesn't
|
trigger signal delivery.
|
free(malloc(0)) would work (triggering the interceptors in malloc), but
|
it also runs a bunch of user-supplied malloc hooks.
|
|
So we choose strncpy(_, _, 0): it requires an extra header,
|
but it's standard and should be very efficient.
|
|
GCC 7 has an unfortunate habit of optimizing out strncpy calls (see
|
https://golang.org/issue/21196), so the arguments here need to be global
|
variables with external linkage in order to ensure that the call traps all the
|
way down into libc.
|
*/
|
strncpy(&x_cgo_yield_strncpy_dst, &x_cgo_yield_strncpy_src,
|
x_cgo_yield_strncpy_n);
|
}
|
|
void(* const _cgo_yield)() = &x_cgo_yield;
|
|
#endif /* GO_TSAN */
|