使用 CGO 链接两个具有相同函数名称的 C 库?
Linking two C libraries with the same function name with CGO?
比如我有两个函数名相同的C库:
hello1.c:
#include <stdio.h>
void hello() {
printf("Hello from C hello1.c\n");
}
hello2.c:
#include <stdio.h>
void hello() {
printf("Hello from C hello2.c\n");
}
我怎样才能link把它变成 Go 可执行二进制文件?
静态或动态都无所谓。
我什至已经尝试过 link 将它静态地放入单独的 Go 插件中,但它不起作用——只有来自 hello1.c
的 hello
函数将从两个 Go 插件中调用:
hello1.go Go插件:
package main
/*
#cgo LDFLAGS: -L${SRCDIR}/lib -lhello1
void hello();
*/
import "C"
import "fmt"
func Hello() {
fmt.Println("Hello from Go plugin hello1.go")
fmt.Println("Call C hello1.c")
C.hello()
}
hello2.go Go插件:
package main
/*
#cgo LDFLAGS: -L${SRCDIR}/lib -lhello2
void hello();
*/
import "C"
import "fmt"
func Hello() {
fmt.Println("Hello from Go plugin hello2.go")
fmt.Println("Call C hello2.c")
C.hello()
}
main.go 可执行二进制文件:
package main
import (
"fmt"
"plugin"
)
func main() {
p1, err := plugin.Open("hello1/hello1.so")
if err != nil {
panic(err)
}
fmt.Println("Load Go plugin hello1.so")
p2, err := plugin.Open("hello2/hello2.so")
if err != nil {
panic(err)
}
fmt.Println("Load Go plugin hello2.so")
h1, err := p1.Lookup("Hello")
if err != nil {
panic(err)
}
h2, err := p2.Lookup("Hello")
if err != nil {
panic(err)
}
fmt.Println("Call Go plugin hello1.go")
h1.(func())()
fmt.Println("\nCall Go plugin hello2.go")
h2.(func())()
}
运行 main
可执行二进制文件后的输出:
./main
Load Go plugin hello1.so
Load Go plugin hello2.so
Call Go plugin hello1.go
Hello from Go plugin hello1.go
Call C hello1.c
Hello from C hello1.c
Call Go plugin hello2.go
Hello from Go plugin hello2.go
Call C hello2.c
Hello from C hello1.c <=== here hello1.c, but I expected hello2.c
更新:
我给golang做了issue得到答案:https://github.com/golang/go/issues/42854
我在 github 期 https://github.com/golang/go/issues/42854
中得到了问题的答案
当我编译 C 库时,我应该使用 -fvisibility=hidden
gcc 标志。
比如我有两个函数名相同的C库:
hello1.c:
#include <stdio.h>
void hello() {
printf("Hello from C hello1.c\n");
}
hello2.c:
#include <stdio.h>
void hello() {
printf("Hello from C hello2.c\n");
}
我怎样才能link把它变成 Go 可执行二进制文件? 静态或动态都无所谓。
我什至已经尝试过 link 将它静态地放入单独的 Go 插件中,但它不起作用——只有来自 hello1.c
的 hello
函数将从两个 Go 插件中调用:
hello1.go Go插件:
package main
/*
#cgo LDFLAGS: -L${SRCDIR}/lib -lhello1
void hello();
*/
import "C"
import "fmt"
func Hello() {
fmt.Println("Hello from Go plugin hello1.go")
fmt.Println("Call C hello1.c")
C.hello()
}
hello2.go Go插件:
package main
/*
#cgo LDFLAGS: -L${SRCDIR}/lib -lhello2
void hello();
*/
import "C"
import "fmt"
func Hello() {
fmt.Println("Hello from Go plugin hello2.go")
fmt.Println("Call C hello2.c")
C.hello()
}
main.go 可执行二进制文件:
package main
import (
"fmt"
"plugin"
)
func main() {
p1, err := plugin.Open("hello1/hello1.so")
if err != nil {
panic(err)
}
fmt.Println("Load Go plugin hello1.so")
p2, err := plugin.Open("hello2/hello2.so")
if err != nil {
panic(err)
}
fmt.Println("Load Go plugin hello2.so")
h1, err := p1.Lookup("Hello")
if err != nil {
panic(err)
}
h2, err := p2.Lookup("Hello")
if err != nil {
panic(err)
}
fmt.Println("Call Go plugin hello1.go")
h1.(func())()
fmt.Println("\nCall Go plugin hello2.go")
h2.(func())()
}
运行 main
可执行二进制文件后的输出:
./main
Load Go plugin hello1.so
Load Go plugin hello2.so
Call Go plugin hello1.go
Hello from Go plugin hello1.go
Call C hello1.c
Hello from C hello1.c
Call Go plugin hello2.go
Hello from Go plugin hello2.go
Call C hello2.c
Hello from C hello1.c <=== here hello1.c, but I expected hello2.c
更新:
我给golang做了issue得到答案:https://github.com/golang/go/issues/42854
我在 github 期 https://github.com/golang/go/issues/42854
中得到了问题的答案当我编译 C 库时,我应该使用 -fvisibility=hidden
gcc 标志。