35. Cooperating Processes and Inter-process Communication#

The process abstraction is designed to give the illusion that a program executes on its own private, isolated (virtual) machine. The OS scheduling mechanism hides the fact that the physical CPU is, in fact, shared by multiple processes. From the perspective of any given process, it appears as though its instructions execute one after the other, in the order that they appear in the program, on a dedicated CPU, without interruption or interference by the execution of other processes. The process also has sole access to a virtual address space, where it can store, and operate on, its data – there is no concern about access to the process’s memory from other processes, or that the process may inadvertently access memory that does not belong to it.

But, sometimes we want multiple processes to work together to solve a problem. How can they cooperate if they are completely isolated from each other? The short answer is, they can’t. For processes to cooperate, they need some way to communicate with each other, so that they can exchange information and coordinate their activities. Let’s think about some ways that multiple processes can work together, and how they can communicate.

Unix command pipelines

Each program is designed to do one thing well, and more complex operations can be created by composing multiple simple programs. Here, communication is supported by the pipe() system call, which creates a communication channel between processes. With some manipulation of file descriptors, the shell can arrange for the standard output of one process to become the standard input of the next process in the pipeline. The pipe provides both a way to pass data from one process to the next and a way to control the order in which processes execute, because a downstream process cannot read its input until after the previous process in the pipeline has started to produce some output.

Complex services

A complex service, such as a banking service, may have multiple logical operations that it needs to carry out, such as handling user input, executing account operations like deposits, calculating daily interest, running fraud detection analytics, etc. It is often convenient to implement the service as a collection of processes, with each process carrying out one of the major tasks of the service and the operating system scheduler ensuring that all tasks make progress concurrently. The processes may communicate by reading and writing shared files in the file system. By

Parallel computation

In this case, the goal is to complete a single task in less time by using multiple physical compute cores. As a trivial example, consider sorting a very large set of values. On a multi-core machine, you could create one process per core, arrange for each process to sort a portion of the input, and then merge the results together after all the individual sorts were complete. Here, processes might communicate using a combination of files in the file system (e.g., to read the original, unsorted input, and write the intermediate results from each sorting process) and pipes (e.g., to notify the merging process when each sorting process was finished).

35.1. Shared Memory#

To support more efficient, fine-grained inter-process communication, the operating system also provides system calls to let processes explicitly share a portion of their virtual address space. This allows processes to exchange information simply by reading or writing variables that are allocated in the shared part of the address space. There are actually two ways to do this on Unix systems. The first comes from UNIX System V:

  • segment_id = shmget(key, size, shmflg) : return the identifier of the System V shared memory segment associated with the value of the argument key; may be used either to obtain the identifier of a previously created shared memory segment (when shmflg is zero and key does not have the value IPC_PRIVATE), or to create a new shared memory segment.

  • shared_startvaddr = shmat(shmid, *shmaddr, shmflg) : attach the System V shared memory segment identified by shmid to the address space of the calling process at shmaddr or an address chosen by the OS if shmaddr is NULL; return the actual starting virtual address of the shared segment in the process address space.

  • err = shmdt(shmaddr) : detach the shared memory segment located at the address specified by shmaddr from the address space of the calling process; the to-be-detached segment must be currently attached with shmaddr equal to the value returned by the attaching shmat() call.

The use of segment identifiers makes it possible for unrelated processes to share memory, however after one process creates a shared memory segment with shmget, it needs to communicate the segment id to the other processes that are allowed to share it. One way to do this is by writing the segment id into a file that can be read by the other processes. The other method for requesting shared memory comes from BSD UNIX and requires a parent process to create a shared mapping in its address space before forking child processes. The mapped segments are then shared with child processes, rather than creating a copy in the child address space.

  • shared_startvaddr = mmap(*addr, length, prot, MAP_SHARED | MAP_ANONYMOUS, ...) : create a new mapping of length bytes in the virtual address space of the calling process starting at addr or an address chosen by the OS if addr is NULL; return the actual starting virtual address of the shared segment in the process address space.

