Go 와 C 언어의 상호작용 cgo

http://tonybai.com/2012/09/26/interoperability-between-go-and-c/
 
// foo.h
 
int count;
void foo();
 
//foo.c
#include "foo.h"
 
int count = 6;
void foo() {
    printf("I am foo!
"); }

 
//foo.go

package main
 
// #cgo LDFLAGS: -L ./ -lfoo
// #include <stdio.h>
// #include <stdlib.h>
// #include "foo.h"
import "C"
import "fmt“
 
func main() {
    fmt.Println(C.count)
    C.foo()
}

정적 라 이브 러 리 사용
$> gcc -c foo.c
$> ar rv libfoo.a foo.o
[diego@localhost ~/GoWork/src/applycation/testCgo]# go build foo.go
[diego@localhost ~/GoWork/src/applycation/testCgo]# ./foo 
6
I am foo!

 
[diego@localhost ~/GoWork/src/applycation/testCgo]# gcc -fPIC -shared -o libfoo.so foo.c
[diego@localhost ~/GoWork/src/applycation/testCgo]# rm li
libfoo.a   libfoo.so* 
[diego@localhost ~/GoWork/src/applycation/testCgo]# rm libfoo.a
[diego@localhost ~/GoWork/src/applycation/testCgo]# go build fo
foo*    foo.c   foo.go  foo.h   foo.o   
[diego@localhost ~/GoWork/src/applycation/testCgo]# go build foo.
foo.c   foo.go  foo.h   foo.o   
[diego@localhost ~/GoWork/src/applycation/testCgo]# go build foo.go 
[diego@localhost ~/GoWork/src/applycation/testCgo]# ./foo 
6
I am foo!
[diego@localhost ~/GoWork/src/applycation/testCgo]#

 
 http://tonybai.com/2012/09/26/interoperability-between-go-and-c/
 
고 에서 C 소스 코드 를 사용 하 는 것 보다 C 에서 고 함 수 를 사용 하 는 경우 가 적다.Go 에서"export+함수 이름"을 사용 하여 Go 함 수 를 C 로 내 보 낼 수 있 습 니 다.간단 한 예 를 보십시오.
 
package main
 
/*
#include
 
extern void GoExportedFunc();
 
void bar() {
        printf("I am bar!");
        GoExportedFunc();
}
*/
import "C"
 
import "fmt"
 
//export GoExportedFunc
func GoExportedFunc() {
        fmt.Println("I am a GoExportedFunc!")
}
 
func main() {
        C.bar()
}
 
그러나 이 Go 파일 을 컴 파일 할 때 다음 과 같은 오류 정 보 를 얻 었 습 니 다.
 
# command-line-arguments
/tmp/go-build163255970/command-line-arguments/_obj/bar.cgo2.o: In function `bar':
./bar.go:7: multiple definition of `bar'
/tmp/go-build163255970/command-line-arguments/_obj/_cgo_export.o:/home/tonybai/test/go/bar.go:7: first defined here
collect2: ld returned 1 exit status
 
코드 는 아무런 문제 가 없 는 것 같 지만 컴 파일 을 통 해'다 중 정의'를 항상 알려 줍 니 다.Cgo 문 서 를 뒤 져 서 실 마 리 를 찾 았 습 니 다.원래
 
There is a limitation: if your program uses any //export directives, then the C code in the comment may only include declarations (extern int f();), not definitions (int f() { return 1; }).
 
//extern int f()와/export f 는 Go 소스 파일 에 넣 을 수 없 는 것 같 습 니 다.우 리 는 bar.go 를 bar 1.go 와 bar 2.go 두 개의 파일 로 나 누 었 다.
 
// bar1.go
 
package main
 
/*
#include
 
extern void GoExportedFunc();
 
void bar() {
        printf("I am bar!");
        GoExportedFunc();
}
*/
import "C"
 
func main() {
        C.bar()
}
 
// bar2.go
 
package main
 
import "C"
import "fmt"
 
//export GoExportedFunc
func GoExportedFunc() {
        fmt.Println("I am a GoExportedFunc!")
}
 
컴 파일 실행:
 
$> go build -o bar bar1.go bar2.go
$> bar
I am bar!
I am a GoExportedFunc!
 
 
 
Go 코드:
func TestCallback() {
    f1 := syscall.NewCallback(PlusOne)
    f2 := syscall.NewCallbackCDecl(PlusTwo)
    var m uint32 = 20
    var n uint32 = 80

    // Func1 __stdcall
    fmt.Println(C.Func1(C.uint(m), (*[0]byte)(unsafe.Pointer(f1)))) // 21

    // Func2 __cdecl
    fmt.Println(C.Func2(C.uint(n), (*[0]byte)(unsafe.Pointer(f2)))) // 82
}

func PlusOne(n uint32) uintptr {
    return uintptr(n + 1)
}

func PlusTwo(n uint32) uintptr {
    return uintptr(n + 2)
}

C.Func 1 의 두 번 째 매개 변수 유형 은 함수 이 므 로*[0]byte 를 입력 해 야 합 니 다.  http://studygolang.com/articles/2629
 
http://blog.giorgis.io/cgo-examples

좋은 웹페이지 즐겨찾기