프로세스 간 통신-명명 된 파이프

파이프는 관련 프로세스 간의 통신을위한 것입니다. 한 터미널에서 클라이언트 프로그램을 실행하고 다른 터미널에서 서버 프로그램을 실행하려는 경우와 같이 관련없는 프로세스 통신에 파이프를 사용할 수 있습니까? 대답은 아니요입니다. 그러면 관련없는 프로세스 통신을 어떻게 달성 할 수 있습니까? 간단한 대답은 Named Pipes입니다. 이것은 관련 프로세스에서 작동하지만 관련 프로세스 통신에 명명 된 파이프를 사용하는 것은 의미가 없습니다.

단방향 통신에는 하나의 파이프를 사용하고 양방향 통신에는 두 개의 파이프를 사용했습니다. 명명 된 파이프에 동일한 조건이 적용됩니까? 대답은 아니오입니다. Named Pipe는 양방향 통신을 지원하므로 양방향 통신 (서버와 클라이언트 간의 통신, 클라이언트와 서버의 동시 통신)에 사용할 수있는 단일 명명 된 파이프를 사용할 수 있습니다.

명명 된 파이프의 또 다른 이름은 FIFO (First-In-First-Out). 일종의 특수 파일 인 명명 된 파이프를 만드는 시스템 호출 (mknod ())을 살펴 보겠습니다.

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>

int mknod(const char *pathname, mode_t mode, dev_t dev);

이 시스템 호출은 일반 파일, 장치 파일 또는 FIFO와 같은 특수 파일 또는 파일 시스템 노드를 만듭니다. 시스템 호출에 대한 인수는 경로 이름, 모드 및 dev입니다. 모드 및 장치 정보의 속성과 함께 경로 이름. 경로 이름은 상대적이며 디렉토리가 지정되지 않은 경우 현재 디렉토리에 생성됩니다. 지정된 모드는 다음 표에 언급 된 파일 유형 및 파일 모드와 같은 파일 유형을 지정하는 파일 모드입니다. dev 필드는 주요 및 보조 장치 번호와 같은 장치 정보를 지정하는 것입니다.

파일 유형 기술 파일 유형 기술
S_IFBLK 특별 블록 S_IFREG 일반 파일
S_IFCHR 캐릭터 스페셜 S_IFDIR 예배 규칙서
S_IFIFO FIFO 스페셜 S_IFLNK 심볼릭 링크
파일 모드 기술 파일 모드 기술
S_IRWXU 소유자 별 읽기, 쓰기, 실행 / 검색 S_IWGRP 쓰기 권한, 그룹
S_IRUSR 읽기 권한, 소유자 S_IXGRP 실행 / 검색 권한, 그룹
S_IWUSR 쓰기 권한, 소유자 S_IRWXO 다른 사람에 의한 읽기, 쓰기, 실행 / 검색
S_IXUSR 실행 / 검색 권한, 소유자 S_IROTH 읽기 권한, 기타
S_IRWXG 그룹 별 읽기, 쓰기, 실행 / 검색 S_IWOTH 쓰기 권한, 기타
S_IRGRP 읽기 권한, 그룹 S_IXOTH 실행 / 검색 권한, 기타

파일 모드는 0XYZ와 같은 8 진수 표기법으로도 나타낼 수 있습니다. 여기서 X는 소유자, Y는 그룹, Z는 기타를 나타냅니다. X, Y 또는 Z 값의 범위는 0에서 7까지입니다. 읽기, 쓰기 및 실행 값은 각각 4, 2, 1입니다. 읽기, 쓰기 및 실행을 조합하여 필요한 경우 그에 따라 값을 추가하십시오.

0640을 언급하면 ​​소유자는 읽기 및 쓰기 (4 + 2 = 6), 그룹은 읽기 (4), 다른 사용자는 권한 없음 (0)을 의미합니다.

이 호출은 성공하면 0을 반환하고 실패하면 -1을 반환합니다. 실패의 원인을 알려면 errno 변수 또는 perror () 함수로 확인하십시오.

#include <sys/types.h>
#include <sys/stat.h>

int mkfifo(const char *pathname, mode_t mode)

