如何在delphi中直接从内存中执行代码?

时间:2012-08-29 06:05:00

标签: windows delphi api loadlibrary

是否有可能模仿loadlibrary功能?我想从BLOB字段加载一个库而不先将其写入临时文件,我需要一个不依赖于特定版本的delphi编译器或Windows的解决方案,并且不会触发防病毒软件。

3 个答案:

答案 0 :(得分:6)

是的,你可以,而且你不需要loadlibrary从内存执行代码 - 你需要使用VirtualAlloc函数分配一个内存,并设置PAGE_EXECUTE标志


更新:这是一个快速而肮脏的代码,用于从32位Delphi的内存执行代码 - 我只测试它是否有效:

type
  TIncMe = procedure(var I: Integer);

var
  IncMeProc: TIncMe;

procedure IncMe(var I: Integer);
begin
  Inc(I);
end;

procedure CopyIncMe;
var
  Size: LongWord;
  Tmp: Pointer;

begin
  Size:= LongWord(@CopyIncMe) - LongWord(@IncMe);
  Tmp:= VirtualAlloc(nil, Size, MEM_COMMIT or MEM_RESERVE, PAGE_EXECUTE_READWRITE);
  Move(Pointer(@IncMe)^, Tmp^, Size);
  IncMeProc:= Tmp;
end;

procedure TForm1.Button1Click(Sender: TObject);
var
  J: Integer;

begin
  J:= 0;
  CopyIncMe;
  while J < 10 do begin
    IncMeProc(J);
    ShowMessage(IntToStr(J));
  end;
  VirtualFree(@IncMeProc, 0, MEM_RELEASE);
end;

答案 1 :(得分:5)

dzlib包含一个现成的对象,用于将资源从资源读取到内存中并使用它而无需将其保存到光盘:

这是主档......

http://sourceforge.net/p/dzlib/code/147/tree/dzlib/trunk/src/u_dzResourceDllLoader.pas

..但它需要来自同一存储库的其他文件。

答案 2 :(得分:4)

有一篇关于delphi.about.com的文章,展示了如何从资源加载dll。

首先将资源加载到内存中,然后使用Memory Module从资源加载dll

您可以使用数据库或任何要从中加载dll的源代替资源。一旦它在内存流中,您可以使用以下代码来加载和执行dll函数,这看起来非常像调用dll的“普通”代码:

var
  btMM: PBTMemoryModule;
begin
  btMM := BTMemoryLoadLibary(mp_DllData, m_DllDataSize);
  try
    if btMM = nil then Abort;
    @m_TestCallstd := BTMemoryGetProcAddress(btMM, 'TestCallstd');
    if @m_TestCallstd = nil then Abort;
    m_TestCallstd('This is a Dll Memory call!');
  except
    Showmessage('An error occoured while loading the dll: ' + BTMemoryGetLastError);
  end;
  if Assigned(btMM) then BTMemoryFreeLibrary(btMM);
end;