Skip to content

Latest commit

 

History

History
160 lines (119 loc) · 9.27 KB

Readme.md

File metadata and controls

160 lines (119 loc) · 9.27 KB

Lines of Code Coverage

InMemoryScanner Plugin

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.

Memory Dumps

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.

Technical Overview

The following sections give a detailed overview about how memory is retrieved.

VAD Tree

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.

VAD Entry

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: alt text

Scanning Exceptions

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.

In Depth Example

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

How to Build

  • 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.

Troubleshooting

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

How to Run

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