이 라이브러리 함수는 명명 된 파이프에 사용되는 FIFO 특수 파일을 생성합니다. 이 함수에 대한 인수는 파일 이름과 모드입니다. 파일 이름은 절대 경로 또는 상대 경로 일 수 있습니다. 전체 경로 이름 (또는 절대 경로)을 지정하지 않으면 실행중인 프로세스의 현재 폴더에 파일이 생성됩니다. 파일 모드 정보는 mknod () 시스템 호출에 설명되어 있습니다.

이 호출은 성공하면 0을 반환하고 실패하면 -1을 반환합니다. 실패의 원인을 알려면 errno 변수 또는 perror () 함수로 확인하십시오.

한 터미널에서 서버를 실행하고 다른 터미널에서 클라이언트를 실행하는 프로그램을 고려해 보겠습니다. 프로그램은 단방향 통신 만 수행합니다. 클라이언트는 사용자 입력을 수락하고 메시지를 서버에 보내고 서버는 출력에 메시지를 인쇄합니다. 사용자가 문자열 "end"를 입력 할 때까지 프로세스가 계속됩니다.

예를 들어 이것을 이해합시다.

Step 1 − 두 개의 프로세스를 생성합니다. 하나는 fifoserver이고 다른 하나는 fifoclient입니다.

Step 2 − 서버 프로세스는 다음을 수행합니다 −

  • 생성되지 않은 경우 이름이 "MYFIFO"인 명명 된 파이프 (시스템 호출 mknod () 사용)를 생성합니다.

  • 읽기 전용 목적으로 명명 된 파이프를 엽니 다.

  • 여기에서 소유자에 대한 읽기 및 쓰기 권한으로 FIFO를 생성했습니다. 그룹에 대한 읽기 및 기타에 대한 권한 없음.

  • 클라이언트의 메시지를 무한히 기다립니다.

  • 클라이언트로부터받은 메시지가 "종료"가 아닌 경우 메시지를 인쇄합니다. 메시지가 "종료"이면 fifo를 닫고 프로세스를 종료합니다.

Step 3 − 클라이언트 프로세스는 다음을 수행합니다 −

  • 쓰기 전용으로 명명 된 파이프를 엽니 다.

  • 사용자의 문자열을받습니다.

  • 사용자가 "종료"를 입력했는지 또는 "종료"가 아닌지 확인합니다. 어느 쪽이든 서버에 메시지를 보냅니다. 그러나 문자열이 "end"인 경우 FIFO를 닫고 프로세스도 종료합니다.

  • 사용자가 문자열 "end"를 입력 할 때까지 무한 반복합니다.

이제 FIFO 서버 파일을 살펴 보겠습니다.

/* Filename: fifoserver.c */
#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>

#define FIFO_FILE "MYFIFO"
int main() {
   int fd;
   char readbuf[80];
   char end[10];
   int to_end;
   int read_bytes;
   
   /* Create the FIFO if it does not exist */
   mknod(FIFO_FILE, S_IFIFO|0640, 0);
   strcpy(end, "end");
   while(1) {
      fd = open(FIFO_FILE, O_RDONLY);
      read_bytes = read(fd, readbuf, sizeof(readbuf));
      readbuf[read_bytes] = '\0';
      printf("Received string: \"%s\" and length is %d\n", readbuf, (int)strlen(readbuf));
      to_end = strcmp(readbuf, end);
      if (to_end == 0) {
         close(fd);
         break;
      }
   }
   return 0;
}

컴파일 및 실행 단계

Received string: "this is string 1" and length is 16
Received string: "fifo test" and length is 9
Received string: "fifo client and server" and length is 22
Received string: "end" and length is 3

이제 FIFO 클라이언트 샘플 코드를 살펴 보겠습니다.

/* Filename: fifoclient.c */
#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>

