Чтение онлайн

ЖАНРЫ

UNIX: разработка сетевых приложений
Шрифт:
Вывод всех полученных ICMP-сообщений при включении параметра verbose

28-32
Если пользователем указан параметр командной строки
– v
, также выводятся поля типа и кода из всех других полученных ICMP-сообщений.

Обработка сообщений ICMPv6 управляется функцией

proc_v6
, приведенной в листинге 28.8. Она аналогична функции
proc_v4
, представленной в листинге 28.6. Однако поскольку символьные сокеты IPv6 не передают процессу заголовок IPv6, ограничение на количество транзитных узлов
приходится получать в виде вспомогательных данных. Для этого нам приходится подготавливать сокет функцией
init_v6
, представленной в листинге 28.7.

Листинг 28.7. Функция init_v6: подготовка сокета

1 void

2 init_v6

3 {

4 #ifdef IPV6

5 int on = 1;

6 if (verbose == 0) {

7 /* установка фильтра, пропускающего только пакеты ICMP6_ECHO_REPLY. если

не включен параметр verbose (вывод всех ICMP-сообщений) */

8 struct icmp6_filter myfilt;

9 ICMP6_FILTER_SETBLOCKALL(&myfilt);

10 ICMP6_FILTER_SETPASS(ICMP6_ECHO_REPLY, &myfilt);

11 setsockopt(sockfd, IPPROTO_IPV6, ICMP6_FILTER, &myfilt,

12 sizeof(myfilt));

13 /* игнорируем ошибку, потому что фильтр - необязательная оптимизация */

14 }

15 /* следующую ошибку тоже игнорируем; придется обойтись без вывода

ограничения на количество транзитных узлов */

16 #ifdef IPV6_RECVHOPLIMIT

17 /* RFC 3542 */

18 setsockopt(sockfd, IPPROTO_IPV6, IPV6_RECVHOPLIMIT, &on, sizeof(on));

19 #else

20 /* RFC 2292 */

21 setsockopt(sockfd, IPPROTO_IPV6, IPV6_HOPLIMIT, &on, sizeof(on));

22 #endif

23 #endif

24 }

Приведенная в листинге 28.8 функция

proc_v6
обрабатывает входящие пакеты.

Листинг 28.8.Функция proc_v6: обработка сообщений ICMPv6

//ping/proc_v6.c

1 #include "ping.h"

2 void

3 proc_v6(char *ptr, ssize_t len, struct msghdr *msg, struct timeval* tvrecv)

4 {

5 #ifdef IPV6

6 double rtt;

7 struct icmp6_hdr *icmp6;

8 struct timeval *tvsend;

9 struct cmsghdr *cmsg;

10 int hlim;

11 icmp6 = (struct icmp6_hdr*)ptr;

12 if (len < 8)

13 return; /*
плохой пакет */

14 if (icmp6->icmp6_type == ICMP6_ECHO_REPLY) {

15 if (icmp6->icmp6_id != pid)

16 return; /* это не ответ на наш ECHO_REQUEST */

17 if (len < 16)

18 return; /* недостаточно данных */

19 tvsend = (struct timeval*)(icmp6 + 1);

20 tv_sub(tvrecv, tvsend);

21 rtt = tvrecv->tv_sec * 1000.0 + tvrecv->tv_usec / 1000.0;

22 hlim = -1;

23 for (cmsg = CMSG_FIRSTHDR(msg); cmsg != NULL;

24 cmsg = CMSG_NXTHDR(msg, cmsg)) {

25 if (cmsg->cmsg_level == IPPROTO_IPV6 &&

26 cmsg->cmsg_type == IPV6_HOPLIMIT) {

27 hlim = *(u_int32_t*)CMSG_DATA(cmsg);

28 break;

29 }

30 }

31 printf("%d bytes from %s; seq=%u, hlim=",

32 len, Sock_ntop__host(pr->sarecv, pr->salen), icmp6->icmp6_seq);

33 if (hlim == -1)

34 printf("???"); /* отсутствуют вспомогательные данные */

35 else

36 printf("%d", hlim);

37 printf(", rtt=%.3f ms\n", rtt);

38 } else if (verbose) {

39 printf(" %d bytes from type = %d, code = %d\n",

40 len, Sock_ntop_host(pr->sarecv, pr->salen);

41 icmp6->icmp6, type, icmp6->icmp6_code);

42 }

43 #endif /* IPV6 */

44 }

Извлечение указателя на заголовок ICMPv6

11-13
Заголовок ICMPv6 возвращается внутри данных при чтении из сокета. (Напомним, что дополнительные заголовки IPv6, если они присутствуют, всегда возвращаются не как стандартные данные, а как вспомогательные.) На рис. 28.4 приведены различные заголовки, указатели и длина, используемые в коде.

Рис. 28.4. Заголовки, указатели и длина при обработке ответов ICMPv6

Поделиться с друзьями: