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

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

Жанры

UNIX: взаимодействие процессов

Стивенс Уильям Ричард

Шрифт:
Проверка очереди на пустоту и отправка уведомления

30-38 Если мы помещаем первое сообщение в пустую очередь, нужно проверить, не зарегистрирован ли какой-нибудь процесс на уведомление об этом событии и нет ли потоков, заблокированных в вызове mq_receive. Для проверки второго условия мы воспользуемся сохраняемым функцией mq_receive счетчиком mqh_nwait, содержащим количество потоков, заблокированных в вызове mq_receive. Если этот счетчик имеет ненулевое значение, мы не отправляем уведомление зарегистрированному процессу. Для отправки сигнала SIGEV_SIGNAL используется функция sigqueue. Затем процесс снимается с регистрации.

ПРИМЕЧАНИЕ

Вызов sigqueue для отправки сигнала приводит к передаче сигнала SI_QUEUE обработчику сигнала в структуре типа siginfo_t (раздел 5.7), что неправильно. Отправка правильного значения si_code (а именно SI_MESGQ) из пользовательского процесса осуществляется в зависимости от реализации. На с. 433 стандарта IEEE 1996 [8] отмечается, что для отправки этого сигнала из пользовательской библиотеки необходимо воспользоваться скрытым интерфейсом механизма отправки сигналов. 

Проверка заполненности очереди

39-48 Если очередь переполнена и установлен флаг O_NONBLOCK, мы возвращаем ошибку с кодом EAGAIN. В противном случае мы ожидаем сигнала по условной переменной mqh_wait, который, как мы увидим, отправляется функцией mq_receive при считывании сообщения из переполненной очереди.

ПРИМЕЧАНИЕ

Наша реализация упрощает ситуацию с возвращением ошибки EINTR при прерывании вызова mq_send сигналом, перехватываемым вызвавшим процессом. Проблема в том, что функция pthread_cond_wait не возвращает ошибки при возврате из обработчика сигнала: она может вернуть либо 0 (что рассматривается как ложное пробуждение), либо вообще не завершить работу. Все эти проблемы можно обойти, но это непросто. 

В листинге 5.26 приведена вторая половина функции mq_send. К моменту ее выполнения мы уже знаем о наличии в очереди свободного места для нашего сообщения.

Листинг 5.25. Функция mq_send: первая половина

//my_pxmsg_mmap/mq_send.с

1 #include "unpipc.h"

2 #include "mqueue.h"

3 int

4 mymq_send(mymqd_t mqd, const char *ptr, size_t len, unsigned int prio)

