| // 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 <sys/types.h> |
| #include <dlfcn.h> |
| #include <errno.h> |
| #include <pthread.h> |
| #include <signal.h> |
| #include <string.h> |
| #include "libcgo.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 = 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_amd64(ts.fn); |
| return nil; |
| } |