Contacto Ransomware Analysis
Introduction
After a short break, I’m back with something new—today, we’re diving into a ransomware sample that a friend sent my way. To be honest, this is my first time reversing a ransomware sample, and I have to say, the experience was both exciting and educational. What made it even better is that the sample is quite straightforward, making it perfect for anyone new to reversing ransomware. If you’re looking to learn, this is a great place to start.
The Contacto Ransomware is a relatively new and modern ransomware that uses advanced techniques to evade security measures. It first surfaced in early January this year, and at first glance, it appears to be a derivative or copied version of another ransomware strain.
The analysis will primarily focus on:
→ Initial Setup of the Ransomware
→ Privilege Enabling Mechanism
→ Persistence and Encryption
Ransomware Setup
→ The ransomware starts by retrieving the console window handle using GetConsoleWindow()
and then hides it using ShowWindow()
. This prevents the user from seeing a command prompt window, keeping the ransomware’s execution discreet.
→ Creates a mutex (ContactoMutex
) to ensure only one instance of the ransomware runs. If the mutex already exists (error 0xb7), it exits.
→ Additionally, several core functions are called, which I’ve renamed for better understanding.
Enabling Privileges
→ The ransomware iterates over a list of predefined privileges and calls the SetPrivileges()
function for each privilege.
→ Here’s what each privilege allows:
SeDebugPrivilege
: Enables debugging and access to other processes.SeRestorePrivilege
: Grants permission to restore files and directories.SeBackupPrivilege
: Allows bypassing file and directory permissions during backups.SeTakeOwnershipPrivilege
: Allows the ransomware to take ownership of files or objects.SeAuditPrivilege
: Enables the modification of system audit settings.SeSecurityPrivilege
: Allows access to sensitive security-related operations.SeIncreaseBasePriorityPrivilege
: Grants the ability to increase the base priority of processes.
→ The SetPrivileges()
function enables a specific privilege for the current process by opening the process token, locating the privilege by name, and then enabling it using the AdjustTokenPrivileges
API. It assumes the ransomware is already running with administrator rights.
Ransomware Arguments
-path
Specifies the target path for encryption.-mode
Defines the encryption mode:full
: Encrypts all files completely.fast
: Encrypts only part of each file for speed.split
: Encrypts files in batches or segments.custom
: Allows a custom encryption pattern.
-priority
Sets the ransomware’s process priority for faster execution.-skip
Specifies a number (e.g., file count or size) to skip during encryption.-power
Defines post-encryption system actions:"restart"
: Restarts the system."shutdown"
: Shuts down the system.
-mft
Enables Master File Table (MFT) scanning.-console
Enables or disables console visibility.-nomutex
Disables the mutex check.-nonetdrive
Skips encrypting files on network drives.-nodel
Prevents self-deletion of the ransomware executable after execution.-nowall
Skips changing the desktop wallpaper with a ransom note.-nologon
Disables setting a ransom note on the Windows logon screen.-noblock
Prevents system block actions, such as disabling Task Manager, etc.
Ransomware Initial Phase
After initializing and setting up its environment, the ransomware executes a series of functions to:
- Ensure persistence.
- Disable Windows Defender.
- Encrypt files on the victim’s machine.
Creating Persistence
→ The ransomware creates a fake scheduled task named Windows Update BETA
with SYSTEM privileges to run on every startup.
→ Writes an empty DLL called MNCS.DLL
, acting as a marker.
Disabling Security Features
→ The ransomware manipulates registry keys such as DisableAntiSpyware
and DisableRealtimeMonitoring
under Windows Defender policies to turn off real-time protection.
→ Deletes Volume Shadow Copies using vssadmin
and backup catalogs using wbadmin
.
→ Clears event logs twice: once using ClearEventLogW
and again using wevtutil
commands.
Emptying Recyle Bin
→ The function starts by calling SHEmptyRecycleBinW
to empty the Recycle Bin for the current user. → : It then loops through a predefined list of drive letters and uses GetDriveTypeW
to check if the drive is a fixed drive (type 3) and then finds deleted files might still be stored on each drive and delete them recursively. It’s essentially trying to ensure that no recoverable files are left on the computer.
Mount Volumes
→ The MountVolumes function works to enumerate, check, and handle mounted volumes on the system. It attempts to mount unmounted volumes and associate them with drive letters.
Encryption
Threading
→ Before we understand how the encryption is done. it’s important to understand how the ransomware sets up multithreaded file encryption across the system. It’s essentially the bread and butter of every ransomware so it’s important to understand. I’ll be going through the code in small snippets to understand better.
→ The function determines the number of processors and sets up twice that number of threads for optimal multithreaded encryption. Here we can see that [eax+eax]
refers to the doubling of threads.
→ This part allocates memory for thread handles and creates multiple worker threads using CreateThread. Each thread runs a worker function called StartAddress
.
→ The function uses an IOCP
(CreateIoCompletionPort) for communication between the main thread and worker threads. It waits for worker threads to complete their encryption tasks.In the end it logs a message indicating encryption completion, resets the system’s thread execution state, and cleans up resources like the IOCP handle etc.
Threading Model Used In Ransomware
Start Address
→ To write about a large function like this, we will focus on breaking it into logical sections based on the flow of the function and highlight key operations only. The function starts by increasing the thread priority to ensure smooth processing and resolves the full path of the targeted directory. → The function attempts to open the MFT and enumerate files directly. If this fails, it falls back to the Win32 API for file enumeration. The code further tries to do many things like heap creation, file enumeration again etc until it lands on the main code responsible which sends the file for encryption.
→ The function enumerates files using FindFirstFileW and checks if the file size is greater than zero using nFileSizeHigh
→ SetFilePermissions
is called to modify the file’s access control list (ACL), ensuring the ransomware has sufficient permissions to manipulate the file.
→ EnsureFileAccessibility
handles access errors like ACCESS_DENIED
or SHARING_VIOLATION
, retrying or resolving conflicts to ensure exclusive access.
→ If accessible, the file undergoes encryption (EncryptFile)
and a global counter (_DAT_0044ee38)
is incremented to track the number of encrypted files.
Encrypt File
→ The function renames the file specified in param_1
to include the .Contacto
extension using MoveFileExW
. If renaming fails, it logs an error and exits.Opens the renamed file with CreateFileW
and retrieves its size using GetFileSizeEx
. If the file size is zero or invalid, the handle is closed, and the function exits.
→ Based on the ransomware_mode_var
, the function determines how to encrypt the file as discusses earlier above.
To understand more about this topic of encryption I’d suggest read this. Wonderful article written from 2004
→ The function CryptFunctions
is invoked to generate random cryptographic keys using either:
bcrypt.dll
for modern, secure algorithms.advapi32.dll
as a fallback, utilizing legacy APIs likeCryptAcquireContextW
andCryptGenRandom
→ Two sets of keys are generated:
- One key
(auStack_2c)
with 32 bytes for initializing the encryption process. - Another key derived from a smaller segment (8 bytes) for further operations.
→ The function EncryptDataChunk
applies cryptographic transformations and is the core cryptographic function. It combines the keys generated above with predefined constants like DAT_004465f4
and DAT_00441954
NOTE
If you go through the code you will find that instead of encrypting the entire file, the ransomware encrypts only parts of the file (header, footer, or random segments) to save time while still rendering files unusable even when the mode is full encryption(XD)
Wallpaper Change
→ The function uses GetDeviceCaps()
** to retrieve display parameters and dynamically creates a compatible bitmap using **CreateCompatibleBitmap()
→ It writes the ransom note text (“Contacto Ransomware…” message) onto the bitmap using DrawTextW()
and renders it with a custom Arial font created via CreateFontW()
→ In the end the malware deletes the scheduled task it created earlier and cleans up by self deleting and clearing all the logs again.
Encryption Process Overview
- Dynamic Key Generation
- The encryption process begins with the generation of two cryptographic keys:
- Primary Key (32 bytes): Generated through a hybrid RNG (Random Number Generator) utilizing system entropy and a pseudo-random number generator (PRNG) seeded with unique initialization vectors.
- Secondary Key (8 bytes): Derived from a cascading hash-based PRNG, ensuring no predictable patterns.
- The encryption process begins with the generation of two cryptographic keys:
- Chunk-Based Data Transformation
- File data is processed in adaptive chunk sizes, dynamically adjusted based on file size and system resources.
- Each chunk undergoes a multi-stage XOR operation with the keys, interspersed with bitwise rotations and modular arithmetic to introduce non-linear transformations.
- Key Evolution with SHA-256
- Both keys are hashed using iterative SHA-256 rounds, producing intermediate digests that are integrated into the encryption stream.
- The process includes salt values embedded in the binary, ensuring per-file uniqueness.
- Layered Obfuscation
- The encryption algorithm applies a layered obfuscation strategy by utilizing:
- Key Whitening: Keys are XORed with pre-defined constants before every operation to mask their actual values.
- Permutation Steps: Each chunk undergoes byte-level shuffling based on a precomputed permutation matrix.
- The encryption algorithm applies a layered obfuscation strategy by utilizing:
- Pipeline Optimization for Speed
- The encryption leverages multi-threaded I/O queues, allowing concurrent processing of multiple files.
- A low-level custom instruction set ensures efficient utilization of CPU caches, speeding up large file encryption.
Indicators of Compromise (IOCs)
1
2
3
4
5
6
MD5
f36c5298b988e68aa15f72223a445e6d
SHA-1
c4f497b7fac36733f445f3f72c392ea7cadcde8c
SHA-256
7ec702b0b999799eb6de4c960814ab46c004536c42085e2cf77a516c4b6ed4e3
Thank you for reading this analysis! ❤️
Feel free to connect with me on:
Discord: somedieyoungzz
Twitter: @IdaNotPro