Note that mmap is often used to access persistent files using a virtual memory interface rather than the file system read() and write() interface. In the example above, MAP_ANONYMOUS indicates that the mapping is for a non-persistent virtual memory region, which is not backed by any file. (Remember that you can use man for the full details of these system calls.)

The shared memory segments are implemented in the operating system using the virtual memory mechanisms. Each process that shares the memory segment has its page tables pointing to the same underlying physical pages, so that any load or store access to a virtual address in the shared segment will be translated to the same physical location in the machine memory.

35.2. Multi-threaded Processes#

Recall from Section 7.3 that a process can have multiple threads, and that all threads in a process share the same virtual address space. This means that multi-threaded processes do not need to do any extra work to communicate using shared memory. The operating system itself is multi-threaded: every user-level process has at least one thread in the kernel, and other threads carry out asynchronous operating system activities, such as writing dirty file system blocks back to the disk. All of these kernel threads share the same operating system code and data structures. We will see later how user-level communication through pipes is implemented in the operating system by threads writing to, and reading from, a shared buffer of memory. In our discussion of concurrency and synchronization, we will use examples in multi-threaded user-level processes, since they can be presented as simple, stand-alone programs. Keep in mind that everything we say about multi-threaded processes also applies to the implementation of the operating system.

Since each thread is an independent execution entity, they need to have their own execution state. This includes a per-thread execution stack to hold local variables. In contrast, global variables and heap-allocated variables are shared by all threads in a process. Note that although each thread has its own execution stack, these are still all part of the same virtual address space, and threads in the same process are not protected from each other by the operating system. Thus, it is possible for a buggy thread to accidentally scribble on another thread’s stack space, leading to bizarre behavior.

35.3. Issues with Concurrent Execution#

Regardless of whether we are using multiple processes, or a single multi-threaded process, whether we are implementing user-level programs or the operating system, we need a way to coordinate multiple concurrent activities.

The operating system scheduler controls when, and for how long, any given thread is allowed to execute. Although each thread’s instructions execute in program order, its execution is interleaved with other threads. And when threads share resources, such as data structures in shared memory, the arbitrary interleaving of operations from multiple threads can lead to unpredictable, and undesirable, outcomes. Synchronization is the mechanism that allows us to restrict the possible interleavings of threads, so that we can control how they use shared resources, reason about their behavior, and prevent undesirable outcomes. There are two main problems that synchronization helps us to solve: how to ensure that threads do not interfere with each other when they use a shared resource, and how to ensure that threads execute their operations in the correct order. For example, consider the simple program in Listing 35.1, below, in which two threads each output a string:

Listing 35.1 output_example.c - Uncontrolled thread interleaving example using output to the terminal.#
 1#include <pthread.h>
 2#include <stdlib.h>
 3#include <sys/types.h>
 4#include <unistd.h>
 5#include <string.h>
 6#include <stdio.h>
 7
 8/* This function is intended so simulate a thread needing to do some other 
 9 * work in between output statements.
10 */
11static long waste_cpu()
12{
13    long dummy = 0;
14    for (int i = 0; i < 10000; i++) {
15        dummy += rand();
16    }
17    return dummy;
18}
19
20/* Start of function executed by each thread.
21 * Each character of the argument string is printed out one-by-one
22 * in a separate system call. 
23 * To increase the number of interleavings, we voluntarily yield the 
24 * CPU after each character is output.
25 */
26static void *write_one_by_one(void *arg)
27{
28    char *mystr = (char *)arg;
29      
30    for (int i = 0; i < strlen(mystr); i++) {
31        write(STDOUT_FILENO, &mystr[i], 1); 
32        waste_cpu();
33    }
34  
35    return (void *)0;       
36}
37/* End of function executed by each thread. */
38
39
40int main(int argc, char **argv)
41{
42    pthread_t tid1, tid2;
43    char *str1 = "Hello";
44    char *str2 = "Goodbye";
45
46    /* Normally we'd want to test the return value of pthread_create. */
47
48    (void)pthread_create(&tid1, NULL, write_one_by_one, (void *)str1 );
49    (void)pthread_create(&tid2, NULL, write_one_by_one, (void *)str2 );
50
51    /* Wait for child threads to finish */
52    pthread_join(tid1, NULL);
53    pthread_join(tid2, NULL);
54
55    (void)write(STDOUT_FILENO, "\n", 1);    
56    return 0;
57}
58

