Five communication methods between processes required for interviews

Five communication methods between processes required for interviews

Inter-Process Communication (IPC) refers to the transmission or exchange of information between different processes.

IPC methods usually include pipes (including unnamed pipes and named pipes), message queues, semaphores, shared storage, Sockets, Streams, etc. Among them, Sockets and Streams support IPC between two processes on different hosts.

[[283876]]

1. Pipeline

Pipes, usually unnamed pipes, are the oldest form of IPC in UNIX systems.

1. Features

  • Half-duplex (data flows in only one direction), with fixed read and write ends
  • Can only be used for communication between parent processes or brother threads (threads with blood relationship)
  • A special file that can be read and written using ordinary read and write functions, but is not an ordinary file and does not belong to any other file system. It only exists in memory.

2. Prototype

  1. #include < unistd.h >  
  2. int pipe(int fd[2]); // Return value: 0 if successful, -1 if failed

When a pipe is created, it creates two file descriptors: fd[0] is opened for reading, and fd[1] is opened for writing. To close the pipe, just close these two file descriptors. As shown below:

3. Examples

A pipe in a single process is almost useless. Therefore, the process that calls pipe usually calls fork, which creates an IPC channel between the parent process and the child process. This is shown in the following figure:

Half-duplex pipe after fork

Pipeline from parent process to child process

If you want the data flow to flow from the parent process to the child process, close the parent process's read end (fd[0]) and the child process's write end (fd[1]); conversely, you can make the data flow from the child process to the parent process.

  1. #include < stdio.h >  
  2. #include < unistd.h >  
  3.  
  4. int main()
  5. {
  6. int fd[2]; // two file descriptors
  7. pid_t pid;
  8. char buff[20];
  9.  
  10. if(pipe(fd) <   0 ) // Create a pipeline
  11. printf("Create Pipe Error!\n");
  12.  
  13. if(( pid = fork ()) <   0 ) // Create a child process
  14. printf("Fork Error!\n");
  15. else if(pid > 0) // parent process
  16. {
  17. close(fd[0]); // Close the read end
  18. write(fd[1], "hello world\n", 12);
  19. }
  20. else
  21. {
  22. close(fd[1]); // Close the write end
  23. read(fd[0], buff, 20);
  24. printf("%s", buff);
  25. }
  26.  
  27. return 0;
  28. }

2. Named Pipe (FIFO)

FIFO, also known as named pipe, is a type of file.

1. Features

  • Unlike unnamed pipes, named pipes can communicate between unrelated processes.
  • FIFO exists in the file system as a special device file with a path name associated with it.

2. Prototype

  1. #include < sys /stat.h >  
  2. int mkfifo(const char *pathname, mode_t mode); // Return value: 0 if successful, -1 if error occurs

The mode parameter is the same as the mode in the open function below.

3. Examples

Wirte:

  1. #include < stdio.h >  
  2. #include < stdlib.h > // exit
  3. #include < fcntl.h > // O_WRONLY
  4. #include < sys /stat.h >  
  5. #include < time.h > // time
  6.  
  7. int main()
  8. {
  9. int fd;
  10. int n, i;
  11. char buf[1024];
  12. time_t tp;
  13.  
  14. printf("I am %d process.\n", getpid()); // Indicates the process ID
  15. //When opening a FIFO, the difference between setting the non-blocking flag (O_NONBLOCK) and not setting it:
  16.  
  17. //If O_NONBLOCK is not specified (the default), a read-only open will block until some other process opens the FIFO for writing. Similarly, a write-only open will block until some other process opens it for reading.
  18.  
  19. If O_NONBLOCK is specified, a read-only open returns immediately, while a write-only open returns -1 on error. If no process has opened the FIFO for reading, its errno is set to ENXIO.
  20. if(( fd = open ("fifo1", O_WRONLY)) <   0 ) // Open a FIFO for writing
  21. {
  22. perror("Open FIFO Failed");
  23. exit(1);
  24. }
  25.  
  26. for( i = 0 ; i < 10 ; ++i)
  27. {
  28. time(&tp); // Get the current system time
  29. n = sprintf (buf,"Process %d's time is %s",getpid(),ctime(&tp));
  30. printf("Send message: %s", buf); // Print
  31. if(write(fd, buf, n+1) <   0 ) // Write to FIFO
  32. {
  33. perror("Write FIFO Failed");
  34. close(fd);
  35. exit(1);
  36. }
  37. sleep(1); // sleep for 1 second
  38. }
  39.  
  40. close(fd); // Close the FIFO file
  41. return 0;
  42. }

