-
Notifications
You must be signed in to change notification settings - Fork 6
/
sysproc.c
208 lines (171 loc) · 3.63 KB
/
sysproc.c
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
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
#include "types.h"
#include "x86.h"
#include "defs.h"
#include "date.h"
#include "param.h"
#include "memlayout.h"
#include "mmu.h"
#include "spinlock.h"
#include "sleeplock.h"
#include "proc.h"
int
sys_fork(void)
{
return fork();
}
// sys_clone function copies the arguments to clone from the user process
// address space to the xv6 kernel memory. The function after copying
// the values passes the control to clone function
int sys_clone(void) {
// actual clone call prototype :
// clone(int (*fun)(void *), void *child_stack), int flags, void *args)
int (*func)(void *);
char *child_stack, *args;
int flags;
// read the function pointer
if(argptr(0, (char **)&func, 0) == -1) {
return -1;
}
// read the child stack address
if(argint(1, (int *)&child_stack) == -1) {
return -1;
}
// read the flags
if(argint(2, &flags) == -1) {
return -1;
}
// read the address of argument
if(argptr(3, &args, 0) == -1) {
return -1;
}
// the arguments passed to clone must have address below size
if(THREAD_LEADER(myproc())->sz < (uint)args) {
return -1;
}
// clone system call supports stack address space which are one page alligned
if(THREAD_LEADER(myproc())->sz <= (uint)child_stack &&
THREAD_LEADER(myproc())->sz < (uint)child_stack - PGSIZE) {
return -1;
}
// call to actual kernel clone function
return clone(func, (void *)child_stack, flags, (void *)args);
}
// sys join copies the contents from user stack to the kernel memory
// sys join calls join which waits for the given thread to be over
int sys_join(void) {
int tid;
// read the thread id to be joined
if(argint(0, &tid) == -1) {
return -1;
}
return join(tid);
}
int
sys_exit(void)
{
exit();
return 0; // not reached
}
int
sys_wait(void)
{
return wait();
}
int
sys_kill(void)
{
int pid;
if(argint(0, &pid) < 0)
return -1;
return kill(pid);
}
// kills paritcular thread in the group
int
sys_tkill(void)
{
int tid;
if(argint(0, &tid) < 0)
return -1;
return tkill(tid);
}
// kills complete thread group
// kills all thread expect group leader
int
sys_tgkill(void)
{
return tgkill();
}
int
sys_getpid(void)
{
return myproc()->pid;
}
// gets thread id of the process
// if process calls gettid pid is returned (single threaded process)
// if thread calls gettid tid is returned (multithreaded process)
int
sys_gettid(void)
{
return myproc()->tid == -1 ? sys_getpid() : myproc()->tid;
}
int
sys_sbrk(void)
{
int addr;
int n;
if(argint(0, &n) < 0)
return -1;
acquire(&(THREAD_LEADER(myproc())->tlock));
addr = THREAD_LEADER(myproc())->sz;
if(growproc(n) < 0) {
release(&(THREAD_LEADER(myproc())->tlock));
return -1;
}
release(&(THREAD_LEADER(myproc())->tlock));
return addr;
}
int
sys_sleep(void)
{
int n;
uint ticks0;
if(argint(0, &n) < 0)
return -1;
acquire(&tickslock);
ticks0 = ticks;
while(ticks - ticks0 < n){
if(myproc()->killed){
release(&tickslock);
return -1;
}
sleep(&ticks, &tickslock);
}
release(&tickslock);
return 0;
}
// return how many clock tick interrupts have occurred
// since start.
int
sys_uptime(void)
{
uint xticks;
acquire(&tickslock);
xticks = ticks;
release(&tickslock);
return xticks;
}
// sys suspend syspends the current thread execution
int
sys_tsuspend(void)
{
return tsuspend();
}
// sys resumes the execution of the thread with given tid
int
sys_tresume(void)
{
int tid;
if(argint(0, &tid) < 0)
return -1;
return tresume(tid);
}