LEFT | RIGHT |
(no file at all) | |
1 // Copyright 2009 The Go Authors. All rights reserved. | 1 // Copyright 2009 The Go Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style | 2 // Use of this source code is governed by a BSD-style |
3 // license that can be found in the LICENSE file. | 3 // license that can be found in the LICENSE file. |
4 | 4 |
5 #include <sys/types.h> | 5 #include <sys/types.h> |
6 #include <dlfcn.h> | 6 #include <dlfcn.h> |
7 #include <errno.h> | 7 #include <errno.h> |
8 #include <pthread.h> | 8 #include <pthread.h> |
9 #include <signal.h> | 9 #include <signal.h> |
10 #include <string.h> | 10 #include <string.h> |
11 #include "libcgo.h" | 11 #include "libcgo.h" |
12 | 12 |
13 static void* threadentry(void*); | 13 static void* threadentry(void*); |
| 14 static void (*setmg_gcc)(void*, void*); |
14 | 15 |
15 // TCB_SIZE is sizeof(struct thread_control_block), | 16 // TCB_SIZE is sizeof(struct thread_control_block), |
16 // as defined in /usr/src/lib/librthread/tcb.h | 17 // as defined in /usr/src/lib/librthread/tcb.h |
17 #define TCB_SIZE (4 * sizeof(void *)) | 18 #define TCB_SIZE (4 * sizeof(void *)) |
18 #define TLS_SIZE (2 * sizeof(void *)) | 19 #define TLS_SIZE (2 * sizeof(void *)) |
19 | 20 |
20 void *__get_tcb(void); | 21 void *__get_tcb(void); |
21 void __set_tcb(void *); | 22 void __set_tcb(void *); |
22 | 23 |
23 static int (*sys_pthread_create)(pthread_t *thread, const pthread_attr_t *attr, | 24 static int (*sys_pthread_create)(pthread_t *thread, const pthread_attr_t *attr, |
(...skipping 51 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
75 errno = ENOMEM; | 76 errno = ENOMEM; |
76 return -1; | 77 return -1; |
77 } | 78 } |
78 p->func = start_routine; | 79 p->func = start_routine; |
79 p->arg = arg; | 80 p->arg = arg; |
80 | 81 |
81 return sys_pthread_create(thread, attr, thread_start_wrapper, p); | 82 return sys_pthread_create(thread, attr, thread_start_wrapper, p); |
82 } | 83 } |
83 | 84 |
84 void | 85 void |
85 x_cgo_init(G *g) | 86 x_cgo_init(G *g, void (*setmg)(void*, void*)) |
86 { | 87 { |
87 pthread_attr_t attr; | 88 pthread_attr_t attr; |
88 size_t size; | 89 size_t size; |
89 void *handle; | 90 void *handle; |
90 | 91 |
| 92 setmg_gcc = setmg; |
91 pthread_attr_init(&attr); | 93 pthread_attr_init(&attr); |
92 pthread_attr_getstacksize(&attr, &size); | 94 pthread_attr_getstacksize(&attr, &size); |
93 g->stackguard = (uintptr)&attr - size + 4096; | 95 g->stackguard = (uintptr)&attr - size + 4096; |
94 pthread_attr_destroy(&attr); | 96 pthread_attr_destroy(&attr); |
95 | 97 |
96 // Locate symbol for the system pthread_create function. | 98 // Locate symbol for the system pthread_create function. |
97 handle = dlopen("libpthread.so", RTLD_LAZY); | 99 handle = dlopen("libpthread.so", RTLD_LAZY); |
98 if(handle == NULL) { | 100 if(handle == NULL) { |
99 fprintf(stderr, "dlopen: failed to load libpthread: %s\n", dlerr
or()); | 101 fprintf(stderr, "dlopen: failed to load libpthread: %s\n", dlerr
or()); |
100 abort(); | 102 abort(); |
(...skipping 47 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
148 | 150 |
149 ts.g->stackbase = (uintptr)&ts; | 151 ts.g->stackbase = (uintptr)&ts; |
150 | 152 |
151 /* | 153 /* |
152 * _cgo_sys_thread_start set stackguard to stack size; | 154 * _cgo_sys_thread_start set stackguard to stack size; |
153 * change to actual guard pointer. | 155 * change to actual guard pointer. |
154 */ | 156 */ |
155 ts.g->stackguard = (uintptr)&ts - ts.g->stackguard + 4096; | 157 ts.g->stackguard = (uintptr)&ts - ts.g->stackguard + 4096; |
156 | 158 |
157 /* | 159 /* |
158 » * Set specific keys. On OpenBSD/ELF, the thread local storage | 160 » * Set specific keys. |
159 » * is just before %fs:0. Our dynamic 6.out's reserve 16 bytes | |
160 » * for the two words g and m at %fs:-16 and %fs:-8. | |
161 */ | 161 */ |
162 » asm volatile ( | 162 » setmg_gcc((void*)ts.m, (void*)ts.g); |
163 » » "movq %0, %%fs:-16\n"» // MOVL g, -16(FS) | 163 |
164 » » "movq %1, %%fs:-8\n"» // MOVL m, -8(FS) | |
165 » » :: "r"(ts.g), "r"(ts.m) | |
166 » ); | |
167 crosscall_amd64(ts.fn); | 164 crosscall_amd64(ts.fn); |
168 return nil; | 165 return nil; |
169 } | 166 } |
LEFT | RIGHT |