Inter Process Communication - Adlandırılmış Borular

Borular, ilgili süreçler arasındaki iletişim içindir. İlişkisiz süreç iletişimi için boruları kullanabilir miyiz, örneğin, istemci programını bir terminalden ve sunucu programını başka bir terminalden yürütmek istiyor muyuz? Cevap Hayır. O halde ilişkisiz süreç iletişimini nasıl başarabiliriz, basit cevap Named Pipes. Bu, ilgili süreçler için çalışsa da, ilgili süreç iletişimi için adlandırılmış kanalları kullanmak anlamsızdır.

Tek yönlü iletişim için bir boru ve iki yönlü iletişim için iki boru kullandık. Aynı durum Adlandırılmış Borular için de geçerli mi? Cevap hayır, Named Pipe çift yönlü iletişimi desteklediğinden, iki yönlü iletişim için (sunucu ile istemci arasındaki iletişim artı aynı anda istemci ve sunucu arasındaki iletişim) kullanılabilen tek adlandırılmış kanal kullanabiliriz.

Adlandırılmış boru için başka bir isim FIFO (First-In-First-Out). Bir tür özel dosya olan adlandırılmış bir kanal oluşturmak için (mknod ()) sistem çağrısını görelim.

#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);

Bu sistem çağrısı, sıradan dosya, aygıt dosyası veya FIFO gibi özel bir dosya veya dosya sistemi düğümü oluşturur. Sistem çağrısının argümanları yol adı, mod ve dev'dir. Yol adı, mod özellikleri ve cihaz bilgileri ile birlikte. Yol adı görecelidir, eğer dizin belirtilmezse mevcut dizinde yaratılır. Belirtilen mod, aşağıdaki tablolarda bahsedildiği gibi dosya türü ve dosya modu gibi dosya türünü belirleyen dosya modudur. Dev alanı, büyük ve küçük cihaz numaraları gibi cihaz bilgilerini belirtmektir.

Dosya tipi Açıklama Dosya tipi Açıklama
S_IFBLK özel blok S_IFREG Normal dosya
S_IFCHR özel karakter S_IFDIR Rehber
S_IFIFO FIFO özel S_IFLNK Sembolik bağlantı
Dosya Modu Açıklama Dosya Modu Açıklama
S_IRWXU Sahibine göre oku, yaz, çalıştır / ara S_IWGRP Yazma izni, grup
S_IRUSR Okuma izni, sahip S_IXGRP Yürütme / arama izni, grup
S_IWUSR Yazma izni, sahibi S_IRWXO Başkaları tarafından okuyun, yazın, çalıştırın / arayın
S_IXUSR Yürütme / arama izni, sahip S_IROTH Okuma izni, diğerleri
S_IRWXG Gruba göre oku, yaz, çalıştır / ara S_IWOTH Yazma izni, diğerleri
S_IRGRP Okuma izni, grup S_IXOTH Yürütme / arama izni, diğerleri

Dosya modu, X'in sahibi, Y'nin grubu ve Z'nin diğerlerini temsil ettiği 0XYZ gibi sekizlik gösterimle de temsil edilebilir. X, Y veya Z'nin değeri 0 ile 7 arasında değişebilir. Okuma, yazma ve çalıştırma değerleri sırasıyla 4, 2, 1'dir. Okuma, yazma ve çalıştırma kombinasyonlarında gerekliyse değerleri uygun şekilde ekleyin.

Diyelim ki 0640'tan bahsediyorsak, bu sahip için oku ve yaz (4 + 2 = 6), grup için oku (4) ve diğerleri için izin (0) yok demektir.

Bu çağrı, başarı durumunda sıfır ve başarısızlık durumunda -1 döndürür. Başarısızlığın nedenini öğrenmek için, errno değişkeni veya perror () işlevi ile kontrol edin.

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

int mkfifo(const char *pathname, mode_t mode)

Bu kütüphane işlevi, adlandırılmış kanal için kullanılan özel bir FIFO dosyası oluşturur. Bu işlevin argümanları dosya adı ve moddur. Dosya adı, mutlak yol veya göreli yol olabilir. Tam yol adı (veya mutlak yol) verilmemişse, dosya yürütme işleminin geçerli klasöründe oluşturulur. Dosya modu bilgisi mknod () sistem çağrısında açıklandığı gibidir.

Bu çağrı, başarı durumunda sıfır ve başarısızlık durumunda -1 döndürür. Başarısızlığın nedenini öğrenmek için, errno değişkeni veya perror () işlevi ile kontrol edin.

Sunucuyu bir terminalde çalıştırıp istemciyi başka bir terminalde çalıştıran bir program düşünelim. Program yalnızca tek yönlü iletişim gerçekleştirir. İstemci, kullanıcı girişini kabul eder ve mesajı sunucuya gönderir, sunucu mesajı çıktıya yazdırır. İşlem, kullanıcı "end" dizesini girene kadar devam eder.

Bunu bir örnekle anlayalım -

