2015年5月,在Red Hat Enterprise Linux(CVE-2018-1111)的多个版本的DHCP客户端软件包提供的NetworkManager脚本中发现了命令注入漏洞,随后,Red Hat提供了相应的补丁。当系统使用含有该漏洞的NetworkManager并配置了DHCP协议时,攻击者可以利用一个恶意的DHCP服务器或者本地网络构造的DHCP响应,在该系统上以root权限执行任意命令。
该漏洞对运行Red Hat Enterprise Linux版本6或7的、含有该漏洞的系统的个人或组织构成了严重威胁,应立即安装补丁。
在本文中,我们将会对该漏洞进行深入的分析,来帮助读者进行风险评估,并加深对该漏洞的了解,最后,我们还提供了相应的安全建议。
漏洞分析
NetworkManager是一个Linux程序,在启用DHCP网络模式的情况下,常使用它来管理系统网络。在这种情况下,NetworkManager将启动dhclient来发送DHCP请求,具体如图1所示。
图1. 通过NetworkManager运行的dhclient进程
在上面的示例中,读者可能会注意到,NetworkManager还向dhclient传递了另外一个配置文件(/var/lib/NetworkManager/dhclient-eth0.conf)。如下例所示,dhclient在默认情况下被配置为请求多个DHCP数据包选项,其中包括WPAD。
图2. 传递给dhclient的配置文件
当dhclient向DHCP服务器发送初始请求时,会在请求中包含这个WPAD(代码252)选项,具体如下图所示:
图3. 包含WPAD选项的DHCP初始请求数据包
利用CVE-2018-1111漏洞,攻击者可以通过格式错误的响应对该DHCP请求进行响应。例如,攻击者可以使用以下数据进行响应,具体如图4所示:
xxx'&touch /tmp/test #
图4. 攻击者利用DHCP响应中畸形的WPAD选项进行响应
收到该响应后,默认的11-dhclient脚本最终会将该数据传递给eval()语句,进而导致touch命令创建/tmp/test。
技术分析
当然,在受害者系统收到畸形的DHCP响应之后到创建/tmp/test之前这段时间内,还会发生许多其他事情。首先,dhclient将调用client_option_envadd()函数将值保存到变量中,这一点可以从下面的源代码中看到。具体来说,这一步是通过client_envadd()函数的3154行代码来完成的。
图5. 处理DHCP数据包选项的dhclient源代码
上面的源代码中,在设置变量之前,会调用pretty_print_option()函数,该函数会通过在特殊字符之前添加反斜杠(“\”)来对这些数据进行“消毒”, 例如:
The ‘ character will be converted to \’
The & character will converted to \&
在我们的示例中,发送的原始数据如下所示:
xxx’&touch /tmp/test #
然后,将转换为下面的样子:
xxx\’\&touch /tmp/test #
从下图可以看出,数据是由下面的函数完成转换的:
图6. pretty_print函数对示例WPAD选项进行相应的转换处理
在完成上述转换后、在将这些值存储到变量之前,会继续调用check_option_values()函数进行相应的检查,以确定在给定某些选项的情况下,是否包含某些特殊字符。例如,当提供HOST_NAME或DOMAIN_NAME时,就需要对特殊字符进行检查,具体如图8、图9所示。
图8. 检查是否提供了特定选项的代码
图9. 在提供NETBIOS_SCOPE选项的情况下所执行的代码
从上面的代码中可以看出,它并没有对WPAD选项进行相应的检查。因此,我们能够通过DHCP响应中的这个选项提供任意数据,因为它处于未被监督的地带。
接下来,dhclient将通过传递相应的参数来启动/usr/libexec/nm-dhcp-helper进程。然后,将这些变量保存到dbus服务中。
此外,还有另一个名为nm-dispatcher的兄弟进程,它是由NetworkManager启动的,之后会从dbus服务中读取这些变量。该进程会将WPAD DHCP选项的值保存到名为DHCP4_WPAD的环境变量中,然后,会启动位于/etc/NetworkManager/dispatcher.d/中的11-dhclient脚本。
在11-dhclient脚本中,包含以下内容:
图10. 11-dhclient脚本的内容
接下来,让我们深入了解这个脚本的作用。
在eval()语句中,执行的第一个命令是“declare”。这个“declare”命令将输出系统上的所有环境变量。读者可能非常熟悉“env”命令,实际上两者的作用非常类似。不过,虽然输出内容相似,但仍然存在一些关键差异,具体如图11所示。
图11. declare和env命令之间的差异
正如您在上面所看到的,“declare”命令还会执行另外两项操作:
- 如果变量包含特殊字符(例如空格或单引号),则会在两侧添加单引号,即'。
- 它会将内部单引号'转换为'\”(将一个字符转换为四个字符)。
由于变量值为xxx\’\&touch /tmp/test #,因此“declare”的输出将变为‘xxx\’\”\&touch /tmp/test #’。
运行“declare”命令后,脚本只会搜索以“DHCP4_”开头的环境变量。接下来,是“read”命令。如果未提供参数,那么该命令将读取转义后的字符,换句话说,\'将成为'。
回到我们通过DHCP响应中的WPAD选项提供的数据,即 ‘xxx\’\”\&touch /tmp/test #’,它将变为‘xxx”’&touch /tmp/test #’。换句话说,由于使用了没有任何参数的“read”命令,以前转义的字符将被取消转义。
其余命令会将解析后的环境变量数据设置为一系列变量。但是,最后一个命令中含有可能被利用的代码。具体来说,有安全问题的代码如下所示:
echo "export $optname=$optvalue"
使用我们的示例响应,可以在系统上执行以下代码:
eval "$(echo "export new_wpad='xxx'''&touch /tmp/test #' ")"
我们也可以在命令行中演示上述过程,具体如图12所示:
图12. 利用DHCP选项WPAD执行代码
由于没有对引号进行转义,并且,后面有一个&符号,所以,这就允许我们在这个eval()语句后面附加一个额外的命令。就本例来说,我们附加的是 ‘touch /tmp/test’命令,这样的话,就会在/tmp/目录中创建一个名为'test'的空文件。
如果引号和&符号被转义的话,我们的攻击将失效,具体如下所示:
图13. 使用转义后的字符执行相同的代码
需要注意的是,其他的字符也能用于该攻击,例如|或;。
漏洞的修复方法
对于这个特定的例子来说,修复方法非常简单——只需在“read”命令中添加“-r”选项即可防止各种字符被转义,该漏洞的补丁程序中的修复代码如下所示:
图14. CVE-2018-1111的补丁
根据“read”命令的文档的介绍,“-r”选项可防止命令将反斜杠作为转义字符读取。换句话说,它将保留提供给它的数据中的所有反斜杠。这样的话,就可以抵御命令注入攻击了。
该漏洞的现状
在漏洞被发现后不久,相关的概念验证(POC)代码就于2018年5月16日通过Twitter公之于众:
图15. 通过Twitter公布的CVE-2018-111 PoC
此外,在GitHub中,也有相关的测试代码:
https://github.com/knqyf263/CVE-2018-1111
结束语
由于NetworkManager的应用非常广泛,并且这个漏洞很容易被攻击者所利用,因此,我们应该将其作为高危漏洞来对待。就目前的发展情况来看,恶意攻击者利用该漏洞的风险正在日益增加。我们建议读者赶在该漏洞被攻击者利用之前安装好相应的补丁。