将 C 风格的整数数组传递给 Ada 函数以检索数组中的第一个元素

doo*_*der 5 c ada googletest

我正在尝试将 google test 链接到 ada95 进行单元测试。我知道 ada 有 aunit 和 ahven,但这只是为了看看这是否可能并且超出了我的问题范围。我已经成功地能够使用基本数据类型执行简单的函数和过程。我想尝试做的下一件事类似于以下内容:

这是 main.cpp 文件:

#include <stdio.h>
#include <gtest/gtest.h>

extern "C" {
  int firstElement(int buffer[]);
}

TEST(tryTest, checkBuffer){
   int buffer[10] = {10,1,6,4,3,2,1,3,4,6};
   ASSERT_EQ(buffer[0],firstElement(buffer));
}

int main(int argc, char ** argv) {
  testing::InitGoogleTest(&argc, argv);
  return RUN_ALL_TESTS();
}
Run Code Online (Sandbox Code Playgroud)

为简单起见,我将只放置广告文件:

Lib_test.ads

with Interfaces.C;
with Interfaces.C.Pointers;
package Lib_Test is

  function FirstElement(a: Interfaces.C.Pointers) return Interfaces.C.Int;
  pragma Export(C,FirstElement,"firstElement");
end Lib_Test;
Run Code Online (Sandbox Code Playgroud)

我知道在 c 中,您传递的是指向第一个元素的指针,而不是函数的整个数组。这就是为什么我尝试使用 Interfaces.C.Pointers 作为数据类型,但出现以下错误

subtype mark required in this context
found "Pointers" declared at i-cpoint.ads:44
Run Code Online (Sandbox Code Playgroud)

除了 char 数组,我还没有找到使用其他数组类型的好例子。有人可以告诉我如何将 Interfaces.C.Pointers 用于整数数组,甚至如何解决这个问题,我相信这只是函数参数中的数据类型。我希望能够访问 ada 函数中的 c 整数数组。

谢谢你们!

Dee*_*Dee 5

根据RM B.3 (70)

具有组件类型 T 的任何模式的数组类型的 Ada 参数作为 at* 参数传递给 C 函数,其中 t 是对应于 Ada 类型 T 的 C 类型。

因此,没有必要使用 package Interfaces.C.Pointers。您可以只使用 Ada 数组类型。一个小例子:

主程序

#include <stdio.h>
#include <gtest/gtest.h>

extern "C" {
  void testinit();
  void testfinal();
  int firstElement(int *buffer);
}

class MyTest : public ::testing::Test {
protected:

  MyTest() {
    testinit();          // Initialize the Ada library
  }

  ~MyTest() override {
    testfinal();         // Finalize the Ada library
  }
};

TEST_F(MyTest, CheckBuffer) {
  int buffer[10] = {10,1,6,4,3,2,1,3,4,6};
  ASSERT_EQ(buffer[0], firstElement(buffer));
}

int main(int argc, char **argv) {
  ::testing::InitGoogleTest(&argc, argv);
  return RUN_ALL_TESTS();
}
Run Code Online (Sandbox Code Playgroud)

测试.gpr

with "libtest";

project Test is

   for Source_Dirs use ("src");
   for Object_Dir use "obj";
   for Main use ("main.cpp");
   for Languages use ("C++");

   package Compiler is
      for Switches ("c++") use ("-I/usr/src/googletest/googletest/include");
   end Compiler;

   package Linker is
      for Switches ("c++") use ("-lgtest", "-lgtest_main", "-pthread", "-ltest");
   end Linker;

end Test;
Run Code Online (Sandbox Code Playgroud)

lib_test.ads

with Interfaces.C;

package Lib_Test is

   package C renames Interfaces.C;
   
   type Seq is array (0 .. 9) of C.Int;
      
   function First_Element (A : Seq) return C.Int;
   pragma Export (C, First_Element, "firstElement");
   
end Lib_Test;
Run Code Online (Sandbox Code Playgroud)

lib_test.adb

package body Lib_Test is
   
   -------------------
   -- First_Element --
   -------------------
   
   function First_Element (A : Seq) return C.Int is
   begin
      return A (A'First);
   end First_Element;

end Lib_Test;
Run Code Online (Sandbox Code Playgroud)

libtest.gpr

library project Libtest is
   for Library_Kind use "dynamic";
   for Library_Name use "test";
   for Library_Interface use ("lib_test");
   for Library_Auto_Init use "False";
   for Library_Dir use "lib";
   for Object_Dir use "obj";
   for Source_Dirs use ("src");   
end Libtest;
Run Code Online (Sandbox Code Playgroud)

输出

$ ./obj/main 
[==========] Running 1 test from 1 test case.
[----------] Global test environment set-up.
[----------] 1 test from MyTest
[ RUN      ] MyTest.CheckBuffer
[       OK ] MyTest.CheckBuffer (0 ms)
[----------] 1 test from MyTest (0 ms total)

[----------] Global test environment tear-down
[==========] 1 test from 1 test case ran. (0 ms total)
[  PASSED  ] 1 test.
Run Code Online (Sandbox Code Playgroud)