问题描述
我正在编写一个读取文件的 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
- 路径名指的是一个不存在的目录,ENOENT
和O_TMPFILE
或O_WRONLY
之一在标志中指定,但 此内核版本不提供 O_TMPFILE 功能。
(剧透警告,O_RDWR
是 2
,所以 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
间接包含此文件)