高效码农

第三章:程序运行

到目前为止,我们已经介绍了 CPU 如何执行从可执行文件加载的机器代码、什么是基于环的安全性以及系统调用如何工作。在本节中,我们将深入了解 Linux 内核,首先了解程序是如何加载和运行的。

我们将专门研究 x86-64 上的 Linux。为什么?

我们学到的大部分内容都可以很好地推广到其他操作系统和体系结构,即使它们在各种特定方面有所不同。

Exec 系统调用的基本行为


演示 exec 系统调用的流程图。 左边是一组标记为“用户空间”的流程图项,右边是一组标记为“内核空间”的流程图项。 从用户空间组开始:用户在终端中运行 ./file.bin,然后运行系统调用 execve("./file.bin", ...)。 这流向正在执行的 SYSCALL 指令,该指令随后指向内核空间组中的第一项:“加载并设置二进制文件”,该指令指向“尝试 binfmt”。 如果支持 binfmt,它将启动新进程(替换当前进程)。 如果没有,它会再次尝试 binfmt。

让我们从一个非常重要的系统调用开始:execve。它加载一个程序,如果成功,则用该程序替换当前进程。还存在其他几个系统调用(execlpexecvpe等),但它们都以各种方式分层execve

在旁边:execveat

execve实际上是建立在 之上的execveat,这是一个更通用的系统调用,它运行带有一些配置选项的程序。为了简单起见,我们主要讨论execve;唯一的区别是它提供了一些默认值execveat

好奇ve代表什么?表示v一个参数是参数 ( ) 的向量(列表)argv,表示e另一个参数是环境变量 ( ) 的向量envp。各种其他 exec 系统调用具有不同的后缀来指定不同的调用签名。in只是“at”,因为它指定了运行的at位置。execveat`execve`

的调用签名execve是:

int execve(const char *filename, char *const argv[], char *const envp[]);

有趣的事实!您知道程序的第一个参数是程序名称的约定吗?这纯粹是一个约定,实际上并不是由execve系统调用本身设置的!execve第一个参数将是作为参数中第一项传递的任何内容argv,即使它与程序名称无关。

有趣的是,execve确实有一些代码假设argv[0]是程序名称。稍后我们讨论解释型脚本语言时会详细介绍这一点。

第0步:定义

我们已经知道系统调用是如何工作的,但我们从未见过现实世界的代码示例!让我们看一下 Linux 内核的源代码,看看execve其底层是如何定义的:

文件系统/exec.c

2105
2106
2107
2108
2109
2110
2111
SYSCALL_DEFINE3(execve,
        const char __user *, filename,
        const char __user *const __user *, argv,
        const char __user *const __user *, envp)
{
    return do_execve(getname(filename), argv, envp);
}

SYSCALL_DEFINE3是一个用于定义 3 参数系统调用代码的宏。

我很好奇为什么元数被硬编码在宏名称中;我用谷歌搜索了一下,了解到这是修复某些安全漏洞的解决方法。

文件名参数被传递给一个getname()函数,该函数将字符串从用户空间复制到内核空间并执行一些使用情况跟踪操作。它返回一个filename结构体,该结构体在include/linux/fs.h. 它在用户空间中存储一个指向原始字符串的指针,以及一个指向复制到内核空间的值的新指针:

包含/linux/fs.h

2294
2295
2296
2297
2298
2299
2300
struct filename {
    const char        *name;    /* pointer to actual string */
    const __user char    *uptr;    /* original userland pointer */
    int            refcnt;
    struct audit_names    *aname;
    const char        iname[];
};

然后系统execve调用调用一个do_execve()函数。反过来,这会调用do_execveat_common()一些默认值。我之前提到的系统调用execveat也调用do_execveat_common(),但会传递更多用户提供的选项。

do_execve在下面的代码片段中,我包含了和 的定义do_execveat

文件系统/exec.c

2028
2029
2030
2031
2032
2033
2034
2035
2036
2037
2038
2039
2040
2041
2042
2043
2044
2045
2046
static int do_execve(struct filename *filename,
    const char __user *const __user *__argv,
    const char __user *const __user *__envp)
{
    struct user_arg_ptr argv = { .ptr.native = __argv };
    struct user_arg_ptr envp = { .ptr.native = __envp };
    return do_execveat_common(AT_FDCWD, filename, argv, envp, 0);
}

static int do_execveat(int fd, struct filename *filename,
        const char __user *const __user *__argv,
        const char __user *const __user *__envp,
        int flags)
{
    struct user_arg_ptr argv = { .ptr.native = __argv };
    struct user_arg_ptr envp = { .ptr.native = __envp };

    return do_execveat_common(fd, filename, argv, envp, flags);
}

[间距原文如此]

