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

signalएक घटना की घटना को इंगित करने वाली एक प्रक्रिया के लिए एक अधिसूचना है। सिग्नल भी कहा जाता हैsoftware interrupt और इसकी घटना को जानने के लिए अनुमानित नहीं है, इसलिए इसे ए भी कहा जाता है asynchronous event

सिग्नल को एक संख्या या एक नाम के साथ निर्दिष्ट किया जा सकता है, आमतौर पर सिग्नल के नाम SIG से शुरू होते हैं। उपलब्ध संकेतों को कमांड किल-एल (लिस्टिंग सिग्नल नामों के लिए l) के साथ चेक किया जा सकता है, जो इस प्रकार है -

जब भी एक सिग्नल उठाया जाता है (या तो प्रोग्रामेटिक या सिस्टम जेनरेट किया गया सिग्नल), एक डिफ़ॉल्ट क्रिया की जाती है। क्या होगा यदि आप डिफ़ॉल्ट कार्रवाई नहीं करना चाहते हैं, लेकिन सिग्नल प्राप्त करने पर अपने स्वयं के कार्यों को करना चाहते हैं? क्या यह सभी संकेतों के लिए संभव है? हां, सिग्नल को संभालना संभव है लेकिन सभी सिग्नल के लिए नहीं। क्या होगा यदि आप संकेतों को अनदेखा करना चाहते हैं, क्या यह संभव है? हां, संकेत को अनदेखा करना संभव है। सिग्नल को नजरअंदाज करने का अर्थ है न तो डिफ़ॉल्ट कार्रवाई करना और न ही सिग्नल को संभालना। लगभग सभी संकेतों को अनदेखा करना या संभालना संभव है। जिन संकेतों को नजरअंदाज नहीं किया जा सकता है या जिन्हें संभाला / पकड़ा जाता है वे हैं SIGSTOP और SIGKILL।

सारांश में, संकेतों के लिए की जाने वाली क्रियाएं इस प्रकार हैं -

  • डिफ़ॉल्ट क्रिया
  • संकेत संभालना
  • संकेत पर ध्यान न दें

जैसा कि चर्चा की गई है कि सिग्नल को डिफ़ॉल्ट कार्रवाई के निष्पादन में फेरबदल किया जा सकता है। सिग्नल हैंडलिंग दोनों तरीकों में से किसी एक में की जा सकती है, सिस्टम कॉल के माध्यम से, सिग्नल () और सिगनेशन ()।

#include <signal.h>

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

सिस्टम कॉल सिग्नल () सिग्नल की पीढ़ी पर पंजीकृत हैंडलर को संकेत के रूप में वर्णित करेगा। हैंडलर या तो SIG_IGN (सिग्नल को नजरअंदाज करना), SIG_DFL (डिफ़ॉल्ट तंत्र पर वापस सिग्नल सेट करना) या उपयोगकर्ता द्वारा परिभाषित सिग्नल हैंडलर या फ़ंक्शन एड्रेस में से एक हो सकता है।

यह सिस्टम सफलता पर कॉल एक फ़ंक्शन का पता देता है जो पूर्णांक तर्क लेता है और जिसका कोई रिटर्न मान नहीं है। यह कॉल त्रुटि के मामले में SIG_ERR देता है।

यद्यपि सिग्नल () के साथ संबंधित सिग्नल हैंडलर को उपयोगकर्ता द्वारा पंजीकृत किया जा सकता है, ठीक ट्यूनिंग जैसे सिग्नल को मास्क करना जैसे कि ब्लॉक किया जाना चाहिए, सिग्नल के व्यवहार को संशोधित करना, और अन्य कार्य संभव नहीं हैं। यह सिगनेशन () सिस्टम कॉल का उपयोग करके संभव है।

#include <signal.h>

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

इस सिस्टम कॉल का उपयोग सिग्नल एक्शन को जांचने या बदलने के लिए किया जाता है। यदि अधिनियम अशक्त नहीं है, तो सिग्नल साइनम के लिए नई कार्रवाई अधिनियम से स्थापित की गई है। यदि पुराना निष्क्रिय नहीं है, तो पिछली कार्रवाई पुराने में बच जाती है।

सिगनेशन संरचना में निम्नलिखित फ़ील्ड शामिल हैं -

Field 1 - हैंडलर का उल्लेख या तो 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

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