read:

  1. #include < stdio.h >  
  2. #include < stdlib.h >  
  3. #include < errno.h >  
  4. #include < fcntl.h >  
  5. #include < sys /stat.h >  
  6.  
  7. int main()
  8. {
  9. int fd;
  10. int len;
  11. char buf[1024];
  12.  
  13. if(mkfifo("fifo1", 0666) <   0 && errno!=EEXIST) // Create FIFO pipe
  14. perror("Create FIFO Failed");
  15.  
  16. if(( fd = open ("fifo1", O_RDONLY)) <   0 ) // Open FIFO for reading
  17. {
  18. perror("Open FIFO Failed");
  19. exit(1);
  20. }
  21.   
  22. while(( len = read (fd, buf, 1024)) > 0) // Read FIFO pipe
  23. printf("Read message: %s", buf);
  24.  
  25. close(fd); // Close the FIFO file
  26. return 0;
  27. }

3. Message Queues

A message queue is a linked list of messages stored in the kernel. A message queue is identified by an identifier (i.e., queue ID).

1. Features

  • Message queues are record-oriented, where messages have a specific format and a specific priority.
  • The message queue is independent of the sending and receiving processes. When the process terminates, the message queue and its contents are not deleted.
  • The message queue can realize random query of messages. Messages do not have to be read in the order of first-in-first-out, but can also be read by message type.

2. Prototype

  1. #include < sys /msg.h >  
  2. // Create or open a message queue: Returns the queue ID if successful, -1 if failed
  3. int msgget(key_t key, int flag);
  4. // Add message: return 0 if successful, -1 if failed
  5. int msgsnd(int msqid, const void *ptr, size_t size, int flag);
  6. // Read message: Return the length of the message data if successful, or -1 if failed
  7. int msgrcv(int msqid, void *ptr, size_t size, long type, int flag);
  8. // Control the message queue: return 0 if successful, -1 if failed
  9. int msgctl(int msqid, int cmd, struct msqid_ds *buf);

In the following two cases, msgget will create a new message queue:

  • If there is no message queue corresponding to the key value key, and flag contains the IPC_CREAT flag.
  • The key parameter is IPC_PRIVATE.

When the msgrcv function reads a message queue, the type parameter has the following conditions:

  • type == 0, returns the first message in the queue;
  • type > 0, returns the first message of type type in the queue;
  • type < 0, returns the message in the queue whose type value is less than or equal to the absolute value of type. If there are multiple messages, take the message with the smallest type value.

As can be seen, when the type value is not 0, it is used to read messages in a non-FIFO order. You can also think of type as a priority weight.

3. Examples

msg_server:

  1. #include < stdio.h >  
  2. #include < stdlib.h >  
  3. #include < sys /msg.h >  
  4.  
  5. // Used to create a unique key
  6. #define MSG_FILE "/etc/passwd"
  7.  
  8. // Message structure
  9. struct msg_form {
  10. long mtype;
  11. char mtext[256];
  12. };
  13.  
  14. int main()
  15. {
  16. int msqid;
  17. key_t key;
  18. struct msg_form msg;
  19.   
  20. // Get the key value
  21. if(( key = ftok (MSG_FILE,'z')) <   0 )
  22. {
  23. perror("ftok error");
  24. exit(1);
  25. }
  26.  
  27. // Print key value
  28. printf("Message Queue - Server key is: %d.\n", key);
  29.  
  30. // Create a message queue
  31. if (( msqid = msgget (key, IPC_CREAT|0777)) == -1)
  32. {
  33. perror("msgget error");
  34. exit(1);
  35. }
  36.  
  37. // Print message queue ID and process ID
  38. printf("My msqid is: %d.\n", msqid);
  39. printf("My pid is: %d.\n", getpid());
  40.  
  41. // Loop to read messages
  42. for(;;)
  43. {
  44. msgrcv(msqid, &msg, 256, 888, 0); // Return the first message of type 888
  45. printf("Server: receive msg.mtext is: %s.\n", msg.mtext);
  46. printf("Server: receive msg.mtype is: %d.\n", msg.mtype);
  47.  
  48. msg.mtype = 999 ; // Message type received by the client
  49. sprintf(msg.mtext, "hello, I'm server %d", getpid());
  50. msgsnd(msqid, &msg, sizeof(msg.mtext), 0);
  51. }
  52. return 0;
  53. }

msg_client:

  1. #include < stdio.h >  
  2. #include < stdlib.h >  
  3. #include < sys /msg.h >  
  4.  
  5. // Used to create a unique key
  6. #define MSG_FILE "/etc/passwd"
  7.  
  8. // Message structure
  9. struct msg_form {
  10. long mtype;
  11. char mtext[256];
  12. };
  13.  
  14. int main()
  15. {
  16. int msqid;
  17. key_t key;
  18. struct msg_form msg;
  19.  
  20. // Get the key value
  21. if (( key = ftok (MSG_FILE, 'z')) <   0 )
  22. {
  23. perror("ftok error");
  24. exit(1);
  25. }
  26.  
  27. // Print key value
  28. printf("Message Queue - Client key is: %d.\n", key);
  29.  
  30. // Open the message queue
  31. if (( msqid = msgget (key, IPC_CREAT|0777)) == -1)
  32. {
  33. perror("msgget error");
  34. exit(1);
  35. }
  36.  
  37. // Print message queue ID and process ID
  38. printf("My msqid is: %d.\n", msqid);
  39. printf("My pid is: %d.\n", getpid());
  40.  
  41. // Add a message of type 888
  42. msg.mtype = 888 ;
  43. sprintf(msg.mtext, "hello, I'm client %d", getpid());
  44. msgsnd(msqid, &msg, sizeof(msg.mtext), 0);
  45.  
  46. // Read messages of type 777
  47. msgrcv(msqid, &msg, 256, 999, 0);
  48. printf("Client: receive msg.mtext is: %s.\n", msg.mtext);
  49. printf("Client: receive msg.mtype is: %d.\n", msg.mtype);
  50. return 0;
  51. }

