2. Возникновение исключительной ситуации.
3. Сигналы, поступающие от пользователя, при нажатии определенных клавиш.
Установить реакцию на поступление сигнала можно с помощью системного вызова signal:
func = signal(snum, function);
где: snum — номер сигнала;
function — адрес функции, которая должна быть выполнена при поступлении указанного сигнала.
Возвращаемое значение — адрес функции, которая будет реагировать на поступление сигнала. Вместо function можно указать ноль или единицу. Если был указан ноль, то при поступлении сигнала snum выполнение процесса будет прервано аналогично вызову exit. Если указать единицу, данный сигнал будет проигнорирован, но это возможно не для всех процессов.
С помощью системного вызова kill можно сгенерировать сигналы и передать их другим процессам. Обычно kill используется для того, чтобы принудительно завершить («убить») процесс:
kill(pid, snum);
где: pid — идентификатор процесса;
snum — номер сигнала, который будет передан процессу (см. табл. 5.1).
Pid состоит из идентификатора группы процессов и идентификатора процесса в группе. Если вместо pid указать нуль, то сигнал snum будет направлен всем процессам, относящимся к данной группе (понятие группы процессов аналогично группе пользователей). В одну группу включаются процессы, имеющие общего предка. Идентификатор группы процесса можно изменить с помощью системного вызова setpgrp. Если вместо pid указать –1, то ядро передаст сигнал всем процессам, идентификатор пользователя которых равен идентификатору текущего выполнения процесса, посылающего сигнал. Номера сигналов приведены в табл. 5.1. Сигналы (точнее, их номера) описаны в файле signal.h.
Номера сигналов Таблица 5.1
Номер Название Описание 01 SIGHUP Освобождение линии (hangup) 02 SIGINT Прерывание (interrupt) 03 SIGQUIT Выход (quit) 04 SIGILL Некорректная команда (illegal instruction). He переустанавливается при перехвате 05 SIGTRAP Трассировочное прерывание (trace trap). He переустанавливается при перехвате 06 SIGIOT или SIGABRT Машинная команда IOT. Останов ввода/вывода 07 SIGBUS Ошибка на шине 08 SIGFPE Исключительная ситуация при выполнении операции с вещественными числами (floating-point exception) 09 SIGKILL Уничтожение процесса (kill). He перехватывается и не игнорируется 10 SIGUSR1 Определяемый пользователем сигнал 1 11 SIGSEGV Некорректное обращение к сегменту памяти (segmentation violation) 12 SIGUSR2 Определяемый пользователем сигнал 2 13 SIGPIPE Запись в канал, из которого некому читать. Обрыв потока 14 SIGALRM Будильник 15 SIGTERM Программный сигнал завершения 16 SIGSTKFLT Сбой стека 17 SIGCHLD (или SIGCLD) Изменение статуса дочернего процесса 18 SIGCONT Продолжение работы после сигнала STOP. He перехватывается и не игнорируется 19 SIGSTOP Сигнал СТОП. Не перехватывается и не игнорируется 20 SIGTSTP Сигнал останова клавиатуры 21 SIGTTIN Фоновое чтение из терминала (tty) 22 SIGTTOU Фоновая запись на терминал (tty) 23 SIGURG Критическое состояние сокета 24 SIGXCPU Превышенный предел процессорного времени 25 SIGXFSZ Превышенный предел размера файла 26 SIGVTALRM Сигнал виртуального будильника 27 SIGPROF Сигнал профилирующего будильника 28 SIGWINCH Изменение размера окна 29 SIGIO Разрешение ввода/вывода 30 SIGPWR Сбой питания 31 SIGSYS Некорректный параметр системного вызова
Для нормального завершения процесса используется вызов:
exit(status)
где status — это целое число, возвращаемое процессу-предку для его информирования о причинах завершения процесса-потомка.
Вызов exit может задаваться в любой точке программы, но может быть и неявным, например, при выходе из функции main (при программировании на С) оператор return 0 будет воспринят как системный вызов exit(0).
5.2. Перенаправление ввода/вывода
Практически все операционные системы обладают механизмом перенаправления ввода/вывода, и Linux не является исключением из этого правила. Обычно программы вводят текстовые данные с консоли (терминала) и выводят данные на консоль. При вводе под консолью подразумевается клавиатура, а при выводе — экран монитора. Клавиатура и экран монитора — это, соответственно, стандартный ввод и вывод (stdin и stdout). Любой ввод/вывод можно интерпретировать как ввод из некоторого файла и вывод в файл. Работа с файлами производится через их дескрипторы.
Для организации ввода/вывода в UNIX используются три файла: stdin (дескриптор 0), stdout (дескриптор 1) и stderr (дескриптор 2).
Символ > («больше») используется для перенаправления стандартного вывода в файл. Например:
$ cat > newfile.txt
В этом примере стандартный вывод команды cat будет перенаправлен в файл newfile.txt, который будет создан после выполнения этой команды. Если файл с этим именем уже существует, то он будет перезаписан. Нажатие Ctrl + D остановит перенаправление и прервет выполнение команды cat.
Символ < («меньше») используется для переназначения стандартного ввода команды. Например, при выполнении команды cat < file.txt в качестве стандартного ввода будет использован файл file.txt, а не клавиатура.
Символ >> используется для присоединения данных в конец файла (append) стандартного вывода команды. Например, в отличие от случая с символом >, выполнение команды cat >> newfile.txt не перезапишет файл в случае его существования, а добавит данные в его конец.
Чтобы перенаправить весь стандартный поток ошибок в какой-нибудь файл, используйте переадресацию 2> имя файла или 2>> имя файла. В первом случае стандартный поток ошибок будет передан в файл или на устройство, а во втором — поток ошибок будет добавлен в файл, если такой существует. При использовании переадресации 2>&1 стандартный поток ошибок будет перенаправлен на стандартный вывод интерпретатора Bourne (здесь 1 и 2 — дескрипторы файлов). Для перенаправления стандартного потока ошибок в файл вы можете также использовать переадресацию >& имя_файла (интерпретатор C-Shell).
В командных интерпретаторах Korn и C-Shell можно использовать переадресацию >! имя_файла. При этом файл не будет перезаписан, если он существует.