百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

MS15-083:Windows SMB内存损坏漏洞分析

zhezhongyun 2025-07-23 19:24 3 浏览

2015年8月11日微软发布了14个安全补丁,其中就包括一个SMB服务器补丁。在本文我将解释我是如何触发该漏洞的。

微软安全公告MS15-083

在所有的修复补丁中,我对“服务器消息块中的漏洞可能允许远程执行代码”很感兴趣。

“当服务器信息块(SMB)不当处理某些日志记录活动,引发一个经过身份验证的远程代码执行漏洞,最终导致内存损坏。”

受影响的版本包括32位和64位的Windows Vista 和 Windows Server 2008,值得注意的是,这是自2011年以来微软发布的第一个SMB服务修复补丁。

安装补丁

下载“
Windows6.0-KB3073921-x86.msu”补丁之后,我试图进行安装,没想到出现了这个

对此我感到有些意外,因为当安装好一个内核补丁后,操作系统都会进行重启。但在这个例子中,补丁安装程序并没有提示我进行重启。

在“c:\windows\system32\drivers”中我发现“srv.sys”和“srvnet.sys”都被变更了。

另外,我还注意到新的“srvnet.sys”文件日期为2011年4月,新的“srv.sys”文件日期显示正常。

差异阶段–Part 1

将新的“srv.sys”版本(v6.0.6002.19438)与2011年发布的MS11-020版本(v6.0.6002.18407)进行对比,惊奇的发现代码根本就没有任何改动!要说有改动也就是编译的字符串改动了。

我寻找着有关该补丁的一些信息,最终在twitter上找到Greg Linares发的一条有关“srvnet.sys”中代码变动的信息。

我联系上Greg之后确认了补丁安装程序的错误,同时感谢其指点,通过使用“expand.exe”命令手动解压了该补丁。

差异阶段–Part 2

补丁解压完成之后,我发现“srv.sys”和“srvnet.sys”的两个版本。将旧的srvnet.sys”版本(v6.0.6002.18462) 和新版本(v6.0.6002.23746)进行差异比较,两个版本都使用相同的补丁安装程序。

在其中我发现7个函数进行了重要改变,更多的则是改动很小:

RfsTableEnumerate

RfsTable64Enumerate

RfsTable64LookupAndEnumerate

SrvGraftName

SrvLibLogError

SrvNetWskEnableInterface

SrvNetWskOpenListenSocket

通过微软安全公告的描述“某些日志记录活动”,所以我重点关注了下“SrvLibLogError”函数,以下为原始代码与新版本的差异比较:

可以清晰的看到修复了一个整数溢出。

关注“IoAllocateErrorLogEntry”调用代码,可以看到这个修复防止了当日志消息大小大于255字节时“message size”被重新初始化为0。

如果这发生在未打补丁的版本上,没有足够的内存分配给日志消息记录就会产生一个堆溢出。出于某些原因,“message size”变量使用的无符号字符型传递参数是不正确的。在新版本代码中,该变量调整为无符号整型。

差异阶段–开盖有奖

检测“SrvLibLogError”函数在两个版本中的变化,我注意到在Windows 7, Windows 8, Windows 8.1 and Windows 10中这个漏洞依旧存在。

下图为“Windows 10” 64位中漏洞基本块图像,“srvnet.sys” v10.0.10240.16384部分:

另一方面要说的是,该问题在Windows 2008 R2中被静默修复。

尽管在这些操作系统中重现漏洞是被禁止的,阐明“SrvLibLogError”函数通过“srvnet.sys”输出,以及当使用SMBv1建立SMB连接依旧调用“srv.sys”是非常重要的。

这也意味着,任何Windows驱动(官方或者第三方)调用这个输出函数,这个漏洞将会重现。

利用该漏洞可能的方法

一旦检测到该漏洞,最大的挑战便是写exp找到正确的输出触发该漏洞,本例中我们借助SMB协议。

以下插图为所有的影响到“SrvLibLogError”函数的方法

在第六个论点中函数接收一个字符串列表进行记录,在第七个论点中其接收数字字符串进行记录。

顺着调用图表一直调用,我发现一个来自“SrvLibLogSpnError”函数的十分有趣的调用,位于相同的库(srvnet.sys)

继续查找,发现该输出函数仅通过函数进行调用,出现在“srv.sys”和“srv2.sys”.

这也就意味着可以使用SMBv1和v2协议进行攻击。有趣的是,在MS11-020补丁中也调用了“SrvLibLogSpnError”函数

触发阶段–Part 1

