Source code for volatility3.plugins.windows.driverscan

# This file is Copyright 2019 Volatility Foundation and licensed under the Volatility Software License 1.0
# which is available at https://www.volatilityfoundation.org/license/vsl-v1.0
#

from typing import Iterable, Optional, Tuple

from volatility3.framework import renderers, interfaces, exceptions
from volatility3.framework.configuration import requirements
from volatility3.framework.renderers import format_hints
from volatility3.plugins.windows import poolscanner, modules


[docs] class DriverScan(interfaces.plugins.PluginInterface): """Scans for drivers present in a particular windows memory image.""" _required_framework_version = (2, 0, 0) _version = (2, 0, 0)
[docs] @classmethod def get_requirements(cls): return [ requirements.ModuleRequirement( name="kernel", description="Windows kernel", architectures=["Intel32", "Intel64"], ), requirements.VersionRequirement( name="poolscanner", component=poolscanner.PoolScanner, version=(3, 0, 0) ), requirements.VersionRequirement( name="modules", component=modules.Modules, version=(3, 0, 0) ), ]
[docs] @classmethod def scan_drivers( cls, context: interfaces.context.ContextInterface, kernel_module_name: str, ) -> Iterable[interfaces.objects.ObjectInterface]: """Scans for drivers using the poolscanner module and constraints. Args: context: The context to retrieve required elements (layers, symbol tables) from layer_name: The name of the layer on which to operate symbol_table: The name of the table containing the kernel symbols Returns: A list of Driver objects as found from the `layer_name` layer based on Driver pool signatures """ kernel = context.modules[kernel_module_name] constraints = poolscanner.PoolScanner.builtin_constraints( kernel.symbol_table_name, [b"Dri\xf6", b"Driv"] ) driver_start_offset = kernel.get_type("_DRIVER_OBJECT").relative_child_offset( "DriverStart" ) kernel_space_start = modules.Modules.get_kernel_space_start( context, kernel_module_name ) for result in poolscanner.PoolScanner.generate_pool_scan( context, kernel_module_name, constraints ): _constraint, mem_object, _header = result scanned_layer = context.layers[mem_object.vol.layer_name] # *Many* _DRIVER_OBJECT instances were found at the end of a page # leading to member access causing backtraces across several plugins # when members were accessed as the next page was paged out. # `DriverStart` is the first member from the beginning of the structure # of interest to plugins, so if it is not accessible then this instance # is not useful or usable during analysis # 8 covers this value 32 and 64 bit systems if scanned_layer.is_valid(mem_object.vol.offset + driver_start_offset, 8): # Many/most rootkits zero out their DriverStart member for anti-forensics # so we accept a driver start that is either 0 or is points into kernel memory (the current layer) if ( mem_object.DriverStart == 0 or mem_object.DriverStart > kernel_space_start ): yield mem_object
[docs] @classmethod def get_names_for_driver( cls, driver ) -> Tuple[Optional[str], Optional[str], Optional[str]]: """ Convenience method for getting the commonly used names associated with a driver Args: driver: A Driver object Returns: A tuple of strings of (driver name, service key, driver alt. name) """ try: driver_name = driver.get_driver_name() except (ValueError, exceptions.InvalidAddressException): driver_name = None try: service_key = driver.DriverExtension.ServiceKeyName.String except exceptions.InvalidAddressException: service_key = None try: name = driver.DriverName.String except exceptions.InvalidAddressException: name = None return driver_name, service_key, name
def _generator(self): for driver in self.scan_drivers( self.context, self.config["kernel"], ): driver_name, service_key, name = self.get_names_for_driver(driver) # Prior to #1481, this plugin reported dozens to hundreds of junk drivers per sample if not driver_name and not service_key and not name: continue yield ( 0, ( format_hints.Hex(driver.vol.offset), format_hints.Hex(driver.DriverStart), format_hints.Hex(driver.DriverSize), service_key or renderers.NotAvailableValue(), driver_name or renderers.NotAvailableValue(), name or renderers.NotAvailableValue(), ), )
[docs] def run(self): return renderers.TreeGrid( [ ("Offset", format_hints.Hex), ("Start", format_hints.Hex), ("Size", format_hints.Hex), ("Service Key", str), ("Driver Name", str), ("Name", str), ], self._generator(), )