A plugin for VMICore capable of scanning as well as dumping process memory from live VMs.
The InMemoryScanner scans each and every process that is terminated during runtime.
As soon as the shutdown of VMICore is requested the InMemoryScanner also scans all processes which are running at this point, except the ones excluded in the config.
When memory dumping is activated the scanner dumps the scanned memory regions to a file. In addition to that a MemoryRegionInformation.json
is created which contains extra information about all dumped regions.
Both dumping and scanning work on exactly the same data. What you see in the dumps is the same data the scanner operates on.
The content of the MemoryRegionInformation.json
looks like the following:
{
"ProcessName": "abcdefghijklmnop",
"ProcessId": 4,
"SharedMemory": false,
"AccessRights": "RWX",
"StartAddress": "0x1234000",
"EndAddress": "0x1234666",
"BeingDeleted": false,
"ProcessBaseImage": false,
"Uid": 0,
"DumpFileName": "abcdefghijklmn-4-RWX-0x1234000-0x1234666-0"
}
In order to simplify automated analysis of memory dumps each json dictionary references the file containing the corresponding memory region in the DumpFileName
key.
The dump filenames adhere to the following naming schema:
{TruncatedProcessName}-{ProcessId}-{AccessRights}-{StartAddress}-{EndAddress}-{Uid}
The value of ProcessName
is truncated to a maximum length of 14 characters and the Uid
key is simply a counter starting at 0 and increasing by 1 for each scanned memory region.
The following values are valid:
Element | Value |
---|---|
SharedMemory |
false for private memory, or true for example if file backed. |
AccessRights |
Any combination of R (read), W (write) / WC (copy on write), X (execute) as set within the VAD tree. It is noteworthy that Windows sets these values only when a new memory region is created. Therefore these extracted permissions do not necessarily reflect the state upon extraction. |
BeingDeleted |
true , false . If this flag is set to true the memory region was in the process of being deleted at the time of the scan, so its contents may be undefined. |
ProcessBaseImage |
If this flag is set to true the memory region represents the base image of the running process. |
The following sections give a detailed overview about how memory is retrieved.
Windows stores information regarding the memory state of each process in the VAD (Virtual Address Descriptor) tree. For additional information see https://www.codemachine.com/article_protopte.html or the book "What makes it page" http://www.opening-windows.com/wmip/overview.htm.
Each VAD entry represents a coherent memory region. The InMemoryScanner extracts every region and scans it independently of other memory regions. This means that Yara rules which need to be applied to several memory regions at once can never match. Within these memory regions not every page has to actually be present in memory. Therefore, non-mapped page ranges are substituted by a single nulled page in order to avoid false positives during Yara rule matching.
For example the following diagram shows the memory padding of one VAD region consisting of 10 pages where 6 are not mapped. These 6 pages are split over 3 not mapped subregions:
Shared memory regions that are not the base image of the process are skipped by default in order to reduce scanning time.
This behavior can be controlled via the scan_all_regions
config option.
Consider the following VAD entry from the vad tree of a process winlogon.exe
with pid 488
:
Element | Value |
---|---|
ProcessName |
winlogon.exe |
ProcessId |
488 |
MemoryType |
private |
AccessRights |
RW |
BeingDeleted |
false |
StartAddress |
1f43b593000 |
EndAddress |
1f43b599000 |
This region has a size of 0x1f43b599000
- 0x1f43b593000
= 0x6000
.
However, the pages from 0x1f43b596000
to 0x1f43b598000
(size 0x2000
) are not mapped into memory.
Therefore, the resulting files will have the size of 0x5000
(mapped size + one zero-page). Note that the start and end address however are the original ones.
winlogon.exe-488-private-RW-1f43b593000-1f43b599000-BeingDeleted_FALSE
-
Install Build Requirements
- g++ or clang
- cmake
- vcpkg
-
Clone this repository
-
Inside the source directory, run:
[user@localhost source_dir]$ cmake -D PROGRAM_BUILD_NUMBER=<Buildnumber> --preset <gcc/clang>-debug
[user@localhost source_dir]$ cmake --build --preset <gcc/clang>-build-debug
Caveats:
- External headers will only be downloaded if they are missing, so a clean rebuild is advised if updates for those are available.
Yara requires openssl as a dependency which in turn requires a perl distribution in order to run its build scripts. If vcpkg is unable to build yara, make sure to have these perl requirements installed specifically:
- perl-Findbin
- perl-IPC-Cmd
The InMemoryScanner has to be used as a plugin in conjunction with the VMICore project. For this, add the following parts to the VMICore config and tweak them to your requirements:
Parameter | Description |
---|---|
directory |
Path to the folder where the compiled VMICore plugins are located. |
dump_memory |
Boolean. If set to true will result in scanned memory being dumped to files. Regions will be dumped to an inmemorydumps subfolder in the output directory. |
ignored_processes |
List with processes that will not be scanned (or dumped) during the final scan. |
output_path |
Optional output path. If this is a relative path it is interpreted relatively to the VMICore results directory. |
plugins |
Add your plugin here by the exact name of your shared library (e.g. libinmemoryscanner.so ). All plugin specific config keys should be added as sub-keys under this name. |
scan_all_regions |
Optional boolean (defaults to false ). Indicates whether to eagerly scan all memory regions as opposed to ignoring shared memory. |
signature_file |
Path to the compiled signatures with which to scan the memory regions. |
scan_timeout |
Timeout in seconds that determines when libyara will cancel the scan process for a single memory region. |
Example configuration:
---
plugin_system:
directory: /usr/local/lib/
plugins:
libinmemoryscanner.so:
signature_file: /usr/local/share/inmemsigs/sigs.sig
dump_memory: false
scan_all_regions: false
output_path: ""
scan_timeout: 10
ignored_processes:
- SearchUI.exe
- system