C套接字引发错误代码22,EINVAL - 无效参数

问题描述:

下面的示例代码与服务器进程一样工作。但是,当我加入一行C套接字引发错误代码22,EINVAL - 无效参数

pid_t childpid; 

下面

struct sockaddr_in servaddr, clientaddr; 

失败的错误代码22,EINVAL线

connectfd = accept(listenfd, (struct sockaddr *) &clientaddr, &clientaddrlen); 

- 无效的参数。我是C新手,我无法理解这个问题,你能帮我解答一下吗?

谢谢。

#include <stdio.h> 
#include <sys/types.h> 
#include <sys/socket.h> 
#include <linux/in.h> 
#include <stdlib.h> 
#include <unistd.h> 
#include <errno.h> 

extern int errno; 

int main() 
{ 
    int clientaddrlen, listenfd, connectfd, bytes_rcvd, listen_queue_size=1; 
    short int port_no = 2000; 
    char buffer[1000]; 
    struct sockaddr_in servaddr, clientaddr; 

    printf("Server running at port #%d\n", port_no); 

    // Create server socket. 
    if ((listenfd = socket(AF_INET, SOCK_STREAM, 0)) < 0) 
    { 
     fprintf(stderr, "Cannot create server socket! errno=%d \n", errno); 
     exit(-1); 
    } 
    printf("Server socket created\n"); 

    // Bind (attach) this process to the server socket. 
    servaddr.sin_family = AF_INET; 
    servaddr.sin_addr.s_addr = htonl(INADDR_ANY); 
    servaddr.sin_port = htons(port_no); 
    bind(listenfd, (struct sockaddr *) &servaddr, sizeof(servaddr)); 
    printf("Server socket is bound to port #%d\n", port_no); 

    // Turn 'listenfd' to a listening socket. Listen queue size is 1. 
    listen(listenfd,listen_queue_size); 
    printf("Server listening with a queue of size %d. \n", listen_queue_size); 

    // Wait for connection(s) from client(s). 
    while (1) 
    { 
     connectfd = accept(listenfd, (struct sockaddr *) &clientaddr, &clientaddrlen); 
     printf("A client has connected\n"); 
     if (recv(connectfd, buffer, sizeof(buffer), 0) > 0) 
      printf("Received message: %s\n", buffer); 
     close(connectfd); 
     printf("Server closed connection to client\n"); 
    } 

    close(listenfd); 
    return 0; 
} 
+3

咦?程序的行为改变仅仅是因为你添加了一个未使用的变量声明?这似乎......不太可能,你粘贴了错误的行吗? – unwind 2009-05-20 08:21:14

+1

另外,您不要在此代码中使用fork(),尽管标题... – 2009-05-20 08:24:02

+0

实际上,此类行为非常普遍。它改变了堆栈帧的大小,因此可能会暴露或隐藏缓冲区溢出等引起的问题。 – 2009-05-20 08:25:20

我没有看到你在哪里初始化clientaddrlen。这是一个输入/输出参数。您必须告知accept()该地址的缓冲区有多大。

添加一个未使用的变量声明应该在正常情况下不会造成接受失败。 Select is not broken

你发布你描述不能表现的代码;你没有检查accept()的返回值,所以你怎么知道它失败?请记住,Unix系统中/ libc中调用通常不设置errno除非有错误发生,所以,除非accept()返回-1,并将errno 可以包含任何

这就是说;如果您已验证accept()失败,并且errno设置为EINVAL,则根据手册页有两种可能性:

  • 套接字未侦听连接。 (你检查从听()的返回码?)
  • addrlen中是无效的(即负)

编辑:最重要的是:发布完整例子编译并演示了你的问题。否则,我们只能猜测问题是什么。