Создание отправляющего сокета
13
Наша функция udp_client
создает отправляющий сокет, обрабатывая два аргумента командной строки, которые задают адрес многоадресной передачи и номер порта. Эта функция также возвращает структуру адреса сокета, готовую к вызовам функции sendto
, и длину этой структуры.
Создание принимающего сокета и связывание (при помощи функции bind) с адресом многоадресной передачи и портом
14-18
Мы создаем принимающий сокет, используя то же семейство адресов, что и при создании отправляющего сокета, и устанавливаем параметр сокета SO_REUSEADDR
, чтобы разрешить множеству экземпляров этой программы одновременно запускаться на узле. Затем мы выделяем в памяти пространство для структуры адреса этого сокета, копируем ее содержимое из структуры адреса отправляющего сокета (адрес и порт которого взяты из аргументов командной строки) и при помощи функции bind
связываем адрес многоадресной передачи и порт с принимающим сокетом.
Присоединение к группе и выключение закольцовки
19-20
Мы вызываем нашу функцию mcast_join
, чтобы присоединиться к группе на получающем сокете, а также нашу функцию mcast_set_loop
, чтобы отключить закольцовку на отправляющем сокете. Для присоединения задаем имя интерфейса в виде пустого указателя и нулевой индекс интерфейса, что указывает ядру на необходимость выбрать интерфейс самостоятельно.
Функция fork и вызов соответствующих функций
21-23
Мы вызываем функцию fork
, после чего дочерним процессом становится получающий цикл, а родительским — отправляющий.
Наша функция sendmail
, отправляющая по одной дейтаграмме многоадресной передачи каждые 5 с, показана в листинге 21.9. Функция main
передает в качестве аргументов дескриптор сокета, указатель на структуру адреса сокета, содержащую адрес получателя многоадресной передачи и порт, и длину структуры.
Листинг 21.9. Отправка дейтаграммы многоадресной передачи каждые 5 с
//mcast/send.c
1 #include "unp.h"
2 #include
3 #define SENDRATE 5 /* отправка дейтаграмм каждые 5 с */
4 void
5 send_all(int sendfd, SA *sadest, socklen_t salen)
6 {
7 static char line[MAXLINE]; /* имя узла и идентификатор процесса */
8 struct utsname myname;
9 if (uname(&myname) < 0)
10 err_sys("uname error");
11 snprintf(line, sizeof(line), "%s, %d\n", myname, nodename, getpid());
12 for (;;) {
13 Sendto(sendfd, line, strlen(line), 0, sadest, salen);
14 sleep(SENDRATE);
15 }
16 }
Получение имени узла и формирование содержимого дейтаграммы
9-11
Мы получаем имя узла из функции uname
и создаем строку вывода, содержащую это имя и идентификатор процесса.
Отправка дейтаграммы, переход в режим ожидания
12-15
Мы отправляем дейтаграмму и с помощью функции sleep
переходим в состояние ожидания на 5 с.
Функция recv_all
, содержащая бесконечный цикл получения, показана в листинге 21.10.
Листинг 21.10. Получение всех дейтаграмм многоадресной передачи для группы, к которой мы присоединились
//mcast/recv.c
1 #include "unp.h"
2 void
3 recv_all(int recvfd, socklen_t salen)
4 {
5 int n;
6 char line[MAXLINE + 1];
7 socklen_t len;
8 struct sockaddr *safrom;
9 safrom = Malloc(salen);
10 for (;;) {
11 len = salen;
12 n = Recvfrom(recvfd, line, MAXLINE, 0, safrom, &len);
13 line[n] = 0; /* завершающий нуль */
14 printf("from %s: %s", Sock_ntop(safrom, len), line);
15 }
16 }
Размещение в памяти структуры адреса сокета
9
При каждом вызове функции recvfrom
в памяти выделяется пространство для структуры адреса сокета, в которую записывается адрес отправителя.
Чтение и вывод дейтаграмм
10-15
Каждая дейтаграмма считывается функцией recvfrom
, дополняется символом конца строки (то есть нулем) и выводится.
Мы запускаем программу в двух системах: freebsd4
и macosx
. Каждая система видит пакеты, отправляемые другой.
freebsd4 % sendrecv 239.255.1.2 8888
from 172.24.37.78:51297: macosx, 21891
from 172.24.37.78:51297: macosx, 21891
from 172.24.37.78:51297: macosx, 21891
from 172.24.37.78:51297: macosx, 21891
macosx % sendrecv 239.255.1.2 8888
from 172.24.37.94.1215: freebsd4, 55372
from 172.24.37.94.1215: freebsd4, 55372
from 172.24.37.94.1215: freebsd4, 55372
from 172.24.37.94.1215: freebsd4, 55372
21.11. SNTP: простой синхронизирующий сетевой протокол
Читать дальше
Конец ознакомительного отрывка
Купить книгу