我有一个完全用C编写的程序,其中使用了多个目标(.o)文件。这些文件都打包在一个存档文件(.a)中,而该存档文件又在程序主(.c)文件的编译时使用。
(.o)
(.a)
(.c)
我想在Go中为此项目编写一个新文件。我的想法是编写此.go文件,然后(.o)从中创建一个目标文件。之后,我想将此对象文件放入已经提到的存档(.a)文件中。
.go
这基本上意味着 我想从C程序调用Go函数。我已经阅读了这个问题,虽然它告诉我可以通过GCCGO实现我想要的东西,但是如何做到这一点还不是100%清楚。
即使使用最基本的测试,在链接阶段也会出现错误。更具体地说,这是这样的基本示例之一:
printString.go
package main import ( "fmt" ) func PrintString(buff string) int { fmt.Printf(buff) return 1 }
c_caller.c
#define _GNU_SOURCE #include <stdio.h> extern int PrintString(char*) __asm__ ("print.main.PrintString"); int main() { char *string_to_pass= NULL; asprintf(&string_to_pass, "This is a test."); int result= PrintString(string_to_pass); if(result) {printf("Everything went as expected!\n");} else {printf("Uh oh, something went wrong!\n");} return result; }
编译中
为了编译Go文件,我使用了以下命令:
gccgo -c printString.go -o printString.o -fgo-prefix=print -Wall -Werror -march=native
为了编译整个事情,我使用了以下命令:
gccgo -o main c_caller.c printString.o -Wall -Werror -march=native
我收到的返回消息是:
/usr/lib64/libgo.so.4.0.0: undefined reference to `main.main' /usr/lib64/libgo.so.4.0.0: undefined reference to `__go_init_main' collect2: error: ld returned 1 exit status
这意味着GCCGO期望Go文件中的主要功能而不是C语言中的主要功能。
使用--static- libgo,-static并且-Wl,-R,/path/to/libgo.so's_folder在所述第二命令选项产生不同的结果:
--static- libgo
-static
-Wl,-R,/path/to/libgo.so's_folder
/usr/bin/ld: cannot find -lgo collect2: error: ld returned 1 exit status
这没有任何意义,因为我有LD_LIBRARY_PATH环境变量正确指向libgo.so的文件夹。
我意识到我在这里可能做错了什么,但我只是看不到那是什么。几乎没有GCCGO及其与C交互的示例,我唯一能找到的参考是此页面,我个人觉得还不够。
谨请您提供一些建议,并感谢您的宝贵时间。:)
这可能不是您想要的,但是在 Go 1.5 中(即将在今年8月推出),您将能够使用go工具构建与C兼容的库。所以有了这个_main.c
_main.c
#include <stdio.h> int main() { char *string_to_pass = NULL; if (asprintf(&string_to_pass, "This is a test.") < 0) { printf("asprintf fail"); return -1; } PrintString(string_to_pass); return 0; }
这在你的 main.go
main.go
package main import "C" import "fmt" //export PrintString func PrintString(cs *C.char) { s := C.GoString(cs) fmt.Println(s) } func main() {}
您可以针对静态库执行以下操作:
go build -buildmode c-archive -o mygopkg.a gcc -o main _main.c mygopkg.a -lpthread
对于共享库:
go build -buildmode c-shared -o mygopkg.so LD_RUN_PATH=$(pwd) gcc -o main _main.c mygopkg.so -lpthread
(LD_RUN_PATH在这里使链接器在您正在构建的同一目录中查找共享库。)
LD_RUN_PATH
有关更多信息,请参见Go执行模式设计文档。