#define FIFO_FILE "MYFIFO"
int main() {
   int fd;
   int end_process;
   int stringlen;
   char readbuf[80];
   char end_str[5];
   printf("FIFO_CLIENT: Send messages, infinitely, to end enter \"end\"\n");
   fd = open(FIFO_FILE, O_CREAT|O_WRONLY);
   strcpy(end_str, "end");
   
   while (1) {
      printf("Enter string: ");
      fgets(readbuf, sizeof(readbuf), stdin);
      stringlen = strlen(readbuf);
      readbuf[stringlen - 1] = '\0';
      end_process = strcmp(readbuf, end_str);
      
      //printf("end_process is %d\n", end_process);
      if (end_process != 0) {
         write(fd, readbuf, strlen(readbuf));
         printf("Sent string: \"%s\" and string length is %d\n", readbuf, (int)strlen(readbuf));
      } else {
         write(fd, readbuf, strlen(readbuf));
         printf("Sent string: \"%s\" and string length is %d\n", readbuf, (int)strlen(readbuf));
         close(fd);
         break;
      }
   }
   return 0;
}

도착한 출력을 살펴 보겠습니다.

컴파일 및 실행 단계

FIFO_CLIENT: Send messages, infinitely, to end enter "end"
Enter string: this is string 1
Sent string: "this is string 1" and string length is 16
Enter string: fifo test
Sent string: "fifo test" and string length is 9
Enter string: fifo client and server
Sent string: "fifo client and server" and string length is 22
Enter string: end
Sent string: "end" and string length is 3

명명 된 파이프를 사용한 양방향 통신

파이프 간의 통신은 단방향을 의미합니다. 파이프는 일반적으로 단방향 통신으로 제한되었으며 양방향 통신을 위해서는 적어도 두 개의 파이프가 필요합니다. 파이프는 상호 관련된 프로세스 전용입니다. 파이프는 관련없는 프로세스 통신에 사용할 수 없습니다. 예를 들어 한 터미널에서 한 프로세스를 실행하고 다른 터미널에서 다른 프로세스를 실행하려는 경우 파이프로는 불가능합니다. 두 프로세스간에 간단한 방식으로 통신 할 수있는 간단한 방법이 있습니까? 대답은 '예'입니다. 명명 된 파이프는 둘 이상의 관련되지 않은 프로세스 간의 통신을 의미하며 양방향 통신을 가질 수도 있습니다.

이미 명명 된 파이프 사이의 단방향 통신, 즉 클라이언트에서 서버로의 메시지를 보았습니다. 이제 양방향 통신, 즉 클라이언트가 서버에 메시지를 전송하고 서버가 메시지를 수신하고 동일한 명명 된 파이프를 사용하여 다른 메시지를 클라이언트에 다시 전송하는 방식을 살펴 보겠습니다.

다음은 예입니다-

Step 1 − 두 개의 프로세스를 생성합니다. 하나는 fifoserver_twoway이고 다른 하나는 fifoclient_twoway입니다.

Step 2 − 서버 프로세스는 다음을 수행합니다 −

  • 생성되지 않은 경우 / tmp 디렉토리에 이름이 "fifo_twoway"인 명명 된 파이프 (라이브러리 함수 mkfifo () 사용)를 생성합니다.

  • 읽기 및 쓰기 목적으로 명명 된 파이프를 엽니 다.

  • 여기에서 소유자에 대한 읽기 및 쓰기 권한으로 FIFO를 생성했습니다. 그룹에 대한 읽기 및 기타에 대한 권한 없음.

  • 클라이언트의 메시지를 무한히 기다립니다.

  • 클라이언트로부터받은 메시지가 "end"가 아닌 경우 메시지를 인쇄하고 문자열을 반대로합니다. 반전 된 문자열은 클라이언트로 다시 전송됩니다. 메시지가 "종료"이면 fifo를 닫고 프로세스를 종료합니다.

Step 3 − 클라이언트 프로세스는 다음을 수행합니다 −

  • 읽기 및 쓰기 목적으로 명명 된 파이프를 엽니 다.

  • 사용자로부터 문자열을받습니다.

  • 사용자가 "종료"를 입력했는지 또는 "종료"가 아닌지 확인합니다. 어느 쪽이든 서버에 메시지를 보냅니다. 그러나 문자열이 "end"인 경우 FIFO를 닫고 프로세스도 종료합니다.

  • 메시지가 "종료"가 아닌 것으로 전송되면 클라이언트로부터 메시지 (반전 된 문자열)를 기다렸다가 반전 된 문자열을 인쇄합니다.

  • 사용자가 문자열 "end"를 입력 할 때까지 무한 반복합니다.

