UNIX: разработка сетевых приложений
Шрифт:
ПРИМЕЧАНИЕ
Эта функция не создает ситуации гонок, описанной в разделе 20.5, благодаря использованию глобального флага.
Извлечение указателя на ICMP-заголовок
31-35
Указатель iр
указывает на начало IPv4-заголовка (напомним, что операция чтения на символьном сокете всегда возвращает IP-заголовок), а указатель icmp
указывает на начало ICMP-заголовка. На рис. 28.5 показаны различные заголовки, указатели и длины, используемые в данном коде. Рис. 28.5.
Обработка ICMP-сообщения о превышении времени передачи
36-50
Если ICMP-сообщение является сообщением «Time exceeded in transit» (Превышено время передачи), вероятно, оно является ответом на один из наших пробных пакетов. Указатель hip
указывает на заголовок IPv4, который возвращается в ICMP-сообщении и следует сразу за 8-байтовым ICMP-заголовком. Указатель udp
указывает на следующий далее UDP-заголовок. Если ICMP-сообщение было сгенерировано UDP-дейтаграммой и если порты отправителя и получателя этой дейтаграммы совпадают с теми значениями, которые мы посылали, то тогда это ответ от промежуточного маршрутизатора на наш пробный пакет. Обработка ICMP-сообщения о недоступности порта
51-68
Если ICMP-сообщение является сообщением «Destination unreachable» (Получатель недоступен), тогда, чтобы узнать, является ли это сообщение ответом на наш пробный пакет, мы смотрим на UDP-заголовок, возвращенный в данном ICMP-сообщении. Если это так и код означает сообщение «Port unreachable» (Порт недоступен), то возвращается значение -1, поскольку достигнут конечный получатель. Если же ICMP-сообщение является ответом на один из наших пробных пакетов, но не является сообщением типа «Destination unreachable» (Получатель недоступен), то тогда возвращается значение ICMP-кода. Обычным примером такого случая является ситуация, когда брандмауэр возвращает какой-либо другой код недоступности для получателя, на который посылается пробный пакет. Обработка других ICMP-сообщений
69-73
Все остальные ICMP-сообщения выводятся, если был задан параметр – v
. Следующая функция, recv_v6, приведена в листинге 28.18 и является IPv6-вepсией ранее описанной функции для IPv4. Эта функция почти идентична функции
recv_v4
, за исключением различий в именах констант и элементов структур. Кроме того, размер заголовка IPv6 является фиксированным и составляет 40 байт, в то время как для получения IP-параметров в заголовке IPv4 необходимо получить поле длины заголовка и умножить его на 4. На рис. 28.6 приведены различные заголовки, указатели и длины, используемые в коде. Рис. 28.6. Заголовки, указатели и длины, используемые при обработке ошибки ICMPv6
Мы определяем две функции
icmpcode_v4
и icmpcode_v6
, которые можно вызывать в конце функции traceloop
для вывода строки описания, соответствующей ICMP-ошибке недоступности получателя. В листинге 28.19 приведена IPv6-функция. IPv4-функция аналогична, хотя и длиннее, поскольку существует большее количество ICMPv4-кодов недоступности получателя (см. табл. А.5). Последней функцией в нашей программе
traceroute
является обработчик сигнала SIGALRM
—
функция sig_alrm
, приведенная в листинге 28.17. Эта функция лишь возвращает ошибку EINTR
из функции recvfrom
, как в случае функции recv_v4
, так и в случае recv_v6
. Листинг 28.17. Функция sig_alrm
//traceroutе/sig_alrm.c
1 #include "trace.h"
2 int gotalarm;
3 void
4 sig_alrm(int signo)
5 {
6 gotalarm = 1; /* установка флага, оповещающего о сигнале */
7 return; /* прерывается работа функции recvfrom */
8 }
Листинг 28.18. Функция recv_v6: чтение и обработка сообщений ICMPv6
//traceroute/recv_v6
1 #include "trace.h"
2 extern int gotalarm;
3 /*
4 * Возвращает; -3 при тайм-ауте
5 * -2 для сообщения ICMP time exceeded in transit (продолжаем поиск
маршрута)
6 * -1 для сообщения ICMP port unreachable (цель достигнута)
7 * неотрицательные значения соответствуют всем прочим ICMP-сообщениям
8 */
9 int
10 recv_v6(int seq, struct timeval *tv)
11 {
12 #ifdef IPV6
13 int hlen2, icmp6len, ret;
14 ssize_t n;
15 socklen_t len;
16 struct ip6_hdr *hip6;
17 struct icmp6_hdr *icmp6;
18 struct udphdr *udp;
19 gotalarm = 0;
20 alarm(3);
21 for (;;) {
22 if (gotalarm)
23 return(-3); /* истек таймер */
24 len = pr->salen;
25 n = recvfrom(recvfd, recvbuf, sizeof(recvbuf), 0, pr->sarecv, &len);
26 if (n < 0) {
27 if (errno == EINTR)
28 continue;
29 else
30 err_sys("recvfrom error");
31 }
32 icmp6 = (struct icmp6_hdr*)recvbuf; /* ICMP-заголовок */
33 if ((icmp6len = n) < 8)
34 continue; /* недостаточно для проверки ICMP-заголовка */
35 if (icmp6->icmp6_type == ICMP6_TIME_EXCEEDED &&
Поделиться с друзьями: