1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
|
#include "u.h"
#include <errno.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <sched.h>
#include <signal.h>
#if !defined(__OpenBSD__)
# if defined(__APPLE__)
# define _XOPEN_SOURCE /* for Snow Leopard */
# endif
#endif
#include <sys/utsname.h>
#include "libc.h"
#include "thread.h"
typedef struct Execjob Execjob;
typedef struct Proc Proc;
typedef struct _Procrendez _Procrendez;
typedef struct Jmp Jmp;
struct Jmp
{
p9jmp_buf b;
};
enum
{
STACK = 8192
};
struct Execjob
{
int *fd;
char *cmd;
char **argv;
char *dir;
Channel *c;
};
struct _Procrendez
{
Lock *l;
int asleep;
pthread_cond_t cond;
};
struct _Thread
{
_Thread *next;
_Thread *prev;
_Thread *allnext;
_Thread *allprev;
void (*startfn)(void*);
void *startarg;
uint id;
pthread_t osprocid;
uchar *stk;
uint stksize;
int exiting;
int mainthread;
Proc *proc;
char name[256];
char state[256];
void *udata;
Alt *alt;
_Procrendez schedrend;
};
extern void _procsleep(_Procrendez*);
extern void _procwakeup(_Procrendez*);
extern void _procwakeupandunlock(_Procrendez*);
struct Proc
{
Proc *next;
Proc *prev;
char msg[128];
pthread_t osprocid;
Lock lock;
int nswitch;
_Thread *thread0;
_Thread *thread;
_Thread *pinthread;
_Threadlist runqueue;
_Threadlist idlequeue;
_Threadlist allthreads;
uint nthread;
uint sysproc;
_Procrendez runrend;
Lock schedlock;
_Thread *schedthread;
void *udata;
Jmp sigjmp;
int mainproc;
};
#define proc() _threadproc()
extern Proc *_threadprocs;
extern Lock _threadprocslock;
extern Proc *_threadexecproc;
extern Channel *_threadexecchan;
extern QLock _threadexeclock;
extern Channel *_dowaitchan;
extern void _procstart(Proc*, void (*fn)(Proc*));
extern _Thread *_threadcreate(Proc*, void(*fn)(void*), void*, uint);
extern void _procexit(void);
extern Proc *_threadproc(void);
extern void _threadsetproc(Proc*);
extern int _threadlock(Lock*, int, ulong);
extern void _threadunlock(Lock*, ulong);
extern void _pthreadinit(void);
extern int _threadspawn(int*, char*, char**, char*);
extern int _runthreadspawn(int*, char*, char**, char*);
extern void _threadsetupdaemonize(void);
extern void _threaddodaemonize(char*);
extern void _threadpexit(void);
extern void _threaddaemonize(void);
extern void *_threadstkalloc(int);
extern void _threadstkfree(void*, int);
extern void _threadpthreadmain(Proc*, _Thread*);
extern void _threadpthreadstart(Proc*, _Thread*);
#define USPALIGN(ucp, align) \
(void*)((((uintptr)(ucp)->uc_stack.ss_sp+(ucp)->uc_stack.ss_size)-(align))&~((align)-1))
|