When this program is run, the main function creates two new threads, passing each a string to output on the terminal. Once created, the threads start executing concurrently in the write_one_by_one function. Each thread loops over its input string, calling the write() system call to output the characters one by one. (We could, of course, call write() to output the entire string at once, or use the C library printf function, but we want to see what could happen when we don’t have the internal synchronization that these functions provide.)

What do you expect the result of running this program to be? Let’s give it a try a few times!

The output doesn’t look great. Not only is the output unreadable, it is also different from one run to the next. (And each time we build this book, we don’t know exactly what output you will be seeing! Maybe some of the runs even produce something reasonable.) What we want to see is either “HelloGoodbye” or “GoodbyeHello” — by adding synchronization, we can restrict the interleavings of the output to just these two possibilities. And, if we wanted to ensure that “Hello” always appeared on the terminal before “Goodbye”, another form of synchronization would allow us to enforce that ordering as well. In fact, the C library printf() family of functions, and the internal implementation of the write() system call, contain synchronization to ensure that output from one call is not interleaved with another, but if we wanted to ensure that “Hello” is displayed before “Goodbye”, we would still need to add more synchronization on top of the write() or printf() functions.

Let’s look at another illustrative example of the problems that can arise when we don’t control concurrent execution.

Listing 35.2 deposit function from bank_deposit.c - Simple bank account example.#
1static void deposit(int account, money_t amount)
2{
3    money_t balance = get_balance(account);
4    balance += amount;
5    put_balance(account, balance);
6
7    return;        
8}

In Listing 35.2 we see a simplified example of a function that is part of a program to maintain a bank account balance at the Bank of Lost Funds. When running on a single thread, the deposit function is trivially correct: after it completes execution, the value of balance stored back to the account will be amount greater than it was before the function was invoked. Problems arise, however, when multiple threads run this function concurrently.

../_images/bank_deposit_interleaving.drawio.png

Fig. 35.1 (a) The body of the deposit() function executed by two threads, (b) one possible interleaving of the execution of the threads, and (c) a second possible interleaving.#

In Fig. 35.1(b), we see one possible interleaving when this function is invoked by two threads nearly simultaneously. In this case Thread 1 is interrupted after it has read the current value of balance, but before it could store the new value back to memory. Thread 2 then gets to run and completes its execution of the deposit() function before being interrupted. Finally, Thread 1 can run again and complete its execution of deposit() by storing its local balance variable back to the account. The result is that the update performed by Thread 2 is lost, being over-written by Thread 1’s computation. After depositing a total of $150 to the account we have a final balance of $50. Context switches are unpredictable, however, and other interleavings of the thread executions are also possible. In Fig. 35.1(c) we see another possible schedule, where Thread 1’s update to the account balance is overwritten by Thread 2.

Let’s try running the bank deposit program a few times and see what happens.


35.4. Race Conditions#

The unpredictable and undesirable outcomes that we have seen in the previous examples arise because of race conditions. A race condition occurs whenever two or more threads access a shared resource (like the terminal device, or the variable that stores the bank account balance), and the outcome depends on the order in which the accesses occur. Clearly, this implies that at least one of the accesses must be a write operation, since the outcome of concurrent read operations is not dependent on the order they occur. Logically, the threads are racing with each other to use the resource, and we don’t know which one will come first. Sometimes, the result of the race will be the one we want, and sometimes things can go horribly wrong. Race condition bugs are extremely tricky to diagnose and fix, since the program might produce the correct results millions of times before failing.

Race conditions happen even on a single CPU because asynchronous events occur arbitrarily during thread execution. For example, the network controller raises an interrupt upon arrival of a network packet, which causes the current thread to be suspended while the interrupt handler runs. Or a timer interrupt occurs and the OS scheduler switches the CPU from the current thread to another thread. With multiple CPUs, threads running in parallel on different CPUs can read and write the same memory location. Multi-threaded programs must be designed so that they can execute correctly in the face of such asynchrony.

