-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathdisastrOS_exit.c
95 lines (83 loc) · 3.17 KB
/
disastrOS_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
#include <assert.h>
#include <unistd.h>
#include <stdio.h>
#include "disastrOS.h"
#include "disastrOS_syscalls.h"
#include "disastrOS_descriptor.h"
// called upon termination
// moves the process to a zombie status
// if the process has children they need to be reparented to
// the init process (pid 1);
// all children should receive SIGHUP
// parent should receive SIGCHLD and be awakened if it was waiting
// for this process to report its value
void internal_exit(){
// 2nd register in pcb contains the exit value
running->return_value=running->syscall_args[0];
assert(init_pcb);
while(running->children.first){
// detach from current list
ListItem* item = List_detach(& running->children, running->children.first);
assert(item);
// attach to init's children list
List_insert(& init_pcb->children, init_pcb->children.last, item);
// send SIGHUP
PCBPtr* pcb_ptr=(PCBPtr*) item;
PCB* pcb=pcb_ptr->pcb;
pcb->signals |= (DSOS_SIGHUP & pcb->signals_mask);
}
running->status=Zombie;
List_insert(&zombie_list, zombie_list.last, (ListItem*) running);
running->parent->signals |= (DSOS_SIGCHLD & running->parent->signals_mask);
// if the parent was waiting for this process to die
if (running->parent->status==Waiting
// since he called wait or waitpid, we wake him up
&& running->parent->syscall_num==DSOS_CALL_WAIT
&& (running->parent->syscall_args[0]==0 ||
running->parent->syscall_args[0]==running->pid)
){
PCB* parent= (PCB*) List_detach(&waiting_list, (ListItem*) running->parent);
assert(parent);
parent->status=Running;
// we remove the process from the parent's children list
PCBPtr* self_in_parent=PCBPtr_byPID(&parent->children, running->pid);
List_detach(&parent->children, (ListItem*) self_in_parent);
// hack the PCB of the parent to put in return value
parent->syscall_retvalue=running->pid;
int* result=(int*)parent->syscall_args[1];
if (result)
*result=running->return_value;
// we release all timers put by this process from the timer list, in case there was someone
// hanging
// WARNING: untested
ListItem* aux=timer_list.first;
while(aux){
TimerItem* timer=(TimerItem*) aux;
aux=aux->next;
if (timer->pcb==running){
ListItem* detach_result=List_detach(&timer_list, (ListItem*)timer);
assert(detach_result);
TimerItem_free(timer);
}
}
// we release all resources of a process upon termination
while(running->descriptors.first) {
Descriptor* des=(Descriptor*) running->descriptors.first;
List_detach(&running->descriptors, (ListItem*) des);
Resource* res=des->resource;
List_detach(&res->descriptors_ptrs, (ListItem*) des->ptr);
DescriptorPtr_free(des->ptr);
Descriptor_free(des);
}
// the process finally dies
ListItem* suppressed_item = List_detach(&zombie_list, (ListItem*) running);
PCB_free((PCB*) suppressed_item);
running=parent;
} else {
// we put the first ready process in running
PCB* next_running=(PCB*) List_detach(&ready_list, ready_list.first);
assert(next_running);
next_running->status=Running;
running=next_running;
}
}