c - STDIN_FILENO 和 STDOUT_FILENO 在 c 中只读吗?

标签 c unix file-descriptor dup2

fd = open("/dev/null", O_RDWR);
if (fd == -1) {
    ngx_log_error(NGX_LOG_EMERG, log, ngx_errno,
                  "open(\"/dev/null\") failed");
    return NGX_ERROR;
}

if (dup2(fd, STDIN_FILENO) == -1) {
    ngx_log_error(NGX_LOG_EMERG, log, ngx_errno, "dup2(STDIN) failed");
    return NGX_ERROR;
}

if (dup2(fd, STDOUT_FILENO) == -1) {
    ngx_log_error(NGX_LOG_EMERG, log, ngx_errno, "dup2(STDOUT) failed");
    return NGX_ERROR;
}


if (fd > STDERR_FILENO) {
    if (close(fd) == -1) {
        ngx_log_error(NGX_LOG_EMERG, log, ngx_errno, "close() failed");
        return NGX_ERROR;
    }
}

man 告诉我 dup2() 使 newfd 成为 oldfd 的副本,必要时先关闭 newfd。:

int dup2(int oldfd, int newfd);

但是 STDIN_FILENOSTDOUT_FILENO 不是只读的吗?

Dump of assembler code for function dup2:
0x00000037aa4c6ac0 <dup2+0>:    mov    $0x21,%eax
0x00000037aa4c6ac5 <dup2+5>:    syscall 
0x00000037aa4c6ac7 <dup2+7>:    cmp    $0xfffffffffffff001,%rax
0x00000037aa4c6acd <dup2+13>:   jae    0x37aa4c6ad0 <dup2+16>
0x00000037aa4c6acf <dup2+15>:   retq   
0x00000037aa4c6ad0 <dup2+16>:   mov    0x28a4d1(%rip),%rcx        # 0x37aa750fa8 <free+3356736>
0x00000037aa4c6ad7 <dup2+23>:   xor    %edx,%edx
0x00000037aa4c6ad9 <dup2+25>:   sub    %rax,%rdx
0x00000037aa4c6adc <dup2+28>:   mov    %edx,%fs:(%rcx)
0x00000037aa4c6adf <dup2+31>:   or     $0xffffffffffffffff,%rax
0x00000037aa4c6ae3 <dup2+35>:   jmp    0x37aa4c6acf <dup2+15>

或者 dup2 根本没有改变 newfd

最佳答案

常量本身(在 POSIX 上,STDIN_FILENO0 并且 STDOUT_FILENO1)确实被读取了-仅,但它们所表征的文件描述符可能会被关闭,而其他东西会在它们的位置打开;它们只是普通的文件描述符(通常设置了一个标志,以便它们在 execve() 系统调用时保持打开状态)。

发生变化的是驻留在操作系统内核中的进程的文件描述符表。看到 syscall 指令了吗?这在这里真的很重要;那是从您的进程进入操作系统的陷阱。

关于c - STDIN_FILENO 和 STDOUT_FILENO 在 c 中只读吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6197944/

相关文章:

c++ - 如何从 POSIX 文件描述符构造 c++ fstream?

c - 在 C 中与 STM32 芯片的内存交互

c - 这是关于 LINUX 中的共享内存

c - IPC管道通过文件/dev/ttyS0和/dev/ttyS1

string - 测试字符串是否包含文字星号

bash - 创建计算器脚本

c - sqlite3事务的检测

c - 使用管道读/写值

C: 如何将 stderr 从系统命令重定向到 stdout 或文件?

c - 在退出时关闭文件描述符是一种好习惯吗