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
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
|
#include "cpython/pthread_stubs.h"
// mutex
int
pthread_mutex_init(pthread_mutex_t *restrict mutex,
const pthread_mutexattr_t *restrict attr)
{
return 0;
}
int
pthread_mutex_destroy(pthread_mutex_t *mutex)
{
return 0;
}
int
pthread_mutex_trylock(pthread_mutex_t *mutex)
{
return 0;
}
int
pthread_mutex_lock(pthread_mutex_t *mutex)
{
return 0;
}
int
pthread_mutex_unlock(pthread_mutex_t *mutex)
{
return 0;
}
// condition
int
pthread_cond_init(pthread_cond_t *restrict cond,
const pthread_condattr_t *restrict attr)
{
return 0;
}
PyAPI_FUNC(int) pthread_cond_destroy(pthread_cond_t *cond)
{
return 0;
}
int
pthread_cond_wait(pthread_cond_t *restrict cond,
pthread_mutex_t *restrict mutex)
{
return 0;
}
int
pthread_cond_timedwait(pthread_cond_t *restrict cond,
pthread_mutex_t *restrict mutex,
const struct timespec *restrict abstime)
{
return 0;
}
int
pthread_cond_signal(pthread_cond_t *cond)
{
return 0;
}
int
pthread_condattr_init(pthread_condattr_t *attr)
{
return 0;
}
int
pthread_condattr_setclock(pthread_condattr_t *attr, clockid_t clock_id)
{
return 0;
}
// pthread
int
pthread_create(pthread_t *restrict thread,
const pthread_attr_t *restrict attr,
void *(*start_routine)(void *),
void *restrict arg)
{
return EAGAIN;
}
int
pthread_detach(pthread_t thread)
{
return 0;
}
int
pthread_join(pthread_t thread, void** value_ptr)
{
if (value_ptr) {
*value_ptr = NULL;
}
return 0;
}
PyAPI_FUNC(pthread_t) pthread_self(void)
{
return 0;
}
int
pthread_exit(void *retval)
{
exit(0);
}
int
pthread_attr_init(pthread_attr_t *attr)
{
return 0;
}
int
pthread_attr_setstacksize(
pthread_attr_t *attr, size_t stacksize)
{
return 0;
}
int
pthread_attr_destroy(pthread_attr_t *attr)
{
return 0;
}
typedef struct py_stub_tls_entry py_tls_entry;
#define py_tls_entries (_PyRuntime.threads.stubs.tls_entries)
int
pthread_key_create(pthread_key_t *key, void (*destr_function)(void *))
{
if (!key) {
return EINVAL;
}
if (destr_function != NULL) {
Py_FatalError("pthread_key_create destructor is not supported");
}
for (pthread_key_t idx = 0; idx < PTHREAD_KEYS_MAX; idx++) {
if (!py_tls_entries[idx].in_use) {
py_tls_entries[idx].in_use = true;
*key = idx;
return 0;
}
}
return EAGAIN;
}
int
pthread_key_delete(pthread_key_t key)
{
if (key < 0 || key >= PTHREAD_KEYS_MAX || !py_tls_entries[key].in_use) {
return EINVAL;
}
py_tls_entries[key].in_use = false;
py_tls_entries[key].value = NULL;
return 0;
}
void *
pthread_getspecific(pthread_key_t key) {
if (key < 0 || key >= PTHREAD_KEYS_MAX || !py_tls_entries[key].in_use) {
return NULL;
}
return py_tls_entries[key].value;
}
int
pthread_setspecific(pthread_key_t key, const void *value)
{
if (key < 0 || key >= PTHREAD_KEYS_MAX || !py_tls_entries[key].in_use) {
return EINVAL;
}
py_tls_entries[key].value = (void *)value;
return 0;
}
// let thread_pthread define the Python API
#include "thread_pthread.h"
|