NASM x86_64:文件打开 (SYS_OPEN) 错误列表?

问题描述

我正在编写一个读取文件的 linux x64 汇编程序,我想处理诸如找不到文件或权限错误之类的错误

在哪里可以找到 SYS_OPEN 错误代码列表?

寻找代码方法(有点有趣)

我打开文件代码

SYS_OPEN    equ 2
O_RDONLY    equ 0

section .data
    filename db "file.txt",0

section .text
    global _start

_start:
    mov rax,SYS_OPEN
    mov rdi,filename
    mov rsi,O_RDONLY
    mov rdx,0644o
    syscall

    [...]

文件成功打开时,RAX 寄存器指向 file descriptor(正整数),如果失败,RAX 指向一个错误(负整数)。我设法通过删除所有用户的所有权限来引发权限错误

chmod 0000 file.txt

这会导致错误代码-13。通过删除文件,我设法得到错误 -2。在哪里可以找到 SYS_OPEN 错误代码列表?

PS:也许我的谷歌搜索技能已经生疏了

解决方法

您可以将其解释为 errno 的值,这里是 table(列出所有代码使用 errno -l),还可以查看 docs .表的一部分:

数量 十六进制 符号 描述
2 0x02 ENOENT 没有那个文件或目录
13 0x0d EACCES 权限被拒绝

描述了这样决定的原因:https://stackoverflow.com/a/6008711/9926721

,

Linux system call return values-4095-1-errno 代码。(Linux 实际定义的实际最高错误编号目前约为 { {1}}、133,但这是官方范围。)

EHWPOISON 可以为您解码它们,因此您在玩弄系统调用时无需在玩具程序中实际编写错误检查。

例如:

strace ./myprog

$ strace touch /tmp/xyjklj/bar ... (dynamic linker / process startup stuff) openat(AT_FDCWD,"/tmp/xyjklj/bar",O_WRONLY|O_CREAT|O_NOCTTY|O_NONBLOCK,0666) = -1 ENOENT (No such file or directory) utimensat(AT_FDCWD,NULL,0) = -1 ENOENT (No such file or directory) ... (more system calls as touch(1) finds a locale-specific set of error messages and prints 是 libc 包装器函数实际返回的内容;errno 代码是从 asm 系统调用返回值中解码出的 strace,glibc 包装器将存储在 -1 中。使用原始系统时在 asm 中调用,你不必浪费指令这样做。但 strace 仍然会说“-1”,而不是数字错误代码)


关于 errno 可能失败的大多数方式的文档

每个系统调用手册页都记录了特定系统调用可能失败的错误代码,以及可能发生的情况。 (这些列表并不完全详尽,例如没有涵盖像 NFS 这样的特定文件系统可能返回的奇怪的东西,比如 SYS_open(见评论)。)

对于您的情况,请参阅 the ERRORs section of the open(2) man page。例如EMULTIHOP 有多个条目,涵盖了可能导致该返回值的所有情况。

  • ENOENT - ENOENT 未设置且命名文件不存在。

  • O_CREAT - 路径名中的目录组件不存在或者是 悬空符号链接。

  • ENOENT - 路径名指的是一个不存在的目录,ENOENTO_TMPFILEO_WRONLY 之一在标志中指定,但 此内核版本不提供 O_TMPFILE 功能。

(剧透警告,O_RDWR2,所以 ENOENT-2。)

当然还有很多其他有趣的方式会导致路径名和文件访问内容(尤其是 -ENOENT)出错,包括:

  • open(2) (EACCES) - 不允许对文件的请求访问,或搜索 中的目录之一的权限被拒绝 -13 的路径前缀,或该文件尚不存在且 不允许对父目录进行写访问。 (看 还有 path_resolution(7).)

  • pathname - EFAULT 点位于您可访问的地址空间之外。

  • pathname - ENAMETOOLONG 太长了。

  • pathname - EBUSY 在标志中指定,O_EXCL 指的是 系统正在使用的块设备(例如,它是 安装)。
    [这需要root,否则你会得到EACCESS]

  • pathname - ETXTBSY 指的是当前正在运行的可执行映像 正在执行并请求写访问权限。

  • pathname - 指定了 EWOULDBLOCK 标志,并且不兼容 租约保留在文件上(请参阅 fcntl(2))。

  • O_NONBLOCK - ENODEV 指的是设备专用文件,没有 存在相应的设备。 (这是一个 Linux 内核错误; 在这种情况下必须返回 pathname。)

  • ENXIO - 在解析时遇到太多符号链接 路径名。 ELOOP - EISDIR 指的是一个目录和请求的访问 涉及写入(即设置了 O_WRONLY 或 O_RDWR)。 pathname - 在 ENOTDIR 中用作目录的组件不是,在 事实上,指定了目录或 pathname 并且 O_DIRECTORY 不是目录。

  • pathname - 指定了 EPERM 标志,但有效用户 调用者的 ID 与文件的所有者不匹配,并且 调用者没有特权。

以及各种限制,例如打开文件的数量(ENFILE、EMFILE)或 ENOSPC 磁盘空间已满。以上是不是一个完整的列表,我只是采取了一种方法来获取许多(但不是全部)错误代码。


根据funnydman 的回答,您可以在手册页中查找数字-> 错误值的符号含义。或者查看 O_NOATIME(完整路径在某些系统上可能不同,您只能通过 /usr/include/asm-generic/errno-base.h 间接包含此文件)