Читаем Linux API. Исчерпывающее руководство полностью

Хотя вероятность генерации сигнала SIGINT в период между началом выполнения критического участка кода (то есть первый вызов функции sigprocmask()) и вызовом функции pause() мала, в вышеприведенном коде это является программной ошибкой. Эта зависящая от времени ошибка может быть примером состояния гонки (см. раздел 5.1). Как правило, состояние гонки (состязательная ситуация) возникает, если два процесса или потока задействуют общие ресурсы. Однако в данном случае основная программа состязается с собственным обработчиком сигнала.

Чтобы избежать возникновения этой проблемы, от нас требуется использование средств автоматического разблокирования сигнала и приостановки процесса. В этом и есть предназначение системного вызова sigsuspend().

#include


int sigsuspend(const sigset_t *mask);

(Обычно) возвращает –1 с установкой errno значения EINTR

Системный вызов sigsuspend() заменяет сигнальную маску процесса набором сигналов, на который указывает аргумент mask, а затем приостанавливает выполнение процесса до тех пор, пока не будет перехвачен сигнал и не будет выполнен возврат из обработчика. После того как обработчик возвращает управление в программу, функция sigsuspend() восстанавливает сигнальную маску процесса к значению до вызова этой функции.

Вызов функции sigsuspend() эквивалентен автоматическому выполнению следующих операций:

sigprocmask(SIG_SETMASK, &mask, &prevMask); /* Назначить новую маску */

pause();

sigprocmask(SIG_SETMASK, &prevMask, NULL); /* Восстановить старую маску */

Несмотря на то что восстановление старой маски (то есть последний шаг в вышеприведенной последовательности) может казаться неуместным, он необходим для избегания состояния гонки в ситуациях, когда нам требуется раз за разом ожидать сигнал. В таких ситуациях сигналы должны оставаться заблокированными, кроме промежутков времени, когда выполняется вызов sigsuspend(). Если впоследствии нам потребуется разблокировать сигналы, которые мы заблокировали перед вызовом функции sigsuspend(), то мы можем воспользоваться еще одним вызовом функции sigprocmask().

Если функция sigsuspend() прерывается доставкой сигнала, она возвращает –1, а переменной errno устанавливается значение EINTR. Если же аргумент mask указывает на недействительный адрес, то вызов sigsuspend() завершается неудачей с ошибкой EFAULT.


Пример программы

В листинге 22.5 продемонстрировано использование функции sigsuspend(). Эта программа выполняет следующие шаги.

1. Выводит изначальное значение сигнальной маски процесса с помощью функции printSigMask() (см. листинг 20.4) .

2. Блокирует сигналы SIGINT и SIGQUIT и сохраняет исходную сигнальную маску процесса .

3. Устанавливает один обработчик для двух сигналов — SIGINT и SIGQUIT . Этот обработчик выводит на экран сообщение, и, если обработчик был инициализирован доставкой сигнала SIGQUIT, происходит установка значения глобальной переменной gotSigquit.

4. Выполняет цикл до тех пор, пока не будет установлено значение переменной gotSigquit . Каждая итерация цикла включает следующие шаги:

1) вывести текущее значение сигнальной маски с помощью нашей функции printSigMask();

2) симулировать на протяжении нескольких секунд критический участок кода выполнением цикла, потребляющим время ЦП;

3) вывести маску ожидающих сигналов с помощью нашей функции printPendingSigs() (см. листинг 20.4);

4) использовать функцию sigsuspend() для разблокирования сигналов SIGINT и SIGQUIT и ожидать сигнал (если таковой уже находится в режиме ожидания).

5. Вызывает функцию sigprocmask() для восстановления исходного значения сигнальной маски процесса , а затем вывести сигнальную маску с помощью функции printSigMask() .


Листинг 22.5. Использование функции sigsuspend()

signals/t_sigsuspend.c

#define _GNU_SOURCE /* Взять объявление strsignal() из */

#include

#include

#include

#include "signal_functions.h" /* Объявления printSigMask() и printPendingSigs() */

#include "tlpi_hdr.h"


static volatile sig_atomic_t gotSigquit = 0;

static void

handler(int sig)

{

printf("Caught signal %d (%s)\n", sig, strsignal(sig));

/* НЕБЕЗОПАСНО (см. подраздел 21.1.2) */


if (sig == SIGQUIT)

gotSigquit = 1;

}


int

main(int argc, char *argv[])