이제 FIFO 서버 샘플 코드를 살펴 보겠습니다.

/* Filename: fifoserver_twoway.c */
#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>

#define FIFO_FILE "/tmp/fifo_twoway"
void reverse_string(char *);
int main() {
   int fd;
   char readbuf[80];
   char end[10];
   int to_end;
   int read_bytes;
   
   /* Create the FIFO if it does not exist */
   mkfifo(FIFO_FILE, S_IFIFO|0640);
   strcpy(end, "end");
   fd = open(FIFO_FILE, O_RDWR);
   while(1) {
      read_bytes = read(fd, readbuf, sizeof(readbuf));
      readbuf[read_bytes] = '\0';
      printf("FIFOSERVER: Received string: \"%s\" and length is %d\n", readbuf, (int)strlen(readbuf));
      to_end = strcmp(readbuf, end);
      
      if (to_end == 0) {
         close(fd);
         break;
      }
      reverse_string(readbuf);
      printf("FIFOSERVER: Sending Reversed String: \"%s\" and length is %d\n", readbuf, (int) strlen(readbuf));
      write(fd, readbuf, strlen(readbuf));
      /*
      sleep - This is to make sure other process reads this, otherwise this
      process would retrieve the message
      */
      sleep(2);
   }
   return 0;
}

void reverse_string(char *str) {
   int last, limit, first;
   char temp;
   last = strlen(str) - 1;
   limit = last/2;
   first = 0;
   
   while (first < last) {
      temp = str[first];
      str[first] = str[last];
      str[last] = temp;
      first++;
      last--;
   }
   return;
}

컴파일 및 실행 단계

FIFOSERVER: Received string: "LINUX IPCs" and length is 10
FIFOSERVER: Sending Reversed String: "sCPI XUNIL" and length is 10
FIFOSERVER: Received string: "Inter Process Communication" and length is 27
FIFOSERVER: Sending Reversed String: "noitacinummoC ssecorP retnI" and length is 27
FIFOSERVER: Received string: "end" and length is 3

이제 FIFO 클라이언트 샘플 코드를 살펴 보겠습니다.

/* Filename: fifoclient_twoway.c */
#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>

#define FIFO_FILE "/tmp/fifo_twoway"
int main() {
   int fd;
   int end_process;
   int stringlen;
   int read_bytes;
   char readbuf[80];
   char end_str[5];
   printf("FIFO_CLIENT: Send messages, infinitely, to end enter \"end\"\n");
   fd = open(FIFO_FILE, O_CREAT|O_RDWR);
   strcpy(end_str, "end");
   
   while (1) {
      printf("Enter string: ");
      fgets(readbuf, sizeof(readbuf), stdin);
      stringlen = strlen(readbuf);
      readbuf[stringlen - 1] = '\0';
      end_process = strcmp(readbuf, end_str);
      
      //printf("end_process is %d\n", end_process);
      if (end_process != 0) {
         write(fd, readbuf, strlen(readbuf));
         printf("FIFOCLIENT: Sent string: \"%s\" and string length is %d\n", readbuf, (int)strlen(readbuf));
         read_bytes = read(fd, readbuf, sizeof(readbuf));
         readbuf[read_bytes] = '\0';
         printf("FIFOCLIENT: Received string: \"%s\" and length is %d\n", readbuf, (int)strlen(readbuf));
      } else {
         write(fd, readbuf, strlen(readbuf));
         printf("FIFOCLIENT: Sent string: \"%s\" and string length is %d\n", readbuf, (int)strlen(readbuf));
         close(fd);
         break;
      }
   }
   return 0;
}

컴파일 및 실행 단계

FIFO_CLIENT: Send messages, infinitely, to end enter "end"
Enter string: LINUX IPCs
FIFOCLIENT: Sent string: "LINUX IPCs" and string length is 10
FIFOCLIENT: Received string: "sCPI XUNIL" and length is 10
Enter string: Inter Process Communication
FIFOCLIENT: Sent string: "Inter Process Communication" and string length is 27
FIFOCLIENT: Received string: "noitacinummoC ssecorP retnI" and length is 27
Enter string: end
FIFOCLIENT: Sent string: "end" and string length is 3