I want to make a process tree like the picture below. I wrote the code below but if you look at the PIDs, you'll find there's a problem!
Process tree if you can't view the image:
//A-B (A parent of B)
//A-C
//B-D
//D-E
//D-F
My code:
void child(int index, int level, pid_t parent) { // this is going to be called recursively by all child processes
pid_t cpid[3];
int i = 0;
pid_t Lparent = getpid();
if (index == 0 && level == 1) { // this is B
if ((cpid[i] = fork()) == 0) { // D is going to execute this
child(i, 2, Lparent);
}
sleep(3);
for (i = 0; i < 1; i++) { // B will kill D
kill(cpid[i], SIGKILL);
}
printf("My PID: %d, My Parent's PID:%d\n", getpid(), parent); // this is B
sleep(7);
} else if (index == 0 && level == 2) { // this is D
for (i = 0; i < 2; i++) {
if ((cpid[i] = fork()) == 0) {
child(i, 3, Lparent);
}
}
sleep(5);
for (i = 0; i < 2; i++) {
kill(cpid[i], SIGKILL);
printf("My PID: %d, My parent's PID:%d\n", getpid(), parent);
sleep(5);
}
} else {
printf("My PID: %d, My Parent's PID:%d\n", getpid(), parent);
sleep(10);
}
}
int main() {
pid_t cpid[2];
pid_t root = getpid();
int i = 0;
for (i = 0; i < 2; i++) {
if ((cpid[i] = fork()) == 0) {
child(i, 1, root);
}
}
sleep(8);
for (i = 0; i < 2; i++) {
kill(cpid[i], SIGKILL);
}
printf("My Pid: %d\n", getpid());
return 0;
}
My output:
My PID: 211, My Parent's PID:206
My PID: 213, My Parent's PID:212
My PID: 214, My Parent's PID:212
My PID: 210, My Parent's PID:206
My Pid: 206
Does anyone know how I can fix this?
A much simpler and easier-to-debug way to create process trees like this in C using fork()
is to create a function for each lettered process and have those functions call fork()
when appropriate. Using loops like you attempt in your proposed solution is possible, but very tricky and hard to read, especially since there doesn't appear to be a useful pattern in this tree.
This is an example of a function to perform the duties of A by calling fork()
twice to produce two distinct child processes, one to perform the duties of B, the other C.
#include <stdio.h>
#include <unistd.h>
void B(int parent){
printf("I am process B. My PID is %d, my parent's PID is %d\n",getpid(),parent);
//Do rest of B's job
}
void C(int parent){
printf("I am process C. My PID is %d, my parent's PID is %d\n",getpid(),parent);
//Do rest of C's job
}
void A(){
//A must fork off a process for B and for C
int A_pid = getpid();
int B_pid = fork();
int C_pid = 0;
if(B_pid == 0) //I'm the first child so I should call B()
B(A_pid);
else if(B_pid > 0){ //I'm the parent so I need to fork() again
C_pid = fork();
if(C_pid == 0) //I'm the second child so I should call C()
C(A_pid);
else if(C_pid > 0) //I'm still the parent
printf("I am process A. My PID is %d and I don't have a parent\n",A_pid);
else
;//Handle fork failing the second time
}
else
;//Handle fork failing the first time
//Do the rest of A's job, wait for children to do their thing etc.
kill(B_PID, SIGKILL);
kill(C_PID, SIGKILL);
}
int main(){
A();
}
Side note: It is very important to remember when using fork()
in UNIX-based operating systems that it can fail, and when it does it returns -1. This can cause very unexpected results if you do not catch and handle this error, because calling kill(-1,<signal>)
sends that signal to every process it has access to, which is likely not what you'd like.