Чтение онлайн

на главную - закладки

Жанры

Linux программирование в примерах
Шрифт:

69 goto retry;

70 } else {

71 strcpy(buf, "\twaitpid failed: ");

72 strcat(buf, strerror(errno));

73 strcat(buf, "\n");

74 write(1, buf, strlen(buf));

75 }

76

77 write(1, exited, strlen(exited));

78 }

Обработчик сигнала похож на показанные ранее. Обратите внимание на список аргументов (строка 39)

и на то, что нет цикла.

Строки 49–54 обрабатывают завершение процесса, включая вызов

manage
для вывода состояния.

Строки 55–60 обрабатывают случай неожиданного завершения потомка. Этого не должно происходить, поскольку обработчику сигнала передается специфическая для определенного порожденного процесса информация.

Строки 61–66 представляют для нас интерес: возвращаемое значение для изменений состояния равно 0.

manage
имеет дело с деталями (строка 66).

Строки 67–69 обрабатывают прерывания, а строки 70–75 распоряжаются ошибками

80 /* child --- что сделать в порожденном процессе */

81

82 void child(void)

83 {

84 raise(SIGCONT); /* должен быть проигнорирован */

85 raise(SIGSTOP); /* заснуть, родитель снова разбудит */

86 printf("\t---> child restarted <---\n");

87 exit(42); /* нормальное завершение, дать возможность родителю получить значение */

88 }

Функция

child
обрабатывает поведение порожденного процесса, предпринимая действия для уведомления родителя [113] . Строка 84 посылает
SIGCONT
, что может вызвать получение родителем события
CLD_CONTINUED
. Строка 85 посылает
SIGSTOP
, который останавливает процесс (сигнал не может быть перехвачен) и вызывает для родителя событие
CLD_STOPPED
. Когда родитель возобновляет порожденный процесс, последний выводит сообщение, что он снова активен, а затем завершается с известным статусом завершения.

113

Возможно, лучшим именем для функции было бы

child_at_school
[ребенок_в_школе] — Примеч. автора.

90 /* main --- установка относящихся к порожденному процессу сведений

и сигналов, создание порожденного процесса */

91

92 int main(int argc, char **argv)

93 {

94 pid_t kid;

95 struct sigaction sa;

96 sigset_t childset, emptyset;

97

98 sigemptyset(&emptyset);

99

100 sa.sa_flags = SA_SIGINFO;

101 sa.sa_sigaction = childhandler;

102 sigfillset(&sa.sa_mask); /*
при вызове обработчика все заблокировать */

103 sigaction(SIGCHLD, &sa, NULL);

104

105 sigemptyset(&childset);

106 sigaddset(&childset, SIGCHLD);

107

108 sigprocmask(SIG_SETMASK, &childset, NULL); /* блокировать его в коде main */

109

110 if ((kid = fork) == 0)

111 child;

112

113 /* здесь выполняется родитель */

114 for (;;) {

115 printf("waiting for signals\n");

116 sigsuspend(&emptyset);

117 }

118

119 return 0;

120 }

Программа

main
все устанавливает. Строки 100–103 помещают на место обработчик. Строка 100 устанавливает флаг
SA_SIGINFO
таким образом, что используется обработчик с тремя аргументами. Строки 105–108 блокируют
SIGCHLD
.

Строка 110 создает порожденный процесс. Строки 113–117 продолжаются в родителе, используя для ожидания входящих сигналов

sigsuspend
.

123 /* manage --- разрешение различных событий, которые могут случиться с потомком */

124

125 void manage(siginfo_t *si)

126 {

127 char buf[100];

128

129 switch (si->si_code) {

130 case CLD_STOPPED:

131 write(1, "\tchild stopped, restarting\n", 27);

132 kill(si->si_pid, SIGCONT);

133 break;

134

135 case CLD_CONTINUED: /* not sent on Linux */

136 write(1, "\tchild continued\n", 17);

137 break;

138

139 case CLD_EXITED:

140 strcpy(buf, "\tchild exited with status ");

141 strcat(buf, format_num(si->si_status));

142 strcat(buf, "\n");

143 write(1, buf, strlen(buf));

144 exit(0); /* we're done */

145 break;

146

147 case CLD_DUMPED:

148 write(1, "\tchild dumped\n", 14);

149 break;

150

151 case CLD_KILLED:

Поделиться:
Популярные книги

Проданная Истинная. Месть по-драконьи

Белова Екатерина
Любовные романы:
любовно-фантастические романы
5.00
рейтинг книги
Проданная Истинная. Месть по-драконьи

Я еще не царь

Дрейк Сириус
25. Дорогой барон!
Фантастика:
юмористическое фэнтези
аниме
попаданцы
5.00
рейтинг книги
Я еще не царь

Наследие Маозари 4

Панежин Евгений
4. Наследие Маозари
Фантастика:
фэнтези
попаданцы
5.00
рейтинг книги
Наследие Маозари 4

Черный Маг Императора 8

Герда Александр
8. Черный маг императора
Фантастика:
юмористическое фэнтези
попаданцы
аниме
5.00
рейтинг книги
Черный Маг Императора 8

Инженерный Парадокс

Cyberdawn
1. Инженерный Парадокс
Фантастика:
боевая фантастика
аниме
фэнтези
5.00
рейтинг книги
Инженерный Парадокс

Кодекс Императора V

Сапфир Олег
5. Кодекс Императора
Фантастика:
аниме
фэнтези
попаданцы
5.00
рейтинг книги
Кодекс Императора V

Культивация зельевара

Крынов Макс
6. Культивация без насилия
Фантастика:
рпг
уся
сказочная фантастика
5.00
рейтинг книги
Культивация зельевара

Я еще князь. Книга XX

Дрейк Сириус
20. Дорогой барон!
Фантастика:
юмористическое фэнтези
попаданцы
аниме
5.00
рейтинг книги
Я еще князь. Книга XX

Гримуар темного лорда II

Грехов Тимофей
2. Гримуар темного лорда
Фантастика:
фэнтези
попаданцы
аниме
5.00
рейтинг книги
Гримуар темного лорда II

Курсант: Назад в СССР 4

Дамиров Рафаэль
4. Курсант
Фантастика:
попаданцы
альтернативная история
7.76
рейтинг книги
Курсант: Назад в СССР 4

Дворянин

Злотников Роман Валерьевич
2. Император и трубочист
Фантастика:
боевая фантастика
альтернативная история
5.00
рейтинг книги
Дворянин

Зеркало силы

Кас Маркус
3. Артефактор
Фантастика:
городское фэнтези
попаданцы
аниме
5.00
рейтинг книги
Зеркало силы

Московское золото и нежная попа комсомолки. Часть Пятая

Хренов Алексей
5. Летчик Леха
Фантастика:
попаданцы
5.00
рейтинг книги
Московское золото и нежная попа комсомолки. Часть Пятая

Матабар

Клеванский Кирилл Сергеевич
1. Матабар
Фантастика:
фэнтези
5.00
рейтинг книги
Матабар