Выбрать главу

34 /* #undef HAVE_PSELECT_PROTO */ /* <sys/select.h> */

35 #define HAVE SHM_OPEN_PROTO 1 /* <sys/mman.h> */

36 #define HAVE_SNPRINTF_PROTO 1 /* <stdio.h> */

37 #define HAVE_THR_SETCONCURRENCY_PROTO 1 /* <thread.h> */

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

39 #define HAVE_SIGINFO_T_STRUCT 1 /* <signal.h> */

40 #define HAVE_TIMESPEC_STRUCT 1 /* <time.h> */

41 /* #undef HAVE_SEMUN_UNION */ /* <sys/sem.h> */

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

43 #define HAVE_DEV_ZERO 1

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

45 /* #undef int8_t */ /* <sys/types.h> */

46 /* #undef intl6_t */ /* <sys/types.h> */

47 /* #undef int32_t */ /* <sys/types.h> */

48 /* #undef uint8_t */ /* <sys/types.h> */

49 /* #undef uintl6_t */ /* <sys/types.h> */

50 /* #undef uint32_t */ /* <sys/types.n> */

51 /* #undef size_t */ /* <sys/types.h> */

52 /* #undef ssize_t */ /* <sys/types.h> */

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); LOG_ERR

В листинге В.3 приведен текст функций из табл. В.1.

Листинг В.3. Функции обработки ошибок

//lib/error.с

1  #include "unpipc.h"

2  #include <stdarg.h> /* заголовочный файл ANSI С */

3  #include <syslog.h> /* для демона syslog() */

4  int daemon_proc; /* устанавливается отличным от нуля daemon_init() */

5  static void err_doit(int, int, const char*, va_list);

6  /* Нефатальная ошибка при системном вызове.

7   * Вывод сообщения и возврат из функции. */

8  void

9  err_ret(const char *fmt, ...)

10 {

11  va_list ар;

12  va_start(ap, fmt);

13  err_doit(1, LOG_INFO, fmt, ap);

14  va_end(ap);

15  return;

16 }

17 /* Фатальная ошибка при системном вызове.

18  * Вывод сообщения и завершение работы. */

19 void

20 err_sys(const char *fmt, ...)

21 {

22  va_list ap;

23  va_start(ap, fmt);

24  err_doit(1, LOG_ERR, fmt, ар);

25  va_end(ap);

26  exit(1);

27 }

28 /* Фатальная ошибка при системном вызове.

29  * Вывод сообщения, сохранение дампа памяти, завершение работы. */

30 void

31 err_dump(const char *fmt, ...)

32 {

33  va_list ар;

34  va_start(ap, fmt);

35  err_doit(1, LOG_ERR, fmt, ap);

36  va_end(ap);

37  abort(); /* сохранение дампа и завершение */

38  exit(1); /* досюда не должно дойти */

39 }

40 /* Нефатальная ошибка не при системном вызове.

41  * Вывод сообщения и возврат. */

42 void

43 err_msg(const char *fmt, ...)

44 {

45  va_list ap;

46  va_start(ap, fmt);

47  err_doit(0, LOG_INFO, fmt, ap);

48  va_end(ap);

49  return;

50 }

51 /* Фатальная ошибка не при системном вызове.

52  * Вывод сообщения и возврат. */

53 void

54 err_quit(const char *fmt, ...)

55 {

56  va_list ap;

57  va_start(ap, fmt);

58  err_doit(0, LOG_ERR, fmt, ap);

59  va_end(ap);

60  exit(1);

61 }

62 /* Вывод сообщения и возврат.

63  * Вызывающий указывает "errnoflag" и "level". */

64 static void

65 err_doit(int errnoflag, int level, const char *fmt, va_list ap)

66 {

67  int errno_save, n;

68  char buf[MAXLINE];

69  errno_save = errno; /* значение может понадобиться вызвавшему */

70 #ifdef HAVE_VSNPRINTF