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

146 #define va_mode_t int

147 #else

148 #define va_mode_t mode_t

149 #endif


150 /* макросы блокировки записей */

151 #define read_lock(fd, offset, whence, len) \

152  lock_reg(fd, F_SETLK, F_RDLCK, offset, whence, len)

153 #define readw_lock(fd, offset, whence, len) \

154  lock_reg(fd, F_SETLKW, F_RDLCK, offset, whence, len)

155 #define write_lock(fd, offset, whence, len) \

156  lock_reg(fd, F_SETLK, F_WRLCK, offset, whence, len)

157 #define writew_lock(fd, offset, whence, len) \

158  lock_reg(fd, F_SETLKW, F_WRLCK, offset, whence, len)

159 #define un_lock(fd, offset, whence, len) \

160  lock_reg(fd, F_SETLK, F_UNLCK, offset, whence, len)

161 #define is_read_lockable(fd, offset, whence, len) \

162  lock_test(fd, F_RDLCK, offset, whence, len)

163 #define is_write_lockable(fd, offset, whence, len) \

164  lock_test(fd. F_WRLCK, offset, whence, len)

B.2. Заголовочный файл config.h

Для подготовки пpoгрaмм в этой книге использовалась утилита GNU autoconf, которая помогает сделать их более переносимыми. Она доступна по адресу ftp://prep.ai.mit.edu/pub/gnu. Утилита создает сценарий configure, который следует запустить после того, как вы зaгрyзитe пpoгрaммy из Сети. Этот сценарий определяет возможности вашей системы: поддерживаются ли очереди System V, определен ли тип uint8_t, определена ли функция gethostname и т. д. В процессе работы он создает заголовочный файл config.h, который включается нашим unpipc.h перед всеми остальными. В листинге В.2 приведен пример заголовочного файла config.h для системы Solaris 2.6 и компилятора gcc.

Строки, начинающиеся с #define, описывают функции, поддерживаемые в системе. Закомментированные строки соответствуют неподдерживаемым функциям.

Листинг В.2. Заголовочный файл config.h для Solaris 2.6

//sparc-sun-solaris2.6/config.h

1  /* config.h. Автоматически создается сценарием configure. */

2  /* Константы следует указывать только тогда, когда есть соответствующий заголовочный файл */

3  #define CPU_VENDOR_OS "sparc-sun-solaris2.6"

4  #define HAVE_DOOR_H 1 /*  */

5  #define HAVE_MQUEUE_H 1 /*  */

6  #define HAVE_POLL_H 1 /**/

7  #define HAVE_PTHREAD_H 1 /*  */

8  #define HAVE_RPC_RPC_H 1 /*  */

9  #define HAVE_SEMAPHORE_H 1 /* semaphore.h> */

10 #define HAVE_STRINGS_H 1 /*  */

11 #define HAVE_SYS_FILIO_H 1 /*  */

12 #define HAVE_SYS_IOCTL_H 1 /*  */

13 #define HAVE_SYS_IPC_H 1 /*  */

14 #define HAVE_SYS_MMAN_H 1 /*  */

15 #define HAVE_SYS_MSG_H 1 /*  */

16 #define HAVE_SYS_SEM_H 1 /*  */

17 #define HAVE_SYS_SHM_H 1 /*  */

18 #define HAVE_SYS_SELECT_H 1 /*  */

19 /* #undef HAVE_SYS_SYSCTL_H */ /*  */

20 #define HAVE_SYS_TIME_H 1 /*  */


21 /* Определена, если можно подключить  вместе с  */

22 #define TIME_WITH_SYS_TIME 1


23 /* Определены, если имеются соответствующие функции */

24 #define HAVE_BZERO 1

25 #define HAVE_FATTACH 1

26 #define HAVE_POLL 1

27 /* #undef HAVE_PSELECT */

28 #define HAVE_SIGWAIT 1

29 #define HAVE_VALLOC 1

30 #define HAVE_VSNPRINTF 1


31 /* Определены, если прототипы функций есть в заголовках */

32 #define HAVE_GETHOSTNAME_PROTO 1 /*  */

33 #define HAVE_GETRUSAGE_PROTO 1 /*  */

34 /* #undef HAVE_PSELECT_PROTO */ /*  */

35 #define HAVE SHM_OPEN_PROTO 1 /*  */

36 #define HAVE_SNPRINTF_PROTO 1 /*  */

37 #define HAVE_THR_SETCONCURRENCY_PROTO 1 /*  */


38 /* Определены, если определены соответствующие структуры */

39 #define HAVE_SIGINFO_T_STRUCT 1 /*  */

40 #define HAVE_TIMESPEC_STRUCT 1 /*  */

41 /* #undef HAVE_SEMUN_UNION */ /*  */


42 /* Устройства */

43 #define HAVE_DEV_ZERO 1


44 /* Для соответствующих типов данных */

45 /* #undef int8_t */ /*  */

46 /* #undef intl6_t */ /*  */

47 /* #undef int32_t */ /*  */

48 /* #undef uint8_t */ /*  */

49 /* #undef uintl6_t */ /*  */

50 /* #undef uint32_t */ /*  */

51 /* #undef size_t */ /*  */

52 /* #undef ssize_t */ /*  */


53 #define POSIX_IPC_PREFIX "/"

54 #define RPCGEN_ANSIC 1 /* определена, если rpcgen поддерживает параметр –С */

В.З. Стандартные функции вывода сообщений об ошибках

Мы определили свой набор функций, используемых во всех программах книги для обработки ситуаций с возникновением ошибок. Причина, по которой мы создаем эти функции, заключается в том, что теперь мы можем писать команды в одну строку:

if (условие_ошибки) err_sys(формат printf с произвольным количеством аргументов);

вместо:

if (условие_ошибки) {

 char buff[200];

 snprintf(buff, sizeof(buff), формат printf с произвольным количеством аргументов);

 perror(buff);

 exit(1);

}

Функции обработки ошибок используют возможности работы со списком аргументов переменной длины, определенные стандартом ANSI С. В разделе 7.3 [ 11 ] вы можете узнать подробности.

В таблице В.1 приведены отличия между различными функциями обработки ошибок. Если глобальное целое daemon_proc отлично от нуля, сообщение передается демону syslog с указанным уровнем (см. главу 12 [24]); в противном случае сообщение выводится в стандартный поток сообщений об ошибках.


Таблица В.1. Функции обработки ошибок

Функцияstrerror(errno)?Завершение?Уровень syslog
err_dumpДаabort();LOG_ERR
err_msgНетreturn;LOG_INFO
err_quitНетexit(1);LOG_ERR
err_retДаreturn;LOG_INFO
err_sysДаexit(1);