U-Boot NFS RCE漏洞(CVE-2019-14192)

原文:https://blog.semmle.com/uboot-rce-nfs-vulnerability/

翻译:看雪翻译小组 - lipss

校对:看雪翻译小组 - Nxe


这篇文章是关于U-Boot引导加载程序中的13个远程执行代码漏洞的,我和我的同事Pavel Avgustinov和Kevin Backhouse发现了这些漏洞。当U-Boot配置为使用网络来获取下一阶段的启动资源时,可以触发这些漏洞。

请注意,该漏洞尚未通过https://gitlab.denx.de/u-boot/u-boot进行修补,并且我应U-Boot的主要托管人Tom Rini的要求将这些漏洞公开。有关更多信息,请查看下面的时间表。

MITER已针对这13个漏洞发布了以下CVE:CVE-2019-14192,CVE-2019-14193,CVE-2019-14194,CVE-2019-14195,CVE-2019-14196,CVE-2019-14197,CVE-2019 -14198,CVE-2019-14199,CVE-2019-14200,CVE-2019-14201,CVE-2019-14202,CVE-2019-14203和CVE-2019-14204

什么是U-Boot?

Das U-Boot(通常称为“通用引导加载程序”)是一种流行的主引导加载程序,广泛用于嵌入式设备中,以从不同来源获取数据并运行下一阶段的代码,通常(但不限于)Linux内核。IoT,Kindle和ARM ChromeOS设备通常使用它。

U-Boot支持从不同的文件分区格式(例如ext4),以及网络(TFTP和NFS)获取下一阶段的代码。请注意,U-boot支持验证启动,在其中检查获取的映像是否被篡改。这样可以减轻使用不安全的明文协议(例如TFTP和NFS)的风险。因此签名检查之前的任何漏洞都可能意味着设备越狱。

我正在使用U-boot,会受到影响吗?

这些漏洞影响非常特定的U-Boot配置,其中指示U-Boot使用网络。这些漏洞中的一些存在于NFS解析代码中,而其他一些则存在于通用TCP / IP堆栈中。

此配置通常用于无盘IoT部署和快速开发过程中。

有什么影响?

通过这些漏洞,同一网络(或控制恶意NFS服务器)中的攻击者可以在U-Boot驱动的设备上执行代码。由于此漏洞的性质,利用似乎并不十分复杂,尽管可以通过使用堆栈cookie,ASLR或其他运行时和编译时的内存保护机制来提高其挑战性。

明白了,这些漏洞是什么?

通过源代码审查在2个非常相似的事件中发现了第一个漏洞,我们使用了Semmle的LGTM.comQL来查找其他漏洞。它是普通的memcpy溢出,攻击者控制的大小来自网络数据包,没有任何验证。

该问题存在于nfs_readlink_reply解析来自网络的nfs答复的函数中。它解析4个字节,并且无需进一步验证,就在两个不同位置中将它们用作memcpy的长度。

  1. static int nfs_readlink_reply(uchar *pkt, unsigned len)
  2. {
  3. [...]
  4. /* new path length */
  5. rlen = ntohl(rpc_pkt.u.reply.data[1 + nfsv3_data_offset]);
  6. if (*((char *)&(rpc_pkt.u.reply.data[2 + nfsv3_data_offset])) != '/') {
  7. int pathlen;
  8. strcat(nfs_path, "/");
  9. pathlen = strlen(nfs_path);
  10. memcpy(nfs_path + pathlen,
  11. (uchar *)&(rpc_pkt.u.reply.data[2 + nfsv3_data_offset]),
  12. rlen);
  13. nfs_path[pathlen + rlen] = 0;
  14. } else {
  15. memcpy(nfs_path,
  16. (uchar *)&(rpc_pkt.u.reply.data[2 + nfsv3_data_offset]),
  17. rlen);
  18. nfs_path[rlen] = 0;
  19. }
  20. return 0;
  21. }

目标缓冲区nfs_path是一个全局缓冲区,最多可容纳2048个字节。

使用QL的变异分析

