UNIX: разработка сетевых приложений - Уильям Стивенс
Шрифт:
Интервал:
Закладка:
Рис. 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 &&
36 icmp6->icmp6_code == ICMP6_TIME_EXCEED_TRANSIT) {
37 if (icmp6len < 8 + sizeof(struct ip6_hdr) + 4)
38 continue; /* недостаточно для проверки внутреннего заголовка */
39 hip6 = (struct ip6_hdr*)(recvbuf + 8);
40 hlen2 = sizeof(struct ip6_hdr);
41 udp = (struct udphdr*)(recvbuf + 8 + hlen2);
42 if (hip6->ip6_nxt == IPPROTO_UDP &&
43 udp->uh_sport == htons(sport) &&
44 udp->uh_dport == htons(dport + seq))
45 ret = -2; /* ответил промежуточный маршрутизатор */
46 break;
47 } else if (icmp6->icmp6_type == ICMP6_DST_UNREACH) {
48 if (icmp6len < 8 + sizeof(struct ip6_hdr) + 4)
49 continue; /* недостаточно для проверки внутреннего заголовка */
50 hip6 = (struct ip6_hdr*)(recvbuf + 8);
51 hlen2 = sizeof(struct ip6_hdr);
52 udp = (struct udphdr*)(recvbuf + 8 + hlen2);
53 if (hip6->ip6_nxt == IPPROTO_UDP &&
54 udp->uh_sport == htons(sport) &&
55 udp->uh_dport == htons(dport + seq)) {
56 if (icmp6->icmp6_code == ICMP6_DST_UNREACH_NOPORT)
57 ret = -1; /* цель достигнута */
58 else
59 ret = icmp6->icmp6_code; /* 0, 1, 2, ... */
60 break;
61 }
62 } else if (verbose) {
63 printf(" (from %s: type = %d, code = %d)n",
64 Sock_ntop_host(pr->sarecv, pr->salen);
65 icmp6->icmp6_type, icmp6->icmp6_code);
66 }
67 /* другая ICMP-ошибка. нужно вызвать recvfrom() */
68 }
69 alarm(0); /* отключаем таймер */
70 Gettimeofday(tv, NULL); /* get time of packet arrival */
71 return(ret);
72 #endif
73 }
Листинг 28.19. Возвращение строки, соответствующей коду недоступности ICMPv6
//traceroute/icmpcode_v6.c
1 #include "trace.h"
2 const char *
3 icmpcode_v6(int code)
4 {
5 #ifdef IPV6
6 static char errbuf[100];
7 switch (code) {
8 case ICMP6_DST_UNREACH_NOROUTE:
9 return("no route to host");
10 case ICMP6_DST_UNREACH_ADMIN:
11 return("administratively prohibited");
12 case ICMP6_DST_UNREACH_NOTNEIGHBOR:
13 return("not a neighbor");
14 case ICMP6_DST_UNREACH_ADDR:
15 return("address unreachable");
15 case ICMP6_DST_UNREACH_NOPORT:
16 return("port unreachable");
17 default:
18 sprintf(errbuf, "[unknown code %d]",. code);
19 return errbuf;
20 }
21 #endif
22 }
Пример
Сначала приведем пример с Ipv4:
freebsd % traceroute www.unpbook.com
traceroute to www.unpbook.com (206.168.112.219): 30 hops max. 24 data bytes
1 12.106.32.1 (12.106.32.1) 0.799 ms 0.719 ms 0.540 ms
2 12.124.47.113 (12.124.47.113) 1.758 ms 1.760 ms 1.839 ms
3 gbr2-p27.sffca.ip.att.net (12.123.195.38) 2.744 ms 2.575 ms 2.648 ms
4 tbr2-p012701.sffca.ip.att.net (12.122.11.85) 3.770 ms 3.689 ms 3.848 ms
5 gbr3-p50.dvmco.ip.att.net (12.122.2.66) 26.202 ms 26.242 ms 26.102 ms
6 gbr2-p20.dvmco.ip.att.net (12.122.5.26) 26 255 ms 26.194 ms 26.470 ms
7 gar2-p370.dvmco.ip.att.net (12.123.36.141) 26.443 ms 26.310 ms 26.427 ms
8 att-46.den.internap.ip.att.net (12.124.158.58) 26.962 ms 27.130 ms 27.279 ms
9 border10 ge3-0-bbnet2.den.pnap.net (216.52.40.79) 27.285 ms 27 293 ms 26.860 ms
10 coop-2.border10.den.pnap.net (216 52.42.118) 28.721 ms 28.991 ms 30.077 ms
11 199.45.130.33 (199.45.130.33) 29.095 ms 29.055 ms 29 378 ms
12 border-to-141-netrack.boulder.со.coop.net (207.174.144.178) 30.875 ms 29.747 ms 30.142 ms
13 linux.unpbook.com (206.168.112.219) 31.713 ms 31.573 ms 33.952 ms
Ниже приведен пример с IPv6. Для лучшей читаемости длинные строки разбиты.
freebsd % traceroute www.kame.net
traceroute to orange.kame.net (2001:200:0:4819:203:47ff:fea5:3085): 30 hops max, 24 data bytes
1 3ffe:b80:3:9ad1::1 (3ffe:b80:3:9ad1::1) 107.437 ms 99.341 ms 103.477 ms
2 Viagenie-gw.int.ipv6.ascc.net (2001:288:3b0::55)
105.129 ms 89.418 ms 90.016 ms
3 gw-Viagenie.int.ipv6.ascc.net (2001:288:3b0::54)
302.300 ms 291.580 ms 289.839 ms
4 c7513-gw.int.ipv6.ascc.net (2001:288:3b0::c)
296.088 ms 298.600 ms 292.196 ms
5 m160-c7513.int.ipv6.ascc.net (2001:288:3b0::1e)
296.266 ms 314.878 ms 302.429 ms
6 m20jp-ml60tw.int.ipv6.ascc.net (2001:288:3b0::1b)
327.637 ms 326.897 ms 347.062 ms
7 hitachi1.otemachi.wide.ad.jp (2001:200:0:1800::9c4:2)
420.140 ms 426.592 ms 422.756 ms
8 pc3.yagami.wide.ad.jp (2001:200:0:1c04::1000:2000)
415.471 ms 418.308 ms 461.654 ms