Книга: UNIX: разработка сетевых приложений

Г.1. Заголовочный файл unp.h

Г.1. Заголовочный файл unp.h

Почти каждая программа в этой книге начинается с подключения заголовочного файла unp.h, показанного в листинге Г.1[1]. Этот файл подключает все стандартные системные заголовочные файлы, необходимые для работы большинства программ, а также некоторые общие системные заголовочные файлы. В нем также определены такие константы, как MAXLINE, прототипы функций ANSI С для тех функций, которые мы определяем в тексте (например, readline), и все используемые нами функции-обёртки. Сами прототипы в приведенном ниже листинге мы не показываем.

Листинг Г.1. Заголовочный файл unp.h

//lib/unp.h
  1 /* Наш собственный заголовочный файл */
  2 #ifndef __unp_h
  3 #define __unp_h
  4 #include "../config.h" /* параметры конфигурации для данной ОС */
  5 /* "../config.h" генерируется сценарием configure */
  6 /* изменив список директив #include,
  7    нужно также изменить файл acsite.m4 */
  8 #include <sys/types.h> /* основные системные типы данных */
  9 #include <sys/socket.h> /* основные определения сокетов */
 10 #include <sys/time.h> /* структура timeval{} для функции select() */
 11 #include <time.h> /* структура timespec{} для функции pselect() */
 12 #include <netinet/in.h> /* структура sockaddr_in{} и другие сетевые
                               определения */
 13 #include <arpa/inet.h> /* inet(3) функции */
 14 #include <errno.h>
 15 #include <fcntl.h> /* для неблокируемых сокетов */
 16 #include <netdb.h>
 17 #include <signal.h>
 18 #include <stdio.h>
 19 #include <stdlib.h>
 20 #include <string.h>
 21 #include <sys/stat.h> /* для констант S_xxx */
 22 #include <sys/uio.h> /* для структуры iovec{} и ready/writev */
 23 #include <unistd.h>
 24 #include <sys/wait.h>
 25 #include <sys/un.h> /* для доменных сокетов Unix */
 26 #ifdef HAVE_SYS_SELECT_H
 27 #include <sys/select.h> /* для удобства */
 28 #endif
 29 #ifdef HAVE_SYS_SYSCTL_H
 30 #include <sys/sysctl.h>
 31 #endif
 32 #ifdef HAVE_POLL_H
 33 #include <poll.h> /* для удобства */
 34 #endif
 35 #ifdef HAVE_SYS_EVENT_H
 36 #include <sys/event.h> /* для kqueue */
 37 #endif
 38 #ifdef HAVE_STRINGS_H
 39 #include <strings.h> /* для удобства */
 40 #endif
 41 /* Три заголовочных файла обычно нужны для вызова ioctl
 42    для сокета/файла: <sys/ioctl.h>, <sys/filio.h>,
 43    <sys/sockio.h> */
 44 #ifdef HAVE_SYS_IOCTL_H
 45 #include <sys/ioctl.h>
 46 #endif
 47 #ifdef HAVE_SYS_FILIO_H
 48 #include <sys/filio.h>
 49 #endif
 50 #ifdef HAVE_SYS_SOCKIO_H
 51 #include <sys/sockio.h>
 52 #endif
 53 #ifdef HAVE_PTHREAD_H
 54 #include <pthread.h>
 55 #endif
 56 #ifdef HAVE_NET_IF_DL_H
 57 #include <net/if_dl.h>
 58 #endif
 59 #ifdef HAVE_NETINET_SCTP_H
 60 #include <netinet/sctp.h>
 61 #endif
 62 /* OSF/1 фактически запрещает recv() и send() в <sys/socket.h> */
 63 #ifdef __osf__
 64 #undef recv
 65 #undef send
 66 #define recv(a,b,c,d) recvfrom(a,b,c,d,0,0)
 67 #define send(a,b,c,d) sendto(a,b,c,d,0,0)
 68 #endif
 69 #ifndef INADDR_NONE
 70 #define INADDR_NONE 0xffffffff /* должно было быть в <netinet/in.h> */
 71 #endif
 72 #ifndef SHUT_RD     /* три новые константы Posix.1g */
 73 #define SHUT_RD   0 /* отключение чтения */
 74 #define SHUT_WR   1 /* отключение записи */
 75 #define SHUT_RDWR 2 /* отключение чтения и записи */
 76 #endif
 77 #ifndef INET_ADDRSTRLEN
 78 #define INET_ADDRSTRLEN 16 /* "ddd.ddd.ddd.ddd"
 79 1234567890123456 */
 80 #endif
 81 /* Нужно, даже если нет поддержки IPv6, чтобы мы всегда могли
 82    разместить в памяти буфер требуемого размера без директив #ifdef */
 83 #ifndef INET6_ADDRSTRLEN
 84 #define INET6_ADDRSTRLEN 46 /* максимальная длина строки адреса IPv6:
 85 "xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:xxxx" или
 86 "xxxx:xxxx:xxxx:xxxx:xxxx:xxxx:ddd.ddd.ddd.ddd"
 87 1234567890123456789012345678901234567890123456 */
 88 #endif
 89 /* Определяем bzero() как макрос, если эта функция отсутствует в
       стандартной библиотеке С */
 90 #ifndef HAVE_BZERO
 91 #define bzero(ptr,n) memset(ptr, 0, n)
 92 #endif
 93 /* В более старых распознавателях отсутствует gethostbyname2() */
 94 #ifndef HAVE_GETHOSTBYNAME2
 95 #define gethostbyname2(host, family) gethostbyname((host))
 96 #endif
 97 /* Структура, возвращаемая функцией recvfrom_flags() */
 98 struct in_pktinfo {
 99  struct in_addr ipi_addr; /* IPv4-адрес получателя */
100  int    ipi_ifindex; /* полученный индекс интерфейса */
101 };
102 /* Нам нужны более новые макросы CMSG_LEN() и CMSG_SPACE(), но в
103    настоящее время их поддерживают далеко не все реализации. Им требуется
104    макрос ALIGN(), но это зависит от реализации */
105 #ifndef CMSG_LEN
106 #define CMSG_LEN(size) (sizeof(struct cmsghdr) + (size))
107 #endif
108 #ifndef CMSG_SPACE
109 #define CMSG_SPACE(size) (sizeof(struct cmsghdr) + (size))
110 #endif
111 /* POSIX требует макрос SUN_LEN(), но он определен
112 не во всех реализациях. Этот макрос 4.4BSD работает
123 независимо от того, имеется ли поле длины */
114 #ifndef SUN_LEN
115 #define SUN_LEN(su)
116  (sizeof(*(su)) - sizeof((su)->sun_path) + strlen((su)->sun_path))
117 #endif
118 /* В POSIX "домен Unix" называется "локальным IPC".
119    Но пока не во всех системах определены AF_LOCAL и PF_LOCAL */
120 #ifndef AF_LOCAL
121 #define AF_LOCAL AF_UNIX
122 #endif
123 #ifndef PF_LOCAL
124 #define PF_LOCAL PF_UNIX
125 #endif
126 /* POSIX требует определения константы INFTIM в <poll.h>, но во многих
127    системах она по-прежнему определяется в <sys/stropts.h>. Чтобы
128    не подключать все функции работы с потоками, определяем ее здесь.
129    Это стандартное значение, но нет гарантии, что оно равно -1 */
130 #ifndef INFTIM
131 #define INFTIM (-1) /* бесконечный тайм-аут */
132 #ifdef HAVE_POLL_H
133 #define INFTIM_UNPH /* надо указать в unpxti.h, что эта константа
                           определена здесь */
134 #endif
135 #endif
136 /* Это значение можно было бы извлечь из SOMAXCONN в <sys/socket.h>,
137    но многие ядра по-прежнему определяют его как 5,
       хотя на самом деле поддерживается гораздо больше */
138 #define LISTENQ 1024 /* второй аргумент функции listen() */
139 /* Различные константы */
140 #define MAXLINE  4096 /* максимальная длина текстовой строки */
141 #define BUFFSIZE 8192 /* размер буфера для чтения и записи */
142 /* Определение номера порта, который может быть использован для
       взаимодействия клиент-сервер */
143 #define SERV_PORT      9877  /* клиенты и серверы TCP и UDP */
144 #define SERV_PORT_STR "9877" /* клиенты и серверы TCP и UDP */
145 #define UNIXSTR_PATH "/tmp/unix.str" /* потоковые клиенты и серверы
                                            домена Unix */
146 #define UNIXDG_PATH "/tmp/unix.dg" /* клиенты и серверы протокола
                                          дейтаграмм домена Unix */
147 /* Дальнейшие определения сокращают преобразования типов
       аргументов-указателей */
148 #define SA struct sockaddr
149 #define HAVE_STRUCT_SOCKADDR_STORAGE
150 #ifndef HAVE_STRUCT_SOCKADDR_STORAGE
151 /*
152  * RFC 3493: протокольно-независимая структура адреса сокета
153  */
154 #define __SS_MAXSIZE 128
155 #define __SS_ALIGNSIZE (sizeof(int64_t))
156 #ifndef HAVE_SOCKADDR_SA_LEN
157 #define __SS_PADS1SIZE (__SS_ALIGNSIZE - sizeof(u_char) -
sizeof(sa_family_t))
158 #else
159 #define _SS_PAD1SIZE (__SS_ALIGNSIZE - sizeof(sa_family_t))
160 #endif
161 #define __SS_PAD2SIZE (__SS_MAXSIZE — 2*__SS_ALIGNSIZE)
162 struct sockaddr_storage {
163 #ifdef HAVE_SOCKADDR_SA_LEN
164  u_char ss_len;
165 #endif
166  sa_family_t ss_family;
167  char        __ss_pad1[__SS_PAD1SIZE];
168  int64_t     ss_align;
169  char        __ss_pad2[_SS_PAD2SIZE];
170 };
171 #endif
172 #define FILE_MODE (S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH)
173 /* заданные по умолчанию разрешения на доступ для новых файлов */
174 #define DIR_MODE (FILE_MODE | S_IXUSR | S_IXGRP | S_IXOTH)
175 /* разрешения по умолчанию на доступ к файлам для новых каталогов */
176 typedef void Sigfunc(int); /* для обработчиков сигналов */
177 #define min(a, b) ((а) < (b) ? (a) : (b))
178 #define max(a, b) ((a) > (b) ? (a) : (b))
179 #ifndef HAVE_ADDRINFO_STRUCT
180 #include "../lib/addrinfo.h"
181 #endif
182 #ifndef HAVE_IF_NAMEINDEX_STRUCT
183 struct if_nameindex {
184  unsigned int if_index; /* 1, 2, ... */
185  char *if_name; /* имя, заканчивающееся нулем: "le0", ... */
186 };
187 #endif
188 #ifndef HAVE_TIMESPEC_STRUCT
189 struct timespec {
190  time_t tv_sec; /* секунды */
191  long tv_nsec;  /* и наносекунды */
192 };
193 #endif

Оглавление книги


Генерация: 1.148. Запросов К БД/Cache: 3 / 0
поделиться
Вверх Вниз