Для доступа к сервису, обеспечиваемому UDP-протоколом, вам следует применять системные вызовы socket
и close
, но вместо использования вызовов read
и write
для сокета вы применяете два системных вызова, характерных для дейтаграмм: sendto
и recvfrom
.
Далее приведена модифицированная версия программы getdate.c, которая получает дату с помощью сервиса UDP-дейтаграмм. Изменения по сравнению с предыдущей версией выделены цветом.
/* Начните с обычных include и объявлений. */
#include
#include
#include
#include
#include
#include
int main(int argc, char *argv[]) {
char *host;
int sockfd;
int len, result;
struct sockaddr_in address;
struct hostent *hostinfo;
struct servent *servinfo;
char buffer[128];
if (argc == 1) host = "localhost";
else host = argv[1];
/* Ищет адрес хоста и сообщает об ошибке, если не находит. */
hostinfo = gethostbyname(host);
if (!hostinfo) {
fprintf(stderr, "no host: %s\n", host);
exit(1);
}
/* Проверяет наличие на компьютере сервиса daytime. */
if (!servinfo) {
fprintf(stderr, "no daytime service\n");
exit(1);
}
printf("daytime port is %d\n", ntohs(servinfo->s_port));
address.sin_family = AF_INET;
address.sin_port = servinfo->s_port;
address.sin_addr = *(struct in_addr*)*hostinfo->h_addr_list;
len = sizeof(address);
buffer [result] = '\0';
printf("read %d bytes: %s", result, buffer);
close(sockfd);
exit(0);
}
Как видите, необходимы лишь незначительные изменения. Как и раньше, вы ищете сервис daytime
с помощью вызова getservbyname
, но задаете дейтаграммный сервис, запрашивая UDP-протокол. Дейтаграммный сокет создается с помощью вызова socket
с параметром SOCK_DGRAM
. Адрес назначения задается, как и раньше, но теперь вместо чтения из сокета вы должны послать дейтаграмму.
Поскольку вы не устанавливаете явное соединение с сервисами на базе UDP, у вас должен быть способ оповещения сервера о том, что вы хотите получить ответ. В данном случае вы посылаете дейтаграмму (в нашем примере вы отправляете один байт из буфера, в который вы хотите получить ответ) сервису и он посылает в ответ дату и время.
Системный вызов sendto
отправляет дейтаграмму из буфера на сокет, используя адрес сокета и длину адреса. У этого вызова фактически следующий прототип:
int sendto(int sockfd, void *buffer, size_t len, int flags,
struct sockaddr *to, socklen_t tolen);
В случае обычного применения параметр flags
можно оставлять нулевым.
Системный вызов recvfrom ожидает дейтаграмму в соединении сокета с заданным адресом и помещает ее в буфер. У этого вызова следующий прототип:
int recvfrom(int sockfd, void *buffer, size_t len, int flags,
struct sockaddr *from, socklen_t *fromlen);
И снова в случае обычного применения параметр flags
можно оставлять нулевым.
Для упрощения примера мы пропустили обработку ошибок. Оба вызова, sendto
и recvfrom
, в случае возникновения ошибки вернут -1 и присвоят переменной errno
соответствующее значение. Возможные ошибки перечислены в табл. 15.6.
Значение errno | Описание |
---|---|
EBADF | Был передан неверный файловый дескриптор |
EINTR | Появился сигнал |