The MemoryDLL unit provides advanced functionality for loading dynamic-link libraries (DLLs) directly from memory in Win64 environments. Unlike traditional methods that involve loading DLLs from the file system, MemoryDLL allows you to load DLLs from byte arrays π or memory streams πΎ, retrieve function addresses, and unload themβall in-memory. This library is ideal for Delphi/FreePascal developers who need to manage DLLs without relying on the filesystem, enhancing both performance β‘ and security π.
- LoadLibrary: Loads a DLL from a memory buffer without writing to the disk π½.
- FreeLibrary: Unloads the DLL from memory, ensuring all associated resources are properly released π.
- GetProcAddress: Retrieves the address of an exported function within the loaded DLL, enabling direct function calls π.
- Comprehensive Error Handling: Manages issues such as invalid DLL data π«, memory allocation failures π¨, and function resolution issues π§.
- Increased Security π: By eliminating the need to store DLLs on disk, MemoryDLL reduces the risk of DLL hijacking and unauthorized access.
- Performance Improvement β‘: Since DLLs are handled in-memory, the overhead of disk I/O operations is avoided, resulting in faster execution π.
- Flexibility π€Ή: Suitable for embedding DLLs in the main executable, loading encrypted π or obfuscated DLLs, and supporting dynamic plugin systems where plugins are provided as in-memory modules.
MemoryDLL uses low-level Windows API functions to manage DLLs entirely from memory:
- Manual Mapping πΊοΈ: The unit manually parses the DLL's Portable Executable (PE) format, maps its sections, and relocates addresses.
- Relocation Handling π: Handles relocation entries to ensure the DLL functions correctly at its loaded memory address.
- Import Table Resolution π: Resolves all imported symbols, making the DLL fully functional once loaded.
Compatibility π€: This unit is designed to be compatible with standard DLL interfaces, making it easy to integrate with existing applications. Additionally, security best practices are incorporated to prevent vulnerabilities like code injection π.
- Embed DLLs directly within your executable. MemoryDLL allows you to store DLLs as resources or encrypted data and load them into memory at runtime, removing the need to distribute them as separate files.
- Enhance application security by storing DLLs in an encrypted form, which can then be decrypted into memory before loading with MemoryDLL. This reduces the risk of reverse engineering.
- Load plugins dynamically as in-memory DLLs. This approach provides a clean and secure method of extending application functionality without relying on the filesystem.
Loads a module from a memory image, mimicking the behavior of the Windows API LoadLibrary
function. It parses the PE format, performs necessary relocations, resolves imports, and initializes the module.
- Parameters:
Data: Pointer
β A pointer to the memory image conforming to the PE format. - Returns:
THandle
representing the loaded module or0
on failure.
Frees a module loaded with LoadLibrary
. It releases all memory and resources associated with the DLL, ensuring clean detachment.
- Parameters:
Module: THandle
β Handle of the module to be freed.
Retrieves the address of an exported function from a loaded DLL, similar to the Windows API GetProcAddress
function.
- Parameters:
Module: THandle
β Handle to the loaded module.Name: PAnsiChar
β Name of the function to retrieve.
- Returns:
Pointer
to the function, ornil
if not found.
To successfully integrate MemoryDLL into your project, please follow these steps:
-
Download the Latest Version π₯
- Visit the official MemoryDLL repository and download the latest release.
-
Unzip the Package π
- Once the download is complete, extract the contents of the zip file to a convenient location on your device's filesystem. The extracted folder should contain the MemoryDLL source code, documentation, and any necessary dependencies.
-
Add MemoryDLL to Your Project β
- Add MemoryDLL to your project's
uses
section. This inclusion will make the MemoryDLL unit available for use in your application. Ensure that the path to the MemoryDLL source file is correctly configured in your project settings to avoid compilation errors.
- Add MemoryDLL to your project's
-
Integration as a Drop-in Replacement π
- The MemoryDLL unit is designed to serve as a drop-in replacement for the Windows API functions
LoadLibrary
,FreeLibrary
, andGetProcAddress
. Simply replace existing calls to these API functions with their MemoryDLL counterparts (LoadFromMemory
,FreeModule
,GetFunctionAddress
). This seamless substitution allows you to benefit from in-memory DLL management without major code modifications.
- The MemoryDLL unit is designed to serve as a drop-in replacement for the Windows API functions
-
Test the Integration β
- It is recommended to thoroughly test your project after integrating MemoryDLL to ensure that all DLLs are being correctly loaded, utilized, and unloaded from memory. Given the in-memory nature of this library, testing will help identify any potential issues related to memory management or function resolution.
- Created/tested with Delphi 12.2, on Windows 11, 64-bit (latest version)
To instantiate MemoryDLL, include the following code at the end of the unitβs implementation section. This code attempts to load the DLL as an embedded resource:
uses
Windows,
MemoryDLL;
...
implementation
{
This code is an example of using MemoryDLL to load an embedded a DLL directly
from an embedded resource in memory, ensuring that no filesystem access is
required. It includes methods for loading, initializing, and unloading the
DLL. The DLL is loaded from a resource with a GUID name, providing a measure
of security by obfuscating the resourceβs identity.
Variables:
- DLLHandle: THandle
- A handle to the loaded DLL. Initialized to 0, indicating the DLL has not been loaded.
It is updated with the handle returned from LoadLibrary when the DLL is successfully
loaded from memory.
Functions:
- LoadDLL: Boolean
- Loads the DLL from an embedded resource and initializes it by retrieving necessary
exported functions. Returns True if the DLL is loaded successfully, otherwise False.
- b6eb28fd6ebe48359ef93aef774b78d1: string
- A GUID-named helper function that returns the resource name for the DLL.
This GUID-like name helps avoid easy detection of the resource.
- UnloadDLL: procedure
- Unloads the DLL by freeing the library associated with DLLHandle. Resets DLLHandle
to 0 to indicate the DLL is unloaded.
Initialization:
- The LoadDLL function is called during initialization, and the program will terminate
with code 1 if the DLL fails to load.
Finalization:
- The UnloadDLL procedure is called upon finalization, ensuring the DLL is unloaded
before program termination.
}
var
DLLHandle: THandle = 0; // Global handle to the loaded DLL, 0 when not loaded.
{
LoadDLL
--------
Attempts to load a DLL directly from a resource embedded within the executable file.
This DLL is expected to be stored as an RCDATA resource under a specific GUID-like name.
Returns:
Boolean - True if the DLL is successfully loaded, False otherwise.
}
function LoadDLL(): Boolean;
var
LResStream: TResourceStream; // Stream to access the DLL data stored in the resource.
{
b6eb28fd6ebe48359ef93aef774b78d1
---------------------------------
Returns the name of the embedded DLL resource. Uses a GUID-like name for obfuscation.
Returns:
string - The name of the resource containing the DLL data.
}
function b6eb28fd6ebe48359ef93aef774b78d1(): string;
const
CValue = 'b87deef5bbfd43c3a07379e26f4dec9b'; // GUID-like resource name for the embedded DLL.
begin
Result := CValue;
end;
begin
Result := False;
// Check if the DLL is already loaded.
if DLLHandle <> 0 then Exit;
// Ensure the DLL resource exists.
if not Boolean((FindResource(HInstance,
PChar(b6eb28fd6ebe48359ef93aef774b78d1()), RT_RCDATA) <> 0)) then Exit;
// Create a stream for the DLL resource data.
LResStream := TResourceStream.Create(HInstance, b6eb28fd6ebe48359ef93aef774b78d1(), RT_RCDATA);
try
// Attempt to load the DLL from the resource stream.
DLLHandle := LoadLibrary(LResStream.Memory);
if DLLHandle = 0 then Exit; // Loading failed.
// Retrieve and initialize any necessary function exports from the DLL.
GetExports(DLLHandle);
Result := True; // Successful load and initialization.
finally
LResStream.Free(); // Release the resource stream.
end;
end;
{
UnloadDLL
---------
Frees the loaded DLL, releasing any resources associated with DLLHandle, and resets DLLHandle to 0.
}
procedure UnloadDLL();
begin
if DLLHandle <> 0 then
begin
FreeLibrary(DLLHandle); // Unload the DLL.
DLLHandle := 0; // Reset DLLHandle to indicate the DLL is no longer loaded.
end;
end;
initialization
// Attempt to load the DLL upon program startup. Halt execution with error code 1 if it fails.
if not LoadDLL() then
begin
Halt(1);
end;
finalization
// Ensure the DLL is unloaded upon program termination.
UnloadDLL();
This project is based on the original Delphi MemoryModule project by Fr0sT-Brutal. We gratefully acknowledge the foundational work that this unit builds upon.
This project is licensed under the BSD-3-Clause License π, which allows for redistribution and use in source and binary forms, with or without modification, provided that certain conditions are met. It strikes a balance between permissiveness and protecting the rights of contributors.
Contributions to MemoryDLL are highly encouraged. Please feel free to submit issues, suggest new features, or create pull requests to expand its capabilities and robustness.