Я изо всех сил пытаюсь понять прототип accept(2)
.Почему `accept (2)` требуется `sockaddr` как отдельный указатель?
У меня есть простой сервер, который принимает подключения и возвращает «Hello world! \ N» для клиентов. Системный вызов accept(2)
принимает указатель на struct sockaddr
и указатель на socklen_t
для хранения длины структуры. Но у sockaddr
уже есть поле под названием sa_len
, которое, кажется, сделано именно для этого.
Кроме того, у меня есть этот простой сервер (компилирует под OSX, надеюсь, Linux тоже), что печатает мой собственный socklen_t
прошел принять, а затем значение sa_len
: они такие же, в данном случае 28, на OSX.
EDIT: после немного более сложного тестирования кажется, что sa_len
не обязательно совпадает с длиной, хранящейся в указателе.
Почему accept(2)
нужна длина как отдельный указатель?
Для справки, я отправляю на пример сервера ниже. Вы можете скомпилировать с:
gcc -Wall -Wextra main.c
А затем запустить:
./a.out
В другом терминале, подключить к нему:
telnet 127.0.0.1 3000
#include <errno.h>
#include <sys/socket.h>
#include <unistd.h>
#include <netdb.h>
#include <stdio.h>
#include <netinet/in.h>
#include <strings.h>
// default port if none if passed in the program arguments
#define FTP_PORT_DEFAULT ("3000")
// number of connections allow to wait for a call to accept
#define FTP_BACKLOG (5)
int main(int ac, char **av)
{
struct addrinfo hints; // hints used to get address information
struct addrinfo *sai; // server address information
int ss; // server socket
char *port;
port = (ac >= 2) ? av[1] : FTP_PORT_DEFAULT;
bzero(&hints, sizeof(hints));
// using AF_INIT6 instead of PF_INET6
// http://stackoverflow.com/questions/6729366/
hints.ai_family = AF_INET6;
hints.ai_socktype = SOCK_STREAM;
// intended for use with `bind`
hints.ai_flags = AI_PASSIVE;
// passing `NULL` as the hostname tells `getaddrinfo` to use
// any available local IP address
if (getaddrinfo(NULL, port, &hints, &sai) != 0)
return (-1);
if ((ss =
socket(sai->ai_family, sai->ai_socktype, sai->ai_protocol))== -1)
return (-1);
if (bind(ss, sai->ai_addr, sai->ai_addrlen) == -1 ||
listen(ss, FTP_BACKLOG) == -1)
{
close(ss);
return (-1);
}
while (1)
{
int cs;
struct sockaddr csockaddr;
unsigned csockaddr_len;
bzero(&csockaddr, sizeof(csockaddr));
cs = accept(ss, &csockaddr, &csockaddr_len);
// handle "fatal" errors that I don't think I can recover from
// other than by relaunching the server
if (cs == EBADF || cs == ECONNABORTED || cs == ENOTSOCK ||
cs == EOPNOTSUPP)
{
break ;
}
printf("my len %u/internal len %u\n",
csockaddr_len, csockaddr.sa_len);
if (fork() == 0)
{
write(cs, "Hello world!\n", 13);
close(cs);
}
break;
}
close(ss);
return 0;
}
'accept' фактически принимает' socklen_t * ', а не' unsigned int * '. В 4.3BSD в ранних версиях POSIX потребовалось 'int *', затем 'size_t *', после чего POSIX передумал и оставил его до реализации, чтобы определить 'socklen_t' как любой из этих типов. –
Вы правы. Я на OSX, который, кажется, использует 'unsigned int'. Исправил его в вопросе. – conradkdotcom
Перед вызовом accept необходимо инициализировать csockaddr_len. На странице согласия (2) man есть детали. – sigjuice