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

ЖАНРЫ

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

37-46
Выводится MTU и те IP-адреса, которые были возвращены.

Если мы запустим эту программу на нашем узле

macosx
(см. рис. 1.7), то получим следующий результат:

macosx % prifinfo inet4 0

lo0: <UP MCAST LOOP >

MTU: 16384

IP addr: 127.0.0.1

en1: <UP BCAST MCAST >

MTU: 1500

IP addr: 172.24.37.78

broadcast addr: 172.24.37.95

Первый

аргумент командной строки
inet4
задает адрес IPv4, а второй, нулевой аргумент указывает, что не должно возвращаться никаких псевдонимов, или альтернативных имен (альтернативные имена IP-адресов мы описываем в разделе А.4). Обратите внимание, что в MacOS X аппаратный адрес интерфейса Ethernet недоступен.

Если мы добавим к интерфейсу Ethernet (

en1
) три альтернативных имени адреса с идентификаторами узла 79, 80 и 81 и изменим второй аргумент командной строки на 1, то получим:

macosx % prifinfo inet4 1

lo0: <UP MCAST LOOP >

MTU: 16384

IP addr: 127.0.0.1

en1: <UP BCAST MCAST >

MTU: 1500

IP addr: 172.24.37.78 первичный IP-адрес

broadcast addr: 172.24.37.95

en1: <UP BCAST MCAST >

MTU: 1500

IP addr: 172.24.37.79 первый псевдоним

broadcast addr: 172.24.37.95

en1: <UP BCAST MCAST >

MTU: 1500

IP addr: 172 24.37.80 второй псевдоним

broadcast addr: 172.24 37.95

en1: <UP BCAST MCAST >

MTU: 1500

IP addr: 172 24.37.81 третий псевдоним

broadcast addr: 172.24.37 95

Если мы запустим ту же программу под FreeBSD, используя реализацию функции

get_ifi_info
, приведенную в листинге 18.9 (которая может легко получить аппаратный адрес), то получим:

freebsd4 % prifinfo inet4 1

de0: <UP BCAST MCAST >

0:80:c8:2b:d9:28

IP addr: 135.197.17.100

broadcast addr: 135.197.17.255

de1: <UP BCAST MCAST >

0:40:5:42:d6:de

IP addr: 172.24.37.94 основной IP-адрес

broadcast addr: 172.24.37.95

ef0: <UP BCAST MCAST >

0:40:5:42:d6:de

IP addr: 172.24.37.93 псевдоним

broadcast addr: 172.24.37.93

lo0: <UP MCAST LOOP >

IP addr: 127.0.0.1

В

этом примере мы указали программе выводить псевдонимы, и мы видим, что один из псевдонимов определен для второго интерфейса Ethernet (
de1
) с идентификатором узла 93.

Теперь мы покажем нашу реализацию функции

get_ifi_info
, использующую вызов
SIOCGIFCONF
функции
ioctl
. В листинге 17.4 показана первая часть этой функции, получающая от ядра конфигурацию интерфейса.

Листинг 17.4. Выполнение вызова SIOCGIFCONF для получения конфигурации интерфейса

//lib/get_if_info.c

1 #include "unpifi.h"

2 struct ifi_info*

3 get_ifi_info(int family, int doaliases)

4 {

5 struct ifi_info *ifi, *ifihead, **ifipnext;

6 int sockfd, len, lastlen, flags, myflags, idx = 0, hlen = 0;

7 char *ptr, *buf, lastname[IFNAMSIZ], *cptr, *haddr, *sdlname;

8 struct ifconf ifc;

9 struct ifreq *ifr, ifrcopy;

10 struct sockaddr_in *sinptr;

11 struct sockaddr_in6 *sin6ptr;

12 sockfd = Socket(AF_INET, SOCK_DGRAM, 0);

13 lastlen = 0;

14 len = 100 * sizeof(struct ifreq); /* начальное приближение к нужному размеру буфера */

15 for (;;) {

16 buf = Mallос(len);

17 ifc.ifc_len = len;

18 ifc.ifc_buf = buf;

19 if (ioctl(sockfd, SIOCGIFCONF, &ifc) < 0) {

20 if (errno != EINVAL || lastlen != 0)

21 err_sys("ioctl error");

22 } else {

23 if (ifc.ifc_len == lastlen)

24 break; /* успех, значение len не изменилось */

25 lastlen = ifc.ifc_len;

26 }

27 len += 10 * sizeof(struct ifreq); /* приращение */

28 free(buf);

29 }

30 ifihead = NULL;

31 ifipnext = &ifihead;

32 lastname[0] = 0;

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