我在Linux下的服務器套接字有問題。出於某種原因,我不知道服務器套接字消失,並且在等待輸入連接的select調用中出現Bad file descriptor
錯誤。當我在不同的線程中關閉不相關的套接字連接時,總是會出現此問題。這發生在2.6.36內核的嵌入式Linux上。Linux服務器套接字 - 錯誤的文件描述符
有誰知道爲什麼會發生這種情況?服務器插座是否可以簡單地消失導致Bad file descriptor
是否正常?
編輯: 其他套接字代碼實現了一個VNC服務器,並運行在一個完全不同的線程中。在其他代碼中唯一特別的是使用setjmp/longjmp
,但這應該不成問題。
是創建服務器套接字的代碼如下:
int server_socket = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP);
struct sockaddr_in saddr;
memset(&saddr, 0, sizeof(saddr));
saddr.sin_family = AF_INET;
saddr.sin_addr.s_addr = htonl(INADDR_ANY);
saddr.sin_port = htons(1234);
const int optionval = 1;
setsockopt(server_socket, SOL_SOCKET, SO_REUSEADDR, &optionval, sizeof(optionval));
if (bind(server_socket, (struct sockaddr *) &saddr, sizeof(saddr)) < 0) {
perror("bind");
return 0;
}
if (listen(server_socket, 1) < 0) {
perror("listen");
return 0;
}
我等待進來的連接使用下面的代碼:
static int WaitForConnection(int server_socket, struct timeval *timeout)
{
fd_set read_fds;
FD_ZERO(&read_fds);
int max_sd = server_socket;
FD_SET(server_socket, &read_fds);
// This select will result in 'EBADFD' in the error case.
// Even though the server socket was not closed with 'close'.
int res = select(max_sd + 1, &read_fds, NULL, NULL, timeout);
if (res > 0) {
struct sockaddr_in caddr;
socklen_t clen = sizeof(caddr);
return accept(server_socket, (struct sockaddr *) &caddr, &clen);
}
return -1;
}
編輯: 當問題的情況下發生我目前只是重新啓動服務器,但我不明白爲什麼服務器套接字ID應該突然變成無效的文件描述符:
int error = 0;
socklen_t len = sizeof (error);
int retval = getsockopt (server_socket, SOL_SOCKET, SO_ERROR, &error, &len);
if (retval < 0) {
close(server_socket);
goto server_start;
}
沒有什麼錯,你發佈的代碼,誤差必須在其他地方。例如,關閉它之後是否使用了套接字? – 2012-07-31 07:22:14
準確使用的線程在哪裏? – 2012-07-31 07:32:10
上面的代碼在一個線程中運行。另一個代碼在另一個也運行線程的模塊中。關閉連接將殺死服務器。我沒有想過如果我沒有關閉服務器套接字就會失效。 – trenki 2012-07-31 07:42:46