{

int loopNum;

time_t startTime;

sigset_t origMask, blockMask;

struct sigaction sa;


printSigMask(stdout, "Initial signal mask is: \n");


sigemptyset(&blockMask);

sigaddset(&blockMask, SIGINT);

sigaddset(&blockMask, SIGQUIT);

if (sigprocmask(SIG_BLOCK, &blockMask, &origMask) == –1)

errExit("sigprocmask — SIG_BLOCK");


sigemptyset(&sa.sa_mask);

sa.sa_flags = 0;

sa.sa_handler = handler;


if (sigaction(SIGINT, &sa, NULL) == –1)

errExit("sigaction");

if (sigaction(SIGQUIT, &sa, NULL) == –1)

errExit("sigaction");


for (loopNum = 1;!gotSigquit; loopNum++) {

printf("=== LOOP %d\n", loopNum);


/* Симуляция критического участка с помощью задержки в несколько секунд */


printSigMask(stdout, "Starting critical section, signal mask is: \n");

for (startTime = time(NULL); time(NULL) < startTime + 4;)

Перейти на страницу:

Похожие книги

C++ Primer Plus
C++ Primer Plus

C++ Primer Plus is a carefully crafted, complete tutorial on one of the most significant and widely used programming languages today. An accessible and easy-to-use self-study guide, this book is appropriate for both serious students of programming as well as developers already proficient in other languages.The sixth edition of C++ Primer Plus has been updated and expanded to cover the latest developments in C++, including a detailed look at the new C++11 standard.Author and educator Stephen Prata has created an introduction to C++ that is instructive, clear, and insightful. Fundamental programming concepts are explained along with details of the C++ language. Many short, practical examples illustrate just one or two concepts at a time, encouraging readers to master new topics by immediately putting them to use.Review questions and programming exercises at the end of each chapter help readers zero in on the most critical information and digest the most difficult concepts.In C++ Primer Plus, you'll find depth, breadth, and a variety of teaching techniques and tools to enhance your learning:• A new detailed chapter on the changes and additional capabilities introduced in the C++11 standard• Complete, integrated discussion of both basic C language and additional C++ features• Clear guidance about when and why to use a feature• Hands-on learning with concise and simple examples that develop your understanding a concept or two at a time• Hundreds of practical sample programs• Review questions and programming exercises at the end of each chapter to test your understanding• Coverage of generic C++ gives you the greatest possible flexibility• Teaches the ISO standard, including discussions of templates, the Standard Template Library, the string class, exceptions, RTTI, and namespaces

Стивен Прата

Программирование, программы, базы данных
1С: Бухгалтерия 8 с нуля
1С: Бухгалтерия 8 с нуля

Книга содержит полное описание приемов и методов работы с программой 1С:Бухгалтерия 8. Рассматривается автоматизация всех основных участков бухгалтерии: учет наличных и безналичных денежных средств, основных средств и НМА, прихода и расхода товарно-материальных ценностей, зарплаты, производства. Описано, как вводить исходные данные, заполнять справочники и каталоги, работать с первичными документами, проводить их по учету, формировать разнообразные отчеты, выводить данные на печать, настраивать программу и использовать ее сервисные функции. Каждый урок содержит подробное описание рассматриваемой темы с детальным разбором и иллюстрированием всех этапов.Для широкого круга пользователей.

Алексей Анатольевич Гладкий

Программирование, программы, базы данных / Программное обеспечение / Бухучет и аудит / Финансы и бизнес / Книги по IT / Словари и Энциклопедии
C# 4.0: полное руководство
C# 4.0: полное руководство

В этом полном руководстве по C# 4.0 - языку программирования, разработанному специально для среды .NET, - детально рассмотрены все основные средства языка: типы данных, операторы, управляющие операторы, классы, интерфейсы, методы, делегаты, индексаторы, события, указатели, обобщения, коллекции, основные библиотеки классов, средства многопоточного программирования и директивы препроцессора. Подробно описаны новые возможности C#, в том числе PLINQ, библиотека TPL, динамический тип данных, а также именованные и необязательные аргументы. Это справочное пособие снабжено массой полезных советов авторитетного автора и сотнями примеров программ с комментариями, благодаря которым они становятся понятными любому читателю независимо от уровня его подготовки. Книга рассчитана на широкий круг читателей, интересующихся программированием на C#.Введите сюда краткую аннотацию

Герберт Шилдт

Программирование, программы, базы данных