// 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 #include #include #include #include #include #include "libcgo.h" #include "libcgo_unix.h" static void* threadentry(void*); static void (*setg_gcc)(void*); // TCB_SIZE is sizeof(struct thread_control_block), as defined in // /usr/src/lib/librthread/tcb.h on OpenBSD 5.9 and earlier. #define TCB_SIZE (4 * sizeof(void *)) // TIB_SIZE is sizeof(struct tib), as defined in // /usr/include/tib.h on OpenBSD 6.0 and later. #define TIB_SIZE (4 * sizeof(void *) + 6 * sizeof(int)) // TLS_SIZE is the size of TLS needed for Go. #define TLS_SIZE (2 * sizeof(void *)) void *__get_tcb(void); void __set_tcb(void *); static int (*sys_pthread_create)(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine)(void *), void *arg); struct thread_args { void *(*func)(void *); void *arg; }; static int has_tib = 0; static void tcb_fixup(int mainthread) { void *tls, *newtcb, *oldtcb; size_t tls_size, tcb_size; // TODO(jsing): Remove once OpenBSD 6.1 is released and OpenBSD 5.9 is // no longer supported. // The OpenBSD ld.so(1) does not currently support PT_TLS. As a result, // we need to allocate our own TLS space while preserving the existing // TCB or TIB that has been setup via librthread. tcb_size = has_tib ? TIB_SIZE : TCB_SIZE; tls_size = TLS_SIZE + tcb_size; tls = malloc(tls_size); if(tls == NULL) abort(); // The signal trampoline expects the TLS slots to be zeroed. bzero(tls, TLS_SIZE); oldtcb = __get_tcb(); newtcb = tls + TLS_SIZE; bcopy(oldtcb, newtcb, tcb_size); if(has_tib) { // Fix up self pointer. *(uintptr_t *)(newtcb) = (uintptr_t)newtcb; } __set_tcb(newtcb); // NOTE(jsing, minux): we can't free oldtcb without causing double-free // problem. so newtcb will be memory leaks. Get rid of this when OpenBSD // has proper support for PT_TLS. } static void * thread_start_wrapper(void *arg) { struct thread_args args = *(struct thread_args *)arg; free(arg); tcb_fixup(0); return args.func(args.arg); } static void init_pthread_wrapper(void) { void *handle; // Locate symbol for the system pthread_create function. handle = dlopen("libpthread.so", RTLD_LAZY); if(handle == NULL) { fprintf(stderr, "runtime/cgo: dlopen failed to load libpthread: %s\n", dlerror()); abort(); } sys_pthread_create = dlsym(handle, "pthread_create"); if(sys_pthread_create == NULL) { fprintf(stderr, "runtime/cgo: dlsym failed to find pthread_create: %s\n", dlerror()); abort(); } // _rthread_init is hidden in OpenBSD librthread that has TIB. if(dlsym(handle, "_rthread_init") == NULL) { has_tib = 1; } dlclose(handle); } static pthread_once_t init_pthread_wrapper_once = PTHREAD_ONCE_INIT; int pthread_create(pthread_t *thread, const pthread_attr_t *attr, void *(*start_routine)(void *), void *arg) { struct thread_args *p; // we must initialize our wrapper in pthread_create, because it is valid to call // pthread_create in a static constructor, and in fact, our test for issue 9456 // does just that. if(pthread_once(&init_pthread_wrapper_once, init_pthread_wrapper) != 0) { fprintf(stderr, "runtime/cgo: failed to initialize pthread_create wrapper\n"); abort(); } p = malloc(sizeof(*p)); if(p == NULL) { errno = ENOMEM; return -1; } p->func = start_routine; p->arg = arg; return sys_pthread_create(thread, attr, thread_start_wrapper, p); } void x_cgo_init(G *g, void (*setg)(void*)) { pthread_attr_t attr; size_t size; setg_gcc = setg; pthread_attr_init(&attr); pthread_attr_getstacksize(&attr, &size); g->stacklo = (uintptr)&attr - size + 4096; pthread_attr_destroy(&attr); if(pthread_once(&init_pthread_wrapper_once, init_pthread_wrapper) != 0) { fprintf(stderr, "runtime/cgo: failed to initialize pthread_create wrapper\n"); abort(); } tcb_fixup(1); } void _cgo_sys_thread_start(ThreadStart *ts) { pthread_attr_t attr; sigset_t ign, oset; pthread_t p; size_t size; int err; sigfillset(&ign); pthread_sigmask(SIG_SETMASK, &ign, &oset); pthread_attr_init(&attr); pthread_attr_getstacksize(&attr, &size); // Leave stacklo=0 and set stackhi=size; mstack will do the rest. ts->g->stackhi = size; err = _cgo_openbsd_try_pthread_create(sys_pthread_create, &p, &attr, threadentry, ts); pthread_sigmask(SIG_SETMASK, &oset, nil); if (err != 0) { fprintf(stderr, "runtime/cgo: pthread_create failed: %s\n", strerror(err)); abort(); } } static void* threadentry(void *v) { ThreadStart ts; tcb_fixup(0); ts = *(ThreadStart*)v; free(v); /* * Set specific keys. */ setg_gcc((void*)ts.g); crosscall_386(ts.fn); return nil; }