// Copyright 2009 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. package main import ( "fmt" "go/ast" "go/printer" "os" "strings" ) func creat(name string) *os.File { f, err := os.Open(name, os.O_WRONLY|os.O_CREAT|os.O_TRUNC, 0666) if err != nil { fatal("%s", err) } return f } // writeDefs creates output files to be compiled by 6g, 6c, and gcc. // (The comments here say 6g and 6c but the code applies to the 8 and 5 tools too.) func (p *Prog) writeDefs() { pkgroot := os.Getenv("GOROOT") + "/pkg/" + os.Getenv("GOOS") + "_" + os.Getenv("GOARCH") path := p.PackagePath if !strings.HasPrefix(path, "/") { path = pkgroot + "/" + path } fgo2 := creat("_cgo_gotypes.go") fc := creat("_cgo_defun.c") // Write second Go output: definitions of _C_xxx. // In a separate file so that the import of "unsafe" does not // pollute the original file. fmt.Fprintf(fgo2, "// Created by cgo - DO NOT EDIT\n") fmt.Fprintf(fgo2, "package %s\n\n", p.Package) fmt.Fprintf(fgo2, "import \"unsafe\"\n\n") fmt.Fprintf(fgo2, "type _ unsafe.Pointer\n\n") for name, def := range p.Typedef { fmt.Fprintf(fgo2, "type %s ", name) printer.Fprint(fgo2, def) fmt.Fprintf(fgo2, "\n") } fmt.Fprintf(fgo2, "type _C_void [0]byte\n") fmt.Fprintf(fc, cProlog, pkgroot, pkgroot, pkgroot, pkgroot) for name, def := range p.Vardef { fmt.Fprintf(fc, "#pragma dynld ·_C_%s %s \"%s.so\"\n", name, name, path) fmt.Fprintf(fgo2, "var _C_%s ", name) printer.Fprint(fgo2, &ast.StarExpr{X: def.Go}) fmt.Fprintf(fgo2, "\n") } fmt.Fprintf(fc, "\n") for name, value := range p.Constdef { fmt.Fprintf(fgo2, "const %s = %s\n", name, value) } for name, value := range p.Enumdef { fmt.Fprintf(fgo2, "const %s = %d\n", name, value) } fmt.Fprintf(fgo2, "\n") for name, def := range p.Funcdef { // Go func declaration. d := &ast.FuncDecl{ Name: ast.NewIdent("_C_" + name), Type: def.Go, } printer.Fprint(fgo2, d) fmt.Fprintf(fgo2, "\n") if name == "CString" || name == "GoString" { // The builtins are already defined in the C prolog. continue } // Construct a gcc struct matching the 6c argument frame. // Assumes that in gcc, char is 1 byte, short 2 bytes, int 4 bytes, long long 8 bytes. // These assumptions are checked by the gccProlog. // Also assumes that 6c convention is to word-align the // input and output parameters. structType := "struct {\n" off := int64(0) npad := 0 for i, t := range def.Params { if off%t.Align != 0 { pad := t.Align - off%t.Align structType += fmt.Sprintf("\t\tchar __pad%d[%d];\n", npad, pad) off += pad npad++ } structType += fmt.Sprintf("\t\t%s p%d;\n", t.C, i) off += t.Size } if off%p.PtrSize != 0 { pad := p.PtrSize - off%p.PtrSize structType += fmt.Sprintf("\t\tchar __pad%d[%d];\n", npad, pad) off += pad npad++ } if t := def.Result; t != nil { if off%t.Align != 0 { pad := t.Align - off%t.Align structType += fmt.Sprintf("\t\tchar __pad%d[%d];\n", npad, pad) off += pad npad++ } structType += fmt.Sprintf("\t\t%s r;\n", t.C) off += t.Size } if off%p.PtrSize != 0 { pad := p.PtrSize - off%p.PtrSize structType += fmt.Sprintf("\t\tchar __pad%d[%d];\n", npad, pad) off += pad npad++ } if len(def.Params) == 0 && def.Result == nil { structType += "\t\tchar unused;\n" // avoid empty struct off++ } structType += "\t}" argSize := off // C wrapper calls into gcc, passing a pointer to the argument frame. // Also emit #pragma to get a pointer to the gcc wrapper. fmt.Fprintf(fc, "#pragma dynld _cgo_%s _cgo_%s \"%s.so\"\n", name, name, path) fmt.Fprintf(fc, "void (*_cgo_%s)(void*);\n", name) fmt.Fprintf(fc, "\n") fmt.Fprintf(fc, "void\n") fmt.Fprintf(fc, "·_C_%s(struct{uint8 x[%d];}p)\n", name, argSize) fmt.Fprintf(fc, "{\n") fmt.Fprintf(fc, "\tcgocall(_cgo_%s, &p);\n", name) fmt.Fprintf(fc, "}\n") fmt.Fprintf(fc, "\n") } fgo2.Close() fc.Close() } // writeOutput creates stubs for a specific source file to be compiled by 6g // (The comments here say 6g and 6c but the code applies to the 8 and 5 tools too.) func (p *Prog) writeOutput(srcfile string) { base := srcfile if strings.HasSuffix(base, ".go") { base = base[0 : len(base)-3] } fgo1 := creat(base + ".cgo1.go") fgcc := creat(base + ".cgo2.c") // Write Go output: Go input with rewrites of C.xxx to _C_xxx. fmt.Fprintf(fgo1, "// Created by cgo - DO NOT EDIT\n") fmt.Fprintf(fgo1, "//line %s:1\n", srcfile) printer.Fprint(fgo1, p.AST) // While we process the vars and funcs, also write 6c and gcc output. // Gcc output starts with the preamble. fmt.Fprintf(fgcc, "%s\n", p.Preamble) fmt.Fprintf(fgcc, "%s\n", gccProlog) for name, def := range p.Funcdef { _, ok := p.OutDefs[name] if name == "CString" || name == "GoString" || ok { // The builtins are already defined in the C prolog, and we don't // want to duplicate function definitions we've already done. continue } p.OutDefs[name] = true // Construct a gcc struct matching the 6c argument frame. // Assumes that in gcc, char is 1 byte, short 2 bytes, int 4 bytes, long long 8 bytes. // These assumptions are checked by the gccProlog. // Also assumes that 6c convention is to word-align the // input and output parameters. structType := "struct {\n" off := int64(0) npad := 0 for i, t := range def.Params { if off%t.Align != 0 { pad := t.Align - off%t.Align structType += fmt.Sprintf("\t\tchar __pad%d[%d];\n", npad, pad) off += pad npad++ } structType += fmt.Sprintf("\t\t%s p%d;\n", t.C, i) off += t.Size } if off%p.PtrSize != 0 { pad := p.PtrSize - off%p.PtrSize structType += fmt.Sprintf("\t\tchar __pad%d[%d];\n", npad, pad) off += pad npad++ } if t := def.Result; t != nil { if off%t.Align != 0 { pad := t.Align - off%t.Align structType += fmt.Sprintf("\t\tchar __pad%d[%d];\n", npad, pad) off += pad npad++ } structType += fmt.Sprintf("\t\t%s r;\n", t.C) off += t.Size } if off%p.PtrSize != 0 { pad := p.PtrSize - off%p.PtrSize structType += fmt.Sprintf("\t\tchar __pad%d[%d];\n", npad, pad) off += pad npad++ } if len(def.Params) == 0 && def.Result == nil { structType += "\t\tchar unused;\n" // avoid empty struct off++ } structType += "\t}" // Gcc wrapper unpacks the C argument struct // and calls the actual C function. fmt.Fprintf(fgcc, "void\n") fmt.Fprintf(fgcc, "_cgo_%s(void *v)\n", name) fmt.Fprintf(fgcc, "{\n") fmt.Fprintf(fgcc, "\t%s *a = v;\n", structType) fmt.Fprintf(fgcc, "\t") if def.Result != nil { fmt.Fprintf(fgcc, "a->r = ") } fmt.Fprintf(fgcc, "%s(", name) for i := range def.Params { if i > 0 { fmt.Fprintf(fgcc, ", ") } fmt.Fprintf(fgcc, "a->p%d", i) } fmt.Fprintf(fgcc, ");\n") fmt.Fprintf(fgcc, "}\n") fmt.Fprintf(fgcc, "\n") } fgo1.Close() fgcc.Close() } const gccProlog = ` // Usual nonsense: if x and y are not equal, the type will be invalid // (have a negative array count) and an inscrutable error will come // out of the compiler and hopefully mention "name". #define __cgo_compile_assert_eq(x, y, name) typedef char name[(x-y)*(x-y)*-2+1]; // Check at compile time that the sizes we use match our expectations. #define __cgo_size_assert(t, n) __cgo_compile_assert_eq(sizeof(t), n, _cgo_sizeof_##t##_is_not_##n) __cgo_size_assert(char, 1) __cgo_size_assert(short, 2) __cgo_size_assert(int, 4) typedef long long __cgo_long_long; __cgo_size_assert(__cgo_long_long, 8) __cgo_size_assert(float, 4) __cgo_size_assert(double, 8) ` const builtinProlog = ` typedef struct { char *p; int n; } _GoString_; _GoString_ GoString(char *p); char *CString(_GoString_); ` const cProlog = ` #include "runtime.h" #include "cgocall.h" #pragma dynld initcgo initcgo "%s/libcgo.so" #pragma dynld libcgo_thread_start libcgo_thread_start "%s/libcgo.so" #pragma dynld _cgo_malloc _cgo_malloc "%s/libcgo.so" #pragma dynld _cgo_free free "%s/libcgo.so" void ·_C_GoString(int8 *p, String s) { s = gostring((byte*)p); FLUSH(&s); } void ·_C_CString(String s, int8 *p) { p = cmalloc(s.len+1); mcpy((byte*)p, s.str, s.len); p[s.len] = 0; FLUSH(&p); } `