In our simple example programs, we added extra code to make it more likely for the negative consequences of race conditions to manifest. Without those added operations, the code executed by each thread is short enough to complete in a single time slice, and the threads are less likely to interfere with each other. However, the race condition is still there, lurking in our code.

Consider the example in Listing 35.4, in which two threads either increment or decrement a shared counter some number of times. The arguments to this program let you try different numbers of threads, and different numbers of iterations for each thread to either increment or decrement the shared counter. The program always creates an equal number of incrementing or decrementing threads, and each thread does the same amount of work, so we would expect the final value of the counter to be the same as its initial value when all the threads are done.

Listing 35.4 counter.c - Two threads increment and decrement a shared counter without synchronization.#
#include <pthread.h>
#include <stdlib.h>
#include <sys/types.h>
#include <unistd.h>
#include <string.h>
#include <stdio.h>

volatile long shared_counter;

static void *increment_thread(void *arg)
{
    long niters = (long)arg;
    for (long i = 0; i < niters; i++) {
        shared_counter++;
    }

    return (void *)0;
}

static void *decrement_thread(void *arg)
{
    long niters = (long)arg;

    for (long i = 0; i < niters; i++) {
        shared_counter--;
    }

    return (void *)0;
}

int main(int argc, char **argv)
{
    pthread_t *tids;
    long niters;
    int nthreads;
    
    if (argc != 3) {
        fprintf(stderr,"Usage: %s <num_threads> <num_iters>\n",argv[0]);
        return 1;
    }

    nthreads = atoi(argv[1]);
    niters = atoi(argv[2]);
    shared_counter = 0;
    
    printf("Main thread: Beginning test with %d threads\n", nthreads);
    
    tids = (pthread_t *)malloc(nthreads * sizeof(pthread_t));
    
    /* We create the same number of increment and decrement threads, each doing the same number of iterations. 
     * When all threads have completed, we expect the final value of the shared counter to be the same as its
     * initial value (i.e., 0).
     */
    for (int i = 0; i < nthreads; i+=2) {
        (void)pthread_create(&tids[i], NULL, increment_thread, (void *)niters );
        (void)pthread_create(&tids[i+1], NULL, decrement_thread, (void *)niters );
    }
    
    /* Wait for child threads to finish */
    for (int i = 0; i < nthreads; i+=2) {
        pthread_join(tids[i], NULL);
        pthread_join(tids[i+1], NULL);
    }
    
    printf("Main thread: Final value of shared counter is %ld\n", shared_counter);

    return 0;
}

Even though it looks like the racing thread accesses to the shared counter variable are each just a single line of code, we still have unpredictable final results. The reason is because that single line of C source code, shared_counter++ or shared_counter-- becomes three lines of machine code when the program is compiled, as shown in Listing 35.5. (Try make counter.s, and see if you can find these lines in the assembly file counter.s.) The highlighted lines load the value of the counter from memory into a register, decrement the value in the register, and then store the register value back to memory. Context switches can occur between any of these statements, leading to exactly the same interleaving problems that we saw in the bank account example.

Listing 35.5 Assembly code snippet of loop in decrement_thread() function.#
 1decrement_thread:
 2.LFB63:
 3	.cfi_startproc
 4	endbr64
 5	testq	%rdi, %rdi
 6	jle	.L7
 7	xorl	%edx, %edx
 8	.p2align 4,,10
 9	.p2align 3
10.L8:
11	movq	shared_counter(%rip), %rax
12	addq	$1, %rdx
13	subq	$1, %rax
14	movq	%rax, shared_counter(%rip)
15	cmpq	%rdx, %rdi
16	jne	.L8
17.L7:
18	xorl	%eax, %eax
19	ret
20	.cfi_endproc

Clearly, uncontrolled sharing can lead to serious problems. In the next chapter, we define the problem a bit more formally. The subsequent chapters look at some ways to solve the problem.