Delphi"const of array"改为"varargs"

HNB*_*HNB 8 arrays delphi variadic-functions

请帮忙!我需要这个转换来为Delphi编写一些C头的包装器.

举个例子:

function pushfstring(fmt: PAnsiChar): PAnsiChar; cdecl; varargs; external;

...

function PushString(fmt: AnsiString; const args: array of const): AnsiString;
begin
  Result := AnsiString(pushfstring(PAnsiString(fmt), args)); // it's incorrect :/
end;
Run Code Online (Sandbox Code Playgroud)

如何将"const of const"转换为"varargs"?

编辑:函数PushString实际上在记录中(我给出了一个简化的例子),我没有直接访问pushfstring.直接呼叫被排除在外.

编辑2:我为Delphi编写了LUA库的单元,这个案例对我来说非常重要.

指定并提供事情的所有细节 - 我在C中有这个功能:

LUA_API const char *(lua_pushfstring) (lua_State *L, const char *fmt, ...);
Run Code Online (Sandbox Code Playgroud)

在Delphi中我有这样的东西:

LuaLibrary.pas

{...}
interface
{...}
function lua_pushfstring(L: lua_State; fmt: PAnsiChar): PAnsiChar; cdecl; varargs;
implementation
{...}
function lua_pushfstring; external 'lua.dll'; // or from OMF *.obj file by $L
Run Code Online (Sandbox Code Playgroud)

dtxLua.pas

uses LuaLibrary;
{...}
type
  TLuaState = packed record
  private
    FLuaState: lua_State;
  public
    class operator Implicit(A: TLuaState): lua_State; inline;
    class operator Implicit(A: lua_State): TLuaState; inline;
    {...}
    // btw. PushFString can't be inline function
    function PushFString(fmt: PAnsiChar; const args: array of const ): PAnsiChar; 
    //... and a lot of 'wrapper functions' for functions like a lua_pushfstring, 
    // where L: lua_State; is the first parameter
  end;
implementation
{...}
function TLuaState.PushFString(fmt: PAnsiChar; const args: array of const )
  : PAnsiChar;
begin
  Result := lua_pushfstring(FLuaState, fmt, args); // it's incorrect :/
end;
Run Code Online (Sandbox Code Playgroud)

在Lua.pas这样的其他单位我只使用dtxLua.pas中的TLuaState(因为LuaLibrary很笨重,dtxLua是我的包装),对于许多有用和酷的东西......

Bar*_*lly 14

我猜这个原型pushfstring有点像这样:

void pushfstring(const char *fmt, va_list args);
Run Code Online (Sandbox Code Playgroud)

如果不是,而是:

void pushfstring(const char *fmt, ...);
Run Code Online (Sandbox Code Playgroud)

...那么我也应该让你满满的.

在C,如果你要在通话通过从一个可变参数函数到另一个,你应该使用va_list,va_startva_end,并调用v该函数的版本.因此,如果您printf自己实现,可能会使用vsprintf格式化字符串 - 您无法sprintf直接调用并传递可变参数列表.你需要va_list和朋友一起使用.

va_list从Delphi 处理C是非常尴尬的,技术上不应该这样做 - 实现va_list是特定于C编译器供应商的运行时.

但是,我们可以试试.假设我们有一个小课程 - 虽然我把它作为易用性的记录:

type
  TVarArgCaller = record
  private
    FStack: array of Byte;
    FTop: PByte;
    procedure LazyInit;
    procedure PushData(Loc: Pointer; Size: Integer);
  public
    procedure PushArg(Value: Pointer); overload;
    procedure PushArg(Value: Integer); overload;
    procedure PushArg(Value: Double); overload;
    procedure PushArgList;
    function Invoke(CodeAddress: Pointer): Pointer;
  end;

procedure TVarArgCaller.LazyInit;
begin
  if FStack = nil then
  begin
    // Warning: assuming that the target of our call doesn't 
    // use more than 8K stack
    SetLength(FStack, 8192);
    FTop := @FStack[Length(FStack)];
  end;
end;

procedure TVarArgCaller.PushData(Loc: Pointer; Size: Integer);
  function AlignUp(Value: Integer): Integer;
  begin
    Result := (Value + 3) and not 3;
  end;
begin
  LazyInit;
  // actually you want more headroom than this
  Assert(FTop - Size >= PByte(@FStack[0]));
  Dec(FTop, AlignUp(Size));
  FillChar(FTop^, AlignUp(Size), 0);
  Move(Loc^, FTop^, Size);
end;

procedure TVarArgCaller.PushArg(Value: Pointer); 
begin
  PushData(@Value, SizeOf(Value));
