aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/libthread/thread.c22
1 files changed, 19 insertions, 3 deletions
diff --git a/src/libthread/thread.c b/src/libthread/thread.c
index 6c31cb76..dfc0acdf 100644
--- a/src/libthread/thread.c
+++ b/src/libthread/thread.c
@@ -77,11 +77,16 @@ procalloc(void)
}
static void
-threadstart(void *v)
+threadstart(uint y, uint x)
{
_Thread *t;
+ ulong z;
+
+ z = x<<16; /* hide undefined 32-bit shift from 32-bit compilers */
+ z <<= 16;
+ z |= y;
+ t = (_Thread*)z;
- t = v;
//print("threadstart %p\n", v);
t->startfn(t->startarg);
//print("threadexits %p\n", v);
@@ -94,6 +99,8 @@ threadalloc(void (*fn)(void*), void *arg, uint stack)
{
_Thread *t;
sigset_t zero;
+ uint x, y;
+ ulong z;
/* allocate the task and stack together */
t = malloc(sizeof *t+stack);
@@ -125,7 +132,16 @@ threadalloc(void (*fn)(void*), void *arg, uint stack)
(char*)t->context.uc.uc_stack.ss_sp
+t->context.uc.uc_stack.ss_size;
#endif
- makecontext(&t->context.uc, (void(*)())threadstart, 1, t);
+ /*
+ * All this magic is because you have to pass makecontext a
+ * function that takes some number of word-sized variables,
+ * and on 64-bit machines pointers are bigger than words.
+ */
+ z = (ulong)t;
+ y = z;
+ z >>= 16; /* hide undefined 32-bit shift from 32-bit compilers */
+ x = z>>16;
+ makecontext(&t->context.uc, (void(*)())threadstart, 2, y, x);
return t;
}