공유 메모리
공유 메모리는 둘 이상의 프로세스간에 공유되는 메모리입니다. 하지만 왜 우리는 기억이나 다른 의사 소통 수단을 공유해야합니까?
반복해서 말하면, 각 프로세스는 자체 주소 공간을 가지고 있습니다. 어떤 프로세스가 자체 주소 공간에서 다른 프로세스로 일부 정보와 통신하려는 경우 IPC (프로세스 간 통신) 기술로만 가능합니다. 이미 알고 있듯이 커뮤니케이션은 관련되거나 관련되지 않은 프로세스간에 이루어질 수 있습니다.
일반적으로 상호 관련된 프로세스 통신은 파이프 또는 명명 된 파이프를 사용하여 수행됩니다. 관련없는 프로세스 (한 터미널에서 실행되는 프로세스와 다른 터미널에서 다른 프로세스) 통신은 명명 된 파이프를 사용하거나 공유 메모리 및 메시지 큐의 인기있는 IPC 기술을 통해 수행 될 수 있습니다.
파이프 및 명명 된 파이프의 IPC 기술을 살펴 보았으므로 이제 나머지 IPC 기술 즉, 공유 메모리, 메시지 큐, 세마포, 신호 및 메모리 매핑을 알아야합니다.
이 장에서는 공유 메모리에 대해 모두 알게 될 것입니다.
우리는 둘 이상의 프로세스 사이에서 통신하기 위해 공유 메모리를 사용한다는 것을 알고 있지만 공유 메모리를 사용하기 전에 시스템 호출로 수행해야하는 작업을 살펴 보겠습니다.
공유 메모리 세그먼트를 생성하거나 이미 생성 된 공유 메모리 세그먼트 (shmget ())를 사용합니다.
이미 생성 된 공유 메모리 세그먼트 (shmat ())에 프로세스를 연결합니다.
이미 연결된 공유 메모리 세그먼트 (shmdt ())에서 프로세스를 분리합니다.
공유 메모리 세그먼트 (shmctl ())에 대한 제어 작업
공유 메모리와 관련된 시스템 호출에 대한 몇 가지 세부 사항을 살펴 보겠습니다.
#include <sys/ipc.h>
#include <sys/shm.h>
int shmget(key_t key, size_t size, int shmflg)
위의 시스템 호출은 System V 공유 메모리 세그먼트를 생성하거나 할당합니다. 전달해야하는 인수는 다음과 같습니다.
그만큼 first argument, key,공유 메모리 세그먼트를 인식합니다. 키는 임의의 값이거나 라이브러리 함수 ftok ()에서 파생 될 수있는 값일 수 있습니다. 키는 또한 IPC_PRIVATE가 될 수 있습니다. 즉, 프로세스를 서버 및 클라이언트로 실행 (상위 및 하위 관계), 즉 상호 관련 프로세스 통신을 의미합니다. 클라이언트가이 키와 함께 공유 메모리를 사용하려면 서버의 하위 프로세스 여야합니다. 또한 부모가 공유 메모리를 얻은 후에 자식 프로세스를 만들어야합니다.
그만큼 second argument, size, PAGE_SIZE의 배수로 반올림 된 공유 메모리 세그먼트의 크기입니다.
그만큼 third argument, shmflg,IPC_CREAT (새 세그먼트 생성) 또는 IPC_EXCL (IPC_CREAT와 함께 사용하여 새 세그먼트를 생성하고 세그먼트가 이미 존재하는 경우 호출이 실패 함)과 같은 필수 공유 메모리 플래그를 지정합니다. 권한도 전달해야합니다.
Note − 권한에 대한 자세한 내용은 이전 섹션을 참조하십시오.
이 호출은 성공하면 유효한 공유 메모리 식별자 (공유 메모리의 추가 호출에 사용됨)를 반환하고 실패하면 -1을 반환합니다. 실패의 원인을 알려면 errno 변수 또는 perror () 함수로 확인하십시오.
#include <sys/types.h>
#include <sys/shm.h>
void * shmat(int shmid, const void *shmaddr, int shmflg)
위의 시스템 호출은 System V 공유 메모리 세그먼트에 대한 공유 메모리 작업을 수행합니다. 즉, 호출 프로세스의 주소 공간에 공유 메모리 세그먼트를 연결합니다. 전달해야하는 인수는 다음과 같습니다.
The first argument, shmid,공유 메모리 세그먼트의 식별자입니다. 이 id는 shmget () 시스템 호출의 반환 값인 공유 메모리 식별자입니다.
The second argument, shmaddr,첨부 주소를 지정하는 것입니다. shmaddr이 NULL이면 시스템은 기본적으로 세그먼트를 연결하는 데 적합한 주소를 선택합니다. shmaddr이 NULL이 아니고 SHM_RND가 shmflg에 지정된 경우 첨부는 SHMLBA (하한 경계 주소)의 가장 가까운 배수의 주소와 같습니다. 그렇지 않으면 shmaddr은 공유 메모리 첨부가 발생 / 시작되는 페이지 정렬 주소 여야합니다.
The third argument, shmflg, SHM_RND (SHMLBA로 주소 반올림) 또는 SHM_EXEC (세그먼트의 내용이 실행되도록 허용) 또는 SHM_RDONLY (읽기 전용 목적으로 세그먼트 연결, 기본적으로 읽기-쓰기)와 같은 필수 공유 메모리 플래그를 지정합니다. 또는 SHM_REMAP (shmaddr에 의해 지정된 범위의 기존 매핑을 대체하고 세그먼트 끝까지 계속됨).
이 호출은 성공하면 연결된 공유 메모리 세그먼트의 주소를 반환하고 실패하면 -1을 반환합니다. 실패의 원인을 알려면 errno 변수 또는 perror () 함수로 확인하십시오.
#include <sys/types.h>
#include <sys/shm.h>
int shmdt(const void *shmaddr)
위의 시스템 호출은 호출 프로세스의 주소 공간에서 공유 메모리 세그먼트를 분리하는 System V 공유 메모리 세그먼트에 대한 공유 메모리 작업을 수행합니다. 전달해야하는 인수는 다음과 같습니다.
인수 shmaddr은 분리 될 공유 메모리 세그먼트의 주소입니다. 분리 될 세그먼트는 shmat () 시스템 호출에 의해 반환 된 주소 여야합니다.
이 호출은 성공하면 0을 반환하고 실패하면 -1을 반환합니다. 실패의 원인을 알려면 errno 변수 또는 perror () 함수로 확인하십시오.
#include <sys/ipc.h>
#include <sys/shm.h>
int shmctl(int shmid, int cmd, struct shmid_ds *buf)
위의 시스템 호출은 System V 공유 메모리 세그먼트에 대한 제어 작업을 수행합니다. 다음 인수를 전달해야합니다-
첫 번째 인수 인 shmid는 공유 메모리 세그먼트의 식별자입니다. 이 id는 shmget () 시스템 호출의 반환 값인 공유 메모리 식별자입니다.
두 번째 인수 cmd는 공유 메모리 세그먼트에서 필요한 제어 작업을 수행하는 명령입니다.
cmd의 유효한 값은 다음과 같습니다.
IPC_STAT− struct shmid_ds의 각 멤버의 현재 값 정보를 buf가 가리키는 전달 된 구조에 복사합니다. 이 명령에는 공유 메모리 세그먼트에 대한 읽기 권한이 필요합니다.
IPC_SET − 구조 buf가 가리키는 사용자 ID, 소유자 그룹 ID, 권한 등을 설정합니다.
IPC_RMID− 파괴 할 세그먼트를 표시합니다. 세그먼트는 마지막 프로세스가 분리 한 후에 만 소멸됩니다.
IPC_INFO − buf가 가리키는 구조의 공유 메모리 제한 및 매개 변수에 대한 정보를 반환합니다.
SHM_INFO − 공유 메모리가 소비 한 시스템 자원에 대한 정보를 포함하는 shm_info 구조를 반환합니다.
세 번째 인수 buf는 struct shmid_ds라는 공유 메모리 구조에 대한 포인터입니다. 이 구조의 값은 cmd에 따라 set 또는 get에 사용됩니다.
이 호출은 전달 된 명령에 따라 값을 반환합니다. IPC_INFO 및 SHM_INFO 또는 SHM_STAT가 성공하면 공유 메모리 세그먼트의 인덱스 또는 식별자 또는 다른 작업의 경우 0을 반환하고 실패의 경우 -1을 반환합니다. 실패의 원인을 알려면 errno 변수 또는 perror () 함수로 확인하십시오.
다음 샘플 프로그램을 살펴 보겠습니다.
두 개의 프로세스를 생성합니다. 하나는 공유 메모리 (shm_write.c)에 쓰기위한 것이고 다른 하나는 공유 메모리 (shm_read.c)에서 읽기위한 것입니다.
프로그램은 쓰기 프로세스 (shm_write.c)를 통해 공유 메모리에 쓰기를 수행하고 읽기 프로세스 (shm_read.c)를 통해 공유 메모리에서 읽기를 수행합니다.
공유 메모리에서 쓰기 프로세스는 1K (및 플래그) 크기의 공유 메모리를 만들고 공유 메모리를 연결합니다.
쓰기 프로세스는 'A'에서 'E'까지 각각 1023 바이트의 알파벳을 공유 메모리에 5 번 씁니다. 마지막 바이트는 버퍼의 끝을 나타냅니다.
읽기 프로세스는 공유 메모리에서 읽고 표준 출력에 씁니다.
읽기 및 쓰기 프로세스 작업이 동시에 수행됩니다.
쓰기 완료 후 쓰기 프로세스가 업데이트되어 공유 메모리에 쓰기 완료를 표시합니다 (struct shmseg의 complete 변수 사용).
읽기 프로세스는 공유 메모리에서 읽기를 수행하고 쓰기 프로세스 완료를 표시 할 때까지 출력에 표시합니다 (struct shmseg의 전체 변수).
단순화를 위해 몇 번 읽기 및 쓰기 프로세스를 수행하고 무한 루프를 방지하고 프로그램을 복잡하게 만듭니다.
다음은 쓰기 프로세스를위한 코드입니다 (공유 메모리에 쓰기 – 파일 : shm_write.c).
/* Filename: shm_write.c */
#include<stdio.h>
#include<sys/ipc.h>
#include<sys/shm.h>
#include<sys/types.h>
#include<string.h>
#include<errno.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#define BUF_SIZE 1024
#define SHM_KEY 0x1234
struct shmseg {
int cnt;
int complete;
char buf[BUF_SIZE];
};
int fill_buffer(char * bufptr, int size);
int main(int argc, char *argv[]) {
int shmid, numtimes;
struct shmseg *shmp;
char *bufptr;
int spaceavailable;
shmid = shmget(SHM_KEY, sizeof(struct shmseg), 0644|IPC_CREAT);
if (shmid == -1) {
perror("Shared memory");
return 1;
}
// Attach to the segment to get a pointer to it.
shmp = shmat(shmid, NULL, 0);
if (shmp == (void *) -1) {
perror("Shared memory attach");
return 1;
}
/* Transfer blocks of data from buffer to shared memory */
bufptr = shmp->buf;
spaceavailable = BUF_SIZE;
for (numtimes = 0; numtimes < 5; numtimes++) {
shmp->cnt = fill_buffer(bufptr, spaceavailable);
shmp->complete = 0;
printf("Writing Process: Shared Memory Write: Wrote %d bytes\n", shmp->cnt);
bufptr = shmp->buf;
spaceavailable = BUF_SIZE;
sleep(3);
}
printf("Writing Process: Wrote %d times\n", numtimes);
shmp->complete = 1;
if (shmdt(shmp) == -1) {
perror("shmdt");
return 1;
}
if (shmctl(shmid, IPC_RMID, 0) == -1) {
perror("shmctl");
return 1;
}
printf("Writing Process: Complete\n");
return 0;
}
int fill_buffer(char * bufptr, int size) {
static char ch = 'A';
int filled_count;
//printf("size is %d\n", size);
memset(bufptr, ch, size - 1);
bufptr[size-1] = '\0';
if (ch > 122)
ch = 65;
if ( (ch >= 65) && (ch <= 122) ) {
if ( (ch >= 91) && (ch <= 96) ) {
ch = 65;
}
}
filled_count = strlen(bufptr);
//printf("buffer count is: %d\n", filled_count);
//printf("buffer filled is:%s\n", bufptr);
ch++;
return filled_count;
}
컴파일 및 실행 단계
Writing Process: Shared Memory Write: Wrote 1023 bytes
Writing Process: Shared Memory Write: Wrote 1023 bytes
Writing Process: Shared Memory Write: Wrote 1023 bytes
Writing Process: Shared Memory Write: Wrote 1023 bytes
Writing Process: Shared Memory Write: Wrote 1023 bytes
Writing Process: Wrote 5 times
Writing Process: Complete
다음은 읽기 프로세스를위한 코드입니다 (공유 메모리에서 읽기 및 표준 출력에 쓰기 – 파일 : shm_read.c).
/* Filename: shm_read.c */
#include<stdio.h>
#include<sys/ipc.h>
#include<sys/shm.h>
#include<sys/types.h>
#include<string.h>
#include<errno.h>
#include<stdlib.h>
#define BUF_SIZE 1024
#define SHM_KEY 0x1234
struct shmseg {
int cnt;
int complete;
char buf[BUF_SIZE];
};
int main(int argc, char *argv[]) {
int shmid;
struct shmseg *shmp;
shmid = shmget(SHM_KEY, sizeof(struct shmseg), 0644|IPC_CREAT);
if (shmid == -1) {
perror("Shared memory");
return 1;
}
// Attach to the segment to get a pointer to it.
shmp = shmat(shmid, NULL, 0);
if (shmp == (void *) -1) {
perror("Shared memory attach");
return 1;
}
/* Transfer blocks of data from shared memory to stdout*/
while (shmp->complete != 1) {
printf("segment contains : \n\"%s\"\n", shmp->buf);
if (shmp->cnt == -1) {
perror("read");
return 1;
}
printf("Reading Process: Shared Memory: Read %d bytes\n", shmp->cnt);
sleep(3);
}
printf("Reading Process: Reading Done, Detaching Shared Memory\n");
if (shmdt(shmp) == -1) {
perror("shmdt");
return 1;
}
printf("Reading Process: Complete\n");
return 0;
}
컴파일 및 실행 단계
segment contains :
"AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA
AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA"
Reading Process: Shared Memory: Read 1023 bytes
segment contains :
"BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB
BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB
BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB
BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB
BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB
BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB
BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB
BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB
BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB
BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB
BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB
BBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBBB"
Reading Process: Shared Memory: Read 1023 bytes
segment contains :
"CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC
CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC"
Reading Process: Shared Memory: Read 1023 bytes
segment contains :
"DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD
DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD
DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD
DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD
DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD
DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD
DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD
DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD
DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD
DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD
DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD
DDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDDD"
Reading Process: Shared Memory: Read 1023 bytes
segment contains :
"EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE
EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE
EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE
EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE
EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE
EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE
EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE
EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE
EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE
EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE
EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE
EEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEEE"
Reading Process: Shared Memory: Read 1023 bytes
Reading Process: Reading Done, Detaching Shared Memory
Reading Process: Complete