在 中execveat,文件描述符(指向某个资源的 id 类型)被传递到系统调用,然后传递到do_execveat_common。这指定了相对于执行程序的目录。

在 中execve,文件描述符参数使用了一个特殊值AT_FDCWD。这是 Linux 内核中的一个共享常量,它告诉函数将路径名解释为相对于当前工作目录的路径名。接受文件描述符的函数通常包括手动检查,例如if (fd == AT_FDCWD) { /* special codepath */ }.

第 1 步:设置

现在我们已经到达了do_execveat_common处理程序执行的核心函数。我们将从盯着代码的角度后退一步,以更全面地了解该函数的作用。

第一个主要工作do_execveat_common是建立一个名为 的结构linux_binprm。我不会包含整个结构定义的副本,但有几个重要的字段需要讨论:

TIL binprm代表二进制程序。)

让我们仔细看看这个缓冲区buf

linux_binprm @ include/linux/binfmts.h

64
    char buf[BINPRM_BUF_SIZE];

正如我们所看到的,它的长度被定义为常数BINPRM_BUF_SIZE。通过在代码库中搜索该字符串,我们可以在以下位置找到该字符串的定义include/uapi/linux/binfmts.h

包括/uapi/linux/binfmts.h

18
19
/* sizeof(linux_binprm->buf) */
#define BINPRM_BUF_SIZE 256

因此,内核将执行文件的开头 256 字节加载到该内存缓冲区中。

旁白:什么是 UAPI?

您可能会注意到上面代码的路径包含/uapi/. 为什么长度没有在与linux_binprm结构体相同的文件中定义include/linux/binfmts.h

UAPI 代表“用户空间 API”。在这种情况下,这意味着有人决定缓冲区的长度应该成为内核公共 API 的一部分。理论上,所有 UAPI 都暴露给用户空间,所有非 UAPI 都是内核代码私有的。

内核和用户空间代码最初共存为一团混乱的代码。2012年,UAPI代码被重构到一个单独的目录中,试图提高可维护性。

第2步:Binfmts

内核的下一个主要工作是迭代一堆“binfmt”(二进制格式)处理程序。fs/binfmt_elf.c这些处理程序在和 等文件中定义fs/binfmt_flat.c内核模块还可以将自己的 binfmt 处理程序添加到池中。

每个处理程序都公开一个load_binary()函数,该函数接受一个linux_binprm结构并检查处理程序是否理解程序的格式。

这通常涉及在缓冲区中查找幻数、尝试解码程序的开头(也来自缓冲区)和/或检查文件扩展名。)如果处理程序确实支持该格式,它将准备执行程序并返回成功代码。否则,它会提前退出并返回错误代码。

内核会尝试load_binary()每个 binfmt 的函数,直到找到一个成功的函数。有时这些会递归运行;例如,如果一个脚本指定了一个解释器,并且该解释器本身就是一个脚本,则层次结构可能是binfmt_script>> (其中 ELF 是链末尾的可执行格式)。binfmt_script`binfmt_elf`

格式亮点:脚本

在 Linux 支持的众多格式中,binfmt_script我想具体谈谈第一个。

您读过或写过Shebang)吗?某些脚本开头的那一行指定解释器的路径?

1
#!/bin/bash

我一直以为这些是由 shell 处理的,但事实并非如此!Shebang 实际上是内核的一项功能,脚本是使用与其他程序相同的系统调用来执行的。电脑太酷了。

看看如何fs/binfmt_script.c检查文件是否以 shebang 开头:

load_script @ fs/binfmt_script.c

40
41
42
    /* Not ours to exec if we don't start with "#!". */
    if ((bprm->buf[0] != '#') || (bprm->buf[1] != '!'))
        return -ENOEXEC;

如果文件确实以 shebang 开头,则 binfmt 处理程序会读取解释器路径以及路径后面的任何以空格分隔的参数。当它到达换行符或缓冲区末尾时它会停止。

这里发生了两件有趣的、奇怪的事情。

首先,还记得其中的缓冲区linux_binprm填充了文件的前 256 个字节吗?它用于可执行格式检测,但同样的缓冲区也是在binfmt_script.

在我的研究过程中,我读到一篇文章,将缓冲区描述为 128 字节长。在那篇文章发表后的某个时候,长度增加了一倍,达到 256 字节!奇怪的是,我检查了 Git 责任 - 编辑某一行代码的每个人的日志 - 查找BINPRM_BUF_SIZELinux 源代码中定义的行。你瞧……

Visual Studio Code 编辑器中的 Git 责任窗口的屏幕截图。 git Blame 显示行“#define BINPRM_BUF_SIZE 128”更改为 256。提交由 Oleg Nesterov,主要文本为“exec:将 BINPRM_BUF_SIZE 增加到 256。大型企业客户端通常在网络文件系统外运行应用程序,其中IT 强制的项目卷布局最终可能会导致路径长度超过 128 个字符。将其提高到下一个两个字符的顺序可以解决除最令人震惊的情况之外的所有问题,同时仍然适合 512b 板。” 该承诺由 Linus Torvalds 等人签署。

