套接字在写操作时没有阻塞:OpenSolaris

时间:2011-08-27 07:03:35

标签: sockets asynchronous tcp solaris nonblocking

我有一个单元测试,用于检查阻塞和非阻塞套接字上的行为 - 服务器写入一个长响应,在某些时候它应该不能再写了它 写下来。

基本上一方写,另一方不读。

在Solaris下某些时候我收到错误“没有足够的空间”(写完75MB后)而不是写入时阻塞:

重现问题的程序:

#include <sys/types.h>
#include <sys/socket.h>
#include <sys/wait.h>
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <signal.h>
#include <arpa/inet.h>
#include <sys/socket.h>
#include <sys/un.h>
#include <netinet/in.h>

char const *address = "127.0.0.1";
#define check(x) do { if( (x) < 0) { perror(#x) ; exit(1); } } while(0)

int main()
{
    signal(SIGPIPE,SIG_IGN);
    struct sockaddr_in inaddr = {};
    inaddr.sin_family = AF_INET;
    inaddr.sin_addr.s_addr = inet_addr(address);
    inaddr.sin_port = htons(8080);

    int res = fork();
    if(res < 0) {
        perror("fork");
        exit(1);
    }
    if(res > 0) {
        int fd = -1;
        int status;
        sleep(1);   
        check(fd = socket(AF_INET,SOCK_STREAM,0));
        check(connect(fd,(sockaddr*)&inaddr,sizeof(inaddr)));
        sleep(5);
        close(fd);

        wait(&status);
        return 0;
    }
    else {
        int acc,fd;
        check(acc = socket(AF_INET,SOCK_STREAM,0));
        int yes = 1;
        check(setsockopt(acc,SOL_SOCKET,SO_REUSEADDR,&yes,sizeof(yes)));
        check(bind(acc,(sockaddr*)&inaddr,sizeof(inaddr)));
        check(listen(acc,10));
        check(fd = accept(acc,0,0));

        char buf[1000];
        long long total= 0;
        do {
            int r = send(fd,buf,sizeof(buf),0);
            if(r < 0) {
                printf("write %s\n",strerror(errno));
                return 0;
            }
            else if(r==0) {
                printf("Got eof\n");
                return 0;
            }
            total += r;
            if(total > 100*1024*1024) {
                printf("Too much!!!!\n");
                return 0;
            }
            printf("%lld\n",total);
        }while(1);
    }
    return 0;
}

Solaris上的输出(最后两行)

75768000
write Not enough space

Linux上的预期输出(最后两行)

271760
write Connection reset by peer

仅当对方关闭套接字时才会发生这种情况。

任何想法为什么以及如何解决它,设置哪些选项?

P.S。:OpenSolaris 2009.06,x86

编辑

  • 添加了可以重现问题的完整C代码

答案:

这似乎是特定版本的Solaris内核libc库中的一个错误。

2 个答案:

答案 0 :(得分:1)

从OpenSolaris源代码中,我担心SO_SNDTIMEO选项不受支持:https://hg.java.net/hg/solaris~on-src/file/tip/usr/src/uts/common/inet/sockmods/socksctp.c#l1233

答案 1 :(得分:0)

如果要阻止没有可用空间,则需要编写代码来执行此操作。 POSIX很清楚,套接字上的write相当于send没有选项,而sendmay fail if ... [i]nsufficient resources were available in the system to perform the operation.