2014-05-13 4 views
2

Я изо всех сил пытаюсь понять прототип 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; 
} 
+2

'accept' фактически принимает' socklen_t * ', а не' unsigned int * '. В 4.3BSD в ранних версиях POSIX потребовалось 'int *', затем 'size_t *', после чего POSIX передумал и оставил его до реализации, чтобы определить 'socklen_t' как любой из этих типов. –

+0

Вы правы. Я на OSX, который, кажется, использует 'unsigned int'. Исправил его в вопросе. – conradkdotcom

+0

Перед вызовом accept необходимо инициализировать csockaddr_len. На странице согласия (2) man есть детали. – sigjuice

ответ

3

sa_len не является переносимым. Это было added to BSD only in 1988, через несколько лет после 4.3BSD, но это не в POSIX, и это не в Linux.