sem11 - Работа с процессами. Перенаправление стандартных потоков. (1114924), страница 3
Текст из файла (страница 3)
Это возможно, только если wait4была вызвана с установленным флагом WUNTRACED.WSTOPSIG(status) — возвращает номер сигнала, который вызвал приостановку выполнения процесса. Этот макрос может использоваться, только если WIFSTOPPED далненулевое значение.Если prusage не равен NULL, информация об использовании процессом ресурсов системы записывается по адресу, указанному аргументом prusage.Структура struct rusage определена в операционной системе Linux следующим образом:struct rusage8{struct timeval ru_utime; /* время в режиме пользователя */struct timeval ru_stime; /* время в режиме ядра */long ru_maxrss;/* максимальный размер в памяти */long ru_ixrss;/* размер всех разделяемых страниц */long ru_idrss;/* размер всех неразделяемых страниц */long ru_isrss;/* размер страниц стека */long ru_minflt;/* сбоев страницы без подкачки */long ru_majflt;/* сбоев страницы с подкачкой */long ru_nswap;/* количество полных откачек */long ru_inblock;/* блочных операций ввода */long ru_oublock;/* блочных операций вывода */long ru_msgsnd;/* послано сообщений */long ru_msgrcv;/* получено сообщений */long ru_nsignals;/* получено сигналов */long ru_nvcsw;/* "добровольных" переключений контекста */long ru_nivcsw;/* "недобровольных" переключений */};Функция wait4 возвращает идентификатор процесса, который вызвал завершение работы функции, либо -1 при ошибке (например, не существует сыновних процессов заданнойкатегории), либо 0, если был использован флаг WNOHANG, и нет процессов, завершившихвыполнение.
В последних двух случаях переменная errno будет содержать код ошибки.Возможные коды ошибок приведены в таблице.ECHILDERESTARTSYSEINTREINVALпроцесс, заданный аргументом pid, не существует или не является сыновним процессом, либо у процесса вообще нет сыновних процессовфлаг WNOHANG не был установлен, и процесс получил неблокируемыйсигнал или SIGCHLD.какой-либо из аргументов имеет недопустимое значение.Таблица 4: Коды ошибок для функций wait, wait41.5 Пример программыНапишем реализацию функции system (стандартная функция, которая выполняет заданную команду).#include#include#include#include<stdio.h><unistd.h><sys/types.h><sys/wait.h>int system(char const *cmd){int pid, status;if ((pid = fork()) < 0) {/* ошибка */perror("fork");return -1;9} else if (!pid) {/* child */execl("/bin/sh", "/bin/sh", "-c", cmd, NULL);/* ошибка */perror("execl");_exit(1);}/* parent */wait(&status);if (WIFSIGNALED(status)) return WTERMSIG(status) + 256;return WEXITSTATUS(status);}2Перенаправление стандартных потоковВ начале работы процесса у него, как правило, уже открыты три файловых дескрипторас номерами 0, 1, 2 — стандартный ввод, стандартный вывод и стандартный поток ошибок.Как правило, эти потоки связаны с терминалом, с которым работает командный интерпретатор, вызвавший процесс.
Однако можно связать со стандартными потоками произвольныеобъекты, работа с которыми ведётся с помощью файловых дескрипторов (каналы, сокеты ипр.).Для копирования открытого файлового дескриптора используется функция dup2, определённая следующим образом:#include <unistd.h>int dup2(int oldfd, int newfd);Функция dup2 создаёт копию файлового дескриптора oldfd. После этого использованиедвух файловых дескрипторов полностью эквивалентно. Оба файловых дескриптора разделяют блокировки файлов, указатель текущего положения в файле и флаги открытия файлов.Например, если текущая позиция в файле была изменена с помощью lseek у одного из дескрипторов, текущая позиция изменится и у другого дескриптора.Флаг «закрытия при exec» не разделятся.Функция dup2 создаёт копию oldfd в дескрипторе с номером newfd, при необходимости предварительно закрывая newfd.Функция возвращает номер нового файлового дескриптора или -1, если функция не смогла создать новый файловый дескриптор.
Переменная errno в этом случае содержит кодошибки.2.1 Пример программыНапишем программу, которая печатает все процессы, идентификатор пользователя которых 0 (root). Для получения списка процессов будем использовать вызов программы ps.Предположим, что второе число в строке, печатаемой командой ps как раз содержит идентификатор пользователя.#include <stdlib.h>10#include#include#include#include#include<unistd.h><stdio.h><sys/types.h><sys/wait.h><string.h>void pexit(char const *str){perror(str);exit(1);}int main(void){char tmppatt[] = "/tmp/XXXXXX";int fd, pid, status;FILE *f;char buf[1024];intv1, v2;/*if/*if/*ifсоздаём временный файл */((fd = mkstemp(tmppatt)) < 0) pexit("mkstemp");сразу удаляем */(unlink(tmppatt) < 0) pexit("unlink");создаём новый процесс */((pid = fork()) < 0) pexit("fork");if (!pid) { /* сын *//* перенаправляем стандартный вывод */if (dup2(fd, 1) != 1) { perror("dup2"); _exit(1); }/* закрываем старый дескриптор */close(fd);/* вызываем другой процесс */execlp("ps", "ps", "axl", 0);perror("execlp");_exit(1);}/* отец *//* ждём */wait(&status);/* ошибка выполнения сыновнего процесса */if (!WIFEXITED(status) || WEXITSTATUS(status) > 0) exit(1);/* перематываем файл на начало */if (lseek(fd, 0, SEEK_SET) < 0) pexit("lseek");/* связываем уже открытый файловый дескриптор с FILE * */if (!(f = fdopen(fd, "r"))) pexit("fdopen");while (fgets(buf, sizeof(buf), f)) {if (strlen(buf) >= sizeof(buf) - 1) {fprintf(stderr, "string too long\n");exit(1);11}/* печатаем все процессы, владелец которых - root */if (sscanf(buf, "%d %d", &v1, &v2) == 2 && !v2)printf("%s", buf);}fclose(f);return 0;}12.