OSLAB notes 2
This time, we are asked to play with the fork()
, signal()
, kill()
and some other POSIX APIs related with process control and communication.
Fork and pass signal
The first task is to use fork
to create 2 child-processes and then use soft interrupt (signal) to make interprocess communication.
Fork
Well, first of all you need to create 2 child-processes. And how to achieve that, this?
1 | pid1 = fork(); |
Well, young man, you’re simply naiveeee.
Before making further explanation, you’d better run this code yourself.
1 | pid_t pid1, pid2; |
The output goes like this,
The father 6281 created, pid1: 6282 pid2: 6283
The father 6282 created, pid1: 0 pid2: 6284
The father 6284 created, pid1: 0 pid2: 0
The father 6283 created, pid1: 6282 pid2: 0
Strange, huh? There is 4 processes created, rather than 3 that we expected; and the child process 6282 has also created one process 6284.
Well, let’s explain this. The fork
will create a duplicate of the running process, and the process created will be the child of the original process. Since the child is a replica, it will have the same local variables ,same PC(program counter) and a bunch of other stuff you may see its man page for details.
See the child and the parent processes are almost identical, the child will have the same PC as the parent’s, which means, it will continue to execute the code after where fork
is called. And that is the reason of the “wrong” behavior we have above. The execution flow goes like this.
- Parent 6281 created child 6282, then 6281 & 6282 will continue to execute
pid2 = fork()
- 6281 created another child 6283, while child 6282 also created one child of its own, 6284.
Then how shall we distinguish the child and the parent? Notice that some pids are 0. After fork
returned the child and the parent will receive 2 different value. The parent will get the pid of its child, meanwhile the child will have a 0. Also -1 for failed duplicate. And we can work on that.
Signal and Kill
In this task, we used software interrupt to achieve interprocess communication.
That is, one process listen for a specific signal (interrupt) and then the other one send that signal. These are done by signal
and kill
respectively.
The signal(signum, handler)
will bind the specified signal with id signum
with the handler function handler
. And kill(pid, sig)
will send signal sig
to process pid
.
Design
Then everything is simple, as the task asked us to let the parent listen for keyboard interrupt and then send signal to kill the children. We shall simply let the parent listen to the SIGINT
, and then bind a handler function that will send signals to the 2 children, the children then wait for the signals to commit suicide.
However, there are few things to note.
Notes
The guide provided is… well, it made few mistakes. Only on some specific OS the break
and delete
will generate the SIGINT
, mostly it’s for Ctrl+c
. And since we are using SIGINT
, it’s vital to overwrite the orignal handler for it. Or it will cause the processes to terminate, though the outcome is same, it’s not what we want.
All in all, it’s my code below.
1 | /** |
Pipe
The task 2 demand us to implement another interprocess communication via pipe
.
Well it’s rather easy, as we already got here. Use pipe to create a communication tunnel, and use that to pass messages. But notice there are 2 process writing to that pipe concurently, so proper lock is needed. Check lockf.
Code here,
1 | /** |