将cdecl函数调用从C转换为使用带有可变参数列表的回调的Pascal

时间:2013-02-24 10:28:16

标签: c delphi header-files pascal

我正在将视频编码器DLL的C头文件转换为Delphi Pascal。

我使用以下函数遇到了一些访问冲突问题,需要帮助解决此问题:

h264venc_tt * MC_EXPORT_API h264OutVideoNew(
  void * (MC_EXPORT_API * get_rc)(const char* name),
  const struct h264_v_settings * set,
  int32_t options,
  int32_t CPU,
  int32_t frame0,
  int32_t nframes);

注意:MC_EXPORT_API = cdecl

get_rc声明如下:

// resource functions dispatcher
void * MC_EXPORT_API get_rc(const char* name)
{
  if (!strcmp(name, "err_printf"))
    return (void*) error_printf;
  else if (!strcmp(name, "prg_printf"))
    return (void*) progress_printf;
  else if (!strcmp(name, "wrn_printf"))
    return (void*) warn_printf;
  else if (!strcmp(name, "inf_printf"))
    return (void*) info_printf;
  return NULL;
}

此函数返回指向另一个具有Variable Argument列表的函数的指针。其中一个被声明为:

void error_printf(const char * fmt, ...)
{
  char lst[256];
  va_list marker;
  va_start(marker,fmt);
  vsprintf(lst,fmt,marker);
  va_end(marker);
  printf("%s\n", lst);
}

我已将函数调用和get_rc转换为此Delphi Pascal代码:

PErrorMessageHandler = function (const Name: String): Pointer; cdecl varargs;

function h264OutVideoNew(
  get_rc: PErrorMessageHandler;
  settings: Ph264_v_settings;
  options: int32;
  CPU: int32;
  frame0: int32;
  nFrames: int32
): Pointer; cdecl; external 'mc_enc_avc.dll' index 4;

不幸的是,我不知道如何实现上面显示的C风格方法error_printf。 任何人都可以帮我指出正确的方向吗?我也很好奇我是否正确地实现了其他功能,因为当我尝试调用h264OutVideoNew()函数时,我遇到了访问冲突。

PS!我没有在这篇文章中包含打包记录Th264_v_settings / P_h264_v_settings,因为这很长并且不是真正的问题。

1 个答案:

答案 0 :(得分:6)

类型char*的C参数是指向8位字符的以空值终止的数组的指针。在Delphi中,等效类型为PAnsiChar。您不能使用string,因为这是一个在C中没有等效的托管Delphi类型。

此外,错误函数原型具有void返回值。你正在返回一个指针,这是一个错误。

你遇到的更大问题是你无法轻易实现在Delphi中接收可变数量参数的C风格函数。您可以声明并调用此类函数,但无法实现。这意味着具有可变参数的此类函数必须是external function

现在,您可以编写自己的汇编程序来完全填充变量参数。但是,这不是我要采取的路线。我会在C中编写该函数,然后将其编译为.obj文件,该文件可以通过$LINK链接到您的Delphi程序中。

如果您实际上不需要读取变量参数,可以忽略它们:

TErrorMessageHandler = procedure(Name: PAnsiChar); cdecl;

请注意,我做了以下更改:

  1. 将类型名称更改为标准的T前缀。
  2. 更正了Name参数的类型。
  3. 从功能改为程序以匹配C声明。
  4. 删除了我们无法在Delphi中实现的varargs,因此忽略了其他参数。
  5. 然后您导入的函数将如下所示:

    function h264OutVideoNew(
      get_rc: TErrorMessageHandler;
      settings: Ph264_v_settings;
      options: int32;
      CPU: int32;
      frame0: int32;
      nFrames: int32
    ): Pointer; cdecl; external 'mc_enc_avc.dll' index 4;
    

    然后你可以像这样实现错误回调函数:

    procedure error_printf(Name: PAnsiChar); cdecl;
    begin
      // do stuff here
    end;