以下查询为我们提供了9个列表,以便手动进行跟踪。查询背后的想法是从任何辅助函数(例如ntohl()/ ntohs()...)到memcpy的size参数来执行数据流分析。

  1. import cpp
  2. import semmle.code.cpp.dataflow.TaintTracking
  3. import semmle.code.cpp.rangeanalysis.SimpleRangeAnalysis
  4. class NetworkByteOrderTranslation extends Expr {
  5. NetworkByteOrderTranslation() {
  6. // On Windows, there are ntoh* functions.
  7. this.(Call).getTarget().getName().regexpMatch("ntoh(l|ll|s)")
  8. or
  9. // On Linux, and in some code bases, these are defined as macros.
  10. this = any(MacroInvocation mi |
  11. mi.getOutermostMacroAccess().getMacroName().regexpMatch("(?i)(^|.*_)ntoh(l|ll|s)")
  12. ).getExpr()
  13. }
  14. }
  15. class NetworkToMemFuncLength extends TaintTracking::Configuration {
  16. NetworkToMemFuncLength() { this = "NetworkToMemFuncLength" }
  17. override predicate isSource(DataFlow::Node source) {
  18. source.asExpr() instanceof NetworkByteOrderTranslation
  19. }
  20. override predicate isSink(DataFlow::Node sink) {
  21. exists (FunctionCall fc |
  22. fc.getTarget().getName().regexpMatch("memcpy|memmove") and
  23. fc.getArgument(2) = sink.asExpr() )
  24. }
  25. }
  26. from Expr ntoh, Expr sizeArg, NetworkToMemFuncLength config
  27. where config.hasFlow(DataFlow::exprNode(ntoh), DataFlow::exprNode(sizeArg))
  28. select ntoh.getLocation(), sizeArg

我们找到任何变体了吗?

我们检查了结果,虽然有些数据在从源到接收器的数据流之间进行了大小检查,但发现有些数据是可利用的。此外,我们通过源代码审查发现了其他一些变体。

nfs_lookup_reply中失败的长度检查导致未限界memcpy

nfs_lookup_reply函数再次解析来自网络的nfs答复时存在此问题。它解析4个字节,并在两个不同位置中将它们用作memcpy的长度。

进行长度检查以确保它不大于分配的缓冲区。不幸的是,可以用负值绕过此检查,这将在以后导致较大的缓冲区溢出。

  1. filefh3_length = ntohl(rpc_pkt.u.reply.data\[1]);
  2. if (filefh3_length > NFS3_FHSIZE)
  3. filefh3_length = NFS3_FHSIZE;
  4. memcpy(filefh, rpc_pkt.u.reply.data + 2, filefh3_length);

目标缓冲区filefh是一个全局缓冲区,最多可容纳64个字节。

nfs_read_reply/store_block中失败的长度检查导致未限界memcpy

nfs_read_reply读取文件并将其存储到另一种介质(闪存或物理存储器)中以供以后处理时,该函数中存在此问题。同样,数据和长度由攻击者完全控制,并且从未验证。

  1. static int nfs_read_reply(uchar *pkt, unsigned len)
  2. { [...]
  3. if (supported_nfs_versions & NFSV2_FLAG) {
  4. rlen = ntohl(rpc_pkt.u.reply.data[18]); // <-- rlen is attacker-controlled could be 0xFFFFFFFF
  5. data_ptr = (uchar *)&(rpc_pkt.u.reply.data[19]);
  6. } else { /* NFSV3_FLAG */
  7. int nfsv3_data_offset =
  8. nfs3_get_attributes_offset(rpc_pkt.u.reply.data);
  9. /* count value */
  10. rlen = ntohl(rpc_pkt.u.reply.data[1 + nfsv3_data_offset]); // <-- rlen is attacker-controlled
  11. /* Skip unused values :
  12. EOF: 32 bits value,
  13. data_size: 32 bits value,
  14. */
  15. data_ptr = (uchar *)
  16. &(rpc_pkt.u.reply.data[4 + nfsv3_data_offset]);
  17. }
  18. if (store_block(data_ptr, nfs_offset, rlen)) // <-- We pass to store_block source and length controlled by the attacker
  19. return -9999;
  20. [...]
  21. }

关注store_block函数的物理内存部分,它尝试使用特定于arch的函数map_physmem保留一些内存,最终调用phys_to_virt。正如您在x86实现中所看到的那样,在保留物理内存时,它显然忽略了长度,并为您提供了原始指针,而没有检查周围区域是否为其他目的保留。

  1. static inline void *phys_to_virt(phys_addr_t paddr)
  2. {
  3. return (void *)(unsigned long)paddr;
  4. }

其后在store_block中有一个攻击者控制的源和长度的memcpy缓冲区溢出。

  1. static inline int store_block(uchar *src, unsigned offset, unsigned len)
  2. {
  3. [...]
  4. void *ptr = map_sysmem(load_addr + offset, len); // <-- essentially this is ptr = load_addr + offset
  5. memcpy(ptr, src, len); // <-- unrestricted overflow happens here
  6. unmap_sysmem(ptr);
  7. [...]
  8. }

flash_write代码路径中也可能存在类似的问题。

由于整数下溢,在解析UDP数据包时出现未限界memcpy

