इंटर प्रोसेस कम्युनिकेशन - सेमफोरस
सबसे पहला सवाल जो दिमाग में आता है, वह यह है कि हमें सेमाफोर की आवश्यकता क्यों है? एक सरल उत्तर, कई प्रक्रियाओं के बीच साझा किए गए महत्वपूर्ण / सामान्य क्षेत्र की रक्षा करना।
आइए मान लेते हैं, कई प्रक्रियाएं समान क्षेत्र के कोड का उपयोग कर रही हैं और यदि सभी समानांतर रूप से एक्सेस करना चाहते हैं तो परिणाम ओवरलैप हो जाता है। उदाहरण के लिए, कई उपयोगकर्ता केवल एक प्रिंटर (सामान्य / महत्वपूर्ण अनुभाग) का उपयोग कर रहे हैं, 3 उपयोगकर्ताओं का कहना है कि एक ही समय में 3 नौकरियां दी जाती हैं, यदि सभी नौकरियां समानांतर रूप से शुरू होती हैं, तो एक उपयोगकर्ता आउटपुट दूसरे के साथ ओवरलैप किया जाता है। इसलिए, हमें इस बात की रक्षा करने की आवश्यकता है कि सेमीफोर का उपयोग करते हुए, महत्वपूर्ण प्रक्रिया को लॉक करते समय, जब एक प्रक्रिया चल रही हो और अनलॉक हो रही हो। यह प्रत्येक उपयोगकर्ता / प्रक्रिया के लिए दोहराया जाएगा ताकि एक नौकरी दूसरी नौकरी के साथ ओवरलैप न हो।
मूल रूप से सेमाफोर को दो प्रकारों में वर्गीकृत किया जाता है -
Binary Semaphores - केवल दो राज्य 0 & 1, यानी, लॉक / अनलॉक या उपलब्ध / अनुपलब्ध, म्यूटेक्स कार्यान्वयन।
Counting Semaphores - सेमाफोर जो मनमाने संसाधन गणना की अनुमति देते हैं, गिनती के सेमाफोर कहलाते हैं।
मान लें कि हमारे पास 5 प्रिंटर हैं (यह समझने के लिए कि 1 प्रिंटर केवल 1 नौकरी स्वीकार करता है) और हमें प्रिंट करने के लिए 3 नौकरियां मिलीं। अब 3 प्रिंटर (1 प्रत्येक) के लिए 3 नौकरी दी जाएगी। फिर से 4 नौकरियां आईं जबकि यह प्रगति पर है। अब, उपलब्ध 2 प्रिंटरों में से, 2 कार्य शेड्यूल किए गए हैं और हमें 2 और नौकरियों के साथ छोड़ दिया गया है, जो संसाधन / प्रिंटर उपलब्ध होने के बाद ही पूरा होगा। संसाधन उपलब्धता के अनुसार इस तरह के शेड्यूलिंग को गणना के रूप में देखा जा सकता है।
सेमाफोर का उपयोग करके सिंक्रनाइज़ेशन करने के लिए, निम्नलिखित कदम हैं -
Step 1 - एक सेमाफोर बनाएं या पहले से मौजूद सेमाफोर (सेगमेंट) से कनेक्ट करें
Step 2 - सेमाफोर पर परिचालन करें अर्थात संसाधनों का आवंटन या विमोचन करें या प्रतीक्षा करें (semop ())
Step 3 - संदेश कतार (semctl ()) पर नियंत्रण संचालन करें
अब, हमारे पास मौजूद सिस्टम कॉल के साथ इसे जांचें।
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
int semget(key_t key, int nsems, int semflg)
यह सिस्टम कॉल सिस्टम V सेमाफोर सेट बनाता है या आवंटित करता है। निम्नलिखित तर्कों को पारित करने की आवश्यकता है -
पहला तर्क, कुंजी, संदेश कतार को पहचानता है। कुंजी या तो एक मनमाना मूल्य हो सकती है या एक जो लाइब्रेरी फंक्शन ftok () से ली जा सकती है।
दूसरा तर्क, nsems, सेमाफोर की संख्या को निर्दिष्ट करता है। यदि बाइनरी है, तो यह 1 है, 1 सेमीफोर सेट की आवश्यकता का अर्थ है, अन्यथा सेमाफोर सेट की संख्या की आवश्यक गणना के अनुसार।
तीसरा तर्क, semflg, आवश्यक semaphore ध्वज / s निर्दिष्ट करता है जैसे IPC_CREAT (यदि यह मौजूद नहीं है तो semaphore बनाना) या IPC_EXCL (Semacore बनाने के लिए IPC_CREAT के साथ उपयोग किया जाता है और कॉल विफल हो जाता है, यदि कोई semaphore पहले से मौजूद है)। साथ ही परमिशन पास करने की जरूरत है।
Note - अनुमतियों के विवरण के लिए पहले के खंड देखें।
यह कॉल वैध अर्ध-पहचानकर्ता (सफलता के लिए अर्धशतक के आगे के कॉल के लिए प्रयुक्त) और -1 विफलता की स्थिति में लौटेगी। विफलता का कारण जानने के लिए, गलत चर या पेरर () फ़ंक्शन के साथ जांचें।
इस कॉल के संबंध में विभिन्न त्रुटियां हैं EACCESS (अनुमति अस्वीकृत), EEXIST (कतार पहले से मौजूद नहीं है), ENOENT (कतार मौजूद नहीं है), ENOMEM (कतार बनाने के लिए पर्याप्त मेमोरी नहीं), ENOSPC (अधिकतम सेट सीमा) पार हो गई), आदि।
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
int semop(int semid, struct sembuf *semops, size_t nsemops)
यह सिस्टम कॉल सिस्टम V सेमीफ़ायर सेट पर परिचालन करता है। संसाधनों को आवंटित करता है, संसाधनों की प्रतीक्षा करता है या संसाधनों को मुक्त करता है। निम्नलिखित तर्कों को पारित करने की आवश्यकता है -
पहला तर्क, semid, semaphore सेट आइडेंटिफ़ायर को semget () द्वारा इंगित करता है।
दूसरा तर्क, वीर्यपात, अर्धचाल सेट पर किए जाने वाले कार्यों की एक सरणी के लिए सूचक है। संरचना इस प्रकार है -
struct sembuf {
unsigned short sem_num; /* Semaphore set num */
short sem_op; /* Semaphore operation */
short sem_flg; /* Operation flags, IPC_NOWAIT, SEM_UNDO */
};
उपरोक्त संरचना में तत्व, sem_op, उस ऑपरेशन को इंगित करता है जिसे करने की आवश्यकता है -
यदि sem_op –ve है, तो संसाधन आवंटित करें या प्राप्त करें। जब तक पर्याप्त संसाधनों को अन्य प्रक्रियाओं द्वारा मुक्त नहीं किया जाता है, तब तक कॉलिंग प्रक्रिया को अवरुद्ध करता है, ताकि यह प्रक्रिया आवंटित हो सके।
यदि sem_op शून्य है, तो कॉलिंग प्रक्रिया प्रतीक्षा करती है या तब तक सोती है जब तक कि semaphore मान 0 तक नहीं पहुँच जाता।
यदि sem_op + ve है, तो संसाधन जारी करें।
उदाहरण के लिए -
संरचना sembuf sem_lock = {0, -1, SEM_UNDO};
संरचना sembuf sem_unlock = {0, 1, SEM_UNDO};
तीसरा तर्क, nsemops, उस सरणी में परिचालनों की संख्या है।
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/sem.h>
int semctl(int semid, int semnum, int cmd, …)
यह सिस्टम कॉल सिस्टम V सेमाफोर के लिए कंट्रोल ऑपरेशन करता है। निम्नलिखित तर्कों को पारित करने की आवश्यकता है -
पहला तर्क, अर्धविराम, अर्धवृत्त का पहचानकर्ता है। यह आईडी सेमीफोर आइडेंटिफायर है, जो कि सेगमेंट () सिस्टम कॉल का रिटर्न वैल्यू है।
दूसरा तर्क, सेमनम, सेमाफोर की संख्या है। सेमाफोर 0 से गिने जाते हैं।
तीसरा तर्क, सेमीड, सेमाफोर पर आवश्यक नियंत्रण ऑपरेशन करने के लिए कमांड है।
चौथा तर्क, प्रकार, यूनियन सेमन, सीएमडी पर निर्भर करता है। कुछ मामलों के लिए, चौथा तर्क लागू नहीं होता है।
आइए संघ सेमिनार की जाँच करें -
union semun {
int val; /* val for SETVAL */
struct semid_ds *buf; /* Buffer for IPC_STAT and IPC_SET */
unsigned short *array; /* Buffer for GETALL and SETALL */
struct seminfo *__buf; /* Buffer for IPC_INFO and SEM_INFO*/
};
Semid_ds डेटा संरचना जिसे sys / sem.h में परिभाषित किया गया है वह इस प्रकार है -
struct semid_ds {
struct ipc_perm sem_perm; /* Permissions */
time_t sem_otime; /* Last semop time */
time_t sem_ctime; /* Last change time */
unsigned long sem_nsems; /* Number of semaphores in the set */
};
Note - कृपया अन्य डेटा संरचनाओं के लिए मैनुअल पेज देखें।
संघ सेमिन अरग; Cmd के लिए मान्य मान हैं -
IPC_STAT- arg.buf द्वारा बताई गई संरचना में संरचनात्मक semid_ds के प्रत्येक सदस्य के वर्तमान मूल्यों की जानकारी की प्रतिलिपि। इस कमांड के लिए सेमाफोर को पढ़ने की अनुमति आवश्यक है।
IPC_SET - उपयोगकर्ता आईडी, स्वामी की ग्रुप आईडी, अनुमतियाँ, आदि संरचना semid_ds द्वारा इंगित करता है।
IPC_RMID - सेमाफोरस सेट को हटाता है।
IPC_INFO - arg .__ buf द्वारा बताए गए संरचना semid_ds में अर्ध-सीमा और मापदंडों की जानकारी देता है।
SEM_INFO - एक अर्धचालक संरचना लौटाता है जिसमें अर्धचालक द्वारा उपभोग किए गए सिस्टम संसाधनों के बारे में जानकारी होती है।
यह कॉल पारित आदेश के आधार पर मान (गैर-नकारात्मक मान) लौटाएगा। सफलता के बाद, IPC_INFO और SEM_INFO या SEM_STAT, सेमाफोर के अनुसार उच्चतम उपयोग किए गए प्रविष्टि के सूचकांक या पहचानकर्ता या GETNCNT के लिए semncnt का मूल्य या GETPID के लिए sempid का मान या सफलता और अन्य कार्यों के लिए GETVAL 0 के लिए सेमेवल के मान को लौटाता है - 1 विफलता के मामले में। विफलता का कारण जानने के लिए, गलत चर या पेरर () फ़ंक्शन के साथ जांचें।
कोड को देखने से पहले, आइए हम इसके कार्यान्वयन को समझते हैं -
बच्चे और माता-पिता का कहना है कि दो प्रक्रियाएं बनाएं।
साझा मेमोरी में पढ़ने / लिखने की प्रक्रिया के अंत को इंगित करने के लिए काउंटर और अन्य झंडे को संग्रहीत करने के लिए मुख्य रूप से आवश्यक साझा मेमोरी बनाएं।
काउंटर को माता-पिता और बच्चे दोनों प्रक्रियाओं द्वारा गणना द्वारा बढ़ाया जाता है। गिनती या तो कमांड लाइन तर्क के रूप में पारित की जाती है या डिफ़ॉल्ट के रूप में ली जाती है (यदि कमांड लाइन तर्क के रूप में पारित नहीं हुई है या मान 10000 से कम है)। माता-पिता और बच्चे दोनों की साझा स्मृति को एक ही समय में, समानांतर में, यह सुनिश्चित करने के लिए निश्चित नींद के समय के साथ बुलाया जाता है।
चूंकि, माता-पिता और बच्चे दोनों द्वारा काउंटर को 1 के चरणों में बढ़ाया जाता है, अंतिम मूल्य काउंटर से दोगुना होना चाहिए। चूंकि, माता-पिता और बच्चे दोनों एक ही समय में ऑपरेशन करते हैं, इसलिए काउंटर को आवश्यकतानुसार नहीं बढ़ाया जाता है। इसलिए, हमें एक प्रक्रिया की पूर्णता सुनिश्चित करने की आवश्यकता है, उसके बाद अन्य प्रक्रिया पूरी होगी।
उपरोक्त सभी कार्यान्वयन shm_write_cntr.c फ़ाइल में किए जाते हैं
काउंटर मूल्य फ़ाइल shm_read_cntr.c में लागू किया गया है या नहीं इसकी जाँच करें
पूर्णता सुनिश्चित करने के लिए, फ़ाइल shm_write_cntr_with_sem.c में सेमाफोर कार्यक्रम कार्यान्वित किया जाता है। पूरी प्रक्रिया को पूरा करने के बाद सेमाफोर को हटा दें (पढ़ने के बाद अन्य प्रोग्राम से किया जाता है)
चूंकि, हमारे पास साझा मेमोरी में काउंटर के मूल्य को पढ़ने के लिए अलग-अलग फाइलें हैं और लिखने से कोई प्रभाव नहीं पड़ता है, पढ़ने का कार्यक्रम एक ही रहता है (shm_read_cntr.c)
एक टर्मिनल में लेखन कार्यक्रम और दूसरे टर्मिनल से रीडिंग प्रोग्राम को निष्पादित करना हमेशा बेहतर होता है। चूंकि, कार्यक्रम लेखन और पढ़ने की प्रक्रिया पूरी होने के बाद ही निष्पादन को पूरा करता है, इसलिए प्रोग्राम को पूरी तरह से निष्पादित करने के बाद प्रोग्राम को चलाना ठीक है। लिखने का कार्यक्रम तब तक इंतजार करेगा जब तक कि पढ़ा हुआ कार्यक्रम न चला जाए और उसके पूरा होने के बाद ही खत्म हो।
कार्यक्रम बिना सेमाफोर के।
/* Filename: shm_write_cntr.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 SHM_KEY 0x12345
struct shmseg {
int cntr;
int write_complete;
int read_complete;
};
void shared_memory_cntr_increment(int pid, struct shmseg *shmp, int total_count);
int main(int argc, char *argv[]) {
int shmid;
struct shmseg *shmp;
char *bufptr;
int total_count;
int sleep_time;
pid_t pid;
if (argc != 2)
total_count = 10000;
else {
total_count = atoi(argv[1]);
if (total_count < 10000)
total_count = 10000;
}
printf("Total Count is %d\n", total_count);
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;
}
shmp->cntr = 0;
pid = fork();
/* Parent Process - Writing Once */
if (pid > 0) {
shared_memory_cntr_increment(pid, shmp, total_count);
} else if (pid == 0) {
shared_memory_cntr_increment(pid, shmp, total_count);
return 0;
} else {
perror("Fork Failure\n");
return 1;
}
while (shmp->read_complete != 1)
sleep(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;
}
/* Increment the counter of shared memory by total_count in steps of 1 */
void shared_memory_cntr_increment(int pid, struct shmseg *shmp, int total_count) {
int cntr;
int numtimes;
int sleep_time;
cntr = shmp->cntr;
shmp->write_complete = 0;
if (pid == 0)
printf("SHM_WRITE: CHILD: Now writing\n");
else if (pid > 0)
printf("SHM_WRITE: PARENT: Now writing\n");
//printf("SHM_CNTR is %d\n", shmp->cntr);
/* Increment the counter in shared memory by total_count in steps of 1 */
for (numtimes = 0; numtimes < total_count; numtimes++) {
cntr += 1;
shmp->cntr = cntr;
/* Sleeping for a second for every thousand */
sleep_time = cntr % 1000;
if (sleep_time == 0)
sleep(1);
}
shmp->write_complete = 1;
if (pid == 0)
printf("SHM_WRITE: CHILD: Writing Done\n");
else if (pid > 0)
printf("SHM_WRITE: PARENT: Writing Done\n");
return;
}
संकलन और निष्पादन कदम
Total Count is 10000
SHM_WRITE: PARENT: Now writing
SHM_WRITE: CHILD: Now writing
SHM_WRITE: PARENT: Writing Done
SHM_WRITE: CHILD: Writing Done
Writing Process: Complete
अब, हम साझा मेमोरी रीडिंग प्रोग्राम की जांच करते हैं।
/* Filename: shm_read_cntr.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>
#define SHM_KEY 0x12345
struct shmseg {
int cntr;
int write_complete;
int read_complete;
};
int main(int argc, char *argv[]) {
int shmid, numtimes;
struct shmseg *shmp;
int total_count;
int cntr;
int sleep_time;
if (argc != 2)
total_count = 10000;
else {
total_count = atoi(argv[1]);
if (total_count < 10000)
total_count = 10000;
}
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;
}
/* Read the shared memory cntr and print it on standard output */
while (shmp->write_complete != 1) {
if (shmp->cntr == -1) {
perror("read");
return 1;
}
sleep(3);
}
printf("Reading Process: Shared Memory: Counter is %d\n", shmp->cntr);
printf("Reading Process: Reading Done, Detaching Shared Memory\n");
shmp->read_complete = 1;
if (shmdt(shmp) == -1) {
perror("shmdt");
return 1;
}
printf("Reading Process: Complete\n");
return 0;
}
संकलन और निष्पादन कदम
Reading Process: Shared Memory: Counter is 11000
Reading Process: Reading Done, Detaching Shared Memory
Reading Process: Complete
यदि आप उपरोक्त आउटपुट का निरीक्षण करते हैं, तो काउंटर 20000 होना चाहिए, हालांकि, एक प्रक्रिया कार्य पूरा होने से पहले अन्य प्रक्रिया भी समानांतर में प्रसंस्करण कर रही है, काउंटर मान अपेक्षित नहीं है। आउटपुट सिस्टम से सिस्टम में अलग-अलग होगा और यह भी प्रत्येक निष्पादन के साथ अलग-अलग होगा। यह सुनिश्चित करने के लिए कि दो कार्य एक कार्य पूरा होने के बाद कार्य करते हैं, इसे सिंक्रोनाइज़ेशन मैकेनिज्म का उपयोग करके लागू किया जाना चाहिए।
अब, हम सेमीफ़ोर्स का उपयोग करके उसी एप्लिकेशन को चेक करते हैं।
Note - रीडिंग प्रोग्राम वही रहता है।
/* Filename: shm_write_cntr_with_sem.c */
#include<stdio.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/shm.h>
#include<sys/sem.h>
#include<string.h>
#include<errno.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
#define SHM_KEY 0x12345
#define SEM_KEY 0x54321
#define MAX_TRIES 20
struct shmseg {
int cntr;
int write_complete;
int read_complete;
};
void shared_memory_cntr_increment(int, struct shmseg*, int);
void remove_semaphore();
int main(int argc, char *argv[]) {
int shmid;
struct shmseg *shmp;
char *bufptr;
int total_count;
int sleep_time;
pid_t pid;
if (argc != 2)
total_count = 10000;
else {
total_count = atoi(argv[1]);
if (total_count < 10000)
total_count = 10000;
}
printf("Total Count is %d\n", total_count);
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;
}
shmp->cntr = 0;
pid = fork();
/* Parent Process - Writing Once */
if (pid > 0) {
shared_memory_cntr_increment(pid, shmp, total_count);
} else if (pid == 0) {
shared_memory_cntr_increment(pid, shmp, total_count);
return 0;
} else {
perror("Fork Failure\n");
return 1;
}
while (shmp->read_complete != 1)
sleep(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");
remove_semaphore();
return 0;
}
/* Increment the counter of shared memory by total_count in steps of 1 */
void shared_memory_cntr_increment(int pid, struct shmseg *shmp, int total_count) {
int cntr;
int numtimes;
int sleep_time;
int semid;
struct sembuf sem_buf;
struct semid_ds buf;
int tries;
int retval;
semid = semget(SEM_KEY, 1, IPC_CREAT | IPC_EXCL | 0666);
//printf("errno is %d and semid is %d\n", errno, semid);
/* Got the semaphore */
if (semid >= 0) {
printf("First Process\n");
sem_buf.sem_op = 1;
sem_buf.sem_flg = 0;
sem_buf.sem_num = 0;
retval = semop(semid, &sem_buf, 1);
if (retval == -1) {
perror("Semaphore Operation: ");
return;
}
} else if (errno == EEXIST) { // Already other process got it
int ready = 0;
printf("Second Process\n");
semid = semget(SEM_KEY, 1, 0);
if (semid < 0) {
perror("Semaphore GET: ");
return;
}
/* Waiting for the resource */
sem_buf.sem_num = 0;
sem_buf.sem_op = 0;
sem_buf.sem_flg = SEM_UNDO;
retval = semop(semid, &sem_buf, 1);
if (retval == -1) {
perror("Semaphore Locked: ");
return;
}
}
sem_buf.sem_num = 0;
sem_buf.sem_op = -1; /* Allocating the resources */
sem_buf.sem_flg = SEM_UNDO;
retval = semop(semid, &sem_buf, 1);
if (retval == -1) {
perror("Semaphore Locked: ");
return;
}
cntr = shmp->cntr;
shmp->write_complete = 0;
if (pid == 0)
printf("SHM_WRITE: CHILD: Now writing\n");
else if (pid > 0)
printf("SHM_WRITE: PARENT: Now writing\n");
//printf("SHM_CNTR is %d\n", shmp->cntr);
/* Increment the counter in shared memory by total_count in steps of 1 */
for (numtimes = 0; numtimes < total_count; numtimes++) {
cntr += 1;
shmp->cntr = cntr;
/* Sleeping for a second for every thousand */
sleep_time = cntr % 1000;
if (sleep_time == 0)
sleep(1);
}
shmp->write_complete = 1;
sem_buf.sem_op = 1; /* Releasing the resource */
retval = semop(semid, &sem_buf, 1);
if (retval == -1) {
perror("Semaphore Locked\n");
return;
}
if (pid == 0)
printf("SHM_WRITE: CHILD: Writing Done\n");
else if (pid > 0)
printf("SHM_WRITE: PARENT: Writing Done\n");
return;
}
void remove_semaphore() {
int semid;
int retval;
semid = semget(SEM_KEY, 1, 0);
if (semid < 0) {
perror("Remove Semaphore: Semaphore GET: ");
return;
}
retval = semctl(semid, 0, IPC_RMID);
if (retval == -1) {
perror("Remove Semaphore: Semaphore CTL: ");
return;
}
return;
}
संकलन और निष्पादन कदम
Total Count is 10000
First Process
SHM_WRITE: PARENT: Now writing
Second Process
SHM_WRITE: PARENT: Writing Done
SHM_WRITE: CHILD: Now writing
SHM_WRITE: CHILD: Writing Done
Writing Process: Complete
अब, हम पढ़ने की प्रक्रिया द्वारा काउंटर मूल्य की जांच करेंगे।
निष्पादन के चरण
Reading Process: Shared Memory: Counter is 20000
Reading Process: Reading Done, Detaching Shared Memory
Reading Process: Complete