अब, हम एक अन्य प्रोग्राम लेते हैं जो सिग्नल को संभालने और अनदेखा करने के लिए प्रदर्शित करता है।

मान लें कि हमने एक सिग्नल बढ़ाकर () का उपयोग किया है, तब क्या होता है? सिग्नल बढ़ाने के बाद, वर्तमान प्रक्रिया का निष्पादन बंद हो जाता है। फिर रुकी हुई प्रक्रिया का क्या होता है? दो परिदृश्य हो सकते हैं - पहला, जब भी आवश्यक हो निष्पादन जारी रखें। दूसरी, प्रक्रिया को (मार कमान के साथ) समाप्त करें।

रुकी हुई प्रक्रिया के निष्पादन को जारी रखने के लिए, उस विशेष प्रक्रिया में SIGCONT भेजें। निष्पादन को जारी रखने के लिए आप fg (अग्रभूमि) या bg (पृष्ठभूमि) आदेश भी जारी कर सकते हैं। यहां, आदेश केवल अंतिम प्रक्रिया के निष्पादन को फिर से शुरू करेंगे। यदि एक से अधिक प्रक्रिया रोक दी जाती है, तो केवल अंतिम प्रक्रिया फिर से शुरू की जाती है। यदि आप पहले से बंद की गई प्रक्रिया को फिर से शुरू करना चाहते हैं, तो जॉब नंबर के साथ नौकरियों (एफजी / बीजी का उपयोग करके) को फिर से शुरू करें।

निम्नलिखित कार्यक्रम का उपयोग सिगस्ट () (फंक्शन) का उपयोग करके संकेत को बढ़ाने के लिए किया जाता है। सीटीजी + जेड (कंट्रोल + जेड) कुंजी के उपयोगकर्ता प्रेस द्वारा सिग्नल सिग्स्टॉप भी उत्पन्न किया जा सकता है। यह संकेत जारी करने के बाद, प्रोग्राम निष्पादित करना बंद कर देगा। निष्पादन जारी रखने के लिए संकेत (SIGCONT) भेजें।

निम्नलिखित उदाहरण में, हम कमांड एफजी के साथ रुकी हुई प्रक्रिया को फिर से शुरू कर रहे हैं।

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

अब तक, हमने उस प्रोग्राम को देखा है जो सिस्टम द्वारा उत्पन्न सिग्नल को संभालता है। अब, प्रोग्राम के माध्यम से उत्पन्न सिग्नल को देखते हैं (उठाना () फ़ंक्शन या किल कमांड के माध्यम से)। यह कार्यक्रम सिग्नल 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

हमने डिफ़ॉल्ट कार्रवाई करने या सिग्नल को संभालने के उदाहरणों को देखा है। अब, संकेत को अनदेखा करने का समय आ गया है। यहाँ, इस नमूना कार्यक्रम में, हम SIG_ST के माध्यम से उपेक्षा करने के लिए संकेत SIGTSTP को पंजीकृत कर रहे हैं और फिर हम संकेत 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 - सिग्नल को पकड़ने या संभालने के लिए एक हैंडलर (हैंडल) को पंजीकृत करता है SIGINT (CTRL + C) या SIGQUIT (CTRL + \)

Step 2 - यदि उपयोगकर्ता सिग्नल SIGQUIT (या तो किल कमांड या CTRL + \ के साथ कीबोर्ड नियंत्रण के माध्यम से) उत्पन्न करता है, तो हैंडलर केवल संदेश को रिटर्न के रूप में प्रिंट करता है।

Step 3 - यदि उपयोगकर्ता पहली बार सीटीआईआईआरटी (सीटीआरएल + सी के साथ किल कमांड या कीबोर्ड कंट्रोल के माध्यम से) सिग्नल उत्पन्न करता है, तो यह अगली बार से डिफ़ॉल्ट कार्रवाई (एसआईजीडीएफएल के साथ) करने के लिए सिग्नल को संशोधित करता है।

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

हम जानते हैं कि एक सिग्नल को संभालने के लिए, हमारे पास दो सिस्टम कॉल हैं, या तो सिग्नल () या सिगनेशन ()। अब तक हमने सिग्नल () सिस्टम कॉल के साथ देखा है, अब यह सिगनेशन () सिस्टम कॉल का समय है। आइए हम उपरोक्त कार्यक्रम को सिगनेशन () का उपयोग करने के लिए संशोधित करते हैं -

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