函数net_process_received_packet未经验证的情况下使用ip->udp_len会导致整数下溢。其后,此字段将用于通过net_set_udp_handler(DNS,dhcp,...)设置的memcpyatnc_input_packet和所有udp数据包处理函数中。

  1. #if defined(CONFIG_NETCONSOLE) && !defined(CONFIG_SPL_BUILD)
  2. nc_input_packet((uchar *)ip + IP_UDP_HDR_SIZE,
  3. src_ip,
  4. ntohs(ip->udp_dst),
  5. ntohs(ip->udp_src),
  6. ntohs(ip->udp_len) - UDP_HDR_SIZE); // <- integer underflow
  7. #endif
  8. /*
  9. * IP header OK. Pass the packet to the current handler.
  10. */
  11. (*udp_packet_handler)((uchar *)ip + IP_UDP_HDR_SIZE,
  12. ntohs(ip->udp_dst),
  13. src_ip,
  14. ntohs(ip->udp_src),
  15. ntohs(ip->udp_len) - UDP_HDR_SIZE); // <- integer underflow

请注意,我们并未审计为不同目的(DNS,DHCP等)设置的所有潜在udp处理程序。但是,我们确实对nfs_handler进行了审计,如下所述。

响应辅助函数nfs_handler中多个基于堆栈的缓冲区溢出

