1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
|
// 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 <pthread.h>
#include "libcgo.h"
static void* threadentry(void*);
static pthread_key_t km, kg;
void
initcgo(void)
{
if(pthread_key_create(&km, nil) < 0) {
fprintf(stderr, "libcgo: pthread_key_create failed\n");
abort();
}
if(pthread_key_create(&kg, nil) < 0) {
fprintf(stderr, "libcgo: pthread_key_create failed\n");
abort();
}
}
void
libcgo_sys_thread_start(ThreadStart *ts)
{
pthread_attr_t attr;
pthread_t p;
size_t size;
pthread_attr_init(&attr);
pthread_attr_getstacksize(&attr, &size);
ts->g->stackguard = size;
pthread_create(&p, &attr, threadentry, ts);
}
static void*
threadentry(void *v)
{
ThreadStart ts;
ts = *(ThreadStart*)v;
free(v);
ts.g->stackbase = (uintptr)&ts;
/*
* libcgo_sys_thread_start set stackguard to stack size;
* change to actual guard pointer.
*/
ts.g->stackguard = (uintptr)&ts - ts.g->stackguard + 4096;
crosscall_amd64(ts.m, ts.g, ts.fn);
return nil;
}
void
libcgo_set_scheduler(void *m, void *g)
{
pthread_setspecific(km, m);
pthread_setspecific(kg, g);
}
struct get_scheduler_args {
void *m;
void *g;
};
void libcgo_get_scheduler(struct get_scheduler_args *)
__attribute__ ((visibility("hidden")));
void
libcgo_get_scheduler(struct get_scheduler_args *p)
{
p->m = pthread_getspecific(km);
p->g = pthread_getspecific(kg);
}
|