libthread: use mmap to allocate OpenBSD stacks
Should fix faults on OpenBSD. Fixes #218. Fixes #226.
This commit is contained in:
parent
4ae529dbfe
commit
8c573cab68
6 changed files with 61 additions and 8 deletions
|
|
@ -16,6 +16,14 @@ makecontext(ucontext_t *uc, void (*fn)(void), int argc, ...)
|
||||||
va_end(arg);
|
va_end(arg);
|
||||||
|
|
||||||
sp = (uintptr*)((char*)uc->uc_stack.ss_sp+uc->uc_stack.ss_size);
|
sp = (uintptr*)((char*)uc->uc_stack.ss_sp+uc->uc_stack.ss_size);
|
||||||
|
/*
|
||||||
|
* Stack pointer at call instruction (before return address
|
||||||
|
* gets pushed) must be 16-byte aligned.
|
||||||
|
*/
|
||||||
|
if((uintptr)sp%4)
|
||||||
|
abort();
|
||||||
|
while((uintptr)sp%16 != 0)
|
||||||
|
sp--;
|
||||||
*--sp = 0; // fn's return address
|
*--sp = 0; // fn's return address
|
||||||
*--sp = (uintptr)fn; // return address of setcontext
|
*--sp = (uintptr)fn; // return address of setcontext
|
||||||
uc->mc.sp = (uintptr)sp;
|
uc->mc.sp = (uintptr)sp;
|
||||||
|
|
|
||||||
13
src/libthread/stkmalloc.c
Normal file
13
src/libthread/stkmalloc.c
Normal file
|
|
@ -0,0 +1,13 @@
|
||||||
|
#include "threadimpl.h"
|
||||||
|
|
||||||
|
void*
|
||||||
|
_threadstkalloc(int n)
|
||||||
|
{
|
||||||
|
return malloc(n);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
_threadstkfree(void *v, int n)
|
||||||
|
{
|
||||||
|
free(v);
|
||||||
|
}
|
||||||
25
src/libthread/stkmmap.c
Normal file
25
src/libthread/stkmmap.c
Normal file
|
|
@ -0,0 +1,25 @@
|
||||||
|
#include <u.h>
|
||||||
|
#include <sys/mman.h>
|
||||||
|
#include "threadimpl.h"
|
||||||
|
|
||||||
|
#ifndef MAP_STACK
|
||||||
|
#define MAP_STACK 0
|
||||||
|
#endif
|
||||||
|
|
||||||
|
void*
|
||||||
|
_threadstkalloc(int n)
|
||||||
|
{
|
||||||
|
void *p;
|
||||||
|
|
||||||
|
p = mmap(nil, n, PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_ANON|MAP_STACK, -1, 0);
|
||||||
|
if(p == (void*)-1)
|
||||||
|
return nil;
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
_threadstkfree(void *v, int n)
|
||||||
|
{
|
||||||
|
if(n > 0)
|
||||||
|
munmap(v, n);
|
||||||
|
}
|
||||||
|
|
@ -6,22 +6,22 @@ tag="$OBJTYPE-$SYSNAME-${SYSVERSION:-`uname -r`}-${CC9:-cc}"
|
||||||
case "$tag" in
|
case "$tag" in
|
||||||
*-Linux-2.[0-5]*)
|
*-Linux-2.[0-5]*)
|
||||||
# will have to fix this for linux power pc
|
# will have to fix this for linux power pc
|
||||||
echo ${SYSNAME}-${OBJTYPE}-asm.o $SYSNAME.o
|
echo ${SYSNAME}-${OBJTYPE}-asm.o $SYSNAME.o stkmalloc.o
|
||||||
;;
|
;;
|
||||||
*-FreeBSD-[0-4].*)
|
*-FreeBSD-[0-4].*)
|
||||||
echo ${SYSNAME}-${OBJTYPE}-asm.o $SYSNAME.o
|
echo ${SYSNAME}-${OBJTYPE}-asm.o $SYSNAME.o stkmalloc.o
|
||||||
;;
|
;;
|
||||||
*-NetBSD-*)
|
*-NetBSD-*)
|
||||||
echo ${SYSNAME}-${OBJTYPE}-asm.o $SYSNAME.o
|
echo ${SYSNAME}-${OBJTYPE}-asm.o $SYSNAME.o stkmalloc.o
|
||||||
;;
|
;;
|
||||||
*-Darwin-10.[5-6].* | *-Darwin-[89].*)
|
*-Darwin-10.[5-6].* | *-Darwin-[89].*)
|
||||||
echo ${SYSNAME}-${OBJTYPE}-asm.o $SYSNAME-${OBJTYPE}.o pthread.o
|
echo ${SYSNAME}-${OBJTYPE}-asm.o $SYSNAME-${OBJTYPE}.o pthread.o stkmalloc.o
|
||||||
;;
|
;;
|
||||||
*-OpenBSD-*)
|
*-OpenBSD-*)
|
||||||
echo ${SYSNAME}-${OBJTYPE}-asm.o ${SYSNAME}-${OBJTYPE}.o pthread.o
|
echo ${SYSNAME}-${OBJTYPE}-asm.o ${SYSNAME}-${OBJTYPE}.o pthread.o stkmmap.o
|
||||||
;;
|
;;
|
||||||
*)
|
*)
|
||||||
echo pthread.o
|
echo pthread.o stkmalloc.o
|
||||||
esac
|
esac
|
||||||
|
|
||||||
case "$OBJTYPE-$SYSNAME" in
|
case "$OBJTYPE-$SYSNAME" in
|
||||||
|
|
|
||||||
|
|
@ -109,7 +109,7 @@ threadalloc(void (*fn)(void*), void *arg, uint stack)
|
||||||
ulong z;
|
ulong z;
|
||||||
|
|
||||||
/* allocate the task and stack together */
|
/* allocate the task and stack together */
|
||||||
t = malloc(sizeof *t+stack);
|
t = malloc(sizeof *t);
|
||||||
if(t == nil)
|
if(t == nil)
|
||||||
sysfatal("threadalloc malloc: %r");
|
sysfatal("threadalloc malloc: %r");
|
||||||
memset(t, 0, sizeof *t);
|
memset(t, 0, sizeof *t);
|
||||||
|
|
@ -122,7 +122,9 @@ threadalloc(void (*fn)(void*), void *arg, uint stack)
|
||||||
/* do a reasonable initialization */
|
/* do a reasonable initialization */
|
||||||
if(stack == 0)
|
if(stack == 0)
|
||||||
return t;
|
return t;
|
||||||
t->stk = (uchar*)(t+1);
|
t->stk = _threadstkalloc(stack);
|
||||||
|
if(t->stk == nil)
|
||||||
|
sysfatal("threadalloc malloc stack: %r");
|
||||||
t->stksize = stack;
|
t->stksize = stack;
|
||||||
memset(&t->context.uc, 0, sizeof t->context.uc);
|
memset(&t->context.uc, 0, sizeof t->context.uc);
|
||||||
sigemptyset(&zero);
|
sigemptyset(&zero);
|
||||||
|
|
@ -353,6 +355,7 @@ Top:
|
||||||
delthreadinproc(p, t);
|
delthreadinproc(p, t);
|
||||||
p->nthread--;
|
p->nthread--;
|
||||||
/*print("nthread %d\n", p->nthread); */
|
/*print("nthread %d\n", p->nthread); */
|
||||||
|
_threadstkfree(t->stk, t->stksize);
|
||||||
free(t);
|
free(t);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -509,6 +512,8 @@ needstack(int n)
|
||||||
_Thread *t;
|
_Thread *t;
|
||||||
|
|
||||||
t = proc()->thread;
|
t = proc()->thread;
|
||||||
|
if(t->stk == nil)
|
||||||
|
return;
|
||||||
|
|
||||||
if((char*)&t <= (char*)t->stk
|
if((char*)&t <= (char*)t->stk
|
||||||
|| (char*)&t - (char*)t->stk < 256+n){
|
|| (char*)&t - (char*)t->stk < 256+n){
|
||||||
|
|
|
||||||
|
|
@ -209,3 +209,5 @@ extern void _threadsetupdaemonize(void);
|
||||||
extern void _threaddodaemonize(char*);
|
extern void _threaddodaemonize(char*);
|
||||||
extern void _threadpexit(void);
|
extern void _threadpexit(void);
|
||||||
extern void _threaddaemonize(void);
|
extern void _threaddaemonize(void);
|
||||||
|
extern void *_threadstkalloc(int);
|
||||||
|
extern void _threadstkfree(void*, int);
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue