Summary
SALTWATER is a backdoor that has been used in the exploitation of the Barracuda 0-day vulnerability CVE-2023-2868. It is a module for the Barracuda SMTP daemon called bsmtpd. The malware hooked the recv, send, and close functions using an open-source hooking library called funchook. The following functionalities are implemented: execute arbitrary commands, download and upload files, proxy functionality, and tunneling functionality.
Technical analysis
SHA256: 1c6cad0ed66cf8fd438974e1eac0bc6dd9119f84892930cb71cb56a5e985f0a4
The malware implements hooks on the recv, send, and close functions in a method called cc_init, as shown below:
data:image/s3,"s3://crabby-images/32368/32368a12c5f39c54990dab8933ab5b157a3713b8" alt=""
It obtains the address of the original functions using the dlsym method. The function called get_symbol_by_name returns the address (see Figure 2).
data:image/s3,"s3://crabby-images/bd60b/bd60b78a864c32f47f4b17e9bb8ca1482843a9dd" alt=""
As Mandiant mentioned in their report, the binary uses the funchook hooking library to construct the hooks. The following functions are called: funchook_create, funchook_prepare, and funchook_install.
data:image/s3,"s3://crabby-images/fc59e/fc59ee8cc41337313f1e1e1ad0fc05b2f5cc26d5" alt=""
data:image/s3,"s3://crabby-images/7ad06/7ad06314258f1707ebf34c4b7bdfb86694ce16a7" alt=""
We will now describe the hooking functions my_recv, my_send, and my_close.
Firstly, my_recv calls the original recv function, as highlighted in Figure 5:
data:image/s3,"s3://crabby-images/0e979/0e9796d50b14ead258a40469763bd4a4d3065bc4" alt=""
The value stored at the gIsRecvAlready_ptr address should be 0. The process allocates a new memory area that will store the buffer to be received:
data:image/s3,"s3://crabby-images/b5b50/b5b502ee41ed33f2634ed9ac477843810eccba2b" alt=""
The server response is copied to the gConnectedData_ptr address and is overwritten with the “quit\r\n” string. The value stored at gIsRecvAlready_ptr is set to 1, which means that a successful receive operation occurred:
data:image/s3,"s3://crabby-images/b4772/b47725263d798f65fa14f7dfa94b1fd465abcdd9" alt=""
The malware retrieves the IP address of the peer connected to the socket passed as a parameter to the send function using getpeername:
data:image/s3,"s3://crabby-images/3dbdc/3dbdc115fb7b5b405d21786c918b97667e85c1af" alt=""
The original send function is called if any error occurred, as shown below:
data:image/s3,"s3://crabby-images/db634/db63447b34d2a33aa18b5802d60f941edf1c2294" alt=""
The process implements a function called CheckRemoteIp, which converts the IP addresses from binary to text form using inet_ntop and then compares them with the value found at the gConnectedAddr_ptr address. It’s worth mentioning that the socket family is also expected to be 0xA (AF_CCITT), which corresponds to CCITT protocols:
data:image/s3,"s3://crabby-images/37c02/37c02d484d6e1e7434f38ee551b05f67b6fa5310" alt=""
data:image/s3,"s3://crabby-images/1d63c/1d63c48a2a33051663281389308836678a7dfd37" alt=""
In the my_close function, the binary searches for the socket descriptor passed as a parameter in a socket descriptors list:
data:image/s3,"s3://crabby-images/824fb/824fba655acf5ad48f9cf955523b4333bf3431e6" alt=""
The malware compares the socket descriptor with the value stored at gConnectedfd_ptr and expects them to be equal; otherwise, it calls the original close method (Figure 13).
data:image/s3,"s3://crabby-images/0b5ae/0b5aee594477651276faa3462b4a49f13993df37" alt=""
A new thread that executes the cc_worker or Connected2Vps functions is created using the pthread_create method:
data:image/s3,"s3://crabby-images/9729d/9729dd8625b4cbe2e05798c04348d04975fa66ea" alt=""
In the Connected2Vps function, the binary implements a method called OpenConnection, which takes two parameters (IP adddress and port number) that are supposed to indicate the threat actor’s VPS infrastructure:
data:image/s3,"s3://crabby-images/91dd7/91dd7f607fc0c1f37c35115e2500629a344b3931" alt=""
The process obtains a structure of type hostent for the IP address using gethostbyname and the addrinfo structure using the getaddrinfo method:
data:image/s3,"s3://crabby-images/f2b29/f2b2966890f430e8e77fbc2de3e6c2bbcd58de46" alt=""
The binary creates a new socket and connects to the C2 server using the connect method:
data:image/s3,"s3://crabby-images/ac1be/ac1bec0f80fbb76e881b8965b17509e1b8e08a4a" alt=""
The malware sends the buffer found at the gConnectedData_ptr address to the server via a function call to write (see Figure 18).
data:image/s3,"s3://crabby-images/8e587/8e58745adb3abcf00375645b47019af0057e7958" alt=""
The select method is utilized to monitor multiple socket file descriptors, waiting for the read file descriptor to be ready for reading:
data:image/s3,"s3://crabby-images/06c73/06c73f84b8b7005b99726909712ad036a5f3da1d" alt=""
Finally, the process reads the server response using the read method:
data:image/s3,"s3://crabby-images/60012/60012ed7e614ac08831a9d84581182bdc671cc34" alt=""
It creates a new SSL structure and sets the socket descriptor as the input/output for network connections:
data:image/s3,"s3://crabby-images/bce99/bce99579da986f271c71e5787952b6143c4c17e1" alt=""
The malware expects a 21-byte structure that contains the command to be executed. SSL_read is utilized to read data from the connections:
data:image/s3,"s3://crabby-images/0615b/0615b090e2c230c63b1908875df3e3f22298e79c" alt=""
data:image/s3,"s3://crabby-images/dad7f/dad7f55b683185e801f3b93316d65f0f67f45ff5" alt=""
The first byte extracted from the response corresponds to the backdoor functionalities named “Channels”:
- 0 (ShellChannel)
- 1 (DownloadChannel)
- 2 (UploadChannel)
- 3 (ProxyChannel)
- 4 (TunnelArgs)
data:image/s3,"s3://crabby-images/9b9d5/9b9d53089daf628a9e4bc4919e851782f07a04d3" alt=""
ShellChannel
The remaining structure received from the C2 server called SHELL has 20 bytes.
The server can specify a command that will be executed on the infected device:
data:image/s3,"s3://crabby-images/a15ca/a15ca444a911fa037e3a76931d9f42071bdf2935" alt=""
The received command is compared with “exit” and “exit\n”, which means that the process just exits. In any other case, the command is passed to the run_cmd function, as highlighted below:
data:image/s3,"s3://crabby-images/383c5/383c58ccc4653d2372b3545a77aa520416a9adde" alt=""
data:image/s3,"s3://crabby-images/8d05c/8d05c55253a785affda01cc7beaf6121402e7e94" alt=""
The popen function is used to run the desired command on the device, and the output is read using the fgets method (Figure 28).
data:image/s3,"s3://crabby-images/4a5cb/4a5cb36f242a7344ff65bc28844e004df8a436df" alt=""
The malware implements a function called MyWriteAll, which calls the SSL_write method. It sends 4 NULL bytes to the C2 server:
data:image/s3,"s3://crabby-images/ebbcc/ebbcca3874a61376092c018bbf9f0be0cbe425e2" alt=""
data:image/s3,"s3://crabby-images/7836a/7836ad533c014aa5ad3e237ec644e3b1d4174023" alt=""
DownloadChannel
The remaining structure received from the C2 server called TRANSFILE has 20 bytes.
The process allocates 5MB of memory and expects that TRANSFILE[0:4], which is the file name’s length that will be created, to be <= 1022:
data:image/s3,"s3://crabby-images/d9130/d9130ed6bc33928a17fee55990cb5c9e461e4775" alt=""
The file name to be created is read using the MyReadAll function, as shown in Figure 32.
data:image/s3,"s3://crabby-images/f2339/f23390223a96ac4d3ddfeb9a895c117886969a21" alt=""
The open64 routine is used to create the file on the device:
data:image/s3,"s3://crabby-images/f2e9c/f2e9c1225015d4f2944a5a3d94156d06db60a3ab" alt=""
The file is populated with content received from the C2 server using the lseek64 and write functions:
data:image/s3,"s3://crabby-images/ae4e9/ae4e94bb6b70447fa16b3966622d01942821420e" alt=""
UploadChannel
The remaining structure received from the C2 server called TRANSFILE has 20 bytes.
The binary expects that TRANSFILE[4:8], which is the file’s length to be exfiltrated, to be <= 5MB. It calls MyReadAll in order to read the file path that will be exfiltrated to the C2 server:
data:image/s3,"s3://crabby-images/f1ac3/f1ac383ef95bd3bbfcef4271afa9d4e70f0f0cc2" alt=""
The file’s length is obtained using a function called GetFileSize, which uses the stat64 method to retrieve it. The malware opens the target file via a function call to open64:
data:image/s3,"s3://crabby-images/f5b55/f5b559b66b969600bf6411a2c695402dbbd6a91a" alt=""
data:image/s3,"s3://crabby-images/6b5fe/6b5fe5587c07d4756e65b70d9c1b1d1dc60e409d" alt=""
The MyWriteAll function is used again to send 21 bytes to the C2 server, as shown in the figure below.
data:image/s3,"s3://crabby-images/da170/da170c394dc6717e7be73f92b532eb6a39f7f7ef" alt=""
Finally, the target file content is read using the lseek64 and read routines, and then sent to the C2 server:
data:image/s3,"s3://crabby-images/48cf6/48cf6e9bc5e2679c4b1981dd60119d9fc5ec7a86" alt=""
data:image/s3,"s3://crabby-images/f7c46/f7c465fb45db3d097c58e1281957c8602c083d6c" alt=""
ProxyChannel
The remaining structure received from the C2 server called PROXY has 20 bytes.
The process allocates 6MB of memory and expects PROXY[12:16] to be 0, as shown below:
data:image/s3,"s3://crabby-images/70fc4/70fc42e3fbb3885481e8e168742ca89fbd7b42f7" alt=""
The first 4 bytes (PROXY[0:4]) are used to construct the Proxy IP address. The Proxy port number is derived from PROXY[10:12]:
data:image/s3,"s3://crabby-images/95340/9534001455251b1b68f812b446f753da1218c110" alt=""
The OpenConnection function presented above uses the recv method to receive data until a new line is received from the socket (see Figure 43).
data:image/s3,"s3://crabby-images/3000e/3000ed3b0f5762ec5437a51bb86f76baf799f6bc" alt=""
The binary creates a new SSL structure, connects it with the socket descriptor initially passed as parameter to the my_close function, and initiates the TLS/SSL handshake with the Proxy server:
data:image/s3,"s3://crabby-images/363a2/363a2a96fbdd2a3f5077da10df8e1278354046be" alt=""
The process requests a 21-byte structure from the C2 server and sends it to the Proxy server using the MyReadAll and MyWriteAll functions:
data:image/s3,"s3://crabby-images/6299e/6299eb170ff7142328aaa6aa1535b476f34c6e32" alt=""
As we’ve already mentioned, the first byte determines the command to be executed. A special case occurs if the byte is equal to 2, which leads to a call to the DownloadByProxyChannel routine:
data:image/s3,"s3://crabby-images/3f35b/3f35b970811d42f7b4f7d87aab7fdc1414996af3" alt=""
In the DownloadByProxyChannel function, the binary reads a buffer from the C2 server and transmits it to the Proxy server:
data:image/s3,"s3://crabby-images/9eed8/9eed840d5e83eb9ce711a0627bb8888b33da0961" alt=""
Moving forward, the malware can read a 21-byte structure corresponding to a command from the Proxy server and sent it to the C2 server (Figure 48).
data:image/s3,"s3://crabby-images/ff99f/ff99f9e6def28b063e3be70355bbbe10174195a7" alt=""
TunnelArgs
The remaining structure received from the C2 server called TUNNEL has 20 bytes.
The value TUNNEL[16:20] is expected to be different than 1 and TUNNEL[12:16] could be 0, 1, or 2:
data:image/s3,"s3://crabby-images/f2afb/f2afb467037383390ac94c47269fb0ac4ab0f1bf" alt=""
In the first case, the bytes extracted from TUNNEL[0:4] and TUNNEL[8:12] are used to construct two IP addresses, and TUNNEL[4:8] is converted to a port number. The resulting values are stored at the gRemoteAddr_ptr, gRemotePort_ptr, and gConnectedAddr_ptr addresses (Figure 50).
data:image/s3,"s3://crabby-images/29f86/29f8623395dadf811310f0fc677282c535daa744" alt=""
The process sends the command ID (4) and a 20-byte buffer that contains NULL bytes to the C2 server:
data:image/s3,"s3://crabby-images/dd571/dd571e035088539ca250628f56aa4bec32877418" alt=""
In the second case, the binary extracts the IP addresses found at gRemoteAddr_ptr and gConnectedAddr_ptr, and the port number found at gRemotePort_ptr and converts them to integer using the atoi function. The 20-byte buffer that is exfiltrated to the C2 server contains the converted values:
data:image/s3,"s3://crabby-images/fe521/fe5210f38b390335d9a8d92b36d46b6b08d4c3b8" alt=""
data:image/s3,"s3://crabby-images/8adce/8adcea6f332b1f6efd90be2de51d059a6fdccfd4" alt=""
In the third case, the values stored at gRemoteAddr_ptr, gRemotePort_ptr, and gConnectedAddr_ptr are overwritten with NULL bytes (Figure 54).
data:image/s3,"s3://crabby-images/d33e4/d33e48b4945f408cdd4506527731584ecd4c787c" alt=""
References
https://www.mandiant.com/resources/blog/barracuda-esg-exploited-globally