使用Windows 7通过执行类似“\\192.168.60.60\shared”命令指向到Windows Server 2008,可以看到通过SMBv2当SMB服务接收到一个“Session Setup Request”包以及“NTLMSSP_AUTH”选项时,“srv2!SrvValidateSecurityBuffer”就被调用了。

以下为影响漏洞函数的第一个要求


_SmbServerNameHardeningLevel”变量的值不同于0,这个值我们可以在注册表“HKEY_LOCAL_MACHINE\SYSTEM\CurrentControlSet\Services\LanmanServer\Parameters\
SmbServerNameHardeningLevel”中查看,该值为“Server SPN target name validation level”策略相关的设定,也是“SMB hardening”的一部分。

触发阶段–Part 2

将值设置完毕之后,新的问题又出现了

“MapSecurityError”函数返回错误代码:0xc00000bb,这也意味着,如果这个值不为0,就不能调用记录函数。

阅读msdn文档,我意识到返回值得意思为“STATUS_NOT_SUPPORTED”

“MapSecurityError”函数接收“QueryContextAttributesW”函数的输出作为参数,位于“ksecdd.sys”

在第二份文档中,我意识到“ulAttribute”参数的值应该设置为0x1b,意思与“
SECPKG_ATTR_CLIENT_SPECIFIED_TARGET”相同。

以下为这个常数的描述:

分析“ksecdd!QueryContextAttributesW”函数的代码,我确认已经不支持该常数了。

这里就矛盾了,因为补丁修复“Windows Vista”和“Windows 2008”中的漏洞,但是这些能够影响到漏洞函数的方法却无法在这些系统中实现!

触发阶段–Part 3

再次浏览微软公告页面,发现了一个参考链接“Extended Protection for Authentication” (EPA).

为了寻找更多有价值的信息,我在“Security Research and Defense Blog”中发现一篇名为Extended Protection for Authentication的文章。

部分阅读:

“微软发布了几个非安全的更新实现身份验证保护延长的机制,用以维护在Windows平台上的身份认证凭证....”

从第一个博客链接中我下载并安装了EPA support for Windows 2008:
https://technet.microsoft.com/library/security/973811

触发阶段–Part 4

EPA安装完成之后,“
SmbServerNameHardeningLevel”注册表键设置为1或者2,启用“File Sharing”选项并关闭“Password protected sharing”选项。“QueryContextAttributesW”函数返回0(“STATUS_SUCCESS”)

当“QueryContextAttributesW”函数的“pBuffer”参数返回这个值,就变得更加有趣了

使用Wireshark进行嗅探,我意识到“pBuffer”参数返回“Target Name”的属性“NTLMv2 Response”结构,包含在“Session Setup AndX Request”包中。

基于连接的SMB版本,这是第三个或者第四个SMB客户端发送的数据包

当我意识到这点,我开始构建和发送“Session Setup AndX Request”数据包,就象这样:

再然后就这样了

利用阶段

微软标记的该漏洞的可利用性分数

现在,我们来看看触发这个漏洞之后到底发生了什么

当尝试释放 “IoAllocateErrorLogEntry”函数分配的当前块(CURRENT CHUNK),“nt!ExFreePoolWithTag”产生了一个内核异常错误。

最重要的是注意到的是产生堆溢出的池类型(NonPagedPool),这里给出一个完整的池类型列表

再细看下

“nt!ExFreePoolWithTag”函数检测到下一个头为CORRUPTED。在本例中,我们看到当前块为0x8c7913b8,下一个为0x8c791478,这就说明下一个头为“41 41 41 41 41 41 41 41”

当前块的头为有效值会发生什么?

分析当前块的头,不难发现以下规律:

9 bits (previous chunk size / 8) + 7 bits (misc) + 9 bits (current chunk size / 8) + 7 bits (allocated|free|misc) + 4 bytes (TAG)

我控制着写入的所有数据,所以我可以设置下一个块第一个字段为有效值。

比如,如果“IoAllocateErrorLogEntry”函数分配256 bytes给块(0×100的16进制值),那么“previous chunk size”的下一个块的头为0×100 / 8 = 0×20

事实上,如果我设置一个正确的值覆盖“previous chunk size”当释放当前块,那么也就不会触发漏洞了,目标也就不会崩溃了。

当我们知道如何绕过第一个BSoD,就可以通过使用各类堆技术实现远程利用的艺术。

利用想法

为了利用这个漏洞,我可以使用一个比较老套的技术“堆合并”。

此外,如果我能够准确控制远程配置,我可以覆盖一个内存对象,这样就获得多个写入的机会。

还有一个选择就是覆盖函数指针的低部分。

我可以覆盖的最大内存大小超过了分配的块,接近2300bytes

