-
Notifications
You must be signed in to change notification settings - Fork 33
/
exit.c
129 lines (114 loc) · 2.8 KB
/
exit.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
/*
* fiwix/kernel/syscalls/exit.c
*
* Copyright 2018-2022, Jordi Sanfeliu. All rights reserved.
* Distributed under the terms of the Fiwix License.
*/
#include <fiwix/asm.h>
#include <fiwix/kernel.h>
#include <fiwix/syscalls.h>
#include <fiwix/process.h>
#include <fiwix/sched.h>
#include <fiwix/mman.h>
#include <fiwix/sleep.h>
#include <fiwix/stdio.h>
#include <fiwix/string.h>
#ifdef CONFIG_SYSVIPC
#include <fiwix/sem.h>
#endif /* CONFIG_SYSVIPC */
void do_exit(int exit_code)
{
int n;
struct proc *p, *init;
#ifdef __DEBUG__
printk("\n");
printk("sys_exit(pid %d, ppid %d)\n", current->pid, current->ppid);
printk("------------------------------\n");
#endif /*__DEBUG__ */
#ifdef CONFIG_SYSVIPC
if(current->semundo) {
semexit();
}
#endif /* CONFIG_SYSVIPC */
release_binary();
current->argv = NULL;
current->envp = NULL;
init = get_proc_by_pid(INIT);
FOR_EACH_PROCESS(p) {
if(SESS_LEADER(current)) {
if(p->sid == current->sid && p->state != PROC_ZOMBIE) {
p->pgid = 0;
p->sid = 0;
p->ctty = NULL;
send_sig(p, SIGHUP);
send_sig(p, SIGCONT);
}
}
/* make INIT inherit the children of this exiting process */
if(p->ppid == current->pid) {
p->ppid = INIT;
init->children++;
current->children--;
if(p->state == PROC_ZOMBIE) {
send_sig(init, SIGCHLD);
if(init->sleep_address == &sys_wait4) {
wakeup_proc(init);
}
}
}
p = p->next;
}
if(SESS_LEADER(current)) {
disassociate_ctty(current->ctty);
}
for(n = 0; n < OPEN_MAX; n++) {
if(current->fd[n]) {
sys_close(n);
}
}
iput(current->root);
current->root = NULL;
iput(current->pwd);
current->pwd = NULL;
current->exit_code = exit_code;
if(!--nr_processes) {
printk("\n");
printk("WARNING: the last user process has exited. The kernel will stop itself.\n");
sync_superblocks(0); /* in all devices */
sync_inodes(0); /* in all devices */
sync_buffers(0); /* in all devices */
stop_kernel();
}
/* notify the parent about the child's death */
if((p = get_proc_by_pid(current->ppid))) {
send_sig(p, SIGCHLD);
if(p->sleep_address == &sys_wait4) {
wakeup_proc(p);
}
}
current->sigpending = 0;
current->sigblocked = 0;
current->sigexecuting = 0;
for(n = 0; n < NSIG; n++) {
current->sigaction[n].sa_mask = 0;
current->sigaction[n].sa_flags = 0;
current->sigaction[n].sa_handler = SIG_IGN;
}
not_runnable(current, PROC_ZOMBIE);
need_resched = 1;
do_sched();
}
int sys_exit(int exit_code)
{
#ifdef __DEBUG__
printk("(pid %d) sys_exit()\n", current->pid);
#endif /*__DEBUG__ */
/* exit code in the second byte.
* 15 8 7 0
* +-------------------+-------------------+
* | exit code (0-255) | 0 |
* +-------------------+-------------------+
*/
do_exit((exit_code & 0xFF) << 8);
return 0;
}