end;

procedure TVarArgCaller.PushArg(Value: Integer); 
begin
  PushData(@Value, SizeOf(Value));
end;

procedure TVarArgCaller.PushArg(Value: Double); 
begin
  PushData(@Value, SizeOf(Value));
end;

procedure TVarArgCaller.PushArgList;
var
  currTop: PByte;
begin
  currTop := FTop;
  PushArg(currTop);
end;

function TVarArgCaller.Invoke(CodeAddress: Pointer): Pointer;
asm
  PUSH EBP
  MOV EBP,ESP

  // Going to do something unpleasant now - swap stack out
  MOV ESP, EAX.TVarArgCaller.FTop
  CALL CodeAddress
  // return value is in EAX
  MOV ESP,EBP

  POP EBP
end;
Run Code Online (Sandbox Code Playgroud)

使用此记录,我们可以手动构建各种C调用所需的调用帧.C在x86上的调用约定是在堆栈中从右向左传递参数,并且调用者清理.这是一个通用C调用例程的框架:

function CallManually(Code: Pointer; const Args: array of const): Pointer;
var
  i: Integer;
  caller: TVarArgCaller;
begin
  for i := High(Args) downto Low(Args) do
  begin
    case Args[i].VType of
      vtInteger: caller.PushArg(Args[i].VInteger);
      vtPChar: caller.PushArg(Args[i].VPChar);
      vtExtended: caller.PushArg(Args[i].VExtended^);
      vtAnsiString: caller.PushArg(PAnsiChar(Args[i].VAnsiString));
      vtWideString: caller.PushArg(PWideChar(Args[i].VWideString));
      vtUnicodeString: caller.PushArg(PWideChar(Args[i].VUnicodeString));
      // fill as needed
    else
      raise Exception.Create('Unknown type');
    end;
  end;
  Result := caller.Invoke(Code);
end;
Run Code Online (Sandbox Code Playgroud)

printf一个例子为例:

function printf(fmt: PAnsiChar): Integer; cdecl; varargs; 
    external 'msvcrt.dll' name 'printf';

const
  // necessary as 4.123 is Extended, and %g expects Double
  C: Double = 4.123;
begin
  // the old-fashioned way
  printf('test of printf %s %d %.4g'#10, PAnsiChar('hello'), 42, C);
  // the hard way
  CallManually(@printf, [AnsiString('test of printf %s %d %.4g'#10), 
                         PAnsiChar('hello'), 42, C]);
end.
Run Code Online (Sandbox Code Playgroud)

调用va_list版本稍微复杂一些,因为va_list参数的位置需要仔细放置在预期的位置:

function CallManually2(Code: Pointer; Fmt: AnsiString;
    const Args: array of const): Pointer;
var
  i: Integer;
  caller: TVarArgCaller;
begin
  for i := High(Args) downto Low(Args) do
  begin
    case Args[i].VType of
      vtInteger: caller.PushArg(Args[i].VInteger);
      vtPChar: caller.PushArg(Args[i].VPChar);
      vtExtended: caller.PushArg(Args[i].VExtended^);
      vtAnsiString: caller.PushArg(PAnsiChar(Args[i].VAnsiString));
      vtWideString: caller.PushArg(PWideChar(Args[i].VWideString));
      vtUnicodeString: caller.PushArg(PWideChar(Args[i].VUnicodeString));
    else
      raise Exception.Create('Unknown type'); // etc.
    end;
  end;
  caller.PushArgList;
  caller.PushArg(PAnsiChar(Fmt));
  Result := caller.Invoke(Code);
end;

function vprintf(fmt: PAnsiChar; va_list: Pointer): Integer; cdecl;
    external 'msvcrt.dll' name 'vprintf';

begin
  // the hard way, va_list
  CallManually2(@vprintf, 'test of printf %s %d %.4g'#10, 
      [PAnsiChar('hello'), 42, C]);
end.
Run Code Online (Sandbox Code Playgroud)

笔记:

  • 以上是Windows上的x86.va_list根据我的实验,Microsoft C,bcc32(Embarcadero C++)和gcc都以相同的方式传递(指向堆栈上第一个可变参数的指针),所以它应该适合你; 但是一旦Windows上的x86被破坏,预计这可能会破坏.

  • 交换堆栈以简化其构造.这可以通过更多工作来避免,但是传递va_list也变得更加棘手,因为它需要指向参数,就好像它们是在堆栈上传递一样.因此,代码需要假设被调用例程使用了多少堆栈; 这个例子假设8K,但这可能太小了.必要时增加.