arrow left
Back to Developer Education

    Fork() in C Programming Language

    Fork() in C Programming Language

    You may have seen a lot of processes in your task manager if you are using Windows. Or in your resource monitor if you are using Linux. Did you ever stop to think how these are created? <!--more--> In this tutorial, we will talk about the fork() function and then implement some examples in the C programming language.

    Prerequisites

    To follow along with this tutorial, you should have:

    • A good understanding of the C programming language.
    • Some familiarity with Unix-like operating systems.

    What is a process?

    According to Wikipedia, a process is the instance of a computer program that is being executed by one or many threads. It contains the program code and its activity. Depending on the operating system (OS), a process may be made up of multiple threads of execution that execute instructions concurrently.

    What is a Fork()?

    In the computing field, fork() is the primary method of process creation on Unix-like operating systems. This function creates a new copy called the child out of the original process, that is called the parent. When the parent process closes or crashes for some reason, it also kills the child process.

    Let's start with the life-cycle of a process:

    Process life-cycle

    Image Source

    The operating system is using a unique id for every process to keep track of all processes. And for that, fork() doesn't take any parameter and return an int value as following:

    • Zero: if it is the child process (the process created).
    • Positive value: if it is the parent process.
    • Negative value: if an error occurred.

    Note: The following code only runs in Linux and UNIX based operating systems. If you are running Windows, then I recommend you to use Cygwin.

    Let's jump into the practical section where we will create examples from the simple level to the advanced one.

    Hello world!

    #include <stdio.h> 
    #include <sys/types.h> 
    #include <unistd.h> 
    int main() 
    { 
        /* fork a process */
        fork(); 
        /* the child and parent will execute every line of code after the fork (each separately)*/
        printf("Hello world!\n"); 
        return 0; 
    } 
    

    The output will be:

    Hello world!
    Hello world!
    

    Where one of the output came from the parent process and the other one from the child process.

    Simple fork

    Simply, we can tell that the result is 2 power of n, where n is the number of fork() system calls.

    For example:

    #include <stdio.h> 
    #include <sys/types.h> 
    #include <unistd.h> 
    #include <stdlib.h>
    int main() 
    { 
        fork(); 
        fork(); 
        fork(); 
        printf("Hello world!\n");
        return 0; 
    } 
    

    The result is:

    Hello world!
    Hello world!
    Hello world!
    Hello world!
    Hello world!
    Hello world!
    Hello world!
    Hello world!
    

    Another example is:

    int main() {
      if(fork() == 0)
        if(fork())
          printf("Hello world!!\n");
      exit(0);
    }
    

    I drew a brief sketch to help you understand the idea:

    Fork

    Inside the first if condition a fork has occurred and it is checking if it is the child process, it then continues to execute its code. Otherwise (if its the parent process) it will not go through that if. Then, in the second if, it will only accept the parent process which holds the positive id.

    As a result, it will print only one "Hello world!".

    Now try to execute the following code and compare your result with ours:

    int doWork(){
    	fork();
    	fork();
    	printf("Hello world!\n");
    }
    int main() {
    	doWork();
    	printf("Hello world!\n");
    	exit(0);
    }
    

    The result will be:

    Hello world!
    Hello world!
    Hello world!
    Hello world!
    Hello world!
    Hello world!
    Hello world!
    Hello world!
    

    Fork explaination

    Because when the process that has been forked inside dowork() prints Hello World! it will continue the main code after the function call and print that Hello World! then exits.

    Advanced example

    When a process creates a new process, then there are two possibilities for the execution exit:

    • The parent continues to execute concurrently with its child.
    • The parent waits until some or all of its children have terminated.
    #include <sys/types.h>
    #include <sys/types.h>
    #include <stdio.h>
    #include <unistd.h>
    #include <sys/wait.h>
    int main(int argc, char *argv[]) {
    
       /* fork a child process */
       pid_t pid = fork();
    
       if (pid < 0) { /* error occurred */
       	fprintf(stderr, "Fork Failed");
       	return 1;
       }
    
       else if (pid == 0) { /* child process */
       	printf("I'm the child \n"); /* you can execute some commands here */
       }
    
       else { /* parent process */
       	/* parent will wait for the child to complete */
       	  wait(NULL);
       	/* When the child is ended, then the parent will continue to execute its code */
       	  printf("Child Complete \n");
       }
    }
    

    The wait call system wait(NULL) will make the parent process wait until the child process has executed all of its commands.

    The result will be:

    I'm the child 
    Child Complete 
    
    

    Another example:

    #include <sys/types.h>
    #include <stdio.h>
    #include <unistd.h>
    #include <sys/wait.h>
    #include <stdlib.h>
    int main(int argc, char *argv[])
    {
       printf("I am: %d\n", (int) getpid());
    
       pid_t pid = fork();
       printf("fork returned: %d\n", (int) pid);
    
       if (pid < 0) { /* error occurred */
       	perror("Fork failed");
       }
       if (pid == 0) { /* child process */
       	printf("I am the child with pid %d\n", (int) getpid());
                   printf("Child process is exiting\n");
                   exit(0);
           }
       /* parent process */
       printf("I am the parent waiting for the child process to end\n");
           wait(NULL);
           printf("parent process is exiting\n");
           return(0);
    } 
    

    The result will be something like:

    I am: 2337
    fork returned: 2338
    I am the parent waiting for the child process to end
    fork returned: 0
    I am the child with pid 2338
    Child process is exiting
    parent process is exiting
    

    That's all for today!🥳

    Conclusion

    We have learned what fork() can do, and how to implement it in the C programming language in unique examples. If you are interested more in the operating system abstractions, and how it is working, then I recommend you to start learning about pipes then semaphores.

    Happy coding!

    Further reading

    1. More about fork

    2. Pipe with fork

    3. What is a semaphore

    4. Introduction to Semaphore


    Peer Review Contributions by: Saiharsha Balasubramaniam

    Published on: Feb 11, 2021
    Updated on: Jul 25, 2024
    CTA

    Cloudzilla is FREE for React and Node.js projects

    Deploy GitHub projects across every major cloud in under 3 minutes. No credit card required.
    Get Started for Free