Crashing Windows CHM parser in seconds using WinAFL
This article is my submission for https://pagedout.institute #3 (UPD: Finally released December 2023)
One day a friend of mine @xina1i asked on the chat if anyone fuzzed .hlp
files. I quickly looked at it, and realized that .hlp
files don’t exist in Windows 10 anymore, but .chm
files are still here. I’ve double clicked on random .chm
file and started to look around.
hh.exe
which is opened by Explorer is very lightweight program it’s only ~16kb, it also takes path to .chm
file as a parameter, which might be helpful for fuzzing with WinAFL. For now, I’ll just enumerate reversing takeaways.
- hh.exe just loads and calls
doWinMain()
fromhhctrl.ocx
file which is just a regular.dll
file. doWinMain()
performs all the job of parsing.chm
file and it also checks a command line for additional options. We’ll be using-decompile
option, which is intended to extract data from .chm archive without GUI.- We could try to patch functionality which is related to writing files to speed up the fuzzing by just targeting
.chm
parsing functionality.
We should be ready to start fuzzing now. Let’s enable full pageheap for the process and pop up a WinAFL. As an input corpus I just placed smallest .chm
file from my system in the r:\fuzz\in
directory.
1
afl-fuzz.exe -M 0 -i r:\fuzz\in -o r:\fuzz\out -D r:\dr\bin32 -t 3000 -- -coverage_module hhctrl.ocx -target_module hhctrl.ocx -target_method doWinMain -call_convention stdcall -nargs 2 -fuzz_iterations 5000 -- hh.exe -decompile r:\fuzz\out_cmd_m0\ @@
As you can see on the screenshot, the speed is extremely slow (~10execs/sec), but WinAFL was able to find two crashes in 10 minutes!
Here are several patches which you can try to improve the fuzzing speed.
- Nop
UninitializeSession()
call indoWinMain()
in order not to call OLE initialization on every fuzzing iteration. - Nop
CFSClient::WriteStorageContents()
call inside of hhctrl’sDeCompile()
which is responsible for writing extracted files to the disk.
By doing so, you should be able to get the first crash in 5 seconds.
Please note, that hhctrl.ocx
actually calls itss.dll
to parse file itself. So, in order to discover more paths specify itss.dll as -coverage_module
.
I’ve reported 4 cases of memory corruption to msrc, but they replied they won’t be fixing it because .chm
files are generally untrusted and when you open .chm
file it’s literally the same as when you open .exe
file. So, beware!
Here is how crash may look like.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
(5260.483c): Access violation - code c0000005 (first chance)
First chance exceptions are reported before any exception handling.
This exception may be expected and handled.
eax=0a606f58 ebx=00b8e3d0 ecx=0a60b000 edx=01000000 esi=0a60afe8 edi=00000000
eip=7bf95e9c esp=00b8e3b0 ebp=00b8e3c8 iopl=0 nv up ei pl zr na pe nc
cs=0023 ss=002b ds=002b es=002b fs=0053 gs=002b efl=00010246
itss!CPathManager1::CImpIPathManager::ReadCacheBlock+0x87:
7bf95e9c 8139504d474c cmp dword ptr [ecx],4C474D50h ds:002b:0a60b000=????????
0:000> k
# ChildEBP RetAddr
00 00b8e3d0 7bf962e1 itss!CPathManager1::CImpIPathManager::ReadCacheBlock+0x87
01 00b8e3f0 7bf9687c itss!CPathManager1::CImpIPathManager::FindCacheBlock+0x47
02 00b8e418 7bf94ebe itss!CPathManager1::CImpIPathManager::FindKeyAndLockBlockSet+0xad
03 00b8eeb0 7bf8e69d itss!CPathManager1::CImpIPathManager::FindEntry+0x7e
04 00b8f130 7bf8e9c2 itss!CITFileSystem::CImpITFileSystem::OpenLockBytes+0xbd
05 00b8f158 7bf8d34b itss!CITFileSystem::CImpITFileSystem::OpenStream+0x32
06 00b8f188 7bf8d6ea itss!CITFileSystem::CImpITFileSystem::OpenSpaceNameList+0x2e
07 00b8f1f8 7bf8c6cf itss!CITFileSystem::CImpITFileSystem::InitOpenOnLockBytes+0x233
08 00b8f210 7bf8c64c itss!CITFileSystem::OpenITFSOnLockBytes+0x57
09 00b8f230 7bf9ef23 itss!CITFileSystem::OpenITFileSystem+0x8a
0a 00b8f240 7c154595 itss!CWarehouse::CImpIWarehouse::StgOpenStorage+0x13
0b 00b8f480 7c154dbe hhctrl!CFileSystem::Open+0x81
0c 00b8f4b8 7c15715f hhctrl!CFSClient::Initialize+0x69
0d 00b8f56c 7c156a84 hhctrl!DeCompile+0x39