Rob Pike | aeb4398 | 2008-06-21 15:36:23 -0700 | [diff] [blame] | 1 | // 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 Cox | 878822f | 2009-03-24 13:06:51 -0700 | [diff] [blame] | 6 | #include "defs.h" |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 7 | #include "os.h" |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 8 | |
Russ Cox | 5963dba | 2010-04-08 18:15:30 -0700 | [diff] [blame^] | 9 | extern SigTab sigtab[]; |
| 10 | |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 11 | static void |
| 12 | unimplemented(int8 *name) |
| 13 | { |
| 14 | prints(name); |
| 15 | prints(" not implemented\n"); |
| 16 | *(int32*)1231 = 1231; |
| 17 | } |
| 18 | |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 19 | // 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. |
| 24 | static void |
| 25 | initsema(uint32 *psema) |
| 26 | { |
| 27 | uint32 sema; |
| 28 | |
| 29 | if(*psema != 0) // already have one |
| 30 | return; |
| 31 | |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 32 | sema = mach_semcreate(); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 33 | if(!cas(psema, 0, sema)){ |
| 34 | // Someone else filled it in. Use theirs. |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 35 | mach_semdestroy(sema); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 36 | return; |
| 37 | } |
| 38 | } |
| 39 | |
| 40 | |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 41 | // 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 Cox | 5ff12f8 | 2008-09-24 10:25:28 -0700 | [diff] [blame] | 52 | // in Plan 9's user-level locks. |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 53 | |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 54 | void |
| 55 | lock(Lock *l) |
| 56 | { |
Russ Cox | 53e69e1 | 2009-01-27 14:01:20 -0800 | [diff] [blame] | 57 | if(m->locks < 0) |
| 58 | throw("lock count"); |
| 59 | m->locks++; |
| 60 | |
Russ Cox | 0365b98 | 2010-01-19 21:14:15 -0800 | [diff] [blame] | 61 | 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 Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 65 | mach_semacquire(l->sema); |
Russ Cox | 0365b98 | 2010-01-19 21:14:15 -0800 | [diff] [blame] | 66 | } |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 67 | } |
| 68 | |
| 69 | void |
| 70 | unlock(Lock *l) |
| 71 | { |
Russ Cox | 1ce1791 | 2009-01-26 17:37:05 -0800 | [diff] [blame] | 72 | m->locks--; |
Russ Cox | 53e69e1 | 2009-01-27 14:01:20 -0800 | [diff] [blame] | 73 | if(m->locks < 0) |
| 74 | throw("lock count"); |
| 75 | |
Russ Cox | 0365b98 | 2010-01-19 21:14:15 -0800 | [diff] [blame] | 76 | 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 Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 80 | mach_semrelease(l->sema); |
Russ Cox | 0365b98 | 2010-01-19 21:14:15 -0800 | [diff] [blame] | 81 | } |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 82 | } |
| 83 | |
Russ Cox | 62d627f | 2010-02-08 21:41:54 -0800 | [diff] [blame] | 84 | void |
| 85 | destroylock(Lock *l) |
| 86 | { |
| 87 | if(l->sema != 0) { |
| 88 | mach_semdestroy(l->sema); |
| 89 | l->sema = 0; |
| 90 | } |
| 91 | } |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 92 | |
Russ Cox | 5ff12f8 | 2008-09-24 10:25:28 -0700 | [diff] [blame] | 93 | // 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. |
| 97 | void |
| 98 | usemacquire(Usema *s) |
| 99 | { |
Russ Cox | 0365b98 | 2010-01-19 21:14:15 -0800 | [diff] [blame] | 100 | if((int32)xadd(&s->u, -1) < 0) { |
| 101 | if(s->k == 0) |
| 102 | initsema(&s->k); |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 103 | mach_semacquire(s->k); |
Russ Cox | 0365b98 | 2010-01-19 21:14:15 -0800 | [diff] [blame] | 104 | } |
Russ Cox | 5ff12f8 | 2008-09-24 10:25:28 -0700 | [diff] [blame] | 105 | } |
| 106 | |
| 107 | void |
| 108 | usemrelease(Usema *s) |
| 109 | { |
Russ Cox | 0365b98 | 2010-01-19 21:14:15 -0800 | [diff] [blame] | 110 | if((int32)xadd(&s->u, 1) <= 0) { |
| 111 | if(s->k == 0) |
| 112 | initsema(&s->k); |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 113 | mach_semrelease(s->k); |
Russ Cox | 0365b98 | 2010-01-19 21:14:15 -0800 | [diff] [blame] | 114 | } |
Russ Cox | 5ff12f8 | 2008-09-24 10:25:28 -0700 | [diff] [blame] | 115 | } |
| 116 | |
| 117 | |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 118 | // Event notifications. |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 119 | void |
Russ Cox | 9682400 | 2008-08-05 14:18:47 -0700 | [diff] [blame] | 120 | noteclear(Note *n) |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 121 | { |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 122 | n->wakeup = 0; |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 123 | } |
| 124 | |
| 125 | void |
Russ Cox | 9682400 | 2008-08-05 14:18:47 -0700 | [diff] [blame] | 126 | notesleep(Note *n) |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 127 | { |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 128 | while(!n->wakeup) |
Russ Cox | 5ff12f8 | 2008-09-24 10:25:28 -0700 | [diff] [blame] | 129 | usemacquire(&n->sema); |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 130 | } |
| 131 | |
| 132 | void |
Russ Cox | 9682400 | 2008-08-05 14:18:47 -0700 | [diff] [blame] | 133 | notewakeup(Note *n) |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 134 | { |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 135 | n->wakeup = 1; |
Russ Cox | 5ff12f8 | 2008-09-24 10:25:28 -0700 | [diff] [blame] | 136 | usemrelease(&n->sema); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 137 | } |
| 138 | |
| 139 | |
| 140 | // BSD interface for threading. |
| 141 | void |
| 142 | osinit(void) |
| 143 | { |
Russ Cox | 133a158 | 2009-10-03 10:37:12 -0700 | [diff] [blame] | 144 | // 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 Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 150 | } |
| 151 | |
| 152 | void |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 153 | newosproc(M *m, G *g, void *stk, void (*fn)(void)) |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 154 | { |
Russ Cox | 3a0df4c | 2009-06-04 11:16:03 -0700 | [diff] [blame] | 155 | m->tls[0] = m->id; // so 386 asm can find it |
Russ Cox | 8522a47 | 2009-06-17 15:15:55 -0700 | [diff] [blame] | 156 | 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 Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 160 | bsdthread_create(stk, m, g, fn); |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 161 | } |
| 162 | |
Russ Cox | a67258f | 2008-09-18 15:56:46 -0700 | [diff] [blame] | 163 | // Called to initialize a new m (including the bootstrap m). |
| 164 | void |
| 165 | minit(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 Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 172 | // Mach IPC, to get at semaphores |
| 173 | // Definitions are in /usr/include/mach on a Mac. |
| 174 | |
| 175 | static void |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 176 | macherror(int32 r, int8 *fn) |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 177 | { |
Russ Cox | 4702c0e | 2009-03-31 15:45:12 -0700 | [diff] [blame] | 178 | printf("mach error %s: %d\n", fn, r); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 179 | throw("mach error"); |
| 180 | } |
| 181 | |
| 182 | enum |
| 183 | { |
| 184 | DebugMach = 0 |
| 185 | }; |
| 186 | |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 187 | static MachNDR zerondr; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 188 | |
| 189 | #define MACH_MSGH_BITS(a, b) ((a) | ((b)<<8)) |
| 190 | |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 191 | static int32 |
| 192 | mach_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 Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 199 | { |
| 200 | // TODO: Loop on interrupt. |
| 201 | return mach_msg_trap(h, op, send_size, rcv_size, rcv_name, timeout, notify); |
| 202 | } |
| 203 | |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 204 | // Mach RPC (MIG) |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 205 | |
| 206 | enum |
| 207 | { |
| 208 | MinMachMsg = 48, |
| 209 | Reply = 100, |
| 210 | }; |
| 211 | |
| 212 | #pragma pack on |
| 213 | typedef struct CodeMsg CodeMsg; |
| 214 | struct CodeMsg |
| 215 | { |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 216 | MachHeader h; |
| 217 | MachNDR NDR; |
| 218 | int32 code; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 219 | }; |
| 220 | #pragma pack off |
| 221 | |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 222 | static int32 |
| 223 | machcall(MachHeader *h, int32 maxsize, int32 rxsize) |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 224 | { |
| 225 | uint32 *p; |
| 226 | int32 i, ret, id; |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 227 | uint32 port; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 228 | CodeMsg *c; |
| 229 | |
| 230 | if((port = m->machport) == 0){ |
| 231 | port = mach_reply_port(); |
| 232 | m->machport = port; |
| 233 | } |
| 234 | |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 235 | 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 Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 239 | |
| 240 | if(DebugMach){ |
| 241 | p = (uint32*)h; |
| 242 | prints("send:\t"); |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 243 | for(i=0; i<h->msgh_size/sizeof(p[0]); i++){ |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 244 | prints(" "); |
Russ Cox | 718be32 | 2010-01-25 18:52:55 -0800 | [diff] [blame] | 245 | ·printpointer((void*)p[i]); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 246 | 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 Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 254 | h->msgh_size, maxsize, port, 0, 0); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 255 | if(ret != 0){ |
| 256 | if(DebugMach){ |
| 257 | prints("mach_msg error "); |
Russ Cox | 718be32 | 2010-01-25 18:52:55 -0800 | [diff] [blame] | 258 | ·printint(ret); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 259 | prints("\n"); |
| 260 | } |
| 261 | return ret; |
| 262 | } |
| 263 | |
| 264 | if(DebugMach){ |
| 265 | p = (uint32*)h; |
| 266 | prints("recv:\t"); |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 267 | for(i=0; i<h->msgh_size/sizeof(p[0]); i++){ |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 268 | prints(" "); |
Russ Cox | 718be32 | 2010-01-25 18:52:55 -0800 | [diff] [blame] | 269 | ·printpointer((void*)p[i]); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 270 | if(i%8 == 7) |
| 271 | prints("\n\t"); |
| 272 | } |
| 273 | if(i%8) |
| 274 | prints("\n"); |
| 275 | } |
| 276 | |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 277 | if(h->msgh_id != id+Reply){ |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 278 | if(DebugMach){ |
| 279 | prints("mach_msg reply id mismatch "); |
Russ Cox | 718be32 | 2010-01-25 18:52:55 -0800 | [diff] [blame] | 280 | ·printint(h->msgh_id); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 281 | prints(" != "); |
Russ Cox | 718be32 | 2010-01-25 18:52:55 -0800 | [diff] [blame] | 282 | ·printint(id+Reply); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 283 | 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 Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 295 | if(h->msgh_size == sizeof(CodeMsg) |
| 296 | && !(h->msgh_bits & MACH_MSGH_BITS_COMPLEX)){ |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 297 | if(DebugMach){ |
| 298 | prints("mig result "); |
Russ Cox | 718be32 | 2010-01-25 18:52:55 -0800 | [diff] [blame] | 299 | ·printint(c->code); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 300 | prints("\n"); |
| 301 | } |
| 302 | return c->code; |
| 303 | } |
| 304 | |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 305 | if(h->msgh_size != rxsize){ |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 306 | if(DebugMach){ |
| 307 | prints("mach_msg reply size mismatch "); |
Russ Cox | 718be32 | 2010-01-25 18:52:55 -0800 | [diff] [blame] | 308 | ·printint(h->msgh_size); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 309 | prints(" != "); |
Russ Cox | 718be32 | 2010-01-25 18:52:55 -0800 | [diff] [blame] | 310 | ·printint(rxsize); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 311 | prints("\n"); |
| 312 | } |
| 313 | return -307; // MIG_ARRAY_TOO_LARGE |
| 314 | } |
| 315 | |
Russ Cox | d28acc4 | 2008-08-04 16:43:49 -0700 | [diff] [blame] | 316 | return 0; |
| 317 | } |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 318 | |
| 319 | |
| 320 | // Semaphores! |
| 321 | |
| 322 | enum |
| 323 | { |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 324 | Tmach_semcreate = 3418, |
| 325 | Rmach_semcreate = Tmach_semcreate + Reply, |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 326 | |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 327 | Tmach_semdestroy = 3419, |
| 328 | Rmach_semdestroy = Tmach_semdestroy + Reply, |
Russ Cox | cd80000 | 2009-06-08 14:03:21 -0700 | [diff] [blame] | 329 | |
Russ Cox | 925183c | 2009-06-08 14:09:04 -0700 | [diff] [blame] | 330 | // Mach calls that get interrupted by Unix signals |
| 331 | // return this error code. We retry them. |
Russ Cox | cd80000 | 2009-06-08 14:03:21 -0700 | [diff] [blame] | 332 | KERN_ABORTED = 14, |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 333 | }; |
| 334 | |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 335 | typedef struct Tmach_semcreateMsg Tmach_semcreateMsg; |
| 336 | typedef struct Rmach_semcreateMsg Rmach_semcreateMsg; |
| 337 | typedef struct Tmach_semdestroyMsg Tmach_semdestroyMsg; |
| 338 | // Rmach_semdestroyMsg = CodeMsg |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 339 | |
| 340 | #pragma pack on |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 341 | struct Tmach_semcreateMsg |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 342 | { |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 343 | MachHeader h; |
| 344 | MachNDR ndr; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 345 | int32 policy; |
| 346 | int32 value; |
| 347 | }; |
| 348 | |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 349 | struct Rmach_semcreateMsg |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 350 | { |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 351 | MachHeader h; |
| 352 | MachBody body; |
| 353 | MachPort semaphore; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 354 | }; |
| 355 | |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 356 | struct Tmach_semdestroyMsg |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 357 | { |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 358 | MachHeader h; |
| 359 | MachBody body; |
| 360 | MachPort semaphore; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 361 | }; |
| 362 | #pragma pack off |
| 363 | |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 364 | uint32 |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 365 | mach_semcreate(void) |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 366 | { |
| 367 | union { |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 368 | Tmach_semcreateMsg tx; |
| 369 | Rmach_semcreateMsg rx; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 370 | uint8 pad[MinMachMsg]; |
| 371 | } m; |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 372 | int32 r; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 373 | |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 374 | 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 Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 378 | m.tx.ndr = zerondr; |
| 379 | |
| 380 | m.tx.policy = 0; // 0 = SYNC_POLICY_FIFO |
| 381 | m.tx.value = 0; |
| 382 | |
Russ Cox | cd80000 | 2009-06-08 14:03:21 -0700 | [diff] [blame] | 383 | while((r = machcall(&m.tx.h, sizeof m, sizeof(m.rx))) != 0){ |
| 384 | if(r == KERN_ABORTED) // interrupted |
| 385 | continue; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 386 | macherror(r, "semaphore_create"); |
Russ Cox | cd80000 | 2009-06-08 14:03:21 -0700 | [diff] [blame] | 387 | } |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 388 | if(m.rx.body.msgh_descriptor_count != 1) |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 389 | unimplemented("mach_semcreate desc count"); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 390 | return m.rx.semaphore.name; |
| 391 | } |
| 392 | |
| 393 | void |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 394 | mach_semdestroy(uint32 sem) |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 395 | { |
| 396 | union { |
Russ Cox | 1f8a40d | 2009-01-22 16:23:44 -0800 | [diff] [blame] | 397 | Tmach_semdestroyMsg tx; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 398 | uint8 pad[MinMachMsg]; |
| 399 | } m; |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 400 | int32 r; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 401 | |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 402 | 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 Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 407 | m.tx.semaphore.name = sem; |
| 408 | m.tx.semaphore.disposition = MACH_MSG_TYPE_MOVE_SEND; |
| 409 | m.tx.semaphore.type = 0; |
| 410 | |
Russ Cox | cd80000 | 2009-06-08 14:03:21 -0700 | [diff] [blame] | 411 | while((r = machcall(&m.tx.h, sizeof m, 0)) != 0){ |
Russ Cox | 3ff5e72 | 2009-07-27 10:59:59 -0700 | [diff] [blame] | 412 | if(r == KERN_ABORTED) // interrupted |
| 413 | continue; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 414 | macherror(r, "semaphore_destroy"); |
Russ Cox | cd80000 | 2009-06-08 14:03:21 -0700 | [diff] [blame] | 415 | } |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 416 | } |
| 417 | |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 418 | // The other calls have simple system call traps in sys.s |
| 419 | int32 mach_semaphore_wait(uint32 sema); |
| 420 | int32 mach_semaphore_timedwait(uint32 sema, uint32 sec, uint32 nsec); |
| 421 | int32 mach_semaphore_signal(uint32 sema); |
| 422 | int32 mach_semaphore_signal_all(uint32 sema); |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 423 | |
| 424 | void |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 425 | mach_semacquire(uint32 sem) |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 426 | { |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 427 | int32 r; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 428 | |
Russ Cox | cd80000 | 2009-06-08 14:03:21 -0700 | [diff] [blame] | 429 | while((r = mach_semaphore_wait(sem)) != 0) { |
| 430 | if(r == KERN_ABORTED) // interrupted |
| 431 | continue; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 432 | macherror(r, "semaphore_wait"); |
Russ Cox | cd80000 | 2009-06-08 14:03:21 -0700 | [diff] [blame] | 433 | } |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 434 | } |
| 435 | |
| 436 | void |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 437 | mach_semrelease(uint32 sem) |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 438 | { |
Russ Cox | 08cfcd1 | 2009-03-24 13:51:48 -0700 | [diff] [blame] | 439 | int32 r; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 440 | |
Russ Cox | cd80000 | 2009-06-08 14:03:21 -0700 | [diff] [blame] | 441 | while((r = mach_semaphore_signal(sem)) != 0) { |
| 442 | if(r == KERN_ABORTED) // interrupted |
| 443 | continue; |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 444 | macherror(r, "semaphore_signal"); |
Russ Cox | cd80000 | 2009-06-08 14:03:21 -0700 | [diff] [blame] | 445 | } |
Russ Cox | 376898c | 2008-09-09 11:50:14 -0700 | [diff] [blame] | 446 | } |
| 447 | |
Russ Cox | 5963dba | 2010-04-08 18:15:30 -0700 | [diff] [blame^] | 448 | void |
| 449 | sigpanic(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 | } |