这是上述漏洞的代码审查变体。在此,当解析很大的ip->udp_len参数的udp数据包时会发生整数下溢,随后又调用nfs_handler。在此函数中,同样没有对长度的验证,我们将调用辅助函数,例如nfs_readlink_reply。该函数盲目使用长度而不进行验证,从而导致基于堆栈的缓冲区溢出。

  1. static int nfs_readlink_reply(uchar *pkt, unsigned len)
  2. {
  3. struct rpc_t rpc_pkt;
  4. [...]
  5. memcpy((unsigned char *)&rpc_pkt, pkt, len);

我们确定了5个不同的易受攻击的函数,它们遵循相同的代码模式,从而导致基于堆栈的缓冲区溢出。除了nfs_readlink_reply之外,还有:

  • rpc_lookup_reply
  • nfs_mount_reply
  • nfs_umountall_reply
  • nfs_lookup_reply

nfs_read_reply中读取越界数据

这与以前的漏洞非常相似。开发人员试图在复制来自套接字的数据时执行大小检查,以保持谨慎。当他们检查以防止缓冲区溢出时,他们没有检查源缓冲区中是否有足够的数据,从而导致潜在的读取越界访问冲突。

  1. static int nfs_read_reply(uchar *pkt, unsigned len)
  2. {
  3. struct rpc_t rpc_pkt;
  4. [...]
  5. memcpy(&rpc_pkt.u.data[0], pkt, sizeof(rpc_pkt.u.reply));

攻击者可以向NFS数据包提供读取请求和发送给套接字的较小数据包请求。

有什么建议吗?

为了缓解这些漏洞,只有两种选择:

  • 补丁发布后立即应用,或者
  • 处于可攻击时,请勿通过NFS或任何U-Boot网络功能挂载文件系统

披露时间表

此漏洞报告受披露政策的约束,在此处可见https://lgtm.com/security/#disclosure_policy

  • 2019年5月15日-Fermín Serna最初发现了两个漏洞,并编写了一个QL查询,以发现另外三个有问题的调用处。
  • 2019年5月16日-Pavel Avgustinov带来了一些QL魔术,概括了查询并找到了更多解析ip和udp标头的函数。
  • 2019年5月23日-Kevin Backhouse通过nfs_handler向Pavel和Fermín警告有关基于堆栈的缓冲区溢出的疏忽。
  • 2019年5月23日-Semmle安全团队结束调查并通过电子邮件联系维护者。
  • 2019年5月24日-Tom Rini(U-Boot的主要托管人)确认收到安全报告。
  • 2019年7月19日-Tom Rini请求在其公共邮件列表u-boot@lists.denx.de上公开此报告。
  • 2019年7月22日-为了避免周末披露,Fermin将该报告通过u-boot@lists.denx.de公开。

U-Boot NFS RCE漏洞(CVE-2019-14192)的更多相关文章

  1. Zimbra无需登录RCE漏洞利用

    2019年3月13号,一名国外的安全研究员在他的博客上公布了zimbra RCE漏洞相关信息,但其中并未提到一些漏洞利用细节. 经过一段时间努力,根据网上各位大牛的分析和我自己的理解,在此我将整个漏洞 ...

  2. WordPress插件Social Warfare<=3.5.2 无需登录RCE漏洞

    该漏洞只存在于Social Warfare插进的3.5.0.3.5.1和3.5.2版本中,其他版本不存在. 2019年3月21日插件作者紧急发布了3.5.3版本以修复高危的RCE漏洞,在<=3. ...

  3. 威胁快报|首爆,新披露Jenkins RCE漏洞成ImposterMiner挖矿木马新“跳板”

    简介 阿里云安全于近日捕获到一起使用Jenkins RCE漏洞进行攻击的挖矿事件.除挖矿外,攻击者还曾植入具有C&C功能的tsunami木马,也预留了反弹shell的功能,给用户带来极大安全隐 ...

  4. ThinkPHP-5.0.23新的RCE漏洞测试和POC

    TP5新RCE漏洞 昨天又是周五,讨厌周五曝漏洞,还得又得加班,算了,还是先验证一波.新的TP5RCE,据说发现者因为上次的RCE,于是又审计了代码,结果发现的.TP5也成了万人轮啊. 测试 环境搭建 ...

  5. GitStack系统RCE漏洞学习

    漏洞简介 漏洞简情 漏洞程序 GitStack 影响版本 <=2.3.10 漏洞类型 RCE 漏洞评价 高危 漏洞编号 CVE-2018-5955 漏洞程序介绍 GitStack是一款基于Pyt ...

  6. Joomla 3.0.0 -3.4.6远程代码执行(RCE)漏洞复现

    Joomla 3.0.0 -3.4.6远程代码执行(RCE)漏洞复现 一.漏洞描述 Joomla是一套内容管理系统,是使用PHP语言加上MYSQL数据库所开发的软件系统,最新版本为3.9.12,官网: ...

  7. Apache Solr Velocity模板注入RCE漏洞复现

    Apache Solr Velocity模板注入RCE漏洞复现 一.Apache Solr介绍 Solr是一个独立的企业级搜索应用服务器,它对外提供类似于web-service的API接口,用户可以通 ...

  8. Apache Shiro<=1.2.4反序列化RCE漏洞

    介绍:Apache Shiro是一个强大且易用的Java安全框架,执行身份验证.授权.密码和会话管理. 漏洞原因:因为shiro对cookie里的rememberme字段进行了反序列化,所以如果知道了 ...

  9. Joomla 3.0.0 - 3.4.6 RCE漏洞分析记录

    0x00  前言 今天早上看到了国内几家安全媒体发了Joomla RCE漏洞的预警,漏洞利用的EXP也在Github公开了.我大致看了一眼描述,觉得是个挺有意思的漏洞,因此有了这篇分析的文章,其实这个 ...

随机推荐

  1. Linux 一款免费的shell工具 MobaXterm_Personal

    一款免费的shell工具 MobaXterm_Personal

  2. CDH 大数据平台搭建

    一.概述 Cloudera版本(Cloudera’s Distribution Including Apache Hadoop,简称“CDH”),基于Web的用户界面,支持大多数Hadoop组件,包括 ...

  3. YAML_10 把监听端口是8080的Apache服务全部停止

    ansible]# vim ad.yml --- - hosts: cache   remote_user: root   tasks:     - shell: netstat -atunlp  | ...

  4. 共享库的使用(.so)文件

    1.共享库的概念 2.创建共享库命令 gcc -shared -fpci -o lib/libmath.so obj/mymath.o 具体加不加 fpci 这个要看平台支持吧支持:具体详情可以查阅 ...

  5. 【知识点】同样是消息队列,Kafka凭什么速度那么快?

    同样是消息队列,Kafka凭什么速度那么快? 作者 | MrZhangxd Kafka的消息是保存或缓存在磁盘上的,一般认为在磁盘上读写数据是会降低性能的,因为寻址会比较消耗时间,但是实际上,Kafk ...

  6. 并发用户 VS TPS

    TPS模式(吞吐量模式)是一种更好的方式衡量服务端系统的能力. 基本概念: 并发用户数:简称VU ,指的是现实系统中操作业务的用户,在性能测试工具中,一般称为虚拟用户数(Virutal User),注 ...

  7. mac安装gmpy2

    brew install libmpc brew install mpfr pip install gmpy2

  8. Control.ImeMode属性简释

    在WINFORM中,我们经常遇到如下问题.文本输入框中输入法有时候需要被禁用,或者某些时候全半角输入自动转换.查阅相关资料,现小结如下. (一)Control.ImeMode 属性:获取或设置控件的输 ...

  9. C排序算法

    几个常用的排序算法:插入排序.快速排序.归并排序 #include <stdio.h> #include <stdlib.h> #include <stdbool.h&g ...

  10. CentOs7设置主机名称,以及主机名称和ip的对应关系

    一.修改主机名称 在CentOS7中有三种定义的主机名:静态的(static).瞬态的(transient).和灵活的(pretty).静态主机名也称为内核主机名,是系统在启动时从/etc/hostn ...