// 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 */