Source code for volatility3.framework.interfaces.context

# 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
#
"""Defines an interface for contexts, which hold the core components that a
plugin will operate upon when running.

These include a `memory` container which holds a series of forest of
layers, and a `symbol_space` which contains tables of symbols that can
be used to interpret data in a layer.  The context also provides some
convenience functions, most notably the object constructor function,
`object`, which will construct a symbol on a layer at a particular
offset.
"""
import copy
from abc import ABCMeta, abstractmethod
from typing import Optional, Union

from volatility3.framework import interfaces


[docs]class ContextInterface(metaclass = ABCMeta): """All context-like objects must adhere to the following interface. This interface is present to avoid import dependency cycles. """ def __init__(self) -> None: """Initializes the context with a symbol_space.""" # ## Symbol Space Functions @property @abstractmethod def config(self) -> 'interfaces.configuration.HierarchicalDict': """Returns the configuration object for this context.""" @property @abstractmethod def symbol_space(self) -> 'interfaces.symbols.SymbolSpaceInterface': """Returns the symbol_space for the context. This object must support the :class:`~volatility3.framework.interfaces.symbols.SymbolSpaceInterface` """ # ## Memory Functions @property @abstractmethod def layers(self) -> 'interfaces.layers.LayerContainer': """Returns the memory object for the context.""" raise NotImplementedError("LayerContainer has not been implemented.")
[docs] def add_layer(self, layer: 'interfaces.layers.DataLayerInterface'): """Adds a named translation layer to the context memory. Args: layer: Layer object to be added to the context memory """ self.layers.add_layer(layer)
# ## Object Factory Functions
[docs] @abstractmethod def object(self, object_type: Union[str, 'interfaces.objects.Template'], layer_name: str, offset: int, native_layer_name: str = None, **arguments): """Object factory, takes a context, symbol, offset and optional layer_name. Looks up the layer_name in the context, finds the object template based on the symbol, and constructs an object using the object template on the layer at the offset. Args: object_type: Either a string name of the type, or a Template of the type to be constructed layer_name: The name of the layer on which to construct the object offset: The address within the layer at which to construct the object native_layer_name: The layer this object references (should it be a pointer or similar) Returns: A fully constructed object """
[docs] def clone(self) -> 'ContextInterface': """Produce a clone of the context (and configuration), allowing modifications to be made without affecting any mutable objects in the original. Memory constraints may become an issue for this function depending on how much is actually stored in the context """ return copy.deepcopy(self)
[docs] def module(self, module_name: str, layer_name: str, offset: int, native_layer_name: Optional[str] = None, size: Optional[int] = None) -> 'ModuleInterface': """Create a module object. A module object is associated with a symbol table, and acts like a context, but offsets locations by a known value and looks up symbols, by default within the associated symbol table. It can also be sized should that information be available. Args: module_name: The name of the module layer_name: The layer the module is associated with (which layer the module lives within) offset: The initial/base offset of the module (used as the offset for relative symbols) native_layer_name: The default native_layer_name to use when the module constructs objects size: The size, in bytes, that the module occupys from offset location within the layer named layer_name Returns: A module object """
[docs]class ModuleInterface(metaclass = ABCMeta): """Maintains state concerning a particular loaded module in memory. This object is OS-independent. """ def __init__(self, context: ContextInterface, module_name: str, layer_name: str, offset: int, symbol_table_name: Optional[str] = None, native_layer_name: Optional[str] = None) -> None: """Constructs a new os-independent module. Args: context: The context within which this module will exist module_name: The name of the module layer_name: The layer within the context in which the module exists offset: The offset at which the module exists in the layer symbol_table_name: The name of an associated symbol table native_layer_name: The default native layer for objects constructed by the module """ self._context = context self._module_name = module_name self._layer_name = layer_name self._offset = offset self._native_layer_name = None if native_layer_name: self._native_layer_name = native_layer_name self.symbol_table_name = symbol_table_name or self._module_name super().__init__() @property def name(self) -> str: """The name of the constructed module.""" return self._module_name @property def offset(self) -> int: """Returns the offset that the module resides within the layer of layer_name.""" return self._offset @property def layer_name(self) -> str: """Layer name in which the Module resides.""" return self._layer_name @property def context(self) -> ContextInterface: """Context that the module uses.""" return self._context
[docs] @abstractmethod def object(self, object_type: str, offset: int = None, native_layer_name: Optional[str] = None, absolute: bool = False, **kwargs) -> 'interfaces.objects.ObjectInterface': """Returns an object created using the symbol_table_name and layer_name of the Module. Args: object_type: The name of object type to construct (using the module's symbol_table) offset: the offset (unless absolute is set) from the start of the module native_layer_name: The native layer for objects that reference a different layer (if not the default provided during module construction) absolute: A boolean specifying whether the offset is absolute within the layer, or relative to the start of the module Returns: The constructed object """
[docs] @abstractmethod def object_from_symbol(self, symbol_name: str, native_layer_name: Optional[str] = None, absolute: bool = False, **kwargs) -> 'interfaces.objects.ObjectInterface': """Returns an object created using the symbol_table_name and layer_name of the Module. Args: symbol_name: The name of a symbol (that must be present in the module's symbol table). The symbol's associated type will be used to construct an object at the symbol's offset. native_layer_name: The native layer for objects that reference a different layer (if not the default provided during module construction) absolute: A boolean specifying whether the offset is absolute within the layer, or relative to the start of the module Returns: The constructed object """
[docs] def get_type(self, name: str) -> 'interfaces.objects.Template': """Returns a type from the module."""
[docs] def get_symbol(self, name: str) -> 'interfaces.symbols.SymbolInterface': """Returns a symbol from the module."""
[docs] def get_enumeration(self, name: str) -> 'interfaces.objects.Template': """Returns an enumeration from the module."""
[docs] def has_type(self, name: str) -> bool: """Determines whether a type is present in the module."""
[docs] def has_symbol(self, name: str) -> bool: """Determines whether a symbol is present in the module."""
[docs] def has_enumeration(self, name: str) -> bool: """Determines whether an enumeration is present in the module."""