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

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

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

नामित पाइप का दूसरा नाम है 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 दूसरों का प्रतिनिधित्व करता है। 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 - सर्वर प्रक्रिया निम्नलिखित कार्य करती है -

  • अगर नाम नहीं बना है तो एक नामित पाइप (सिस्टम कॉल माकनॉड ()) का नाम "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