Литмир - Электронная Библиотека
Содержание  
A
A

Обратите внимание, что табл. 11.3 описывает только обработку адресов IPv4 и IPv6 функцией

getaddrinfo
, то есть количество и тип адресов, возвращаемых процессу в различных ситуациях. Реальное количество структур
addrinfo
зависит также от типа сокета и имени службы, о чем уже говорилось в связи с табл. 11.1.

11.10. Функция getaddrinfo: примеры

Теперь мы покажем некоторые примеры работы функции

getaddrinfo
, используя тестовую программу, которая позволяет нам вводить все параметры: имя узла, имя службы, семейство адресов, тип сокета и флаги
AI_CANONNAME
и
AI_PASSIVE
. (Мы не показываем эту тестовую программу, поскольку она содержит около 350 строк малоинтересного кода. Ее можно получить тем же способом, что и прочие исходные коды для этой книги.) Тестовая программа выдает информацию о переменном числе возвращаемых структур
addrinfo
, показывая аргументы вызова функции
socket
и адрес в каждой структуре адреса сокета. Сначала показываем тот же пример, что и на рис. 11.3:

freebsd % <b>testga -f inet -c -h freebsd4 -s domain</b>

socket(AF_INET, SOCK_DGRAM, 17) ai_canonname = freebsd4.unpbook.com

      address: 135.197.17.100:53

socket(AF_INET, SOCK_DGRAM, 17)

      address: 172:24.37.94:53

socket(AF_INET, SOCK_STREAM, 6) ai_canonname = freebsd4.unpbook.com

      address: 135.197.17.100:53

socket(AF_INET, SOCK_STREAM, 6)

      address: 172.24.37.94:53

Параметр

-f inet
задает семейство адресов, -с указывает, что нужно возвратить каноническое имя,
-h freebsd4
задает имя узла,
-s domain
задает имя службы.

Типичный сценарий клиента — задать семейство адресов, тип сокета (параметр

-t
), имя узла и имя службы. Следующий пример показывает это для узла с несколькими сетевыми интерфейсами с шестью адресами Ipv4:

freebsd % <b>testga -f inet -t stream -h gateway.tuc.noao.edu -s daytime</b>

socket(AF_INET, SOCK_STREAM, 6)

      address: 140.252.108.1:13

socket(AF_INET, SOCK_STREAM, 6)

      address: 140.252.1.4:13

socket(AF_INET, SOCK_STREAM, 6)

      address: 140.252.104.1:13

socket(AF_INET, SOCK_STREAM, 0)

      address: 140.252.3.6.13

socket(AF_INET, SOCK_STREAM, 0)

      address: 140.252.4.100.13

socket(AF_INET, SOCK_STREAM, 0)

      address: 140.252.1.4.13

Затем мы задаем наш узел

aix
, у которого имеется и запись типа AAAA, и запись типа А, не указывая семейства адресов. Имя службы —
ftp
, которая предоставляется только TCP.

freebsd % <b>testga -h aix -s ftp -t stream</b>

socket(AF_NET6, SOCK_STREAM, 6)

      address: [3ffe:b80:1f8d:2:204:acff:fe17:bf38]:21

socket(AF_INET, SOCK_STREAM, 6)

      address: 192.168.42.2:21

Поскольку мы не задали семейство адресов и запустили этот пример на узле, который поддерживает и IPv4, и IPv6, возвращаются две структуры: одна для IPv6 и одна для IPv4.

Затем мы задаем флаг

AI_PASSIVE
(параметр
), не указываем ни семейства адресов, ни имени узла (подразумевая универсальный адрес), задаем номер порта 8888 и не указываем тип сокета.

freebsd % <b>testga -р -s 8888 -t stream</b>

socket(AF_INET6, SOCK_STREAM, 6)

address: [::]:8888

socket(AF_INET, SOCK_STREAM, 6)

address: 0.0.0.0:8888

Возвращаются две структуры. Поскольку мы запустили эту программу на узле, поддерживающем и IPv4, и IPv6, не задав семейства адресов, функция

getaddrinfo
возвращает универсальный адрес IPv6 и универсальный адрес IPv4. Структура IPv6 возвращается перед структурой IPv4, поскольку, как мы увидим в главе 12, клиент или сервер IPv6 на узле с двойным стеком может взаимодействовать с собеседниками по IPv6 и по IPv4.

11.11. Функция host_serv

Наш первый интерфейс функции

getaddrinfo
не требует от вызывающего процесса размещать в памяти структуру рекомендаций и заполнять ее. Вместо этого аргументами нашей функции
host_serv
будут интересующие нас поля — семейство адресов и тип сокета.

#include &quot;unp.h&quot;

struct addrinfo *host_serv(const char *<i>hostname</i>, const char *<i>service</i>, int <i>family</i>, int <i>socktype</i>);

<i>Возвращает: в случае успешного выполнения указатель на структуру addrinfo. NULL в случае ошибки</i>

В листинге 11.3 показан исходный код этой функции.

Листинг 11.3. Функция host_serv

//lib/host_serv.c

 1 #include &quot;unp.h&quot;

 2 struct addrinfo*

 3 host_serv(const char *host, const char *serv, int family, int socktype)

 4 {

 5  int n;

 6  struct addrinfo hints, *res;

 7  bzero(&amp;hints, sizeof(struct addrinfo));

 8  hints.ai_flags = AI_CANONNAME; /* всегда возвращает каноническое имя */

 9  hints.ai_family = family; /* AF_UNSPEC, AF_INET, AF_INET6, ... */

130
{"b":"225366","o":1}