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, ap);
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 + 1];
69 errno_save = errno; /* значение может понадобиться вызвавшему
процессу */
70 #ifdef HAVE_VSNPRINTF
71 vsnprintf(buf, MAXLINE, fmt, ap); /* защищенный вариант */
72 #else
73 vsprintf(buf, fmt, ap); /* незащищенный вариант */
74 #endif
75 n = strlen(buf);
76 if (errnoflag)
77 snprintf(buf + n, MAXLINE - n, ": %s", strerror(errno_save));
78 strcat(buf, "\n");
79 if (daemon_proc) {
80 syslog(level, buf);
81 } else {
82 fflush(stdout); /* если stdout и stderr совпадают */
83 fputs(buf, stderr);
84 fflush(stderr);
85 }
86 return;
87 }
Приложение Д
Решения некоторых упражнений
Глава 1
1.3. В операционной системе Solaris получаем:
solaris %
daytimetcpcli 127.0.0.1
socket error: Protocol not supported
Для получения дополнительной информации об этой ошибке сначала используем программу
grep
, чтобы найти строку
Protocol not supported
в заголовочном файле
sys/errno.h
.
solaris %
grep 'Protocol not supported' /usr/include/sys/errno.h
#define EPROTONOSUPPORT 120 /* Protocol not supported */
Это значение
errno
возвращается функцией
socket
. Далее смотрим в руководство пользователя:
solaris %
man socket
В большинстве руководств пользователя в конце под заголовком «Errors» приводится дополнительная, хотя и лаконичная информация об ошибках.
1.4. Заменяем первое описание на следующее:
int sockfd, n, counter = 0;
Добавляем оператор
counter++;
в качестве первого оператора цикла
while
. Наконец, прежде чем прервать программу, выполняем
printf("counter = %d\n", counter);
На экран всегда выводится значение 1.