为什么文件描述符中省略了 20

问题描述

编辑

lsof 显示它是由 ptmx 打开的。感谢@zwol


我下面的代码打印了从 open 返回的文件描述符。我注意到缺少 20。据我所知,没有类似的问题。

背景:

  • 文件系统:ext4
  • WSL2 上的 Ubuntu 20.04

代码

int main()
{
    char name[2] = "a";
    for (int i = 0; i < 52; i++) {
        int fd = open(name,O_RDWR | O_CREAT,0644);
        printf("fd is %d\n",fd);
    }
    return 0;
}

输出

$ ./a.out 
fd is 3
fd is 4
fd is 5
...
fd is 18
fd is 19
fd is 21 <-- here
fd is 22
...

lsof

...
a.out   1815 ryan   19u   REG   8,16        0 42321 /tmp/tmp/a
a.out   1815 ryan   20u   CHR    5,2      0t0 15832 /dev/ptmx
a.out   1815 ryan   21u   REG   8,16        0 42321 /tmp/tmp/a
...

我有两个问题:

  1. 背后的原因是什么? (是什么触发了ptmx
  2. 如果我一直打开文件,会不会有更多的索引丢失? (更多程序,如 ptmx?)

解决方法

POSIX 规定每个分配文件描述符的系统调用都必须使用尚未使用的最低编号。因此,20 号描述符必须已经打开。你的示例程序在循环之前没有打开任何东西,所以它一定是从你的 shell 继承的,或者在 main 之前被 C 库打开。

您可以通过让程序打印其 PID,然后在循环之后长时间休眠,然后在它休眠时对其运行 lsof 来了解更多信息。