2015-10-19 2 views
3

Рассмотрим следующий hello.c:Как обернуть «мир привет» с помощью argv с помощью cgo?

#include <stdio.h> 
int main(int argc, char* argv[]) { 
    printf("Hello, world! argv[0]=%s\n", argv[0]); 
} 

Makefile содержит только одну строку: all: hello. Как я могу обернуть (и скомпилировать) эту функцию вокруг программы Go, например, в этом (нерабочем) примере?

package main 
// int main(int argc, char* argv[]); 
import "C" 
func main() { 
argc := C.int(1); 
argv := make([]*C.char, 1) 
argv[0] = C.CString("hai"); 
C.main(argc, &argv[0]) 
} 

Ошибка я получаю:

(virtualenv)[14:16:21][/tmp/cgo]$ ./main 
fatal error: unexpected signal during runtime execution 
[signal 0xb code=0x1 addr=0x1c000000000 pc=0x41d563] 

runtime stack: 
runtime.throw(0x48b400, 0x2a) 
/usr/lib/golang/src/runtime/panic.go:527 +0x90 
runtime.sigpanic() 
/usr/lib/golang/src/runtime/sigpanic_unix.go:12 +0x5a 
runtime.mHeap_Grow(0x6c7c20, 0x8, 0x0) 
/usr/lib/golang/src/runtime/mheap.go:647 +0x243 
runtime.mHeap_AllocSpanLocked(0x6c7c20, 0x1, 0x0) 
/usr/lib/golang/src/runtime/mheap.go:532 +0x5f1 
runtime.mHeap_Alloc_m(0x6c7c20, 0x1, 0x15, 0x0) 
/usr/lib/golang/src/runtime/mheap.go:425 +0x1ac 
runtime.mHeap_Alloc.func1() 
/usr/lib/golang/src/runtime/mheap.go:484 +0x41 
runtime.systemstack(0x7ffcb3aab958) 
/usr/lib/golang/src/runtime/asm_amd64.s:278 +0xab 
runtime.mHeap_Alloc(0x6c7c20, 0x1, 0x10000000015, 0x40a954) 
/usr/lib/golang/src/runtime/mheap.go:485 +0x63 
runtime.mCentral_Grow(0x6d0898, 0x0) 
/usr/lib/golang/src/runtime/mcentral.go:190 +0x93 
runtime.mCentral_CacheSpan(0x6d0898, 0x6c1180) 
/usr/lib/golang/src/runtime/mcentral.go:86 +0x4d4 
runtime.mCache_Refill(0x7fbc9971d050, 0x15, 0x40ca28) 
/usr/lib/golang/src/runtime/mcache.go:118 +0xcf 
runtime.mallocgc.func2() 
/usr/lib/golang/src/runtime/malloc.go:611 +0x2b 
runtime.systemstack(0x7ffcb3aabb28) 
/usr/lib/golang/src/runtime/asm_amd64.s:278 +0xab 
runtime.mallocgc(0x180, 0x476220, 0x0, 0x800000000) 
/usr/lib/golang/src/runtime/malloc.go:612 +0x842 
runtime.newobject(0x476220, 0x6c0c40) 
/usr/lib/golang/src/runtime/malloc.go:760 +0x42 
runtime.malg(0x8000, 0x6c0e40) 
/usr/lib/golang/src/runtime/proc1.go:2188 +0x27 
runtime.mpreinit(0x6c1180) 
/usr/lib/golang/src/runtime/os1_linux.go:197 +0x1f 
runtime.mcommoninit(0x6c1180) 
/usr/lib/golang/src/runtime/proc1.go:114 +0x100 
runtime.schedinit() 
/usr/lib/golang/src/runtime/proc1.go:57 +0x79 
runtime.rt0_go(0xc82002e008, 0x1, 0xc82002e008, 0x1, 0x40122a, 0xc82004fef8, 0x0, 0x8, 0x44fbaa, 0xc82004fe80, ...) 
/usr/lib/golang/src/runtime/asm_amd64.s:109 +0x132 

goroutine 1 [syscall, locked to thread]: 
runtime.cgocall(0x401210, 0xc82004fef8, 0x0) 
/usr/lib/golang/src/runtime/cgocall.go:120 +0x11b fp=0xc82004feb0 sp=0xc82004fe80 
main._Cfunc_main(0x1, 0xc82002e008, 0x0) 
command-line-arguments/_obj/_cgo_gotypes.go:49 +0x39 fp=0xc82004fef8 sp=0xc82004feb0 
main.main() 
/tmp/cgo/main.go:8 +0xb4 fp=0xc82004ff50 sp=0xc82004fef8 
runtime.main() 
/usr/lib/golang/src/runtime/proc.go:111 +0x2b0 fp=0xc82004ffa0 sp=0xc82004ff50 
runtime.goexit() 
/usr/lib/golang/src/runtime/asm_amd64.s:1696 +0x1 fp=0xc82004ffa8 sp=0xc82004ffa0 

goroutine 17 [syscall, locked to thread]: 
runtime.goexit() 
/usr/lib/golang/src/runtime/asm_amd64.s:1696 +0x1 
+1

Программа c с 'main' обычно не используется в качестве библиотеки. Почему вы пытаетесь это сделать? – JimB

+0

@JimB: Чтобы иметь возможность частично переносить программу C на Go - так, как я ее понимаю, мне нужно сначала вызвать программу во время выполнения Go, а затем я должен уметь говорить C и Go в обоих направлениях. – d33tah

+1

Вам нужна ваша «главная» точка входа в Go. Вероятно, вы должны изменить 'main' в программе C на что-то еще. – JimB

ответ

7

я получаю тот же результат при использовании go run main.go, но при использовании go build ., вы получите лучшее сообщение об ошибке:

/tmp/go-build887090083/cgoSO/_obj/hello.o: In function `main': 
/usr/include/x86_64-linux-gnu/bits/stdio2.h:104: multiple definition of `main' 
/tmp/go-build887090083/cgoSO/_obj/_cgo_main.o:/tmp/go-build887090083/cgoSO/_obj/_cgo_main.c:1: first defined here 
collect2: error: ld returned 1 exit status 

I затем добавили флаг -work в go build, чтобы сохранить рабочее пространство tmp, и вот содержание /tmp/go-build465152107/cgoSO/_obj/_cgo_main.c:

int main() { return 0; } 
void crosscall2(void(*fn)(void*, int), void *a, int c) { } 
void _cgo_wait_runtime_init_done() { } 
char* _cgo_topofstack(void) { return (char*)0; } 
void _cgo_allocate(void *a, int c) { } 
void _cgo_panic(void *a, int c) { } 
void _cgo_reginit(void) { } 

Это первая линия там написано: https://github.com/golang/go/blob/f2e4c8b5fb3660d793b2c545ef207153db0a34b1/src/cmd/cgo/out.go#L49

И нет никаких условий, поэтому он не может быть предотвращено.

Заключение: вы не можете импортировать программу на C, вы можете импортировать библиотеки (это действительно имеет смысл).

Если C - ваш код, то просто измените main на что-нибудь еще, и он будет работать.

Смежные вопросы