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

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

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

#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 - लिखने के लिए मूल प्रक्रिया और पढ़ने के लिए बच्चे की प्रक्रिया के लिए pip1 बनाएं।

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