我目前正在尝试使用命名管道作为IPC来运行服务器以同时参加多个客户端(#l ;;本地进程)请求。客户端能够在其上写入,但似乎服务器上的select()函数无法正常工作,它始终返回0。
这是服务器的主要代码:
int main (int argc, char const *argv[]){
fd_set set; //fds to monitor
Request * r;
struct timeval tv; //timeout
Connection *c; //where the fd will be saved
fd_set foo; //debugging purpose
//opens the NamedPipe and saves de ReadOnly fd on c
if( (c = openConnection()) == NULL) {
return ERROR_OPEN_REQUEST_QUEUE;
}
//sets up the fds to monitor FD_ZERO and FD_SET
setConnection(c, &set);
FD_ZERO(&foo);
tv.tv_sec = 2;
tv.tv_usec = 0;
while(1){
int fdCount = select(2, &set, &foo, &foo, &tv);
//it seems select directly modifies this value
tv.tv_sec = 2;
//saw it on another post
setConnection(c, &set);
if( fdCount > 0){
r = getRequest(c);
if( r != NULL ){
TODO processRequest(r);
}
} else {
printf("No requests to process\n");
}
}
return 0;
}
服务器和客户端都使用openConnection从NamedPipe获取fd。 openConnections调用此函数并创建一个连接对象:
int * openNamedPipe(char * name) {
char origin[] = "/tmp/";
char myfifo[80];
int * fd;
fd = malloc(sizeof(int)*2);
strcpy(myfifo,origin);
strcat(myfifo,name);
mkfifo(myfifo, 0777);
fd[0] = open(myfifo, O_RDONLY|O_NONBLOCK);
fcntl(fd[0], F_SETFL, fcntl(fd[0], F_GETFL) &~O_NONBLOCK);
fd[1] = open(myfifo, O_WRONLY);
return fd;
}
我的问题如下:
我用cat手动检查fifo,我能够从shell中读取内容。因此,如果客户端能够写入,则服务器应该能够使用ReadOnly fd。
进行读取添加setConnection函数以防万一:
void setConnection(Connection * connection, fd_set* set){
FD_ZERO(set);
FD_SET(connection -> np -> fd, set);
return;
}
答案 0 :(得分:4)
select
的第一个参数应该是编号最高的文件描述符+1。您传递值2,因此您的选择仅关注fds 0和1(如果它们在传递的集合中设置)。你的烟斗真的使用那些吗?如果没有,您需要在最高的代码中进行测试并传递选择。
另一件事,似乎你应该只传递NULL而不是foo
,如果你不想看那些事件。