c - 如何使用 NativeCall 处理可变参数

标签 c raku nativecall

我正在为 Editline 编写绑定(bind);它的功能之一, history ,为库的这一部分完成大部分工作,但有几个可能的签名:

:(Pointer[Internal], Pointer[Event], int32 --> int32)
:(Pointer[Internal], Pointer[Event], int32, int32 --> int32)
:(Pointer[Internal], Pointer[Event], int32, Str --> int32)
# etc.

第三个参数是一个标志,用于确定哪个函数 history应该使用给定的参数调用,但由于这些符号没有被导出,我无法使用它们来代替。如何找到使用该功能的方法?我不能使用多子或使用 cglobal将其转换为 Pointer ,然后到具有正确签名的函数。

编辑:我知道 is symbol ,但我想知道是否有更好的方法来解决这个问题,因为有 9 个不同的签名要写。

最佳答案

您可以使用 Raku 函数包装 native 函数:

sub history ( Pointer[Internal] $a, Pointer[Event] $b, int32 $flag, $c? --> int32 ){
  given $flag {
    when 0|2 {
      sub history (Pointer[Internal], Pointer[Event], int32 --> int32) is native(…) {}
      history( $a, $b, $flag )
    }
    when 1|3 {
      sub history (Pointer[Internal], Pointer[Event], int32, int32 --> int32) is native(…) {}
      history( $a, $b, $flag, $c )
    }
    when 4 {
      sub history (Pointer[Internal], Pointer[Event], int32, Str --> int32) is native(…) {}
      history( $a, $b, $flag, $c )
    }
  }
}

或者您可以将每个版本包装在自己的多重中:

# Note the flag value ---------------------------------------+
#                                                            |
#                                                            V
multi sub history ( Pointer[Internal] $a, Pointer[Event] $b, 0 --> int32 ){
  sub history (Pointer[Internal], Pointer[Event], int32 --> int32) is native(…) {}
  history( $a, $b, 0 )
}
multi sub history ( Pointer[Internal] $a, Pointer[Event] $b, 2 --> int32 ){
  sub history (Pointer[Internal], Pointer[Event], int32 --> int32) is native(…) {}
  history( $a, $b, 2 )
}

multi sub history ( Pointer[Internal] $a, Pointer[Event] $b, int32 $flag where 1|3, int32 $c --> int32 ){
  sub history (Pointer[Internal], Pointer[Event], int32, int32 --> int32) is native(…) {}
  history( $a, $b, $flag, $c )
}

multi sub history ( Pointer[Internal] $a, Pointer[Event] $b, 4, Str:D $c --> int32 ){
  sub history (Pointer[Internal], Pointer[Event], int32, Str --> int32) is native(…) {}
  history( $a, $b, 4, $c )
}

您甚至可以根据参数确定要给出的正确标志。这只有在每个签名只有一个标志时才有效。
(因此,以下示例不遵循上述 02 具有相同签名的示例。)

multi sub history ( Pointer[Internal] $a, Pointer[Event] $b --> int32 ){
  sub history (Pointer[Internal], Pointer[Event], int32 --> int32) is native(…) {}
  history( $a, $b, 0 )
}

multi sub history ( Pointer[Internal] $a, Pointer[Event] $b, int32 $c --> int32 ){
  sub history (Pointer[Internal], Pointer[Event], int32 --> int32) is native(…) {}
  history( $a, $b, 1, $c )
}

multi sub history ( Pointer[Internal] $a, Pointer[Event] $b, Str:D $c --> int32 ){
  sub history (Pointer[Internal], Pointer[Event], int32, Str --> int32) is native(…) {}
  history( $a, $b, 4, $c )
}

您可以编写具有不同名称的包装器:

sub foo ( Pointer[Internal] $a, Pointer[Event] $b --> int32 ){
  sub history (Pointer[Internal], Pointer[Event], int32 --> int32) is native(…) {}
  history( $a, $b, 0 )
}

sub bar ( Pointer[Internal] $a, Pointer[Event] $b, int32 $c --> int32 ){
  sub history (Pointer[Internal], Pointer[Event], int32 --> int32) is native(…) {}
  history( $a, $b, 1, $c )
}

sub baz ( Pointer[Internal] $a, Pointer[Event] $b, Str:D $c --> int32 ){
  sub history (Pointer[Internal], Pointer[Event], int32, Str --> int32) is native(…) {}
  history( $a, $b, 4, $c )
}

你可以通过使用 is symbol 来做到这一点而无需包装器。 .

sub foo (Pointer[Internal], Pointer[Event], int32        --> int32) is native(…) is symbol<history> {}
sub bar (Pointer[Internal], Pointer[Event], int32, int32 --> int32) is native(…) is symbol<history> {}
sub baz (Pointer[Internal], Pointer[Event], int32, Str   --> int32) is native(…) is symbol<history> {}

您也可以使用 nativecast以及包装函数中生成的 Signature 对象:

sub history ( Pointer[Internal] $a, Pointer[Event] $b, int32 $flag, $c? --> int32 ){
  my \I32 = Parameter.new( type => int32 );
  my \PI  = Parameter.new( type => Pointer[Internal] );
  my \PE  = Parameter.new( type => Pointer[Event] );
  my \STR = Parameter.new( type => Str );

  my @params = ( PI, PE, I32 );
  given $flag {
    when 0|2 {
    }
    when 1|3 {
      @params.push( I32 );
    }
    when 4 {
      @params.push( STR );
    }
  }

  my \signature = Signature.new( params => @params.List, returns => int32 );

  # fill this out -----------V
  my \history-ptr = cglobal( …, 'history', Pointer );

  my &history = nativecast( signature, history-ptr );

  history( $a, $b, $flag, ($c if +@params == 4) );
}

这就是您可以为 C 的 printf 编写包装器的方式。或 scanf .

无论您使用上述哪种方法,您都需要某种方法来确定需要调用的变体。

即使在 C 中,也必须有一种方法可以根据当前或先前的参数来确定参数的数量和类型。

printf参数的数量和类型基于格式的第一个参数。

C 函数可以通过尾随空参数来确定参数的数量。在这种情况下,你必须告诉 Raku。

不管底层外部函数如何处理va_args ,您必须以某种形式将该算法复制到 Raku 中。它不能只是猜测。

如果你有几个工作相似的外部函数,你可以为它们创建一个包装器生成器。

关于c - 如何使用 NativeCall 处理可变参数,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/49586031/

相关文章:

types - 在 Raku 中, "Node"类型或区分值是什么?

winapi - 将复杂结构传递给 Windows API

c - 这是否被认为是未定义行为 x =++x % 5;

raku - Perl 6错误消息: Malformed UTF-8 in block <unit>

c++ - 为什么结构的 sizeof 不等于每个成员的 sizeof 之和?

arrays - `@$array` 和其他结构的含义

raku - NativeCall 中的位域

Raku 如何将指向 Buf 的指针传递给本地调用进行写入

c++ - 在 C++ 程序中使用 C 头文件

c - 为什么“while(!feof(file))”总是错误的?