CVE-2017-11882及利用樣本分析
1.本文由複眼小組ERFZE師傅原創
2.本文略微偏向基礎,首先介紹了該漏洞的成因,並且分析了該漏洞在蔓靈花,摩訶草,響尾蛇APT組織用於實際攻擊活動中的詳細除錯過程
3.本文全文字數共2234字,圖片95張 預計閱讀時間14分鐘
0x01 漏洞描述
成因:Windows的公式編輯器EQNEDT32.EXE
讀入包含MathType的OLE資料,在拷貝公式字型名稱時沒有對名稱長度進行校驗,使得攻擊者可以通過刻意構造的資料內容覆蓋棧上的函式返回地址,從而劫持程式流程。
影響版本:Microsoft Office 2007 Service Pack 3, Microsoft Office 2010 Service Pack 2, Microsoft Office 2013 Service Pack 1, Microsoft Office 2016
POC:https://github.com/Ridter/CVE-2017-11882
0x02 漏洞分析
筆者復現及分析環境:Windows 7 Service Pack 1、Microsoft Office 2010、x32dbg、IDA 7.0
EQUATION.exe
存在:
data:image/s3,"s3://crabby-images/b8848/b88485a7be69a617a6b323f2b7aca0e3d0afb16c" alt="圖片1 Equation.exe"
設定登錄檔項HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Image File Execution Options\EQNEDT32.EXE
:
data:image/s3,"s3://crabby-images/72686/7268648d3a3a4314ecefdb603c79e3c40f5d82fe" alt="圖片2 regedit.exe"
Debugger
鍵值為x32dbg路徑。
生成POC:
data:image/s3,"s3://crabby-images/d95d3/d95d3e863da13b30d65f471bb97a4da473100b11" alt="圖片3 POC"
開啟該文件,於WinExec()
函式處設斷:
data:image/s3,"s3://crabby-images/5cdeb/5cdebae2e55eb9359d6fde9789ec9c988f8b1408" alt="圖片4 WinExec"
成功斷下後,檢視棧中返回地址:
data:image/s3,"s3://crabby-images/37a9a/37a9a37e8e103e18ece79790b65dc96eba938a53" alt="圖片5 ReturnAdd"
繼續向上檢視棧,發現呼叫WinExec()
的函式:
data:image/s3,"s3://crabby-images/c6da9/c6da94e7c018c6d6d76e0ffa29ac52abe91d578b" alt="圖片6 Stack"
通過IDA分析sub_4115A7
功能:
data:image/s3,"s3://crabby-images/e93e2/e93e266b7198a0d07f8fb5698d4320567706eba3" alt="圖片7 IDA"
跟進sub_41160F
檢視:
data:image/s3,"s3://crabby-images/ab796/ab7961765e1b87908fbfce6a94381b9878835341" alt="圖片8 sub41160F"
未校驗長度,直接使用strcpy()
函式,此處應該就是漏洞觸發位置。進一步確定具體位置:
data:image/s3,"s3://crabby-images/1737f/1737f92fbee76569f1261b83d630c799b8732c3d" alt="圖片9 strcpy"
於0x411658
處設斷,重新執行。第二次成功斷下後,檢視ESI暫存器指向記憶體內容:
data:image/s3,"s3://crabby-images/76ac8/76ac81d2ad5291f983c2e50b73ff6e5ab2360cd5" alt="圖片10 ESI"
此時ECX暫存器值為0xC,即複製48個位元組到EDI暫存器指向記憶體,而var_28
實際大小隻有36個位元組:
data:image/s3,"s3://crabby-images/5af7a/5af7ac1d5afa23891778d7b3b4e3c4cc0a2f2db5" alt="圖片11 EDI"
到達函式結束處:
data:image/s3,"s3://crabby-images/b3533/b35334732d88a020292f89250a5cf381909e9ce5" alt="圖片12 FunEnd"
leave
指令執行完畢後,棧頂0x18F1D0
處值為0x430C12
,即呼叫WinExec()
。而傳遞引數正是0x18F350
指向記憶體中的cmd指令:
data:image/s3,"s3://crabby-images/d2d4d/d2d4dceae93eacf1ce88c1f89f90f7402881e868" alt="圖片13 shellcode"
成功彈出計算器:
data:image/s3,"s3://crabby-images/454a3/454a3027643b6b2742395391b4d841bcc474e9e5" alt="圖片14 calc.exe"
下面對使用到的POC進行簡要分析。各變數含義由命名可知,RTF文件格式並非本文重點,如讀者此前對RTF文件格式沒有了解,建議先閱讀文末參考連結中有關RTF文件格式的文章後再看POC原始碼。
data:image/s3,"s3://crabby-images/3745c/3745ca542aac7a03f646cfb80c1e24984d0c66a1" alt="圖片15 RTF"
首先判斷命令長度是否小於43,而43這個數字是因為:
data:image/s3,"s3://crabby-images/f82da/f82da982ca0859b8551f73603e5ad34e8aba2748" alt="圖片16 CmdLen"
上圖選中部分是插入命令處,具體偏移由POC中COMMAND_OFFSET(0x949*2)
變數給出。
將命令插入到構造資料中之後,函式返回拼接好的OLE。下面將OLE嵌入到RTF文件中:
data:image/s3,"s3://crabby-images/1e6d7/1e6d7c2e7812acb6126f50d8881fcc9e15d9c73f" alt="圖片17 OLE"
0x03 摩訶草(APT-C-09)組織某樣本分析
MD5:0D38ADC0B048BAB3BD91861D42CD39DF
於0x411658
處設斷,在第二次斷下時,各暫存器值如下:
data:image/s3,"s3://crabby-images/ce374/ce3743a1742ae30763c9c71bc66687b7849e6be0" alt="圖片18 register"
繼續執行到函式結束處leave
指令:
data:image/s3,"s3://crabby-images/1e72c/1e72ce934764b9104e78713532de80fa4b1db27f" alt="圖片19 leave"
0x18F230
地址處值0x430C47
即覆蓋後的函式返回地址:
data:image/s3,"s3://crabby-images/e332c/e332c0dc942bb3e37bd3f645a23eb64694905f13" alt="圖片20 FunReturn"
而該地址處指令是ret
,有些出乎意料。繼續向下執行,來到0x18F3B0
處,正是0x18F234
地址處值:
data:image/s3,"s3://crabby-images/d8fc3/d8fc31b9d6beb48e6f38fe31b554fbd02ccf01cb" alt="圖片21 ret"
這方才是構造者意欲執行的指令。經過藍色方框中的一系列運算後,EBX指向是真正的Shellcode:
data:image/s3,"s3://crabby-images/14377/14377be4cec6b704c13e7f800f5248557d6d3432" alt="圖片22 shellcode"
上述內容均可在OLE中檢視(路徑\xl\embeddings
):
data:image/s3,"s3://crabby-images/24066/24066018fb919306344831a1c7fc12970de830aa" alt="圖片23 OLE"
data:image/s3,"s3://crabby-images/9b018/9b01838e27f329e476cf5a1a07c3bb6ab62c84fa" alt="圖片24 OLE"
將OLE0x1000
—0x1520
中資料複製到一bin檔案後,通過IDA檢視。sub_247
功能如下:
data:image/s3,"s3://crabby-images/8aa8a/8aa8a79733b4040d2e074b17fcb1ff91bfa2e404" alt="圖片25 IDA"
該函式接受的第二個引數即上文提到的EBX指向地址,於OLE中位置是0x1040
,而0x1040
+0x558
處內容如下:
data:image/s3,"s3://crabby-images/eb620/eb620b828d7a904330c6b97a8753469a30cd1822" alt="圖片26 PE"
故該函式第一個功能是修正PE檔案頭。第二個功能流程如下:
data:image/s3,"s3://crabby-images/dcf2a/dcf2a2d7f3ee637e5990c605dcff6a032c0e981a" alt="圖片27 PEWrite"
data:image/s3,"s3://crabby-images/a002f/a002fe29fbc26d10a96e317ea4b4cb4bd1fa312e" alt="圖片28 PEWrite"
data:image/s3,"s3://crabby-images/e9238/e92381a74ae756b6488f4281faa852b6a2bd21a5" alt="圖片29 PEWrite"
data:image/s3,"s3://crabby-images/a6e7d/a6e7d6e986a570c9ea26edbeed44710d51e9794e" alt="圖片30 PEWrite"
將0x1040
+0x558
後的PE檔案資料寫入到%APPDATA%\MSBuild.exe
中。第三個功能流程如下:
data:image/s3,"s3://crabby-images/0b7f3/0b7f31dc6976733bb89c66b56f8cde93d55d5c99" alt="圖片31 RegeditWrite"
data:image/s3,"s3://crabby-images/cfb38/cfb38bca23ea5cb103572898467c6fed8817f14a" alt="圖片32 RegeditWrite"
data:image/s3,"s3://crabby-images/b28ae/b28ae123162fe1c71190524245fa733702ff481b" alt="圖片33 RegeditWrite"
將%APPDATA%\MSBuild.exe
寫入登錄檔run項鍵值lollipop
中。
0x04 響尾蛇(SideWinder)組織某樣本分析
將文件拖進WinHex檢視:
data:image/s3,"s3://crabby-images/9f0b2/9f0b215aa27e85ca4c51dae00ba2b8e49fd49d4f" alt="圖片34 WinHex"
可以看出該文件實質是一RTF格式文件。
用rtfobj.py
分析如下:
data:image/s3,"s3://crabby-images/bef54/bef548fbf33d89361279b10acea62cb01a8744de" alt="圖片35 rtfobj"
Package後文會提到,先來看其CVE-2017-11882利用部分。
同樣是第二次斷下時:
data:image/s3,"s3://crabby-images/33ab7/33ab7b517ecf5c77b594f41b02e4968bb06a9f27" alt="圖片36 break"
其後的執行流程與上一樣本相似:
data:image/s3,"s3://crabby-images/f5bbe/f5bbe9b1782f64c2cbc6f373fb4f99cb60676290" alt="圖片37 leave"
data:image/s3,"s3://crabby-images/d93f0/d93f01a4408ed2347dac3182b2fab7768c994079" alt="圖片38 ret"
經過綠色方框中的一系列運算後,呼叫GlobalLock()
函式,傳遞引數如下:
data:image/s3,"s3://crabby-images/a1d05/a1d0548a3d939cceff705ab7a683a6f152f495b0" alt="圖片39 GlobalLock"
接下來跳轉到GlobalLock()
函式返回記憶體區域中:
data:image/s3,"s3://crabby-images/b9b39/b9b39bc9a2f114992fd647d322ba598308fb1f11" alt="圖片40 jmp"
經過兩次call
呼叫:
data:image/s3,"s3://crabby-images/70fc0/70fc0b53c9d8d5959eaf1d08fbc63eb6afcac9d8" alt="圖片41 call"
data:image/s3,"s3://crabby-images/0e411/0e411f3b79f175c1d083675fdc9469f6002fc3e1" alt="圖片42 call"
修正記憶體中的字串:
data:image/s3,"s3://crabby-images/e4a0c/e4a0c9a90c405cd9ee8e504c44cf2ff5dd704781" alt="圖片43 EditString"
接下來定址kernel32.dll
:
data:image/s3,"s3://crabby-images/d3570/d3570fcdf9c7fc000e8004e2a6aeb90b80fbca04" alt="圖片44 kernel32"
其所呼叫的函式功能如下:
data:image/s3,"s3://crabby-images/2d2c5/2d2c5f275e488f479989b53fe8dff4f2ac9be4f0" alt="圖片45 sub298"
兩次call
呼叫之後:
data:image/s3,"s3://crabby-images/7bfc2/7bfc2498916ec2d3ad2d1ee7d240bf12323e36bf" alt="圖片46 call"
data:image/s3,"s3://crabby-images/be1aa/be1aa83ea3c9b70beb0465a8a5f3f6ff331970a0" alt="圖片47 call"
其功能為返回某函式呼叫地址,此次是LoadLibrayW()
:
data:image/s3,"s3://crabby-images/87143/8714334232e807fad5f919821db278bb94af2f66" alt="圖片48 LoadLibrary"
data:image/s3,"s3://crabby-images/f0717/f071732222823af25447f310d83df3286cdac36d" alt="圖片49 Loadlibrary"
接下來,返回GetProcAddress()
呼叫地址:
data:image/s3,"s3://crabby-images/8ff8b/8ff8b6058c841729c7a72e6613dd33a93bf386c6" alt="圖片50 GetProcAddress"
data:image/s3,"s3://crabby-images/346cb/346cb10374b2ce8f04ee123dab838a901228c816" alt="圖片51 GetProcAddress"
繼續call
呼叫:
data:image/s3,"s3://crabby-images/1af74/1af7449ed2e1721d37b99f720ed744f3d65abd40" alt="圖片52 call"
其後流程如圖所示:
data:image/s3,"s3://crabby-images/58487/584871a0f7fd96852fb26d014c07f061e841e04c" alt="圖片53 GetCommandLine"
data:image/s3,"s3://crabby-images/29de0/29de08c2ab3cf1ccc4dd4ebd1c978c332653ade0" alt="圖片54 GetCommandLine"
data:image/s3,"s3://crabby-images/0ee78/0ee78804e5db80a189e715998fce8e342d2a8018" alt="圖片55 call"
下面將字串解密,並覆蓋原CommandLine內容:
data:image/s3,"s3://crabby-images/7ff2d/7ff2d2c5420e10be051c5f172f85946fa7de7ac6" alt="圖片56 DecryptStr"
data:image/s3,"s3://crabby-images/174b8/174b8429950f066181da5a4b1dc47f536482ebe0" alt="圖片57 DecryptStr"
執行完結果如下:
data:image/s3,"s3://crabby-images/d4518/d4518a31636a2b25c9fcda2788024ead11ab9407" alt="圖片58 StrResult"
最後實際執行部分:
javascript:eval("sa=ActiveXObject;ab=new sa(\"Scripting.FileSystemObject\");
eval(ab.OpenTextFile(ab.GetSpecialFolder(2)+\"\\\\1.a\",1).ReadAll());windowclose()")
其後呼叫RunHTMLApplication()
:
data:image/s3,"s3://crabby-images/b373e/b373efb7bd8cb78af78dbe47fcd3f5ab9766429b" alt="圖片59 RunHTMLApplicaton"
data:image/s3,"s3://crabby-images/f82e7/f82e71e46d54e48abc7ba3816798bf5cf14ca40c" alt="圖片60 RunHTMLApplicaton"
data:image/s3,"s3://crabby-images/9b2fe/9b2fef57e2be5e8550a239a73f868846d9795588" alt="圖片61 RunHTMLApplicaton圖片60 RunHTMLApplicaton圖片60 RunHTMLApplicaton"
data:image/s3,"s3://crabby-images/162f9/162f9eaa531e950faa51a9221dc031f2f0415f33" alt="圖片62 RunHTMLApplicaton圖片60 RunHTMLApplicaton"
data:image/s3,"s3://crabby-images/17d7c/17d7c67e9c592fe13b1cbebdeb43bfda92930817" alt="圖片63 RunHTMLApplicaton"
1.a就是之前提到RTF文件中的Package,其實質是一JS檔案:
data:image/s3,"s3://crabby-images/3ad8c/3ad8c86bc5c7df9e3f320585fc3c184c4696e0dc" alt="圖片64 JS"
data:image/s3,"s3://crabby-images/97dd9/97dd90d90ec18243e09f05ca1d9c6982ee5f636b" alt="圖片65 JS"
最後,其執行結果大體如下圖所示:
data:image/s3,"s3://crabby-images/d7bdd/d7bddea9a3a8597474c427ae6bd54393a8ce6c13" alt="圖片66 result"
0x05 蔓靈花(Bitter)組織某樣本分析
通過遠端模板注入的方式下載一RTF格式文件:
data:image/s3,"s3://crabby-images/226ac/226ac49df4eb8ffe4cdc3a42b40e588d57ff168b" alt="圖片67 downloadRTF"
拖進WinHex檢視,可以確認其格式為RTF文件格式:
data:image/s3,"s3://crabby-images/c6ee8/c6ee8514d60f33958c0f1647e244fd82ff0d878d" alt="圖片68 Winhex"
新增副檔名後,開啟該文件。同樣是于于0x411658
處第二次斷下時:
data:image/s3,"s3://crabby-images/893fd/893fd16610fab069bf83d8eb7d1a51efe86a8df1" alt="圖片69 breakpoint"
data:image/s3,"s3://crabby-images/07834/07834b9f2b2ce98794c50e1390d9b1d020c9f197" alt="圖片70 ret"
data:image/s3,"s3://crabby-images/a82b5/a82b532b6b1875b967f16625b2511f757642806c" alt="圖片71 shellcode"
跳轉之後經過綠色方框中一系列計算,接著跳轉:
data:image/s3,"s3://crabby-images/dddec/dddec079a08bf4bf93422277b68e4295e57fe012" alt="圖片72 jmp"
fldpi
將π的值載入到FPU堆疊:
data:image/s3,"s3://crabby-images/9fbd8/9fbd889f061b570d0502d089df40714e308fbcad" alt="圖片73 fldpi"
執行完後fpu_instruction_pointer
指向fldpi
指令,其後的fnstenv
指令將FpuSaveState
結構體儲存到esp-0xC
處:
data:image/s3,"s3://crabby-images/27035/27035c61357e3467be0eda5f4b1fceb341c5794b" alt="圖片74 fnstenv"
如此一來,pop ebp
後EBP暫存器的值是fpu_instruction_pointer
——fldpi
指令位置:
data:image/s3,"s3://crabby-images/90c16/90c1699e5bcc142926efea950d1c771890a96fd1" alt="圖片75 EBP"
由EBP計算出需要解密的資料起始位置,EDX中儲存的是資料長度(0x315):
data:image/s3,"s3://crabby-images/34088/34088cceafbe0115a5e86b2e502d97b5de7f402a" alt="圖片76 Decrypt"
接著執行解密後的指令:
data:image/s3,"s3://crabby-images/5a503/5a503876f0ef4d063fd15f8255fc1c9cb6bf95b3" alt="圖片77 Execute"
data:image/s3,"s3://crabby-images/2d74b/2d74b315dd621d96ffe6f50f24ee040bb91606b0" alt="圖片78 Execute"
跳轉後,執行相應指令,接下來call
呼叫:
data:image/s3,"s3://crabby-images/f9e5f/f9e5f399b4c41b4f9385e1f58bb439cfd10a063d" alt="圖片79 call"
sub_562B2F
功能是獲取指定的系統函式呼叫地址,此次是kernel32.VirtualAlloc()
:
data:image/s3,"s3://crabby-images/e6819/e6819f4dfd4e5b6c204506d472aa864cf17fefa9" alt="圖片80 ReturnVirtualAddr"
data:image/s3,"s3://crabby-images/9fb0e/9fb0e75bad92a10581958f9d033e5874808f6029" alt="圖片81 ReturnVirtualAddr"
之後呼叫VirtualAlloc()
申請記憶體空間:
data:image/s3,"s3://crabby-images/e1f25/e1f259579d0c3032b5a4b5d5499bf8a7ea30784b" alt="圖片82 VirtualAlloc"
向申請的記憶體空間中寫入資料:
data:image/s3,"s3://crabby-images/710f6/710f655633df755162446ae2f664d882647f570e" alt="圖片83 WriteMem"
呼叫sub_562B2F
獲取kernel32.Wow64DisableWow64FsRedirection()
呼叫地址:
data:image/s3,"s3://crabby-images/48581/485812b7f3a31d441e3e98ea8ae5030fd191f19f" alt="圖片84 ReturnWow64DisableWow64FsRedirectionAddr"
LoadLibrary(shell32)
:
data:image/s3,"s3://crabby-images/0abd6/0abd66b8c29fde5a50a3d886e78131f469e44d19" alt=""
傳遞引數給sub_562B2F
,獲取shell32.ShellExcute()
呼叫地址:
data:image/s3,"s3://crabby-images/e8e96/e8e96fdd0590b1fd4d062139f67f9890a0bfeac0" alt="圖片86 ReturnShellExcuteAddr"
data:image/s3,"s3://crabby-images/87308/87308bdf49134e062f5c273d4fdcc10bd0d0b47e" alt="圖片87 ReturnShellExcuteAddr"
LoadLibrary(urlmon)
:
data:image/s3,"s3://crabby-images/a91f0/a91f007f9b93cbd34910c6d7fe1e0f4d7ba10ae1" alt=""
獲取urlmon.URLDownloadToFile()
呼叫地址:
data:image/s3,"s3://crabby-images/2758f/2758f508d0e22e4d546108c0126afdd579d0b1fa" alt="圖片89 ReturnURLDownloadToFileAddr"
data:image/s3,"s3://crabby-images/ac75d/ac75de4e49a80b7cc1eabb4fa3bf52c84c810fce" alt="圖片90 ReturnURLDownloadToFileAddr"
呼叫URLDownloadToFile()
,其傳遞引數如圖:
data:image/s3,"s3://crabby-images/09c9e/09c9e8e8391fcbb866a3cbd51ab7453aa7703c5d" alt="圖片91 URLDownloadToFile"
data:image/s3,"s3://crabby-images/23f60/23f602b77821c2d40e95ca87b185d5e1625f7824" alt="圖片92 URLDownloadToFile"
讀取檔案:
data:image/s3,"s3://crabby-images/b38b7/b38b79418b63227796703d3145eca710f71f2513" alt="圖片93 CreateFile"
data:image/s3,"s3://crabby-images/3008e/3008e1da5ac02f3e20e95a14793997e698a63ad9" alt="圖片94 ReadFile"
由於沒有獲取到檔案,計算出的EBX值錯誤:
data:image/s3,"s3://crabby-images/541b3/541b381388c105c012b6bf63b1bccb95b398e4af" alt="圖片95 End"
故至此結束。
0x06 參考連結
Office惡意檔案解析與混淆研究 -https://zhuanlan.zhihu.com/p/31345299
https://github.com/Ridter/CVE-2017-11882
Office檔案格式基礎知識 -https://www.anquanke.com/post/id/175548