Inne procesy

Do tej pory dyskutowaliśmy o procesach, ich tworzeniu, procesach nadrzędnych i potomnych itp. Dyskusja będzie niekompletna bez omówienia innych powiązanych procesów, takich jak proces osierocony, proces zombie i proces demona.

Proces osierocony

Jak wskazuje nazwa, sierota oznacza proces bez rodziców. Kiedy uruchamiamy program lub aplikację, procesem nadrzędnym dla aplikacji jest powłoka. Kiedy tworzymy proces za pomocą fork (), nowo utworzony proces jest procesem potomnym, a proces, który utworzył dziecko, jest procesem nadrzędnym. Z kolei procesem macierzystym tego jest powłoka. Oczywiście rodzicem wszystkich procesów jest init process (Process ID → 1).

Powyższy scenariusz jest zwykłym scenariuszem, jednak co się dzieje, gdy proces nadrzędny kończy pracę przed procesem potomnym. W rezultacie proces potomny staje się teraz procesem osieroconym. A co z jego rodzicem, jego nowy rodzic jest rodzicem wszystkich procesów, który jest niczym innym jak procesem inicjującym (identyfikator procesu - 1).

Spróbujmy to zrozumieć na poniższym przykładzie.

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

Kroki kompilacji i wykonywania

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

Proces zombie

Mówiąc prosto, załóżmy, że masz dwa procesy, a mianowicie proces rodzica i proces potomny. Do obowiązków procesu nadrzędnego należy czekanie na proces potomny, a następnie czyszczenie wpisu procesu potomnego z tabeli procesów. Co się stanie, jeśli proces nadrzędny nie jest gotowy do oczekiwania na proces potomny, a w międzyczasie proces potomny kończy swoje zadanie i kończy pracę? Teraz proces potomny stałby się procesem zombie. Oczywiście proces zombie jest czyszczony, gdy proces nadrzędny jest gotowy.

Zrozummy to na przykładzie.

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

Kroki kompilacji i wykonywania

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

Proces demona

Mówiąc prościej, proces, który nie ma skojarzonej powłoki ani terminala, jest nazywany procesem demona. Dlaczego to jest potrzebne? Są to procesy, które działają w tle, aby wykonywać akcje w określonych odstępach czasu, a także reagować na określone zdarzenia. Proces demona nie powinien mieć żadnych interakcji z użytkownikiem, ponieważ działa jako proces w tle.

Wewnętrzne procesy demona Linuksa zwykle kończą się literą „d”, na przykład demony jądra (ksoftirqd, kblockd, kswapd itp.), Demony drukowania (cupsd, lpd itp.), Demony usług plików (smbd, nmbd itp.) , Administracyjne demony baz danych (ypbind, ypserv itp.), Demony poczty elektronicznej (sendmail, popd, smtpd itp.), Demony zdalnego logowania i wykonywania poleceń (sshd, in.telnetd itp.), Demony uruchamiania i konfiguracji (dhcpd , udevd itp.), proces init (init), demon cron, demon atd itp.

Zobaczmy teraz, jak stworzyć proces demona. Oto kroki -

Step 1- Utwórz proces potomny. Teraz mamy dwa procesy - proces nadrzędny i proces potomny

Zwykle hierarchia procesów to POWŁOKA → PROCES DLA RODZICÓW → PROCES DZIECIĘCY

Step 2- Zakończ proces nadrzędny, wychodząc. Proces potomny staje się teraz procesem osieroconym i jest przejmowany przez proces init.

Teraz hierarchia to INIT PROCESS → CHILD PROCESS

Step 3- Wywołanie wywołania systemowego setsid () tworzy nową sesję, jeśli wywołujący proces nie jest liderem grupy procesów. Teraz osoba dzwoniąca staje się liderem grupy w nowej sesji. Ten proces będzie jedynym procesem w tej nowej grupie procesów i w tej nowej sesji.

Step 4 - Ustaw identyfikator grupy procesów i identyfikator sesji na PID procesu wywołującego.

Step 5 - Zamknij domyślne deskryptory plików (standardowe wejście, standardowe wyjście i standardowy błąd) procesu, ponieważ terminal i powłoka są teraz odłączone od aplikacji.

/ * Nazwa pliku: 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