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