4

我可以通过这个程序执行整数除法:

package main

import "fmt"

func main() {
 a := 10
 b := 5
    fmt.Println(a/b)
}

然后我在 go 中编写了一个程序,它具有 +、-、* 和 / 的功能。我用 C 语言编写了一个程序,它调用这些函数中的每一个并执行算术运算。除除法外,代码工作正常。

带有函数的 go 文件是:(calc.go)

package main

func Add(a, b int) int {
    return a + b
}
func Sub(a, b int) int {
    return a - b
}
func Mul(a, b int) int {
    return a * b
}
func Div(a, b int) int {
    return a / b
}

调用这些函数的 C 程序是:(calcc.c)

#include <stdio.h>

extern int go_add(int, int) __asm__ ("main.Add");
extern int go_sub(int, int) __asm__ ("main.Sub");
extern int go_mul(int, int) __asm__ ("main.Mul");
extern int go_div(int, int) __asm__ ("main.Div");

int menu()
{
  int op;
  printf("\n1 add");
  printf("\n2 sub");
  printf("\n3 mul");
  printf("\n4 div");
  printf("\nEnter your choice : ");
  scanf("%d", &op);
  return op;
}
int main() {


  int op, ch, result, a, b;

  do{ 
    op= menu();

    printf("First number  : ");
    scanf("%d", &a);
    printf("Second number : ");
    scanf("%d", &b);

    switch(op)
    {
       case 1:
        result = go_add(a, b);
    printf("Result : %d" , result);
        break;
       case 2:
        result = go_sub(a, b);
    printf("Result : %d" , result);
        break;
       case 3:
        result = go_mul(a, b);
    printf("Result : %d" , result);
    break;
       case 4:
        result = go_div(a, b);
    printf("Result : %d" , result);
    break;
       default:
        printf("Invalid choice ! ");
    }
    printf("\nAnother operation? (1 if yes) : ");
    scanf("%d", &ch);
  } while(ch==1);
  printf("\nThank you!");
}

我使用以下命令在终端上编译:

gccgo -c calc.go

gcc calc.o calcc.c -o 主

并得到了这个错误: undefined reference to `__go_runtime_error' collect2: error: ld returned 1 exit status

我应该如何解决这个问题?

4

2 回答 2

1

您需要链接 usinggccgo而不是 normal gcc。普通 gcc 不知道它应该链接到 go 运行时(libgo)。

根据您的配置,您可能还需要指定可以找到运行时库的位置。例如,通过静态嵌入或使其在LD_LIBRARY_PATH环境变量中可用。例子:

gccgo -static-libgo calc.o calcc.o -o main

有关更多信息,请查看设置和使用 gccgo

于 2015-03-20T08:34:35.340 回答
1

我相信您的使用方法__asm__是 gccgo 特定的(我以前从未见过)。

将Go 函数导出到 C的标准方法//export name是通过Go 代码中的“”注释。

此外,通过 cgo 的标准 Go<->C 要求 C 代码链接到 Go 和 Go 的主要运行,而不是相反。这是为了让 Go 运行时完全运行。否则 goroutines、垃圾收集器等将不会运行。当然,Go 的 main 可能只是对 C 伪 main 函数的简单调用,该函数完成所有工作并仅在需要时回调 Go。

鉴于这些要点,您尝试使用标准cgo 和完全go build-able 的一个小例子是这样的:

计算:

package main

// /* could be in a declared in a header file instead */
// extern void pseudo_main(void);
import "C"

//export Add
func Add(a, b int) int {
    return a + b
}

// … etc …

//export Div
func Div(a, b int) int {
    return a / b
}

// Main needs to be Go so that the go runtime
// gets started so you can use goroutines, the
// garbage collector, etc,etc.
//
// It can just be a trivial call into a C main like
// function.
func main() {
    C.pseudo_main()
}

和 calc.c:

#include <stdio.h>
#include "_cgo_export.h" // file auto-generated by cgo from Go's "//export func" comments

// passing argc, argv, envp like arguments
// if desired is left as an excersise :)
void pseudo_main(void) {
    int x, y, z;

    printf("Hello from C\n");
    x = 42;
    y = 6;
    z = Add(x, y);
    printf("%d + %d = %d\n", x, y, z);
    z = Div(x, y);
    printf("%d / %d = %d\n", x, y, z);
}

构建和运行(在类似 Unix 的主机上):

% go build -o calc
% ./calc

注意:通常你不会使用-o,你会让工具根据包或目录名称来选择名称。我在-o这里使用了列出精确且可重复的命令,而没有指定文件所在的目录。进一步注意,对于 Microsoft Windows,它会有所不同。此外,如果您对 cgo 幕后发生的事情感兴趣,请尝试go build -x.

输出:

Hello from C
42 + 6 = 48
42 / 6 = 7

gist.github.com

另请参阅:Go 博客:C? 去?加油!

于 2015-03-20T17:41:47.103 回答