Linux select()没有阻止

时间:2019-11-05 08:34:20

标签: c linux select system-calls file-handling

我试图更好地理解select()和poll()之间的区别。为此,我试图实现一个简单的程序,该程序将以只读方式打开文件,将其文件描述符添加到读取集中,然后执行select,以希望该功能在授予读取权限之前将一直阻塞。 由于这没有用(据我所知,这是预期的行为),因此我在select()执行之前尝试使用flock阻止对文件的访问。尽管如此,该程序仍未阻止其执行。

我的示例代码如下:

#include <stdio.h>
#include <poll.h>
#include <sys/file.h>
#include <errno.h>
#include <sys/select.h>


int main(int argc, char **argv)
{
   printf("[+] Select minimal example\n");
   int max_number_fds = FOPEN_MAX;
   int select_return;
   int cnt_pollfds;
   struct pollfd pfds_array[max_number_fds];
   struct pollfd *pfds = pfds_array;

   fd_set fds;
   int fd_file = open("./poll_text.txt",  O_WRONLY);

   struct timeval tv;
   tv.tv_sec = 10;
   tv.tv_usec = 0;

   printf("\t[+] Textfile fd: %d\n", fd_file);

   //create and set fds set    
   FD_ZERO(&fds);
   FD_SET(fd_file, &fds);

   printf("[+] Locking file descriptor!\n");
   if(flock(fd_file,LOCK_EX) == -1)
   {
       int error_nr = errno;
       printf("\t[+] Errno: %d\n", error_nr);
   }  

   printf("[+] Executing select()\n");
   select_return = select(fd_file+1, &fds, NULL, NULL, &tv);
   if(select_return == -1){
       int error_nr = errno;
       printf("[+] Select Errno: %d\n", error_nr);
   }

   printf("[+] Select return: %d\n", select_return); 
 }

有人可以在此代码中看到我的错误吗?另外:我首先尝试使用添加到读取列表中的两个FD执行此代码。当试图锁定它们时,我不得不使用flock(fd_file,LOCK_SH),因为我不能只用LOCK_EX锁定两个FD。如何锁定同一文件的两个FD(只有一个fd)有区别

我也不确定为什么将添加到读取集中的文件以“只写”方式打开时,select不会被阻塞。该程序永远无法(未经许可更改)从fd读取数据,所以据我所知select应该阻止执行,对吧?

作为一个澄清:我要解决的“问题”是,我必须检查是否能够用poll()替换现有的select()调用(存在于以下方面:我不会重写select()调用代码,但可以访问select。)的参数。为了检查这一点,我想实现一个测试,该测试将强制select阻止其执行,因此我可以稍后检查poll是否将以相同的方式工作(当给出相似的指令时,即要检查的相同FD)。

因此,我的“工作流程”将是:针对不同的选择行为(即阻止而不是阻止)编写测试,针对民意调查(也阻止而不是阻止)编写类似的测试,并检查是否可以强制执行轮询/如何严格执行选择是在做。

谢谢您的提示!

1 个答案:

答案 0 :(得分:3)

select告诉您可以读取文件描述符时,这并不一定意味着您可以读取数据。这仅意味着read调用不会阻塞。 read调用在返回EOF或错误条件时也不会阻塞。

在您的情况下,我希望read将立即返回-1并将errno设置为EBADF(fd不是有效的文件描述符或未打开以供读取)或也许{ {1}}(fd附加到不适合阅读的对象上...)

编辑:根据评论的要求提供其他信息:

如果需要进行一些物理操作(例如,需要一些时间),则文件可以处于阻止状态。如果读取缓冲区为空,并且必须从磁盘读取(新)数据,如果文件已连接到终端并且用户尚未输入任何(更多)数据,或者文件是套接字或管道,并且EINVAL将不得不等待(新)数据到达...

read同样适用:如果发送缓冲区已满,则write将阻塞。如果发送缓冲区中的剩余空间小于您的数据量,则它可能仅写入当前适合缓冲区的部分。

如果将文件设置为非阻止模式,则writeread不会阻止,但会告诉您阻止。

如果出于测试目的而遇到阻塞情况,则需要控制提供或使用数据的过程或硬件。当您不输入任何数据或在写入过程未写入任何数据的管道中,建议从终端(write)使用read。当读取过程没有从管道中读取时,您也可以在管道上填充写入缓冲区。