mirror of
https://github.com/9fans/plan9port.git
synced 2025-01-24 11:41:58 +00:00
add threaddaemonize(), more bug fixes
This commit is contained in:
parent
929fcfe0e3
commit
2c87dda8f8
7 changed files with 56 additions and 14 deletions
|
@ -21,6 +21,11 @@ void _threadsleep(Rendez*);
|
|||
_Thread *_threadwakeup(Rendez*);
|
||||
#define yield threadyield
|
||||
|
||||
/*
|
||||
* daemonize
|
||||
*/
|
||||
void threaddaemonize(void);
|
||||
|
||||
/*
|
||||
* per proc and thread data
|
||||
*/
|
||||
|
|
|
@ -8,6 +8,9 @@
|
|||
#include "thread.h"
|
||||
#include "threadimpl.h"
|
||||
|
||||
int ngetpid;
|
||||
|
||||
|
||||
/*
|
||||
* spin locks
|
||||
*/
|
||||
|
@ -99,15 +102,12 @@ again:
|
|||
sigdelset(&mask, SIGUSR1);
|
||||
sigsuspend(&mask);
|
||||
|
||||
//print("%d %d awake again\n", time(0), getpid());
|
||||
|
||||
/*
|
||||
* We're awake. Make USR1 not interrupt system calls.
|
||||
*/
|
||||
lock(r->l);
|
||||
ignusr1(1);
|
||||
if(r->asleep && r->pid == getpid()){
|
||||
//print("resleep %d\n", getpid());
|
||||
/* Didn't really wake up - signal from something else */
|
||||
goto again;
|
||||
}
|
||||
|
@ -160,10 +160,14 @@ dofreestacks(void)
|
|||
next = sf->next;
|
||||
if(sf->pid >= 1 && kill(sf->pid, 0) < 0 && errno == ESRCH){
|
||||
free(sf);
|
||||
if(last)
|
||||
last->next = next;
|
||||
else
|
||||
stackfree = next;
|
||||
sf = last;
|
||||
}
|
||||
}
|
||||
unlock(&stacklock);
|
||||
}
|
||||
|
||||
static int
|
||||
|
@ -240,7 +244,7 @@ static char *threadexitsmsg;
|
|||
void
|
||||
sigusr2handler(int s)
|
||||
{
|
||||
print("%d usr2 %d\n", time(0), getpid());
|
||||
/* fprint(2, "%d usr2 %d\n", time(0), getpid()); */
|
||||
if(threadexitsmsg)
|
||||
_exits(threadexitsmsg);
|
||||
}
|
||||
|
@ -341,6 +345,7 @@ _threadsetproc(Proc *p)
|
|||
{
|
||||
Perproc *pp;
|
||||
|
||||
if(p)
|
||||
p->osprocid = getpid();
|
||||
pp = newperproc();
|
||||
pp->proc = p;
|
||||
|
|
|
@ -20,6 +20,8 @@ static void
|
|||
xioproc(void *a)
|
||||
{
|
||||
Ioproc *io, *x;
|
||||
|
||||
threadsetname("ioproc");
|
||||
io = a;
|
||||
/*
|
||||
* first recvp acquires the ioproc.
|
||||
|
|
|
@ -6,6 +6,7 @@ LIB=libthread.a
|
|||
OFILES=\
|
||||
$SYSOFILES\
|
||||
channel.$O\
|
||||
daemonize.$O\
|
||||
exec.$O\
|
||||
ioproc.$O\
|
||||
iorw.$O\
|
||||
|
|
|
@ -3,6 +3,6 @@
|
|||
SHORTLIB=thread 9
|
||||
OFILES=
|
||||
|
||||
TARG=tprimes tspawn tspawnloop
|
||||
TARG=tprimes tspawn tspawnloop tdaemon
|
||||
|
||||
<$PLAN9/src/mkmany
|
||||
|
|
|
@ -20,6 +20,25 @@ static void delthreadinproc(Proc*, _Thread*);
|
|||
static void contextswitch(Context *from, Context *to);
|
||||
static void scheduler(Proc*);
|
||||
|
||||
static void
|
||||
_threaddebug(char *fmt, ...)
|
||||
{
|
||||
va_list arg;
|
||||
char buf[128];
|
||||
_Thread *t;
|
||||
|
||||
return;
|
||||
|
||||
va_start(arg, fmt);
|
||||
vsnprint(buf, sizeof buf, fmt, arg);
|
||||
va_end(arg);
|
||||
t = proc()->thread;
|
||||
if(t)
|
||||
fprint(2, "%d.%d: %s\n", getpid(), t->id, buf);
|
||||
else
|
||||
fprint(2, "%d._: %s\n", getpid(), buf);
|
||||
}
|
||||
|
||||
static _Thread*
|
||||
getthreadnow(void)
|
||||
{
|
||||
|
@ -50,6 +69,7 @@ threadstart(void *v)
|
|||
|
||||
t = v;
|
||||
t->startfn(t->startarg);
|
||||
memset(&v, 0xff, 32); /* try to cut off stack traces */
|
||||
threadexits(nil);
|
||||
}
|
||||
|
||||
|
@ -81,7 +101,7 @@ threadalloc(void (*fn)(void*), void *arg, uint stack)
|
|||
/* call makecontext to do the real work. */
|
||||
/* leave a few words open on both ends */
|
||||
t->context.uc.uc_stack.ss_sp = t->stk+8;
|
||||
t->context.uc.uc_stack.ss_size = t->stksize-16;
|
||||
t->context.uc.uc_stack.ss_size = t->stksize-64;
|
||||
makecontext(&t->context.uc, (void(*)())threadstart, 1, t);
|
||||
|
||||
return t;
|
||||
|
@ -185,6 +205,7 @@ scheduler(Proc *p)
|
|||
_Thread *t;
|
||||
|
||||
setproc(p);
|
||||
_threaddebug("scheduler enter");
|
||||
// print("s %p %d\n", p, gettid());
|
||||
lock(&p->lock);
|
||||
for(;;){
|
||||
|
@ -192,15 +213,15 @@ scheduler(Proc *p)
|
|||
if(p->nthread == 0)
|
||||
goto Out;
|
||||
p->runrend.l = &p->lock;
|
||||
//print("%d sleep for jobs %d\n", time(0), getpid());
|
||||
_threaddebug("scheduler sleep");
|
||||
_procsleep(&p->runrend);
|
||||
//print("%d wake from jobs %d\n", time(0), getpid());
|
||||
_threaddebug("scheduler wake");
|
||||
}
|
||||
delthread(&p->runqueue, t);
|
||||
unlock(&p->lock);
|
||||
p->thread = t;
|
||||
p->nswitch++;
|
||||
// print("run %s %d\n", t->name, t->id);
|
||||
_threaddebug("run %d (%s)", t->id, t->name);
|
||||
contextswitch(&p->schedcontext, &t->context);
|
||||
p->thread = nil;
|
||||
lock(&p->lock);
|
||||
|
@ -212,6 +233,7 @@ scheduler(Proc *p)
|
|||
}
|
||||
|
||||
Out:
|
||||
_threaddebug("scheduler exit");
|
||||
delproc(p);
|
||||
lock(&threadnproclock);
|
||||
if(p->sysproc)
|
||||
|
@ -447,11 +469,18 @@ threadmainstart(void *v)
|
|||
threadmain(threadargc, threadargv);
|
||||
}
|
||||
|
||||
void
|
||||
threadlinklibrary(void)
|
||||
{
|
||||
}
|
||||
|
||||
int
|
||||
main(int argc, char **argv)
|
||||
{
|
||||
Proc *p;
|
||||
|
||||
_threadsetupdaemonize();
|
||||
|
||||
threadargc = argc;
|
||||
threadargv = argv;
|
||||
|
||||
|
|
|
@ -5,8 +5,6 @@ typedef struct Execjob Execjob;
|
|||
typedef struct Proc Proc;
|
||||
typedef struct _Procrendez _Procrendez;
|
||||
|
||||
|
||||
|
||||
typedef struct Jmp Jmp;
|
||||
struct Jmp
|
||||
{
|
||||
|
@ -106,3 +104,5 @@ extern void _threadunlock(Lock*, ulong);
|
|||
extern void _pthreadinit(void);
|
||||
extern int _threadspawn(int*, char*, char**);
|
||||
extern int _runthreadspawn(int*, char*, char**);
|
||||
extern void _threadsetupdaemonize(void);
|
||||
extern void _threaddodaemonize(char*);
|
||||
|
|
Loading…
Reference in a new issue