5 {

6 int n;

7 long index, freeindex;

8 int8_t *mptr;

9 struct sigevent *sigev;

10 struct mymq_hdr *mqhdr;

11 struct mymq_attr *attr;

12 struct mymsg_hdr *msghdr, *nmsghdr, *pmsghdr;

13 struct mymq_info *mqinfo;

14 mqinfo = mqd;

15 if (mqinfo->mqi_magic != MQI_MAGIC) {

16 errno = EBADF;

17 return(-1);

18 }

19 mqhdr = mqinfo->mqi_hdr; /* указатель типа struct */

20 mptr = (int8_t *) mqhdr; /* указатель на байт */

21 attr = &mqhdr->mqh_attr;

22 if ((n = pthread_mutex_lock(&mqhdr->mqh_lock)) != 0) {

23 errno = n;

24 return(-1);

25 }

26 if (len > attr->mq_msgsize) {

27 errno = EMSGSIZE;

28 goto err;

29 }

30 if (attr->mq_curmsgs == 0) {

31 if (mqhdr->mqh_pid != 0 && mqhdr->mqh_nwait == 0) {

32 sigev = &mqhdr->mqh_event;

33 if (sigev->sigev_notify == SIGEV_SIGNAL) {

34 sigqueue(mqhdr->mqh_pid, sigev->sigev_signo,

35 sigev->sigev_value);

36 }

37 mqhdr->mqh_pid = 0; /* снятие с регистрации */

38 }

39 } else if (attr->mq_curmsgs >= attr->mq_maxmsg) {

40 /* 4queue is full */

41 if (mqinfo->mqi_flags & O_NONBLOCK) {

32 errno = EAGAIN;

43 goto err;

44 }

45 /* ожидание освобождения места в очереди */

46 while (attr->mq_curmsgs >= attr->mq_maxmsg)

47 pthread_cond_wait(&mqhdr->mqh_wait, &mqhdr->mqh_lock);

48 }

Листинг 5.25. Функция mq_send: вторая половина

//my_pxmsg_mmap/mq_send.с

49 /* nmsghdr будет указывать на новое сообщение*/

50 if ((freeindex = mqhdr->mqh_free) == 0)

51 err_dump("mymq_send: curmsgs = %ld; free = 0", attr->mq_curmsgs);

52 nmsghdr = (struct mymsg_hdr *) &mptr[freeindex];

53 nmsghdr->msg_prio = prio;

54 nmsghdr->msg_len = len;

55 memcpy(nmsghdr + 1, ptr, len); /* копирование сообщения в очередь */

56 mqhdr->mqh_free = nmsghdr->msg_next; /* новое начало списка пустых сообщений */

57 /* поиск места в списке для нового сообщения */

58 index = mqhdr->mqh_head;

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

Кодекс Крови. Книга IV

Борзых М.
4. РОС: Кодекс Крови
Фантастика:
фэнтези
попаданцы
аниме
5.00
рейтинг книги
Кодекс Крови. Книга IV

Печать Пожирателя

Соломенный Илья
4. Пожиратель
Фантастика:
аниме
сказочная фантастика
фэнтези
попаданцы
5.00
рейтинг книги
Печать Пожирателя

Наследник

Кулаков Алексей Иванович
1. Рюрикова кровь
Фантастика:
научная фантастика
попаданцы
альтернативная история
8.69
рейтинг книги
Наследник

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

Панежин Евгений
5. Наследие Маозари
Фантастика:
фэнтези
юмористическое фэнтези
5.00
рейтинг книги
Наследие Маозари 5

Страж

Иевлев Геннадий Васильевич
1. Страж
Фантастика:
научная фантастика
6.00
рейтинг книги
Страж

Я уже барон

Дрейк Сириус
2. Дорогой барон!
Фантастика:
боевая фантастика
попаданцы
аниме
5.00
рейтинг книги
Я уже барон

Мл. сержант. Назад в СССР. Книга 3

Гаусс Максим
3. Второй шанс
Фантастика:
альтернативная история
6.40
рейтинг книги
Мл. сержант. Назад в СССР. Книга 3

Сердце Дракона. Том 10

Клеванский Кирилл Сергеевич
10. Сердце дракона
Фантастика:
фэнтези
героическая фантастика
боевая фантастика
7.14
рейтинг книги
Сердце Дракона. Том 10

Газлайтер. Том 21

Володин Григорий Григорьевич
21. История Телепата
Фантастика:
боевая фантастика
аниме
попаданцы
5.00
рейтинг книги
Газлайтер. Том 21

Наследник старого рода

Шелег Дмитрий Витальевич
1. Живой лёд
Фантастика:
фэнтези
8.19
рейтинг книги
Наследник старого рода

Дочь моего друга

Тоцка Тала
2. Айдаровы
Любовные романы:
современные любовные романы
эро литература
5.00
рейтинг книги
Дочь моего друга

Я уже граф. Книга VII

Дрейк Сириус
7. Дорогой барон!
Фантастика:
боевая фантастика
попаданцы
аниме
5.00
рейтинг книги
Я уже граф. Книга VII

Товарищ "Чума" 6

lanpirot
6. Товарищ "Чума"
Фантастика:
городское фэнтези
попаданцы
альтернативная история
5.00
рейтинг книги
Товарищ Чума 6

Воин-Врач

Дмитриев Олег
1. Воин-Врач
Фантастика:
попаданцы
альтернативная история
историческое фэнтези
6.00
рейтинг книги
Воин-Врач