मेमोरी मैपिंग

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