在这个漏洞,最重要的利用过程就是利用失效或者Windows内核崩溃,目标将会自动重启。

很明显,利用很棘手。但是我不确定微软给出的漏洞利用性是否正确。

后记

2015年9月8日再次更新了这个漏洞补丁

目前补丁安装程序正常运行

但是,进修复了“SrvLibLogError”函数

另一方面Windows XP和Windows 2003由于停止维护,漏洞依旧存在

相关推荐

C#.NET NLog 详解(c#nuget)

简介NLog是.NET平台上最流行的开源日志框架之一,特色是灵活的配置、丰富的输出目标(Target),以及高性能的异步写入能力。适用场景:从控制台、文件、数据库、网络到Elastic...

WPF中datagrid单元格背景颜色(wpf datagrid单元格编辑)

datagrid中AutoGenerateColumns="true",使用viewmodel中绑定的数据源。后台代码找到目标字段,重写IValueConverter值转换器接口,根...

Unity Profiler实战指南:从卡顿到丝滑的性能优化之旅

当玩家说"这游戏卡爆了"时,你该怎么办?"角色移动时帧率从60掉到20,技能特效一放直接卡成PPT"——这是《幻境冒险》项目上线前测试阶段收到的玩家反馈。作为主程的我知...

Windows下取文件属性特例(windows文件属性快捷键)

今日碰到有程序在我们产品系统环境下无法正常运行某些功能,使用ProcessMonitor加反复测试发现,与产品中创建的symlink(软链接)有关。具体来讲,symlink文件是一个软链接文件,它的...

展开说说,DOS有哪些常用、实用的命令?

晚上好,我是老杨,今天来聊聊常用的DOS命令。虽然是老古董,但不妨碍它的好用程度。可能一些新手不晓得,但是和老杨一个年纪的,一定对DOS系统不陌生。尽管现在大多数电脑的操作系统是Windows,但在W...

webservice更改返回信息节点名称(webservice返回值)

问题详情:<!--访问webservice中,返回的信息--><soap:Envelopexmlns:soap="http://schemas.xmlsoap.org/s...

Windows Server 2019 基线检查表 (1)

ControlSetCorrectlyYesNo1AccountPolicies1.1PasswordPolicy1.1.1(L1)Ensure'Enforcepasswordh...

MapStruct架构设计(mapstruct官方文档)

MapStruct架构原理及改造一、前言4二、什么是语法树(AST)42.1Java编译时的三个阶段4三、什么是JSR26953.1使用步骤53.2流程图6四、源码架构分析64.1...

Excel常用技能分享与探讨(6-实战小功能分享 三)

书接上文,上一篇主要写了如何用代码动态创建控件,这一章讲讲如何具体实现我们需要的功能。五、功能性代码我们添加两个OptionButton的目的就是为了切换到对应的工具,所以,我们需要的是在点击了相对应...

30天学会Python编程:8. Python面向对象编程

8.1OOP基础概念8.1.1面向对象三大特性8.1.2类与对象关系核心概念:类(Class):对象的蓝图/模板对象(Object):类的具体实例属性(Attribute):对象的状态/数据方法...

环境变量设置被禁止临时方案(为什么设置了环境变量还是会出现)

1,到官网下载ant安装包。官网下载地址:ApacheAnt-BinaryDistributions2,解压到本地不带中文目录下,我这儿是D盘并且改名为ant3,设置环镜变量时,发现云桌面系统...

一文读懂 JavaScript依赖注入(java依赖注入简单理解)

大家好,我是Echa。依赖注入DI(DependencyInjection)是编程领域中一个非常常见的设计模式,它指的是将应用程序所需的依赖关系(如服务或其他组件)通过构造函数参数或属性自动...

TypeScript 熟练度自测:6 道题检验你的 TS 功底!

这些题目既可以测试基本的类型知识,也能考察面向对象编程、泛型、类型推导和高级类型等方面的能力。以下是几个我会出题的方向和具体题目:1.类型推导与基础类型目的:考察应聘者对TypeScript类型推...

怀旧服实用宏整理,猎人篇(怀旧服实用宏整理,猎人篇怎么用)

关于宏的贴子不少,这里我去芜存菁,整理并留下了觉得比较实用的宏命令,希望对大家有用。/m打开宏命令设置窗口宝宝清图腾宏/scriptlocalt,n,i,_={"根基\","...

MS15-083:Windows SMB内存损坏漏洞分析

2015年8月11日微软发布了14个安全补丁,其中就包括一个SMB服务器补丁。在本文我将解释我是如何触发该漏洞的。微软安全公告MS15-083在所有的修复补丁中,我对“服务器消息块中的漏洞可能允许远程...