github.com/aloncn/graphics-go@v0.0.1/src/runtime/cgo/gcc_netbsd_arm.c (about) 1 // Copyright 2013 The Go Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style 3 // license that can be found in the LICENSE file. 4 5 // +build cgo 6 7 #include <sys/types.h> 8 #include <pthread.h> 9 #include <signal.h> 10 #include <string.h> 11 #include "libcgo.h" 12 13 static void *threadentry(void*); 14 15 static void (*setg_gcc)(void*); 16 17 void 18 x_cgo_init(G *g, void (*setg)(void*)) 19 { 20 pthread_attr_t attr; 21 size_t size; 22 23 setg_gcc = setg; 24 pthread_attr_init(&attr); 25 pthread_attr_getstacksize(&attr, &size); 26 g->stacklo = (uintptr)&attr - size + 4096; 27 pthread_attr_destroy(&attr); 28 } 29 30 31 void 32 _cgo_sys_thread_start(ThreadStart *ts) 33 { 34 pthread_attr_t attr; 35 sigset_t ign, oset; 36 pthread_t p; 37 size_t size; 38 int err; 39 40 sigfillset(&ign); 41 pthread_sigmask(SIG_SETMASK, &ign, &oset); 42 43 pthread_attr_init(&attr); 44 pthread_attr_getstacksize(&attr, &size); 45 // Leave stacklo=0 and set stackhi=size; mstack will do the rest. 46 ts->g->stackhi = size; 47 err = pthread_create(&p, &attr, threadentry, ts); 48 49 pthread_sigmask(SIG_SETMASK, &oset, nil); 50 51 if (err != 0) { 52 fprintf(stderr, "runtime/cgo: pthread_create failed: %s\n", strerror(err)); 53 abort(); 54 } 55 } 56 57 extern void crosscall_arm1(void (*fn)(void), void (*setg_gcc)(void*), void *g); 58 static void* 59 threadentry(void *v) 60 { 61 ThreadStart ts; 62 stack_t ss; 63 64 ts = *(ThreadStart*)v; 65 free(v); 66 67 // On NetBSD, a new thread inherits the signal stack of the 68 // creating thread. That confuses minit, so we remove that 69 // signal stack here before calling the regular mstart. It's 70 // a bit baroque to remove a signal stack here only to add one 71 // in minit, but it's a simple change that keeps NetBSD 72 // working like other OS's. At this point all signals are 73 // blocked, so there is no race. 74 memset(&ss, 0, sizeof ss); 75 ss.ss_flags = SS_DISABLE; 76 sigaltstack(&ss, nil); 77 78 crosscall_arm1(ts.fn, setg_gcc, (void*)ts.g); 79 return nil; 80 }