blob: d9acfa8d30697d17393051702207b814dca50cdd [file] [log] [blame]
Rob Pikeaeb43982008-06-21 15:36:23 -07001// Copyright 2009 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#include "runtime.h"
Russ Cox878822f2009-03-24 13:06:51 -07006#include "defs.h"
Russ Cox08cfcd12009-03-24 13:51:48 -07007#include "os.h"
Russ Coxd28acc42008-08-04 16:43:49 -07008
Russ Cox5963dba2010-04-08 18:15:30 -07009extern SigTab sigtab[];
10
Russ Coxd28acc42008-08-04 16:43:49 -070011static void
12unimplemented(int8 *name)
13{
14 prints(name);
15 prints(" not implemented\n");
16 *(int32*)1231 = 1231;
17}
18
Russ Cox376898c2008-09-09 11:50:14 -070019// Thread-safe allocation of a semaphore.
20// Psema points at a kernel semaphore key.
21// It starts out zero, meaning no semaphore.
22// Fill it in, being careful of others calling initsema
23// simultaneously.
24static void
25initsema(uint32 *psema)
26{
27 uint32 sema;
28
29 if(*psema != 0) // already have one
30 return;
31
Russ Cox1f8a40d2009-01-22 16:23:44 -080032 sema = mach_semcreate();
Russ Cox376898c2008-09-09 11:50:14 -070033 if(!cas(psema, 0, sema)){
34 // Someone else filled it in. Use theirs.
Russ Cox1f8a40d2009-01-22 16:23:44 -080035 mach_semdestroy(sema);
Russ Cox376898c2008-09-09 11:50:14 -070036 return;
37 }
38}
39
40
Russ Cox376898c2008-09-09 11:50:14 -070041// Blocking locks.
42
43// Implement Locks, using semaphores.
44// l->key is the number of threads who want the lock.
45// In a race, one thread increments l->key from 0 to 1
46// and the others increment it from >0 to >1. The thread
47// who does the 0->1 increment gets the lock, and the
48// others wait on the semaphore. When the 0->1 thread
49// releases the lock by decrementing l->key, l->key will
50// be >0, so it will increment the semaphore to wake up
51// one of the others. This is the same algorithm used
Russ Cox5ff12f82008-09-24 10:25:28 -070052// in Plan 9's user-level locks.
Russ Cox376898c2008-09-09 11:50:14 -070053
Russ Coxd28acc42008-08-04 16:43:49 -070054void
55lock(Lock *l)
56{
Russ Cox53e69e12009-01-27 14:01:20 -080057 if(m->locks < 0)
58 throw("lock count");
59 m->locks++;
60
Russ Cox0365b982010-01-19 21:14:15 -080061 if(xadd(&l->key, 1) > 1) { // someone else has it; wait
62 // Allocate semaphore if needed.
63 if(l->sema == 0)
64 initsema(&l->sema);
Russ Cox1f8a40d2009-01-22 16:23:44 -080065 mach_semacquire(l->sema);
Russ Cox0365b982010-01-19 21:14:15 -080066 }
Russ Coxd28acc42008-08-04 16:43:49 -070067}
68
69void
70unlock(Lock *l)
71{
Russ Cox1ce17912009-01-26 17:37:05 -080072 m->locks--;
Russ Cox53e69e12009-01-27 14:01:20 -080073 if(m->locks < 0)
74 throw("lock count");
75
Russ Cox0365b982010-01-19 21:14:15 -080076 if(xadd(&l->key, -1) > 0) { // someone else is waiting
77 // Allocate semaphore if needed.
78 if(l->sema == 0)
79 initsema(&l->sema);
Russ Cox1f8a40d2009-01-22 16:23:44 -080080 mach_semrelease(l->sema);
Russ Cox0365b982010-01-19 21:14:15 -080081 }
Russ Coxd28acc42008-08-04 16:43:49 -070082}
83
Russ Cox62d627f2010-02-08 21:41:54 -080084void
85destroylock(Lock *l)
86{
87 if(l->sema != 0) {
88 mach_semdestroy(l->sema);
89 l->sema = 0;
90 }
91}
Russ Cox376898c2008-09-09 11:50:14 -070092
Russ Cox5ff12f82008-09-24 10:25:28 -070093// User-level semaphore implementation:
94// try to do the operations in user space on u,
95// but when it's time to block, fall back on the kernel semaphore k.
96// This is the same algorithm used in Plan 9.
97void
98usemacquire(Usema *s)
99{
Russ Cox0365b982010-01-19 21:14:15 -0800100 if((int32)xadd(&s->u, -1) < 0) {
101 if(s->k == 0)
102 initsema(&s->k);
Russ Cox1f8a40d2009-01-22 16:23:44 -0800103 mach_semacquire(s->k);
Russ Cox0365b982010-01-19 21:14:15 -0800104 }
Russ Cox5ff12f82008-09-24 10:25:28 -0700105}
106
107void
108usemrelease(Usema *s)
109{
Russ Cox0365b982010-01-19 21:14:15 -0800110 if((int32)xadd(&s->u, 1) <= 0) {
111 if(s->k == 0)
112 initsema(&s->k);
Russ Cox1f8a40d2009-01-22 16:23:44 -0800113 mach_semrelease(s->k);
Russ Cox0365b982010-01-19 21:14:15 -0800114 }
Russ Cox5ff12f82008-09-24 10:25:28 -0700115}
116
117
Russ Cox376898c2008-09-09 11:50:14 -0700118// Event notifications.
Russ Coxd28acc42008-08-04 16:43:49 -0700119void
Russ Cox96824002008-08-05 14:18:47 -0700120noteclear(Note *n)
Russ Coxd28acc42008-08-04 16:43:49 -0700121{
Russ Cox376898c2008-09-09 11:50:14 -0700122 n->wakeup = 0;
Russ Coxd28acc42008-08-04 16:43:49 -0700123}
124
125void
Russ Cox96824002008-08-05 14:18:47 -0700126notesleep(Note *n)
Russ Coxd28acc42008-08-04 16:43:49 -0700127{
Russ Cox376898c2008-09-09 11:50:14 -0700128 while(!n->wakeup)
Russ Cox5ff12f82008-09-24 10:25:28 -0700129 usemacquire(&n->sema);
Russ Coxd28acc42008-08-04 16:43:49 -0700130}
131
132void
Russ Cox96824002008-08-05 14:18:47 -0700133notewakeup(Note *n)
Russ Coxd28acc42008-08-04 16:43:49 -0700134{
Russ Cox376898c2008-09-09 11:50:14 -0700135 n->wakeup = 1;
Russ Cox5ff12f82008-09-24 10:25:28 -0700136 usemrelease(&n->sema);
Russ Cox376898c2008-09-09 11:50:14 -0700137}
138
139
140// BSD interface for threading.
141void
142osinit(void)
143{
Russ Cox133a1582009-10-03 10:37:12 -0700144 // Register our thread-creation callback (see {amd64,386}/sys.s)
145 // but only if we're not using cgo. If we are using cgo we need
146 // to let the C pthread libary install its own thread-creation callback.
147 extern void (*libcgo_thread_start)(void*);
148 if(libcgo_thread_start == nil)
149 bsdthread_register();
Russ Coxd28acc42008-08-04 16:43:49 -0700150}
151
152void
Russ Cox376898c2008-09-09 11:50:14 -0700153newosproc(M *m, G *g, void *stk, void (*fn)(void))
Russ Coxd28acc42008-08-04 16:43:49 -0700154{
Russ Cox3a0df4c2009-06-04 11:16:03 -0700155 m->tls[0] = m->id; // so 386 asm can find it
Russ Cox8522a472009-06-17 15:15:55 -0700156 if(0){
157 printf("newosproc stk=%p m=%p g=%p fn=%p id=%d/%d ostk=%p\n",
158 stk, m, g, fn, m->id, m->tls[0], &m);
159 }
Russ Cox376898c2008-09-09 11:50:14 -0700160 bsdthread_create(stk, m, g, fn);
Russ Coxd28acc42008-08-04 16:43:49 -0700161}
162
Russ Coxa67258f2008-09-18 15:56:46 -0700163// Called to initialize a new m (including the bootstrap m).
164void
165minit(void)
166{
167 // Initialize signal handling.
168 m->gsignal = malg(32*1024); // OS X wants >=8K, Linux >=2K
169 signalstack(m->gsignal->stackguard, 32*1024);
170}
171
Russ Cox376898c2008-09-09 11:50:14 -0700172// Mach IPC, to get at semaphores
173// Definitions are in /usr/include/mach on a Mac.
174
175static void
Russ Cox08cfcd12009-03-24 13:51:48 -0700176macherror(int32 r, int8 *fn)
Russ Coxd28acc42008-08-04 16:43:49 -0700177{
Russ Cox4702c0e2009-03-31 15:45:12 -0700178 printf("mach error %s: %d\n", fn, r);
Russ Cox376898c2008-09-09 11:50:14 -0700179 throw("mach error");
180}
181
182enum
183{
184 DebugMach = 0
185};
186
Russ Cox08cfcd12009-03-24 13:51:48 -0700187static MachNDR zerondr;
Russ Cox376898c2008-09-09 11:50:14 -0700188
189#define MACH_MSGH_BITS(a, b) ((a) | ((b)<<8))
190
Russ Cox08cfcd12009-03-24 13:51:48 -0700191static int32
192mach_msg(MachHeader *h,
193 int32 op,
194 uint32 send_size,
195 uint32 rcv_size,
196 uint32 rcv_name,
197 uint32 timeout,
198 uint32 notify)
Russ Cox376898c2008-09-09 11:50:14 -0700199{
200 // TODO: Loop on interrupt.
201 return mach_msg_trap(h, op, send_size, rcv_size, rcv_name, timeout, notify);
202}
203
Russ Cox376898c2008-09-09 11:50:14 -0700204// Mach RPC (MIG)
Russ Cox376898c2008-09-09 11:50:14 -0700205
206enum
207{
208 MinMachMsg = 48,
209 Reply = 100,
210};
211
212#pragma pack on
213typedef struct CodeMsg CodeMsg;
214struct CodeMsg
215{
Russ Cox08cfcd12009-03-24 13:51:48 -0700216 MachHeader h;
217 MachNDR NDR;
218 int32 code;
Russ Cox376898c2008-09-09 11:50:14 -0700219};
220#pragma pack off
221
Russ Cox08cfcd12009-03-24 13:51:48 -0700222static int32
223machcall(MachHeader *h, int32 maxsize, int32 rxsize)
Russ Cox376898c2008-09-09 11:50:14 -0700224{
225 uint32 *p;
226 int32 i, ret, id;
Russ Cox08cfcd12009-03-24 13:51:48 -0700227 uint32 port;
Russ Cox376898c2008-09-09 11:50:14 -0700228 CodeMsg *c;
229
230 if((port = m->machport) == 0){
231 port = mach_reply_port();
232 m->machport = port;
233 }
234
Russ Cox08cfcd12009-03-24 13:51:48 -0700235 h->msgh_bits |= MACH_MSGH_BITS(MACH_MSG_TYPE_COPY_SEND, MACH_MSG_TYPE_MAKE_SEND_ONCE);
236 h->msgh_local_port = port;
237 h->msgh_reserved = 0;
238 id = h->msgh_id;
Russ Cox376898c2008-09-09 11:50:14 -0700239
240 if(DebugMach){
241 p = (uint32*)h;
242 prints("send:\t");
Russ Cox08cfcd12009-03-24 13:51:48 -0700243 for(i=0; i<h->msgh_size/sizeof(p[0]); i++){
Russ Cox376898c2008-09-09 11:50:14 -0700244 prints(" ");
Russ Cox718be322010-01-25 18:52:55 -0800245 ·printpointer((void*)p[i]);
Russ Cox376898c2008-09-09 11:50:14 -0700246 if(i%8 == 7)
247 prints("\n\t");
248 }
249 if(i%8)
250 prints("\n");
251 }
252
253 ret = mach_msg(h, MACH_SEND_MSG|MACH_RCV_MSG,
Russ Cox08cfcd12009-03-24 13:51:48 -0700254 h->msgh_size, maxsize, port, 0, 0);
Russ Cox376898c2008-09-09 11:50:14 -0700255 if(ret != 0){
256 if(DebugMach){
257 prints("mach_msg error ");
Russ Cox718be322010-01-25 18:52:55 -0800258 ·printint(ret);
Russ Cox376898c2008-09-09 11:50:14 -0700259 prints("\n");
260 }
261 return ret;
262 }
263
264 if(DebugMach){
265 p = (uint32*)h;
266 prints("recv:\t");
Russ Cox08cfcd12009-03-24 13:51:48 -0700267 for(i=0; i<h->msgh_size/sizeof(p[0]); i++){
Russ Cox376898c2008-09-09 11:50:14 -0700268 prints(" ");
Russ Cox718be322010-01-25 18:52:55 -0800269 ·printpointer((void*)p[i]);
Russ Cox376898c2008-09-09 11:50:14 -0700270 if(i%8 == 7)
271 prints("\n\t");
272 }
273 if(i%8)
274 prints("\n");
275 }
276
Russ Cox08cfcd12009-03-24 13:51:48 -0700277 if(h->msgh_id != id+Reply){
Russ Cox376898c2008-09-09 11:50:14 -0700278 if(DebugMach){
279 prints("mach_msg reply id mismatch ");
Russ Cox718be322010-01-25 18:52:55 -0800280 ·printint(h->msgh_id);
Russ Cox376898c2008-09-09 11:50:14 -0700281 prints(" != ");
Russ Cox718be322010-01-25 18:52:55 -0800282 ·printint(id+Reply);
Russ Cox376898c2008-09-09 11:50:14 -0700283 prints("\n");
284 }
285 return -303; // MIG_REPLY_MISMATCH
286 }
287
288 // Look for a response giving the return value.
289 // Any call can send this back with an error,
290 // and some calls only have return values so they
291 // send it back on success too. I don't quite see how
292 // you know it's one of these and not the full response
293 // format, so just look if the message is right.
294 c = (CodeMsg*)h;
Russ Cox08cfcd12009-03-24 13:51:48 -0700295 if(h->msgh_size == sizeof(CodeMsg)
296 && !(h->msgh_bits & MACH_MSGH_BITS_COMPLEX)){
Russ Cox376898c2008-09-09 11:50:14 -0700297 if(DebugMach){
298 prints("mig result ");
Russ Cox718be322010-01-25 18:52:55 -0800299 ·printint(c->code);
Russ Cox376898c2008-09-09 11:50:14 -0700300 prints("\n");
301 }
302 return c->code;
303 }
304
Russ Cox08cfcd12009-03-24 13:51:48 -0700305 if(h->msgh_size != rxsize){
Russ Cox376898c2008-09-09 11:50:14 -0700306 if(DebugMach){
307 prints("mach_msg reply size mismatch ");
Russ Cox718be322010-01-25 18:52:55 -0800308 ·printint(h->msgh_size);
Russ Cox376898c2008-09-09 11:50:14 -0700309 prints(" != ");
Russ Cox718be322010-01-25 18:52:55 -0800310 ·printint(rxsize);
Russ Cox376898c2008-09-09 11:50:14 -0700311 prints("\n");
312 }
313 return -307; // MIG_ARRAY_TOO_LARGE
314 }
315
Russ Coxd28acc42008-08-04 16:43:49 -0700316 return 0;
317}
Russ Cox376898c2008-09-09 11:50:14 -0700318
319
320// Semaphores!
321
322enum
323{
Russ Cox1f8a40d2009-01-22 16:23:44 -0800324 Tmach_semcreate = 3418,
325 Rmach_semcreate = Tmach_semcreate + Reply,
Russ Cox376898c2008-09-09 11:50:14 -0700326
Russ Cox1f8a40d2009-01-22 16:23:44 -0800327 Tmach_semdestroy = 3419,
328 Rmach_semdestroy = Tmach_semdestroy + Reply,
Russ Coxcd800002009-06-08 14:03:21 -0700329
Russ Cox925183c2009-06-08 14:09:04 -0700330 // Mach calls that get interrupted by Unix signals
331 // return this error code. We retry them.
Russ Coxcd800002009-06-08 14:03:21 -0700332 KERN_ABORTED = 14,
Russ Cox376898c2008-09-09 11:50:14 -0700333};
334
Russ Cox1f8a40d2009-01-22 16:23:44 -0800335typedef struct Tmach_semcreateMsg Tmach_semcreateMsg;
336typedef struct Rmach_semcreateMsg Rmach_semcreateMsg;
337typedef struct Tmach_semdestroyMsg Tmach_semdestroyMsg;
338// Rmach_semdestroyMsg = CodeMsg
Russ Cox376898c2008-09-09 11:50:14 -0700339
340#pragma pack on
Russ Cox1f8a40d2009-01-22 16:23:44 -0800341struct Tmach_semcreateMsg
Russ Cox376898c2008-09-09 11:50:14 -0700342{
Russ Cox08cfcd12009-03-24 13:51:48 -0700343 MachHeader h;
344 MachNDR ndr;
Russ Cox376898c2008-09-09 11:50:14 -0700345 int32 policy;
346 int32 value;
347};
348
Russ Cox1f8a40d2009-01-22 16:23:44 -0800349struct Rmach_semcreateMsg
Russ Cox376898c2008-09-09 11:50:14 -0700350{
Russ Cox08cfcd12009-03-24 13:51:48 -0700351 MachHeader h;
352 MachBody body;
353 MachPort semaphore;
Russ Cox376898c2008-09-09 11:50:14 -0700354};
355
Russ Cox1f8a40d2009-01-22 16:23:44 -0800356struct Tmach_semdestroyMsg
Russ Cox376898c2008-09-09 11:50:14 -0700357{
Russ Cox08cfcd12009-03-24 13:51:48 -0700358 MachHeader h;
359 MachBody body;
360 MachPort semaphore;
Russ Cox376898c2008-09-09 11:50:14 -0700361};
362#pragma pack off
363
Russ Cox08cfcd12009-03-24 13:51:48 -0700364uint32
Russ Cox1f8a40d2009-01-22 16:23:44 -0800365mach_semcreate(void)
Russ Cox376898c2008-09-09 11:50:14 -0700366{
367 union {
Russ Cox1f8a40d2009-01-22 16:23:44 -0800368 Tmach_semcreateMsg tx;
369 Rmach_semcreateMsg rx;
Russ Cox376898c2008-09-09 11:50:14 -0700370 uint8 pad[MinMachMsg];
371 } m;
Russ Cox08cfcd12009-03-24 13:51:48 -0700372 int32 r;
Russ Cox376898c2008-09-09 11:50:14 -0700373
Russ Cox08cfcd12009-03-24 13:51:48 -0700374 m.tx.h.msgh_bits = 0;
375 m.tx.h.msgh_size = sizeof(m.tx);
376 m.tx.h.msgh_remote_port = mach_task_self();
377 m.tx.h.msgh_id = Tmach_semcreate;
Russ Cox376898c2008-09-09 11:50:14 -0700378 m.tx.ndr = zerondr;
379
380 m.tx.policy = 0; // 0 = SYNC_POLICY_FIFO
381 m.tx.value = 0;
382
Russ Coxcd800002009-06-08 14:03:21 -0700383 while((r = machcall(&m.tx.h, sizeof m, sizeof(m.rx))) != 0){
384 if(r == KERN_ABORTED) // interrupted
385 continue;
Russ Cox376898c2008-09-09 11:50:14 -0700386 macherror(r, "semaphore_create");
Russ Coxcd800002009-06-08 14:03:21 -0700387 }
Russ Cox08cfcd12009-03-24 13:51:48 -0700388 if(m.rx.body.msgh_descriptor_count != 1)
Russ Cox1f8a40d2009-01-22 16:23:44 -0800389 unimplemented("mach_semcreate desc count");
Russ Cox376898c2008-09-09 11:50:14 -0700390 return m.rx.semaphore.name;
391}
392
393void
Russ Cox08cfcd12009-03-24 13:51:48 -0700394mach_semdestroy(uint32 sem)
Russ Cox376898c2008-09-09 11:50:14 -0700395{
396 union {
Russ Cox1f8a40d2009-01-22 16:23:44 -0800397 Tmach_semdestroyMsg tx;
Russ Cox376898c2008-09-09 11:50:14 -0700398 uint8 pad[MinMachMsg];
399 } m;
Russ Cox08cfcd12009-03-24 13:51:48 -0700400 int32 r;
Russ Cox376898c2008-09-09 11:50:14 -0700401
Russ Cox08cfcd12009-03-24 13:51:48 -0700402 m.tx.h.msgh_bits = MACH_MSGH_BITS_COMPLEX;
403 m.tx.h.msgh_size = sizeof(m.tx);
404 m.tx.h.msgh_remote_port = mach_task_self();
405 m.tx.h.msgh_id = Tmach_semdestroy;
406 m.tx.body.msgh_descriptor_count = 1;
Russ Cox376898c2008-09-09 11:50:14 -0700407 m.tx.semaphore.name = sem;
408 m.tx.semaphore.disposition = MACH_MSG_TYPE_MOVE_SEND;
409 m.tx.semaphore.type = 0;
410
Russ Coxcd800002009-06-08 14:03:21 -0700411 while((r = machcall(&m.tx.h, sizeof m, 0)) != 0){
Russ Cox3ff5e722009-07-27 10:59:59 -0700412 if(r == KERN_ABORTED) // interrupted
413 continue;
Russ Cox376898c2008-09-09 11:50:14 -0700414 macherror(r, "semaphore_destroy");
Russ Coxcd800002009-06-08 14:03:21 -0700415 }
Russ Cox376898c2008-09-09 11:50:14 -0700416}
417
Russ Cox08cfcd12009-03-24 13:51:48 -0700418// The other calls have simple system call traps in sys.s
419int32 mach_semaphore_wait(uint32 sema);
420int32 mach_semaphore_timedwait(uint32 sema, uint32 sec, uint32 nsec);
421int32 mach_semaphore_signal(uint32 sema);
422int32 mach_semaphore_signal_all(uint32 sema);
Russ Cox376898c2008-09-09 11:50:14 -0700423
424void
Russ Cox08cfcd12009-03-24 13:51:48 -0700425mach_semacquire(uint32 sem)
Russ Cox376898c2008-09-09 11:50:14 -0700426{
Russ Cox08cfcd12009-03-24 13:51:48 -0700427 int32 r;
Russ Cox376898c2008-09-09 11:50:14 -0700428
Russ Coxcd800002009-06-08 14:03:21 -0700429 while((r = mach_semaphore_wait(sem)) != 0) {
430 if(r == KERN_ABORTED) // interrupted
431 continue;
Russ Cox376898c2008-09-09 11:50:14 -0700432 macherror(r, "semaphore_wait");
Russ Coxcd800002009-06-08 14:03:21 -0700433 }
Russ Cox376898c2008-09-09 11:50:14 -0700434}
435
436void
Russ Cox08cfcd12009-03-24 13:51:48 -0700437mach_semrelease(uint32 sem)
Russ Cox376898c2008-09-09 11:50:14 -0700438{
Russ Cox08cfcd12009-03-24 13:51:48 -0700439 int32 r;
Russ Cox376898c2008-09-09 11:50:14 -0700440
Russ Coxcd800002009-06-08 14:03:21 -0700441 while((r = mach_semaphore_signal(sem)) != 0) {
442 if(r == KERN_ABORTED) // interrupted
443 continue;
Russ Cox376898c2008-09-09 11:50:14 -0700444 macherror(r, "semaphore_signal");
Russ Coxcd800002009-06-08 14:03:21 -0700445 }
Russ Cox376898c2008-09-09 11:50:14 -0700446}
447
Russ Cox5963dba2010-04-08 18:15:30 -0700448void
449sigpanic(void)
450{
451 switch(g->sig) {
452 case SIGBUS:
453 if(g->sigcode0 == BUS_ADRERR && g->sigcode1 < 0x1000)
454 panicstring("invalid memory address or nil pointer dereference");
455 break;
456 case SIGSEGV:
457 if((g->sigcode0 == 0 || g->sigcode0 == SEGV_MAPERR) && g->sigcode1 < 0x1000)
458 panicstring("invalid memory address or nil pointer dereference");
459 break;
460 case SIGFPE:
461 switch(g->sigcode0) {
462 case FPE_INTDIV:
463 panicstring("integer divide by zero");
464 case FPE_INTOVF:
465 panicstring("integer overflow");
466 }
467 panicstring("floating point error");
468 }
469 panicstring(sigtab[g->sig].name);
470}