4. Semaphore

Semaphore is different from the IPC structure already introduced. It is a counter. Semaphore is used to achieve mutual exclusion and synchronization between processes, rather than to store inter-process communication data.

1. Features

  • Semaphores are used for synchronization between processes. If you want to transfer data between processes, you need to combine shared memory
  • Semaphores are based on the PV operation of the operating system, and the program's operations on semaphores are all atomic operations.
  • Each PV operation on the semaphore is not limited to adding or subtracting 1 from the semaphore value, but can also add or subtract any positive integer.
  • Support semaphore groups

2. Prototype

The simplest semaphore is a variable that can only take values ​​of 0 and 1. This is also the most common form of semaphore, called a binary semaphore. A semaphore that can take multiple positive integers is called a universal semaphore.

The semaphore functions under Linux all operate on a general semaphore array rather than on a single binary semaphore.

  1. #include < sys /sem.h >  
  2. // Create or get a semaphore group: if successful, returns the semaphore set ID, if failed, returns -1
  3. int semget(key_t key, int num_sems, int sem_flags);
  4. // Operate on the semaphore group to change the value of the semaphore: return 0 if successful, -1 if failed
  5. int semop(int semid, struct sembuf semoparray[], size_t numops);
  6. // Control semaphore information
  7. int semctl(int semid, int sem_num, int cmd, ...);

When semget creates a new semaphore set, the number of semaphores in the set (num_sems) must be specified, usually 1; if an existing set is referenced, num_sems is specified as 0.

In the semop function, the sembuf structure is defined as follows:

  1. struct sembuf
  2. {
  3. short sem_num; // The corresponding sequence number in the semaphore group, 0 to sem_nums-1
  4. short sem_op; // The amount of change in the semaphore value in one operation
  5. short sem_flg; // IPC_NOWAIT, SEM_UNDO
  6. }

5. Shared Memory

1. Features

  • Shared memory is the fastest type of IPC because the process directly accesses the memory
  • Because multiple processes can operate simultaneously, synchronization is required
  • Semaphores + shared memory are often used together, semaphores are used to synchronize access to shared memory

2. Prototype

  1. #include < sys /shm.h >  
  2. // Create or get a shared memory: return the shared memory ID if successful, or -1 if failed
  3. int shmget(key_t key, size_t size, int flag);
  4. // Connect shared memory to the address space of the current process: return a pointer to the shared memory if successful, or -1 if failed
  5. void *shmat(int shm_id, const void *addr, int flag);
  6. // Disconnect from shared memory: return 0 if successful, -1 if failed
  7. int shmdt(void *addr);
  8. // Control the related information of shared memory: return 0 if successful, return -1 if failed
  9. int shmctl(int shm_id, int cmd, struct shmid_ds *buf);

When creating a shared memory segment using the shmget function, its size must be specified; if an existing shared memory segment is referenced, the size must be specified as 0.

Once a shared memory is created, it cannot be accessed by any process. You must use the shmat function to connect the shared memory to the address space of the current process. After the connection is successful, the shared memory area object is mapped to the address space of the calling process, and then it can be accessed like a local space.

The shmdt function is used to disconnect the connection established by shmat. Note that this does not delete the shared memory from the system, but only makes the current process no longer able to access the shared memory.

The shmctl function can perform various operations on shared memory, and performs corresponding operations according to the parameter cmd. The commonly used one is IPC_RMID (delete the shared memory from the system).

<<:  How to configure basic IPv6 addresses? Learn in one minute

>>:  The rise and fall of online storage in the past decade: persistence is not easy, and free storage is not sustainable

Recommend

Smartphones supporting Wi-Fi 6/6E will dominate the market by 2025

Wi-Fi 6E will be commercially available in 2021. ...

GSMA: Global 5G connections will reach 1.8 billion by 2025

According to a new study from GSMA, global 5G con...

The basics of optical fiber you must know

1. Classification of optical fiber Optical fibers...

What is the difference between HTTP and RPC?

HTTP (Hyper Text Transfer Protocol), also known a...

Snapchat QUIC Practice: Small Protocol Solve Big Problems

Friends who are familiar with the Internet should...

Ten underutilized SD-WAN features

SD-WAN is more than just an alternative to Multip...