使用 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.chello 函数将从两个 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 标志。