Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/tinybiggames/memorydll

In-Memory Win64 DLL Loading & Execution for Pascal.
https://github.com/tinybiggames/memorydll

delphi dll-loading dynamic-link-library fpc high-performance in-memory-dll memory-dll no-filesystem object-pascal pascal software-library win64 windows-10 windows-11

Last synced: 2 months ago
JSON representation

In-Memory Win64 DLL Loading & Execution for Pascal.

Awesome Lists containing this project

README

        

![MemoryDLL](media/memorydll.png)
[![Chat on Discord](https://img.shields.io/discord/754884471324672040?style=for-the-badge)](https://discord.gg/tPWjMwK) [![Twitter Follow](https://img.shields.io/twitter/follow/tinyBigGAMES?style=for-the-badge)](https://twitter.com/tinyBigGAMES)

### In-Memory Win64 DLL Loading & Execution for Pascal πŸ’»πŸ”„

### Overview πŸ“

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 πŸ”’.

### Features ✨

- **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 πŸ”§.

### Key Benefits 🌟

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

### Implementation Details πŸ”

**MemoryDLL** uses low-level Windows API functions to manage DLLs entirely from memory:

1. **Manual Mapping πŸ—ΊοΈ**: The unit manually parses the DLL's Portable Executable (PE) format, maps its sections, and relocates addresses.
2. **Relocation Handling πŸ”„**: Handles relocation entries to ensure the DLL functions correctly at its loaded memory address.
3. **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 πŸ’‰.

### Usage Scenarios 🎯

#### Embedding DLLs πŸ“¦

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

#### Encrypted DLL Loading πŸ”

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

#### Dynamic Plugin Systems πŸ”Œ

- Load plugins dynamically as in-memory DLLs. This approach provides a clean and secure method of extending application functionality without relying on the filesystem.

### Public Functions πŸ“–

#### LoadLibrary πŸ“œ

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 or `0` on failure.

#### FreeLibrary 🚫

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.

#### GetProcAddress πŸ”

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, or `nil` if not found.

### Installation πŸ› οΈ

To successfully integrate **MemoryDLL** into your project, please follow these steps:

1. **Download the Latest Version πŸ“₯**
- Visit the official **MemoryDLL** repository and download the latest release.

2. **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.

3. **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.

4. **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`, and `GetProcAddress`. 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.

5. **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)

### πŸ“– Example Usage

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:

```delphi

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 successfullyloaded 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
// GUID-like resource name for the embedded DLL.
CValue = 'b87deef5bbfd43c3a07379e26f4dec9b';
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();

```

### Acknowledgments πŸ™

This project is based on the original **Delphi MemoryModule** project by [Fr0sT-Brutal](https://github.com/Fr0sT-Brutal/Delphi_MemoryModule). We gratefully acknowledge the foundational work that this unit builds upon.

### License πŸ“œ

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.

### How to Contribute 🀝

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.

### Support πŸ“§

- Issues
- Discussions
- Learn Delphi
- FreePascal Wiki


Delphi


Made with :heart: in Delphi