Step 1 - Biri beşli sunucu ve diğeri elli istemci olmak üzere iki işlem oluşturun.

Step 2 - Sunucu işlemi aşağıdakileri gerçekleştirir -

  • Oluşturulmamışsa, “MYFIFO” adıyla adlandırılmış bir kanal oluşturur (mknod () sistem çağrısını kullanarak).

  • Adlandırılmış boruyu salt okunur amaçlarla açar.

  • Burada, Sahip için okuma ve yazma izinlerine sahip FIFO oluşturuldu. Grup için okuyun ve Diğerleri için izin yok.

  • Müşteriden gelen mesajı sonsuza kadar bekler.

  • İstemciden alınan mesaj "bitmemiş" ise, mesajı yazdırır. Mesaj "son" ise, fifo'yu kapatır ve işlemi bitirir.

Step 3 - İstemci süreci aşağıdakileri gerçekleştirir -

  • Adlandırılmış boruyu yalnızca yazma amacıyla açar.

  • Kullanıcıdan gelen dizeyi kabul eder.

  • Kullanıcının "end" veya "end" dışında girdi girip girmediğini kontrol eder. Her iki durumda da sunucuya bir mesaj gönderir. Bununla birlikte, dizge "son" ise, bu FIFO'yu kapatır ve işlemi de sonlandırır.

  • Kullanıcı "end" dizesini girene kadar sonsuz sayıda tekrar eder.

Şimdi FIFO sunucu dosyasına bir göz atalım.

/* 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;
}

Derleme ve Yürütme Adımları

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

Şimdi, FIFO istemci örnek koduna bir göz atalım.

/* 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;
}

Gelen çıktıya bir bakalım.

Derleme ve Yürütme Adımları

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

Adlandırılmış Borular Kullanarak İki yönlü İletişim

Borular arasındaki iletişimin tek yönlü olması amaçlanmıştır. Borular genel olarak tek yönlü iletişimle sınırlıydı ve iki yönlü iletişim için en az iki boruya ihtiyaç duyuyordu. Borular yalnızca birbiriyle ilişkili işlemler içindir. Borular, ilgisiz süreç iletişimi için kullanılamaz, örneğin bir işlemi bir terminalden ve başka bir işlemi başka bir terminalden yürütmek istiyorsak, borularla bu mümkün değildir. İki süreç arasında iletişim kurmanın basit bir yolu var mı, mesela ilgisiz süreçleri basit bir şekilde? Cevap Evet. Adlandırılmış kanal, iki veya daha fazla ilgisiz süreç arasındaki iletişim içindir ve ayrıca iki yönlü iletişime de sahip olabilir.

Şimdiden, adlandırılmış kanallar arasındaki tek yönlü iletişimi, yani istemciden sunucuya giden mesajları gördük. Şimdi, iki yönlü iletişime bir göz atalım, yani sunucuya mesaj gönderen istemci ve mesajı alan sunucu ve aynı adlandırılmış boruyu kullanarak istemciye başka bir mesaj gönderiyor.

Aşağıda bir örnek verilmiştir -

Step 1 - Biri fifoserver_twoway ve diğeri fifoclient_twoway olmak üzere iki süreç oluşturun.

Step 2 - Sunucu işlemi aşağıdakileri gerçekleştirir -

  • Oluşturulmamışsa, / tmp dizininde "fifo_twoway" adıyla adlandırılmış bir kanal oluşturur (mkfifo () kitaplık işlevini kullanarak).

  • Adlandırılmış boruyu okuma ve yazma amacıyla açar.

  • Burada, Sahip için okuma ve yazma izinlerine sahip FIFO oluşturuldu. Grup için okuyun ve Diğerleri için izin yok.

  • Müşteriden gelen bir mesajı sonsuza kadar bekler.

  • İstemciden alınan mesaj "son" değilse, mesajı yazdırır ve dizeyi tersine çevirir. Tersine çevrilen dizi istemciye geri gönderilir. Mesaj "son" ise, fifo'yu kapatır ve işlemi bitirir.

Step 3 - İstemci süreci aşağıdakileri gerçekleştirir -

  • Adlandırılmış boruyu okuma ve yazma amacıyla açar.

  • Kullanıcıdan gelen dizeyi kabul eder.

  • Kullanıcının "end" veya "end" dışında girdi girip girmediğini kontrol eder. Her iki durumda da sunucuya bir mesaj gönderir. Bununla birlikte, dizge "son" ise, bu FIFO'yu kapatır ve işlemi de sonlandırır.

  • Mesaj "bitmedi" olarak gönderilirse, istemciden gelen mesajı (ters dizge) bekler ve ters çevrilmiş dizeyi yazdırır.

  • Kullanıcı "end" dizesini girene kadar sonsuz sayıda tekrar eder.

Şimdi, FIFO sunucusu örnek koduna bir göz atalım.

/* 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;
}

Derleme ve Yürütme Adımları

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

Şimdi, FIFO istemci örnek koduna bir göz atalım.

/* 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;
}

Derleme ve Yürütme Adımları

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