电脑太酷了!

由于 shebang 由内核处理,并且从文件中提取buf而不是加载整个文件,因此它们总是被截断为buf. 显然,4 年前,有人对内核截断超过 128 个字符的路径感到恼火,他们的解决方案是通过将缓冲区大小加倍来将截断点加倍!今天,在您自己的 Linux 机器上,如果您的 shebang 行长度超过 256 个字符,则超过 256 个字符的所有内容都将完全丢失


描绘 shebang 截断的图表。 来自名为 file.bin 的文件的大字节数组。 前 256 个字节突出显示并标记为“已加载到 buf”,而其余字节是半透明的并标记为“已忽略,过去 256 个字节”。

想象一下因此而出现错误。想象一下试图找出破坏代码的根本原因。想象一下,当发现问题出在 Linux 内核深处时,会是什么感觉。大型企业中的下一个 IT 人员发现路径的一部分神秘地丢失了,这对他们来说是不幸的。

第二件奇怪的事情:还记得它是程序名称的唯一约定argv[0]吗?调用者如何将argv他们想要的任何内容传递给 exec 系统调用,并且它将不受监管地传递?

碰巧这是假定程序名称的binfmt_script地方之一。它总是删除,然后将以下内容添加到 的开头: argv[0]`argv[0]`argv

示例:参数修改

让我们看一个示例execve调用:

// Arguments: filename, argv, envp
execve("./script", [ "A", "B", "C" ], []);

该假设script文件的第一行包含以下 shebang:

脚本

1
#!/usr/bin/node --experimental-module

最终传递给 Node 解释器的修改argv将是:

[ "/usr/bin/node", "--experimental-module", "./script", "B", "C" ]

更新后,处理程序通过设置解释器路径(在本例中为 Node 二进制文件)argv完成执行文件的准备。linux_binprm.interp最后,它返回 0 表示成功准备程序执行。

格式亮点:杂项口译员

另一个有趣的处理程序是binfmt_misc. 它提供了通过用户空间配置添加一些有限格式的能力,通过在/proc/sys/fs/binfmt_misc/. 程序可以对此目录中的文件执行特殊格式的写入以添加自己的处理程序。每个配置条目指定:

binfmt_misc系统通常由 Java 安装使用,配置为通过0xCAFEBABE魔术字节检测类文件,通过扩展名检测 JAR 文件。在我的特定系统上,配置了一个处理程序,通过其 .pyc 扩展名检测 Python 字节码并将其传递给适当的处理程序。

这是一种非常酷的方式,可以让程序安装程序添加对自己格式的支持,而无需编写高特权的内核代码。

最后(不是林肯公园的歌曲)

exec 系统调用始终会出现在以下两个路径之一:

如果您曾经使用过类 Unix 系统,您可能会注意到,从终端运行的 shell 脚本如果没有 shebang 行或.sh扩展名,仍然会执行。如果您手头有非 Windows 终端,您现在就可以测试一下:

外壳会话

$ echo "echo hello" > ./file
$ chmod +x ./file
$ ./file
hello

chmod +x告诉操作系统文件是可执行文件。否则您将无法运行它。)

那么,为什么shell脚本会以shell脚本的形式运行呢?内核的格式处理程序应该没有明确的方法来检测没有任何可辨别标签的 shell 脚本!

好吧,事实证明这种行为不是内核的一部分。这实际上是shell处理故障情况的常见方法。

当您使用 shell 执行文件并且 exec 系统调用失败时,大多数 shell 将通过执行以文件名作为第一个参数的 shell 来重试将文件作为 shell 脚本执行。Bash 通常会使用自身作为解释器,而 ZSH 使用任何sh解释器,通常是Bourne shell

这种行为非常常见,因为它是在POSIX中指定的,POSIX 是一种旧标准,旨在使代码在 Unix 系统之间可移植。虽然大多数工具或操作系统并未严格遵循 POSIX,但它的许多约定仍然是共享的。

如果 [an exec syscall] 由于与该错误等效的错误而失败[ENOEXEC]则 shell 应执行一个命令,相当于使用该命令名称作为其第一个操作数调用 shell,并将任何剩余参数传递给新 shell。如果可执行文件不是文本文件,shell 可能会绕过此命令执行。在这种情况下,它应写入一条错误消息并返回退出状态 126。

来源:Shell 命令语言,POSIX.1-2017

电脑真是太酷了!

当前页面是本站的「Google AMP」版。查看和发表评论请点击:完整版 »