Merge pull request #881 from cpunion/cgo-extra
cgo: support macros and callbacks (fpvars)
This commit is contained in:
1
_demo/cgobasic
Symbolic link
1
_demo/cgobasic
Symbolic link
@@ -0,0 +1 @@
|
|||||||
|
../cl/_testgo/cgobasic
|
||||||
1
_demo/cgocfiles
Symbolic link
1
_demo/cgocfiles
Symbolic link
@@ -0,0 +1 @@
|
|||||||
|
../cl/_testgo/cgocfiles
|
||||||
1
_demo/cgodefer
Symbolic link
1
_demo/cgodefer
Symbolic link
@@ -0,0 +1 @@
|
|||||||
|
../cl/_testgo/cgodefer
|
||||||
1
_demo/cgofull
Symbolic link
1
_demo/cgofull
Symbolic link
@@ -0,0 +1 @@
|
|||||||
|
../cl/_testgo/cgofull
|
||||||
1
_demo/cgomacro
Symbolic link
1
_demo/cgomacro
Symbolic link
@@ -0,0 +1 @@
|
|||||||
|
../cl/_testgo/cgomacro
|
||||||
1
_demo/cgopython
Symbolic link
1
_demo/cgopython
Symbolic link
@@ -0,0 +1 @@
|
|||||||
|
../cl/_testgo/cgopython
|
||||||
404
cl/_testgo/cgobasic/out.ll
Normal file
404
cl/_testgo/cgobasic/out.ll
Normal file
@@ -0,0 +1,404 @@
|
|||||||
|
; ModuleID = 'main'
|
||||||
|
source_filename = "main"
|
||||||
|
|
||||||
|
%"github.com/goplus/llgo/internal/runtime.eface" = type { ptr, ptr }
|
||||||
|
%"github.com/goplus/llgo/internal/runtime.String" = type { ptr, i64 }
|
||||||
|
%"github.com/goplus/llgo/internal/runtime.Slice" = type { ptr, i64, i64 }
|
||||||
|
%"github.com/goplus/llgo/internal/runtime.iface" = type { ptr, ptr }
|
||||||
|
|
||||||
|
@"github.com/goplus/llgo/internal/runtime.cgoAlwaysFalse" = external global i1, align 1
|
||||||
|
@_cgo_62905c3ec377_Cfunc__Cmalloc = external global i8, align 1
|
||||||
|
@_cgo_62905c3ec377_Cfunc_cos = external global i8, align 1
|
||||||
|
@_cgo_62905c3ec377_Cfunc_free = external global i8, align 1
|
||||||
|
@_cgo_62905c3ec377_Cfunc_log = external global i8, align 1
|
||||||
|
@_cgo_62905c3ec377_Cfunc_puts = external global i8, align 1
|
||||||
|
@_cgo_62905c3ec377_Cfunc_sin = external global i8, align 1
|
||||||
|
@_cgo_62905c3ec377_Cfunc_sqrt = external global i8, align 1
|
||||||
|
@main._cgo_62905c3ec377_Cfunc__Cmalloc = global ptr null, align 8
|
||||||
|
@main._cgo_62905c3ec377_Cfunc_cos = global ptr null, align 8
|
||||||
|
@main._cgo_62905c3ec377_Cfunc_free = global ptr null, align 8
|
||||||
|
@main._cgo_62905c3ec377_Cfunc_log = global ptr null, align 8
|
||||||
|
@main._cgo_62905c3ec377_Cfunc_puts = global ptr null, align 8
|
||||||
|
@main._cgo_62905c3ec377_Cfunc_sin = global ptr null, align 8
|
||||||
|
@main._cgo_62905c3ec377_Cfunc_sqrt = global ptr null, align 8
|
||||||
|
@"main.init$guard" = global i1 false, align 1
|
||||||
|
@__llgo_argc = global i32 0, align 4
|
||||||
|
@__llgo_argv = global ptr null, align 8
|
||||||
|
@0 = private unnamed_addr constant [13 x i8] c"Hello, World!", align 1
|
||||||
|
@1 = private unnamed_addr constant [29 x i8] c"Converted back to Go string: ", align 1
|
||||||
|
@2 = private unnamed_addr constant [23 x i8] c"Length-limited string: ", align 1
|
||||||
|
@3 = private unnamed_addr constant [33 x i8] c"Converted back to Go byte slice: ", align 1
|
||||||
|
@_llgo_float64 = linkonce global ptr null, align 8
|
||||||
|
@4 = private unnamed_addr constant [14 x i8] c"sqrt(%v) = %v\0A", align 1
|
||||||
|
@5 = private unnamed_addr constant [13 x i8] c"sin(%v) = %v\0A", align 1
|
||||||
|
@6 = private unnamed_addr constant [13 x i8] c"cos(%v) = %v\0A", align 1
|
||||||
|
@7 = private unnamed_addr constant [13 x i8] c"log(%v) = %v\0A", align 1
|
||||||
|
@_llgo_byte = linkonce global ptr null, align 8
|
||||||
|
@"[]_llgo_byte" = linkonce global ptr null, align 8
|
||||||
|
@_llgo_Pointer = linkonce global ptr null, align 8
|
||||||
|
|
||||||
|
define double @main._Cfunc_cos(double %0) {
|
||||||
|
_llgo_0:
|
||||||
|
%1 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 8)
|
||||||
|
%2 = load ptr, ptr @main._cgo_62905c3ec377_Cfunc_cos, align 8
|
||||||
|
%3 = load ptr, ptr %2, align 8
|
||||||
|
%4 = call double %3(double %0)
|
||||||
|
ret double %4
|
||||||
|
}
|
||||||
|
|
||||||
|
define [0 x i8] @main._Cfunc_free(ptr %0) {
|
||||||
|
_llgo_0:
|
||||||
|
%1 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 8)
|
||||||
|
%2 = load ptr, ptr @main._cgo_62905c3ec377_Cfunc_free, align 8
|
||||||
|
%3 = load ptr, ptr %2, align 8
|
||||||
|
%4 = call [0 x i8] %3(ptr %0)
|
||||||
|
ret [0 x i8] %4
|
||||||
|
}
|
||||||
|
|
||||||
|
define double @main._Cfunc_log(double %0) {
|
||||||
|
_llgo_0:
|
||||||
|
%1 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 8)
|
||||||
|
%2 = load ptr, ptr @main._cgo_62905c3ec377_Cfunc_log, align 8
|
||||||
|
%3 = load ptr, ptr %2, align 8
|
||||||
|
%4 = call double %3(double %0)
|
||||||
|
ret double %4
|
||||||
|
}
|
||||||
|
|
||||||
|
define i32 @main._Cfunc_puts(ptr %0) {
|
||||||
|
_llgo_0:
|
||||||
|
%1 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 8)
|
||||||
|
%2 = load ptr, ptr @main._cgo_62905c3ec377_Cfunc_puts, align 8
|
||||||
|
%3 = load ptr, ptr %2, align 8
|
||||||
|
%4 = call i32 %3(ptr %0)
|
||||||
|
ret i32 %4
|
||||||
|
}
|
||||||
|
|
||||||
|
define double @main._Cfunc_sin(double %0) {
|
||||||
|
_llgo_0:
|
||||||
|
%1 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 8)
|
||||||
|
%2 = load ptr, ptr @main._cgo_62905c3ec377_Cfunc_sin, align 8
|
||||||
|
%3 = load ptr, ptr %2, align 8
|
||||||
|
%4 = call double %3(double %0)
|
||||||
|
ret double %4
|
||||||
|
}
|
||||||
|
|
||||||
|
define double @main._Cfunc_sqrt(double %0) {
|
||||||
|
_llgo_0:
|
||||||
|
%1 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 8)
|
||||||
|
%2 = load ptr, ptr @main._cgo_62905c3ec377_Cfunc_sqrt, align 8
|
||||||
|
%3 = load ptr, ptr %2, align 8
|
||||||
|
%4 = call double %3(double %0)
|
||||||
|
ret double %4
|
||||||
|
}
|
||||||
|
|
||||||
|
define ptr @main._Cgo_ptr(ptr %0) {
|
||||||
|
_llgo_0:
|
||||||
|
ret ptr %0
|
||||||
|
}
|
||||||
|
|
||||||
|
declare void @runtime.cgoUse(%"github.com/goplus/llgo/internal/runtime.eface")
|
||||||
|
|
||||||
|
declare void @runtime.cgoCheckResult(%"github.com/goplus/llgo/internal/runtime.eface")
|
||||||
|
|
||||||
|
define void @main.init() {
|
||||||
|
_llgo_0:
|
||||||
|
%0 = load i1, ptr @"main.init$guard", align 1
|
||||||
|
br i1 %0, label %_llgo_2, label %_llgo_1
|
||||||
|
|
||||||
|
_llgo_1: ; preds = %_llgo_0
|
||||||
|
store i1 true, ptr @"main.init$guard", align 1
|
||||||
|
call void @syscall.init()
|
||||||
|
call void @fmt.init()
|
||||||
|
call void @"main.init$after"()
|
||||||
|
store ptr @_cgo_62905c3ec377_Cfunc_cos, ptr @main._cgo_62905c3ec377_Cfunc_cos, align 8
|
||||||
|
store ptr @_cgo_62905c3ec377_Cfunc_free, ptr @main._cgo_62905c3ec377_Cfunc_free, align 8
|
||||||
|
store ptr @_cgo_62905c3ec377_Cfunc_log, ptr @main._cgo_62905c3ec377_Cfunc_log, align 8
|
||||||
|
store ptr @_cgo_62905c3ec377_Cfunc_puts, ptr @main._cgo_62905c3ec377_Cfunc_puts, align 8
|
||||||
|
store ptr @_cgo_62905c3ec377_Cfunc_sin, ptr @main._cgo_62905c3ec377_Cfunc_sin, align 8
|
||||||
|
store ptr @_cgo_62905c3ec377_Cfunc_sqrt, ptr @main._cgo_62905c3ec377_Cfunc_sqrt, align 8
|
||||||
|
store ptr @_cgo_62905c3ec377_Cfunc__Cmalloc, ptr @main._cgo_62905c3ec377_Cfunc__Cmalloc, align 8
|
||||||
|
br label %_llgo_2
|
||||||
|
|
||||||
|
_llgo_2: ; preds = %_llgo_1, %_llgo_0
|
||||||
|
ret void
|
||||||
|
}
|
||||||
|
|
||||||
|
define i32 @main(i32 %0, ptr %1) {
|
||||||
|
_llgo_0:
|
||||||
|
store i32 %0, ptr @__llgo_argc, align 4
|
||||||
|
store ptr %1, ptr @__llgo_argv, align 8
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.init"()
|
||||||
|
call void @main.init()
|
||||||
|
%2 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 8)
|
||||||
|
%3 = call ptr @"github.com/goplus/llgo/internal/runtime.CString"(%"github.com/goplus/llgo/internal/runtime.String" { ptr @0, i64 13 })
|
||||||
|
store ptr %3, ptr %2, align 8
|
||||||
|
%4 = load ptr, ptr %2, align 8
|
||||||
|
%5 = call i32 @main._Cfunc_puts(ptr %4)
|
||||||
|
%6 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 24)
|
||||||
|
%7 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 4)
|
||||||
|
%8 = getelementptr inbounds i8, ptr %7, i64 0
|
||||||
|
store i8 65, ptr %8, align 1
|
||||||
|
%9 = getelementptr inbounds i8, ptr %7, i64 1
|
||||||
|
store i8 66, ptr %9, align 1
|
||||||
|
%10 = getelementptr inbounds i8, ptr %7, i64 2
|
||||||
|
store i8 67, ptr %10, align 1
|
||||||
|
%11 = getelementptr inbounds i8, ptr %7, i64 3
|
||||||
|
store i8 68, ptr %11, align 1
|
||||||
|
%12 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" undef, ptr %7, 0
|
||||||
|
%13 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" %12, i64 4, 1
|
||||||
|
%14 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" %13, i64 4, 2
|
||||||
|
store %"github.com/goplus/llgo/internal/runtime.Slice" %14, ptr %6, align 8
|
||||||
|
%15 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 8)
|
||||||
|
%16 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 8)
|
||||||
|
%17 = getelementptr inbounds { ptr }, ptr %16, i32 0, i32 0
|
||||||
|
store ptr %6, ptr %17, align 8
|
||||||
|
%18 = insertvalue { ptr, ptr } { ptr @"main.main$1", ptr undef }, ptr %16, 1
|
||||||
|
%19 = extractvalue { ptr, ptr } %18, 1
|
||||||
|
%20 = extractvalue { ptr, ptr } %18, 0
|
||||||
|
%21 = call ptr %20(ptr %19)
|
||||||
|
store ptr %21, ptr %15, align 8
|
||||||
|
%22 = load ptr, ptr %2, align 8
|
||||||
|
%23 = call %"github.com/goplus/llgo/internal/runtime.String" @"github.com/goplus/llgo/internal/runtime.GoString"(ptr %22)
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.PrintString"(%"github.com/goplus/llgo/internal/runtime.String" { ptr @1, i64 29 })
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.PrintByte"(i8 32)
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.PrintString"(%"github.com/goplus/llgo/internal/runtime.String" %23)
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.PrintByte"(i8 10)
|
||||||
|
%24 = load ptr, ptr %2, align 8
|
||||||
|
%25 = call %"github.com/goplus/llgo/internal/runtime.String" @"github.com/goplus/llgo/internal/runtime.GoStringN"(ptr %24, i32 5)
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.PrintString"(%"github.com/goplus/llgo/internal/runtime.String" { ptr @2, i64 23 })
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.PrintByte"(i8 32)
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.PrintString"(%"github.com/goplus/llgo/internal/runtime.String" %25)
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.PrintByte"(i8 10)
|
||||||
|
%26 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 8)
|
||||||
|
%27 = getelementptr inbounds { ptr }, ptr %26, i32 0, i32 0
|
||||||
|
store ptr %15, ptr %27, align 8
|
||||||
|
%28 = insertvalue { ptr, ptr } { ptr @"main.main$2", ptr undef }, ptr %26, 1
|
||||||
|
%29 = extractvalue { ptr, ptr } %28, 1
|
||||||
|
%30 = extractvalue { ptr, ptr } %28, 0
|
||||||
|
%31 = call %"github.com/goplus/llgo/internal/runtime.Slice" %30(ptr %29)
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.PrintString"(%"github.com/goplus/llgo/internal/runtime.String" { ptr @3, i64 33 })
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.PrintByte"(i8 32)
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.PrintSlice"(%"github.com/goplus/llgo/internal/runtime.Slice" %31)
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.PrintByte"(i8 10)
|
||||||
|
%32 = call double @main._Cfunc_sqrt(double 2.000000e+00)
|
||||||
|
%33 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 32)
|
||||||
|
%34 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.eface", ptr %33, i64 0
|
||||||
|
%35 = load ptr, ptr @_llgo_float64, align 8
|
||||||
|
%36 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %35, 0
|
||||||
|
%37 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %36, ptr inttoptr (i64 4611686018427387904 to ptr), 1
|
||||||
|
store %"github.com/goplus/llgo/internal/runtime.eface" %37, ptr %34, align 8
|
||||||
|
%38 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.eface", ptr %33, i64 1
|
||||||
|
%39 = load ptr, ptr @_llgo_float64, align 8
|
||||||
|
%40 = bitcast double %32 to i64
|
||||||
|
%41 = inttoptr i64 %40 to ptr
|
||||||
|
%42 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %39, 0
|
||||||
|
%43 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %42, ptr %41, 1
|
||||||
|
store %"github.com/goplus/llgo/internal/runtime.eface" %43, ptr %38, align 8
|
||||||
|
%44 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" undef, ptr %33, 0
|
||||||
|
%45 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" %44, i64 2, 1
|
||||||
|
%46 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" %45, i64 2, 2
|
||||||
|
%47 = call { i64, %"github.com/goplus/llgo/internal/runtime.iface" } @fmt.Printf(%"github.com/goplus/llgo/internal/runtime.String" { ptr @4, i64 14 }, %"github.com/goplus/llgo/internal/runtime.Slice" %46)
|
||||||
|
%48 = call double @main._Cfunc_sin(double 2.000000e+00)
|
||||||
|
%49 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 32)
|
||||||
|
%50 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.eface", ptr %49, i64 0
|
||||||
|
%51 = load ptr, ptr @_llgo_float64, align 8
|
||||||
|
%52 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %51, 0
|
||||||
|
%53 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %52, ptr inttoptr (i64 4611686018427387904 to ptr), 1
|
||||||
|
store %"github.com/goplus/llgo/internal/runtime.eface" %53, ptr %50, align 8
|
||||||
|
%54 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.eface", ptr %49, i64 1
|
||||||
|
%55 = load ptr, ptr @_llgo_float64, align 8
|
||||||
|
%56 = bitcast double %48 to i64
|
||||||
|
%57 = inttoptr i64 %56 to ptr
|
||||||
|
%58 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %55, 0
|
||||||
|
%59 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %58, ptr %57, 1
|
||||||
|
store %"github.com/goplus/llgo/internal/runtime.eface" %59, ptr %54, align 8
|
||||||
|
%60 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" undef, ptr %49, 0
|
||||||
|
%61 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" %60, i64 2, 1
|
||||||
|
%62 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" %61, i64 2, 2
|
||||||
|
%63 = call { i64, %"github.com/goplus/llgo/internal/runtime.iface" } @fmt.Printf(%"github.com/goplus/llgo/internal/runtime.String" { ptr @5, i64 13 }, %"github.com/goplus/llgo/internal/runtime.Slice" %62)
|
||||||
|
%64 = call double @main._Cfunc_cos(double 2.000000e+00)
|
||||||
|
%65 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 32)
|
||||||
|
%66 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.eface", ptr %65, i64 0
|
||||||
|
%67 = load ptr, ptr @_llgo_float64, align 8
|
||||||
|
%68 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %67, 0
|
||||||
|
%69 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %68, ptr inttoptr (i64 4611686018427387904 to ptr), 1
|
||||||
|
store %"github.com/goplus/llgo/internal/runtime.eface" %69, ptr %66, align 8
|
||||||
|
%70 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.eface", ptr %65, i64 1
|
||||||
|
%71 = load ptr, ptr @_llgo_float64, align 8
|
||||||
|
%72 = bitcast double %64 to i64
|
||||||
|
%73 = inttoptr i64 %72 to ptr
|
||||||
|
%74 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %71, 0
|
||||||
|
%75 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %74, ptr %73, 1
|
||||||
|
store %"github.com/goplus/llgo/internal/runtime.eface" %75, ptr %70, align 8
|
||||||
|
%76 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" undef, ptr %65, 0
|
||||||
|
%77 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" %76, i64 2, 1
|
||||||
|
%78 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" %77, i64 2, 2
|
||||||
|
%79 = call { i64, %"github.com/goplus/llgo/internal/runtime.iface" } @fmt.Printf(%"github.com/goplus/llgo/internal/runtime.String" { ptr @6, i64 13 }, %"github.com/goplus/llgo/internal/runtime.Slice" %78)
|
||||||
|
%80 = call double @main._Cfunc_log(double 2.000000e+00)
|
||||||
|
%81 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 32)
|
||||||
|
%82 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.eface", ptr %81, i64 0
|
||||||
|
%83 = load ptr, ptr @_llgo_float64, align 8
|
||||||
|
%84 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %83, 0
|
||||||
|
%85 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %84, ptr inttoptr (i64 4611686018427387904 to ptr), 1
|
||||||
|
store %"github.com/goplus/llgo/internal/runtime.eface" %85, ptr %82, align 8
|
||||||
|
%86 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.eface", ptr %81, i64 1
|
||||||
|
%87 = load ptr, ptr @_llgo_float64, align 8
|
||||||
|
%88 = bitcast double %80 to i64
|
||||||
|
%89 = inttoptr i64 %88 to ptr
|
||||||
|
%90 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %87, 0
|
||||||
|
%91 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %90, ptr %89, 1
|
||||||
|
store %"github.com/goplus/llgo/internal/runtime.eface" %91, ptr %86, align 8
|
||||||
|
%92 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" undef, ptr %81, 0
|
||||||
|
%93 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" %92, i64 2, 1
|
||||||
|
%94 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" %93, i64 2, 2
|
||||||
|
%95 = call { i64, %"github.com/goplus/llgo/internal/runtime.iface" } @fmt.Printf(%"github.com/goplus/llgo/internal/runtime.String" { ptr @7, i64 13 }, %"github.com/goplus/llgo/internal/runtime.Slice" %94)
|
||||||
|
%96 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 8)
|
||||||
|
%97 = getelementptr inbounds { ptr }, ptr %96, i32 0, i32 0
|
||||||
|
store ptr %2, ptr %97, align 8
|
||||||
|
%98 = insertvalue { ptr, ptr } { ptr @"main.main$3", ptr undef }, ptr %96, 1
|
||||||
|
%99 = extractvalue { ptr, ptr } %98, 1
|
||||||
|
%100 = extractvalue { ptr, ptr } %98, 0
|
||||||
|
call void %100(ptr %99)
|
||||||
|
%101 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 8)
|
||||||
|
%102 = getelementptr inbounds { ptr }, ptr %101, i32 0, i32 0
|
||||||
|
store ptr %15, ptr %102, align 8
|
||||||
|
%103 = insertvalue { ptr, ptr } { ptr @"main.main$4", ptr undef }, ptr %101, 1
|
||||||
|
%104 = extractvalue { ptr, ptr } %103, 1
|
||||||
|
%105 = extractvalue { ptr, ptr } %103, 0
|
||||||
|
call void %105(ptr %104)
|
||||||
|
ret i32 0
|
||||||
|
}
|
||||||
|
|
||||||
|
define ptr @"main.main$1"(ptr %0) {
|
||||||
|
_llgo_0:
|
||||||
|
%1 = load { ptr }, ptr %0, align 8
|
||||||
|
%2 = extractvalue { ptr } %1, 0
|
||||||
|
%3 = load %"github.com/goplus/llgo/internal/runtime.Slice", ptr %2, align 8
|
||||||
|
%4 = load ptr, ptr @_llgo_byte, align 8
|
||||||
|
%5 = load ptr, ptr @"[]_llgo_byte", align 8
|
||||||
|
%6 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 24)
|
||||||
|
store %"github.com/goplus/llgo/internal/runtime.Slice" %3, ptr %6, align 8
|
||||||
|
%7 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %5, 0
|
||||||
|
%8 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %7, ptr %6, 1
|
||||||
|
%9 = call ptr @"github.com/goplus/llgo/internal/runtime.CBytes"(%"github.com/goplus/llgo/internal/runtime.Slice" %3)
|
||||||
|
ret ptr %9
|
||||||
|
}
|
||||||
|
|
||||||
|
define %"github.com/goplus/llgo/internal/runtime.Slice" @"main.main$2"(ptr %0) {
|
||||||
|
_llgo_0:
|
||||||
|
%1 = load { ptr }, ptr %0, align 8
|
||||||
|
%2 = extractvalue { ptr } %1, 0
|
||||||
|
%3 = load ptr, ptr %2, align 8
|
||||||
|
%4 = load ptr, ptr @_llgo_Pointer, align 8
|
||||||
|
%5 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %4, 0
|
||||||
|
%6 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %5, ptr %3, 1
|
||||||
|
%7 = call %"github.com/goplus/llgo/internal/runtime.Slice" @"github.com/goplus/llgo/internal/runtime.GoBytes"(ptr %3, i32 4)
|
||||||
|
ret %"github.com/goplus/llgo/internal/runtime.Slice" %7
|
||||||
|
}
|
||||||
|
|
||||||
|
define void @"main.main$3"(ptr %0) {
|
||||||
|
_llgo_0:
|
||||||
|
%1 = load { ptr }, ptr %0, align 8
|
||||||
|
%2 = extractvalue { ptr } %1, 0
|
||||||
|
%3 = load ptr, ptr %2, align 8
|
||||||
|
%4 = load ptr, ptr @_llgo_Pointer, align 8
|
||||||
|
%5 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %4, 0
|
||||||
|
%6 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %5, ptr %3, 1
|
||||||
|
%7 = call [0 x i8] @main._Cfunc_free(ptr %3)
|
||||||
|
ret void
|
||||||
|
}
|
||||||
|
|
||||||
|
define void @"main.main$4"(ptr %0) {
|
||||||
|
_llgo_0:
|
||||||
|
%1 = load { ptr }, ptr %0, align 8
|
||||||
|
%2 = extractvalue { ptr } %1, 0
|
||||||
|
%3 = load ptr, ptr %2, align 8
|
||||||
|
%4 = load ptr, ptr @_llgo_Pointer, align 8
|
||||||
|
%5 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %4, 0
|
||||||
|
%6 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %5, ptr %3, 1
|
||||||
|
%7 = call [0 x i8] @main._Cfunc_free(ptr %3)
|
||||||
|
ret void
|
||||||
|
}
|
||||||
|
|
||||||
|
declare void @runtime.throw(%"github.com/goplus/llgo/internal/runtime.String")
|
||||||
|
|
||||||
|
declare ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64)
|
||||||
|
|
||||||
|
declare void @syscall.init()
|
||||||
|
|
||||||
|
declare void @fmt.init()
|
||||||
|
|
||||||
|
declare void @"github.com/goplus/llgo/internal/runtime.init"()
|
||||||
|
|
||||||
|
declare ptr @"github.com/goplus/llgo/internal/runtime.CString"(%"github.com/goplus/llgo/internal/runtime.String")
|
||||||
|
|
||||||
|
declare ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64)
|
||||||
|
|
||||||
|
declare %"github.com/goplus/llgo/internal/runtime.String" @"github.com/goplus/llgo/internal/runtime.GoString"(ptr)
|
||||||
|
|
||||||
|
declare void @"github.com/goplus/llgo/internal/runtime.PrintString"(%"github.com/goplus/llgo/internal/runtime.String")
|
||||||
|
|
||||||
|
declare void @"github.com/goplus/llgo/internal/runtime.PrintByte"(i8)
|
||||||
|
|
||||||
|
declare %"github.com/goplus/llgo/internal/runtime.String" @"github.com/goplus/llgo/internal/runtime.GoStringN"(ptr, i64)
|
||||||
|
|
||||||
|
declare void @"github.com/goplus/llgo/internal/runtime.PrintSlice"(%"github.com/goplus/llgo/internal/runtime.Slice")
|
||||||
|
|
||||||
|
define void @"main.init$after"() {
|
||||||
|
_llgo_0:
|
||||||
|
%0 = load ptr, ptr @_llgo_float64, align 8
|
||||||
|
%1 = icmp eq ptr %0, null
|
||||||
|
br i1 %1, label %_llgo_1, label %_llgo_2
|
||||||
|
|
||||||
|
_llgo_1: ; preds = %_llgo_0
|
||||||
|
%2 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 46)
|
||||||
|
store ptr %2, ptr @_llgo_float64, align 8
|
||||||
|
br label %_llgo_2
|
||||||
|
|
||||||
|
_llgo_2: ; preds = %_llgo_1, %_llgo_0
|
||||||
|
%3 = load ptr, ptr @_llgo_byte, align 8
|
||||||
|
%4 = icmp eq ptr %3, null
|
||||||
|
br i1 %4, label %_llgo_3, label %_llgo_4
|
||||||
|
|
||||||
|
_llgo_3: ; preds = %_llgo_2
|
||||||
|
%5 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 40)
|
||||||
|
store ptr %5, ptr @_llgo_byte, align 8
|
||||||
|
br label %_llgo_4
|
||||||
|
|
||||||
|
_llgo_4: ; preds = %_llgo_3, %_llgo_2
|
||||||
|
%6 = load ptr, ptr @"[]_llgo_byte", align 8
|
||||||
|
%7 = icmp eq ptr %6, null
|
||||||
|
br i1 %7, label %_llgo_5, label %_llgo_6
|
||||||
|
|
||||||
|
_llgo_5: ; preds = %_llgo_4
|
||||||
|
%8 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 40)
|
||||||
|
%9 = call ptr @"github.com/goplus/llgo/internal/runtime.SliceOf"(ptr %8)
|
||||||
|
store ptr %9, ptr @"[]_llgo_byte", align 8
|
||||||
|
br label %_llgo_6
|
||||||
|
|
||||||
|
_llgo_6: ; preds = %_llgo_5, %_llgo_4
|
||||||
|
%10 = load ptr, ptr @_llgo_Pointer, align 8
|
||||||
|
%11 = icmp eq ptr %10, null
|
||||||
|
br i1 %11, label %_llgo_7, label %_llgo_8
|
||||||
|
|
||||||
|
_llgo_7: ; preds = %_llgo_6
|
||||||
|
%12 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 58)
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.SetDirectIface"(ptr %12)
|
||||||
|
store ptr %12, ptr @_llgo_Pointer, align 8
|
||||||
|
br label %_llgo_8
|
||||||
|
|
||||||
|
_llgo_8: ; preds = %_llgo_7, %_llgo_6
|
||||||
|
ret void
|
||||||
|
}
|
||||||
|
|
||||||
|
declare ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64)
|
||||||
|
|
||||||
|
declare { i64, %"github.com/goplus/llgo/internal/runtime.iface" } @fmt.Printf(%"github.com/goplus/llgo/internal/runtime.String", %"github.com/goplus/llgo/internal/runtime.Slice")
|
||||||
|
|
||||||
|
declare ptr @"github.com/goplus/llgo/internal/runtime.SliceOf"(ptr)
|
||||||
|
|
||||||
|
declare ptr @"github.com/goplus/llgo/internal/runtime.CBytes"(%"github.com/goplus/llgo/internal/runtime.Slice")
|
||||||
|
|
||||||
|
declare void @"github.com/goplus/llgo/internal/runtime.SetDirectIface"(ptr)
|
||||||
|
|
||||||
|
declare %"github.com/goplus/llgo/internal/runtime.Slice" @"github.com/goplus/llgo/internal/runtime.GoBytes"(ptr, i64)
|
||||||
180
cl/_testgo/cgocfiles/out.ll
Normal file
180
cl/_testgo/cgocfiles/out.ll
Normal file
@@ -0,0 +1,180 @@
|
|||||||
|
; ModuleID = 'main'
|
||||||
|
source_filename = "main"
|
||||||
|
|
||||||
|
%"github.com/goplus/llgo/internal/runtime.eface" = type { ptr, ptr }
|
||||||
|
%main._Ctype_struct___3 = type { i32 }
|
||||||
|
%main._Ctype_struct___4 = type { i32, i32 }
|
||||||
|
%main._Ctype_struct___0 = type { i32, i32, i32 }
|
||||||
|
%main._Ctype_struct___1 = type { i32, i32, i32, i32 }
|
||||||
|
%main._Ctype_struct___2 = type { i32, i32, i32, i32, i32 }
|
||||||
|
%"github.com/goplus/llgo/internal/runtime.Slice" = type { ptr, i64, i64 }
|
||||||
|
%"github.com/goplus/llgo/internal/runtime.iface" = type { ptr, ptr }
|
||||||
|
%"github.com/goplus/llgo/internal/runtime.String" = type { ptr, i64 }
|
||||||
|
|
||||||
|
@"github.com/goplus/llgo/internal/runtime.cgoAlwaysFalse" = external global i1, align 1
|
||||||
|
@_cgo_35faaf752e93_Cfunc_test_structs = external global i8, align 1
|
||||||
|
@main._cgo_35faaf752e93_Cfunc_test_structs = global ptr null, align 8
|
||||||
|
@"main.init$guard" = global i1 false, align 1
|
||||||
|
@__llgo_argc = global i32 0, align 4
|
||||||
|
@__llgo_argv = global ptr null, align 8
|
||||||
|
@_llgo_main._Ctype_int = linkonce global ptr null, align 8
|
||||||
|
@0 = private unnamed_addr constant [15 x i8] c"main._Ctype_int", align 1
|
||||||
|
@_llgo_int32 = linkonce global ptr null, align 8
|
||||||
|
@1 = private unnamed_addr constant [4 x i8] c"main", align 1
|
||||||
|
@2 = private unnamed_addr constant [10 x i8] c"_Ctype_int", align 1
|
||||||
|
@3 = private unnamed_addr constant [19 x i8] c"test_structs failed", align 1
|
||||||
|
@_llgo_string = linkonce global ptr null, align 8
|
||||||
|
|
||||||
|
define i32 @main._Cfunc_test_structs(ptr %0, ptr %1, ptr %2, ptr %3, ptr %4) {
|
||||||
|
_llgo_0:
|
||||||
|
%5 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 8)
|
||||||
|
%6 = load ptr, ptr @main._cgo_35faaf752e93_Cfunc_test_structs, align 8
|
||||||
|
%7 = load ptr, ptr %6, align 8
|
||||||
|
%8 = call i32 %7(ptr %0, ptr %1, ptr %2, ptr %3, ptr %4)
|
||||||
|
ret i32 %8
|
||||||
|
}
|
||||||
|
|
||||||
|
define ptr @main._Cgo_ptr(ptr %0) {
|
||||||
|
_llgo_0:
|
||||||
|
ret ptr %0
|
||||||
|
}
|
||||||
|
|
||||||
|
declare void @runtime.cgoUse(%"github.com/goplus/llgo/internal/runtime.eface")
|
||||||
|
|
||||||
|
declare void @runtime.cgoCheckResult(%"github.com/goplus/llgo/internal/runtime.eface")
|
||||||
|
|
||||||
|
define void @main.init() {
|
||||||
|
_llgo_0:
|
||||||
|
%0 = load i1, ptr @"main.init$guard", align 1
|
||||||
|
br i1 %0, label %_llgo_2, label %_llgo_1
|
||||||
|
|
||||||
|
_llgo_1: ; preds = %_llgo_0
|
||||||
|
store i1 true, ptr @"main.init$guard", align 1
|
||||||
|
call void @syscall.init()
|
||||||
|
call void @fmt.init()
|
||||||
|
call void @"main.init$after"()
|
||||||
|
store ptr @_cgo_35faaf752e93_Cfunc_test_structs, ptr @main._cgo_35faaf752e93_Cfunc_test_structs, align 8
|
||||||
|
br label %_llgo_2
|
||||||
|
|
||||||
|
_llgo_2: ; preds = %_llgo_1, %_llgo_0
|
||||||
|
ret void
|
||||||
|
}
|
||||||
|
|
||||||
|
define i32 @main(i32 %0, ptr %1) {
|
||||||
|
_llgo_0:
|
||||||
|
store i32 %0, ptr @__llgo_argc, align 4
|
||||||
|
store ptr %1, ptr @__llgo_argv, align 8
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.init"()
|
||||||
|
call void @main.init()
|
||||||
|
%2 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 4)
|
||||||
|
%3 = getelementptr inbounds %main._Ctype_struct___3, ptr %2, i32 0, i32 0
|
||||||
|
store i32 1, ptr %3, align 4
|
||||||
|
%4 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 8)
|
||||||
|
%5 = getelementptr inbounds %main._Ctype_struct___4, ptr %4, i32 0, i32 0
|
||||||
|
%6 = getelementptr inbounds %main._Ctype_struct___4, ptr %4, i32 0, i32 1
|
||||||
|
store i32 1, ptr %5, align 4
|
||||||
|
store i32 2, ptr %6, align 4
|
||||||
|
%7 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 12)
|
||||||
|
%8 = getelementptr inbounds %main._Ctype_struct___0, ptr %7, i32 0, i32 0
|
||||||
|
%9 = getelementptr inbounds %main._Ctype_struct___0, ptr %7, i32 0, i32 1
|
||||||
|
%10 = getelementptr inbounds %main._Ctype_struct___0, ptr %7, i32 0, i32 2
|
||||||
|
store i32 1, ptr %8, align 4
|
||||||
|
store i32 2, ptr %9, align 4
|
||||||
|
store i32 3, ptr %10, align 4
|
||||||
|
%11 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 16)
|
||||||
|
%12 = getelementptr inbounds %main._Ctype_struct___1, ptr %11, i32 0, i32 0
|
||||||
|
%13 = getelementptr inbounds %main._Ctype_struct___1, ptr %11, i32 0, i32 1
|
||||||
|
%14 = getelementptr inbounds %main._Ctype_struct___1, ptr %11, i32 0, i32 2
|
||||||
|
%15 = getelementptr inbounds %main._Ctype_struct___1, ptr %11, i32 0, i32 3
|
||||||
|
store i32 1, ptr %12, align 4
|
||||||
|
store i32 2, ptr %13, align 4
|
||||||
|
store i32 3, ptr %14, align 4
|
||||||
|
store i32 4, ptr %15, align 4
|
||||||
|
%16 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 20)
|
||||||
|
%17 = getelementptr inbounds %main._Ctype_struct___2, ptr %16, i32 0, i32 0
|
||||||
|
%18 = getelementptr inbounds %main._Ctype_struct___2, ptr %16, i32 0, i32 1
|
||||||
|
%19 = getelementptr inbounds %main._Ctype_struct___2, ptr %16, i32 0, i32 2
|
||||||
|
%20 = getelementptr inbounds %main._Ctype_struct___2, ptr %16, i32 0, i32 3
|
||||||
|
%21 = getelementptr inbounds %main._Ctype_struct___2, ptr %16, i32 0, i32 4
|
||||||
|
store i32 1, ptr %17, align 4
|
||||||
|
store i32 2, ptr %18, align 4
|
||||||
|
store i32 3, ptr %19, align 4
|
||||||
|
store i32 4, ptr %20, align 4
|
||||||
|
store i32 5, ptr %21, align 4
|
||||||
|
%22 = call i32 @main._Cfunc_test_structs(ptr %2, ptr %4, ptr %7, ptr %11, ptr %16)
|
||||||
|
%23 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64 16)
|
||||||
|
%24 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.eface", ptr %23, i64 0
|
||||||
|
%25 = load ptr, ptr @_llgo_main._Ctype_int, align 8
|
||||||
|
%26 = sext i32 %22 to i64
|
||||||
|
%27 = inttoptr i64 %26 to ptr
|
||||||
|
%28 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %25, 0
|
||||||
|
%29 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %28, ptr %27, 1
|
||||||
|
store %"github.com/goplus/llgo/internal/runtime.eface" %29, ptr %24, align 8
|
||||||
|
%30 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" undef, ptr %23, 0
|
||||||
|
%31 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" %30, i64 1, 1
|
||||||
|
%32 = insertvalue %"github.com/goplus/llgo/internal/runtime.Slice" %31, i64 1, 2
|
||||||
|
%33 = call { i64, %"github.com/goplus/llgo/internal/runtime.iface" } @fmt.Println(%"github.com/goplus/llgo/internal/runtime.Slice" %32)
|
||||||
|
%34 = icmp ne i32 %22, 35
|
||||||
|
br i1 %34, label %_llgo_1, label %_llgo_2
|
||||||
|
|
||||||
|
_llgo_1: ; preds = %_llgo_0
|
||||||
|
%35 = load ptr, ptr @_llgo_string, align 8
|
||||||
|
%36 = call ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64 16)
|
||||||
|
store %"github.com/goplus/llgo/internal/runtime.String" { ptr @3, i64 19 }, ptr %36, align 8
|
||||||
|
%37 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" undef, ptr %35, 0
|
||||||
|
%38 = insertvalue %"github.com/goplus/llgo/internal/runtime.eface" %37, ptr %36, 1
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.Panic"(%"github.com/goplus/llgo/internal/runtime.eface" %38)
|
||||||
|
unreachable
|
||||||
|
|
||||||
|
_llgo_2: ; preds = %_llgo_0
|
||||||
|
ret i32 0
|
||||||
|
}
|
||||||
|
|
||||||
|
declare ptr @"github.com/goplus/llgo/internal/runtime.AllocZ"(i64)
|
||||||
|
|
||||||
|
declare void @syscall.init()
|
||||||
|
|
||||||
|
declare void @fmt.init()
|
||||||
|
|
||||||
|
declare void @"github.com/goplus/llgo/internal/runtime.init"()
|
||||||
|
|
||||||
|
define void @"main.init$after"() {
|
||||||
|
_llgo_0:
|
||||||
|
%0 = call ptr @"github.com/goplus/llgo/internal/runtime.NewNamed"(%"github.com/goplus/llgo/internal/runtime.String" { ptr @0, i64 15 }, i64 5, i64 4, i64 0, i64 0)
|
||||||
|
store ptr %0, ptr @_llgo_main._Ctype_int, align 8
|
||||||
|
%1 = load ptr, ptr @_llgo_int32, align 8
|
||||||
|
%2 = icmp eq ptr %1, null
|
||||||
|
br i1 %2, label %_llgo_1, label %_llgo_2
|
||||||
|
|
||||||
|
_llgo_1: ; preds = %_llgo_0
|
||||||
|
%3 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 37)
|
||||||
|
store ptr %3, ptr @_llgo_int32, align 8
|
||||||
|
br label %_llgo_2
|
||||||
|
|
||||||
|
_llgo_2: ; preds = %_llgo_1, %_llgo_0
|
||||||
|
%4 = load ptr, ptr @_llgo_int32, align 8
|
||||||
|
call void @"github.com/goplus/llgo/internal/runtime.InitNamed"(ptr %0, %"github.com/goplus/llgo/internal/runtime.String" { ptr @1, i64 4 }, %"github.com/goplus/llgo/internal/runtime.String" { ptr @2, i64 10 }, ptr %4, { ptr, i64, i64 } zeroinitializer, { ptr, i64, i64 } zeroinitializer)
|
||||||
|
%5 = load ptr, ptr @_llgo_string, align 8
|
||||||
|
%6 = icmp eq ptr %5, null
|
||||||
|
br i1 %6, label %_llgo_3, label %_llgo_4
|
||||||
|
|
||||||
|
_llgo_3: ; preds = %_llgo_2
|
||||||
|
%7 = call ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64 24)
|
||||||
|
store ptr %7, ptr @_llgo_string, align 8
|
||||||
|
br label %_llgo_4
|
||||||
|
|
||||||
|
_llgo_4: ; preds = %_llgo_3, %_llgo_2
|
||||||
|
ret void
|
||||||
|
}
|
||||||
|
|
||||||
|
declare ptr @"github.com/goplus/llgo/internal/runtime.NewNamed"(%"github.com/goplus/llgo/internal/runtime.String", i64, i64, i64, i64)
|
||||||
|
|
||||||
|
declare ptr @"github.com/goplus/llgo/internal/runtime.Basic"(i64)
|
||||||
|
|
||||||
|
declare void @"github.com/goplus/llgo/internal/runtime.InitNamed"(ptr, %"github.com/goplus/llgo/internal/runtime.String", %"github.com/goplus/llgo/internal/runtime.String", ptr, %"github.com/goplus/llgo/internal/runtime.Slice", %"github.com/goplus/llgo/internal/runtime.Slice")
|
||||||
|
|
||||||
|
declare { i64, %"github.com/goplus/llgo/internal/runtime.iface" } @fmt.Println(%"github.com/goplus/llgo/internal/runtime.Slice")
|
||||||
|
|
||||||
|
declare ptr @"github.com/goplus/llgo/internal/runtime.AllocU"(i64)
|
||||||
|
|
||||||
|
declare void @"github.com/goplus/llgo/internal/runtime.Panic"(%"github.com/goplus/llgo/internal/runtime.eface")
|
||||||
1
cl/_testgo/cgodefer/out.ll
Normal file
1
cl/_testgo/cgodefer/out.ll
Normal file
@@ -0,0 +1 @@
|
|||||||
|
;
|
||||||
@@ -6,7 +6,9 @@ package main
|
|||||||
#cgo windows,amd64 CFLAGS: -D_WIN64
|
#cgo windows,amd64 CFLAGS: -D_WIN64
|
||||||
#cgo linux,amd64 CFLAGS: -D_LINUX64
|
#cgo linux,amd64 CFLAGS: -D_LINUX64
|
||||||
#cgo !windows,amd64 CFLAGS: -D_UNIX64
|
#cgo !windows,amd64 CFLAGS: -D_UNIX64
|
||||||
|
#cgo pkg-config: python3-embed
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
#include <Python.h>
|
||||||
#include "foo.h"
|
#include "foo.h"
|
||||||
typedef struct {
|
typedef struct {
|
||||||
int a;
|
int a;
|
||||||
@@ -71,9 +73,52 @@ static void test_macros() {
|
|||||||
printf("UNIX64 is defined\n");
|
printf("UNIX64 is defined\n");
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#define MY_VERSION "1.0.0"
|
||||||
|
#define MY_CODE 0x12345678
|
||||||
|
|
||||||
|
static void test_void() {
|
||||||
|
printf("test_void\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef int (*Cb)(int);
|
||||||
|
|
||||||
|
extern int go_callback(int);
|
||||||
|
|
||||||
|
extern int c_callback(int i);
|
||||||
|
|
||||||
|
static void test_callback(Cb cb) {
|
||||||
|
printf("test_callback, cb: %p, go_callback: %p, c_callback: %p\n", cb, go_callback, c_callback);
|
||||||
|
printf("test_callback, *cb: %p, *go_callback: %p, *c_callback: %p\n", *(void**)cb, *(void**)(go_callback), *(void**)(c_callback));
|
||||||
|
printf("cb result: %d\n", cb(123));
|
||||||
|
printf("done\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
extern int go_callback_not_use_in_go(int);
|
||||||
|
|
||||||
|
static void run_callback() {
|
||||||
|
test_callback(c_callback);
|
||||||
|
test_callback(go_callback_not_use_in_go);
|
||||||
|
}
|
||||||
*/
|
*/
|
||||||
import "C"
|
import "C"
|
||||||
import "fmt"
|
import (
|
||||||
|
"fmt"
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
"github.com/goplus/llgo/cl/_testgo/cgofull/pymod1"
|
||||||
|
"github.com/goplus/llgo/cl/_testgo/cgofull/pymod2"
|
||||||
|
)
|
||||||
|
|
||||||
|
//export go_callback_not_use_in_go
|
||||||
|
func go_callback_not_use_in_go(i C.int) C.int {
|
||||||
|
return i + 1
|
||||||
|
}
|
||||||
|
|
||||||
|
//export go_callback
|
||||||
|
func go_callback(i C.int) C.int {
|
||||||
|
return i + 1
|
||||||
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
runPy()
|
runPy()
|
||||||
@@ -86,10 +131,27 @@ func main() {
|
|||||||
if r != 35 {
|
if r != 35 {
|
||||||
panic("test_structs failed")
|
panic("test_structs failed")
|
||||||
}
|
}
|
||||||
|
fmt.Println(C.MY_VERSION)
|
||||||
|
fmt.Println(int(C.MY_CODE))
|
||||||
|
C.test_void()
|
||||||
|
|
||||||
|
println("call run_callback")
|
||||||
|
C.run_callback()
|
||||||
|
|
||||||
|
// test _Cgo_ptr and _cgoCheckResult
|
||||||
|
println("call with go_callback")
|
||||||
|
C.test_callback((C.Cb)(C.go_callback))
|
||||||
|
|
||||||
|
println("call with c_callback")
|
||||||
|
C.test_callback((C.Cb)(C.c_callback))
|
||||||
}
|
}
|
||||||
|
|
||||||
func runPy() {
|
func runPy() {
|
||||||
Initialize()
|
Initialize()
|
||||||
defer Finalize()
|
defer Finalize()
|
||||||
Run("print('Hello, Python!')")
|
Run("print('Hello, Python!')")
|
||||||
|
C.PyObject_Print((*C.PyObject)(unsafe.Pointer(pymod1.Float(1.23))), C.stderr, 0)
|
||||||
|
C.PyObject_Print((*C.PyObject)(unsafe.Pointer(pymod2.Long(123))), C.stdout, 0)
|
||||||
|
// test _Cgo_use
|
||||||
|
C.PyObject_Print((*C.PyObject)(unsafe.Pointer(C.PyComplex_FromDoubles(C.double(1.23), C.double(4.56)))), C.stdout, 0)
|
||||||
}
|
}
|
||||||
@@ -1,6 +1,12 @@
|
|||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include "foo.h"
|
#include "foo.h"
|
||||||
|
|
||||||
void print_foo(Foo* f) {
|
void print_foo(Foo *f)
|
||||||
|
{
|
||||||
printf("print_foo: %d\n", f->a);
|
printf("print_foo: %d\n", f->a);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int c_callback(int i)
|
||||||
|
{
|
||||||
|
return i + 1;
|
||||||
|
}
|
||||||
1
cl/_testgo/cgofull/out.ll
Normal file
1
cl/_testgo/cgofull/out.ll
Normal file
@@ -0,0 +1 @@
|
|||||||
|
;
|
||||||
11
cl/_testgo/cgofull/pymod1/pymod1.go
Normal file
11
cl/_testgo/cgofull/pymod1/pymod1.go
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
package pymod1
|
||||||
|
|
||||||
|
/*
|
||||||
|
#cgo pkg-config: python3-embed
|
||||||
|
#include <Python.h>
|
||||||
|
*/
|
||||||
|
import "C"
|
||||||
|
|
||||||
|
func Float(f float64) *C.PyObject {
|
||||||
|
return C.PyFloat_FromDouble(C.double(f))
|
||||||
|
}
|
||||||
11
cl/_testgo/cgofull/pymod2/pymod2.go
Normal file
11
cl/_testgo/cgofull/pymod2/pymod2.go
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
package pymod2
|
||||||
|
|
||||||
|
/*
|
||||||
|
#cgo pkg-config: python3-embed
|
||||||
|
#include <Python.h>
|
||||||
|
*/
|
||||||
|
import "C"
|
||||||
|
|
||||||
|
func Long(l int64) *C.PyObject {
|
||||||
|
return C.PyLong_FromLongLong(C.longlong(l))
|
||||||
|
}
|
||||||
31
cl/_testgo/cgomacro/cgomacro.go
Normal file
31
cl/_testgo/cgomacro/cgomacro.go
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
/*
|
||||||
|
#cgo pkg-config: python3-embed
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <Python.h>
|
||||||
|
|
||||||
|
void test_stdout() {
|
||||||
|
printf("stdout ptr: %p\n", stdout);
|
||||||
|
fputs("outputs to stdout\n", stdout);
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
import "C"
|
||||||
|
import (
|
||||||
|
"unsafe"
|
||||||
|
|
||||||
|
"github.com/goplus/llgo/c"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
C.test_stdout()
|
||||||
|
C.fputs((*C.char)(unsafe.Pointer(c.Str("hello\n"))), C.stdout)
|
||||||
|
C.Py_Initialize()
|
||||||
|
defer C.Py_Finalize()
|
||||||
|
C.PyObject_Print(C.Py_True, C.stdout, 0)
|
||||||
|
C.fputs((*C.char)(unsafe.Pointer(c.Str("\n"))), C.stdout)
|
||||||
|
C.PyObject_Print(C.Py_False, C.stdout, 0)
|
||||||
|
C.fputs((*C.char)(unsafe.Pointer(c.Str("\n"))), C.stdout)
|
||||||
|
C.PyObject_Print(C.Py_None, C.stdout, 0)
|
||||||
|
C.fputs((*C.char)(unsafe.Pointer(c.Str("\n"))), C.stdout)
|
||||||
|
}
|
||||||
1
cl/_testgo/cgomacro/out.ll
Normal file
1
cl/_testgo/cgomacro/out.ll
Normal file
@@ -0,0 +1 @@
|
|||||||
|
;
|
||||||
1
cl/_testgo/cgopython/out.ll
Normal file
1
cl/_testgo/cgopython/out.ll
Normal file
@@ -0,0 +1 @@
|
|||||||
|
;
|
||||||
@@ -1,64 +0,0 @@
|
|||||||
; ModuleID = 'main'
|
|
||||||
source_filename = "main"
|
|
||||||
|
|
||||||
%"github.com/goplus/llgo/internal/runtime.String" = type { ptr, i64 }
|
|
||||||
|
|
||||||
@"main.init$guard" = global i1 false, align 1
|
|
||||||
@0 = private unnamed_addr constant [5 x i8] c"hello", align 1
|
|
||||||
@__llgo_argc = global i32 0, align 4
|
|
||||||
@__llgo_argv = global ptr null, align 8
|
|
||||||
|
|
||||||
define i64 @main.Foo(%"github.com/goplus/llgo/internal/runtime.String" %0) {
|
|
||||||
_llgo_0:
|
|
||||||
%1 = extractvalue %"github.com/goplus/llgo/internal/runtime.String" %0, 1
|
|
||||||
ret i64 %1
|
|
||||||
}
|
|
||||||
|
|
||||||
define void @main.Test() {
|
|
||||||
_llgo_0:
|
|
||||||
br label %_llgo_3
|
|
||||||
|
|
||||||
_llgo_1: ; preds = %_llgo_3
|
|
||||||
%0 = alloca %"github.com/goplus/llgo/internal/runtime.String", align 8
|
|
||||||
%1 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %0, i32 0, i32 0
|
|
||||||
store ptr @0, ptr %1, align 8
|
|
||||||
%2 = getelementptr inbounds %"github.com/goplus/llgo/internal/runtime.String", ptr %0, i32 0, i32 1
|
|
||||||
store i64 5, ptr %2, align 4
|
|
||||||
%3 = load %"github.com/goplus/llgo/internal/runtime.String", ptr %0, align 8
|
|
||||||
%4 = call i64 @main.Foo(%"github.com/goplus/llgo/internal/runtime.String" %3)
|
|
||||||
%5 = add i64 %6, 1
|
|
||||||
br label %_llgo_3
|
|
||||||
|
|
||||||
_llgo_2: ; preds = %_llgo_3
|
|
||||||
ret void
|
|
||||||
|
|
||||||
_llgo_3: ; preds = %_llgo_1, %_llgo_0
|
|
||||||
%6 = phi i64 [ 0, %_llgo_0 ], [ %5, %_llgo_1 ]
|
|
||||||
%7 = icmp slt i64 %6, 1000000
|
|
||||||
br i1 %7, label %_llgo_1, label %_llgo_2
|
|
||||||
}
|
|
||||||
|
|
||||||
define void @main.init() {
|
|
||||||
_llgo_0:
|
|
||||||
%0 = load i1, ptr @"main.init$guard", align 1
|
|
||||||
br i1 %0, label %_llgo_2, label %_llgo_1
|
|
||||||
|
|
||||||
_llgo_1: ; preds = %_llgo_0
|
|
||||||
store i1 true, ptr @"main.init$guard", align 1
|
|
||||||
br label %_llgo_2
|
|
||||||
|
|
||||||
_llgo_2: ; preds = %_llgo_1, %_llgo_0
|
|
||||||
ret void
|
|
||||||
}
|
|
||||||
|
|
||||||
define i32 @main(i32 %0, ptr %1) {
|
|
||||||
_llgo_0:
|
|
||||||
store i32 %0, ptr @__llgo_argc, align 4
|
|
||||||
store ptr %1, ptr @__llgo_argv, align 8
|
|
||||||
call void @"github.com/goplus/llgo/internal/runtime.init"()
|
|
||||||
call void @main.init()
|
|
||||||
call void @main.Test()
|
|
||||||
ret i32 0
|
|
||||||
}
|
|
||||||
|
|
||||||
declare void @"github.com/goplus/llgo/internal/runtime.init"()
|
|
||||||
@@ -1,12 +1,23 @@
|
|||||||
; ModuleID = 'main'
|
; ModuleID = 'main'
|
||||||
source_filename = "main"
|
source_filename = "main"
|
||||||
|
|
||||||
|
%"github.com/goplus/llgo/internal/runtime.eface" = type { ptr, ptr }
|
||||||
|
|
||||||
@"github.com/goplus/llgo/internal/runtime.cgoAlwaysFalse" = external global i1, align 1
|
@"github.com/goplus/llgo/internal/runtime.cgoAlwaysFalse" = external global i1, align 1
|
||||||
@main.format = global [10 x i8] zeroinitializer, align 1
|
@main.format = global [10 x i8] zeroinitializer, align 1
|
||||||
@"main.init$guard" = global i1 false, align 1
|
@"main.init$guard" = global i1 false, align 1
|
||||||
@__llgo_argc = global i32 0, align 4
|
@__llgo_argc = global i32 0, align 4
|
||||||
@__llgo_argv = global ptr null, align 8
|
@__llgo_argv = global ptr null, align 8
|
||||||
|
|
||||||
|
define ptr @main._Cgo_ptr(ptr %0) {
|
||||||
|
_llgo_0:
|
||||||
|
ret ptr %0
|
||||||
|
}
|
||||||
|
|
||||||
|
declare void @runtime.cgoUse(%"github.com/goplus/llgo/internal/runtime.eface")
|
||||||
|
|
||||||
|
declare void @runtime.cgoCheckResult(%"github.com/goplus/llgo/internal/runtime.eface")
|
||||||
|
|
||||||
define void @main.init() {
|
define void @main.init() {
|
||||||
_llgo_0:
|
_llgo_0:
|
||||||
%0 = load i1, ptr @"main.init$guard", align 1
|
%0 = load i1, ptr @"main.init$guard", align 1
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
source_filename = "main"
|
source_filename = "main"
|
||||||
|
|
||||||
%main.Foo = type { i32, i1 }
|
%main.Foo = type { i32, i1 }
|
||||||
|
%"github.com/goplus/llgo/internal/runtime.eface" = type { ptr, ptr }
|
||||||
|
|
||||||
@"github.com/goplus/llgo/internal/runtime.cgoAlwaysFalse" = external global i1, align 1
|
@"github.com/goplus/llgo/internal/runtime.cgoAlwaysFalse" = external global i1, align 1
|
||||||
@main.format = global [10 x i8] zeroinitializer, align 1
|
@main.format = global [10 x i8] zeroinitializer, align 1
|
||||||
@@ -35,6 +36,15 @@ _llgo_0:
|
|||||||
ret void
|
ret void
|
||||||
}
|
}
|
||||||
|
|
||||||
|
define ptr @main._Cgo_ptr(ptr %0) {
|
||||||
|
_llgo_0:
|
||||||
|
ret ptr %0
|
||||||
|
}
|
||||||
|
|
||||||
|
declare void @runtime.cgoUse(%"github.com/goplus/llgo/internal/runtime.eface")
|
||||||
|
|
||||||
|
declare void @runtime.cgoCheckResult(%"github.com/goplus/llgo/internal/runtime.eface")
|
||||||
|
|
||||||
define void @main.init() {
|
define void @main.init() {
|
||||||
_llgo_0:
|
_llgo_0:
|
||||||
%0 = load i1, ptr @"main.init$guard", align 1
|
%0 = load i1, ptr @"main.init$guard", align 1
|
||||||
|
|||||||
@@ -1,6 +1,8 @@
|
|||||||
; ModuleID = 'main'
|
; ModuleID = 'main'
|
||||||
source_filename = "main"
|
source_filename = "main"
|
||||||
|
|
||||||
|
%"github.com/goplus/llgo/internal/runtime.eface" = type { ptr, ptr }
|
||||||
|
|
||||||
@"github.com/goplus/llgo/internal/runtime.cgoAlwaysFalse" = external global i1, align 1
|
@"github.com/goplus/llgo/internal/runtime.cgoAlwaysFalse" = external global i1, align 1
|
||||||
@main.format = global [10 x i8] zeroinitializer, align 1
|
@main.format = global [10 x i8] zeroinitializer, align 1
|
||||||
@"main.init$guard" = global i1 false, align 1
|
@"main.init$guard" = global i1 false, align 1
|
||||||
@@ -23,6 +25,15 @@ _llgo_2: ; preds = %_llgo_1, %_llgo_0
|
|||||||
ret void
|
ret void
|
||||||
}
|
}
|
||||||
|
|
||||||
|
define ptr @main._Cgo_ptr(ptr %0) {
|
||||||
|
_llgo_0:
|
||||||
|
ret ptr %0
|
||||||
|
}
|
||||||
|
|
||||||
|
declare void @runtime.cgoUse(%"github.com/goplus/llgo/internal/runtime.eface")
|
||||||
|
|
||||||
|
declare void @runtime.cgoCheckResult(%"github.com/goplus/llgo/internal/runtime.eface")
|
||||||
|
|
||||||
define void @main.init() {
|
define void @main.init() {
|
||||||
_llgo_0:
|
_llgo_0:
|
||||||
%0 = load i1, ptr @"main.init$guard", align 1
|
%0 = load i1, ptr @"main.init$guard", align 1
|
||||||
|
|||||||
@@ -110,10 +110,11 @@ type context struct {
|
|||||||
inCFunc bool
|
inCFunc bool
|
||||||
skipall bool
|
skipall bool
|
||||||
|
|
||||||
cgoCalled bool
|
cgoCalled bool
|
||||||
cgoArgs []llssa.Expr
|
cgoArgs []llssa.Expr
|
||||||
cgoRet llssa.Expr
|
cgoRet llssa.Expr
|
||||||
cgoFuncs map[string][]string
|
cgoSymbols []string
|
||||||
|
cgoExports map[string]string
|
||||||
}
|
}
|
||||||
|
|
||||||
type pkgState byte
|
type pkgState byte
|
||||||
@@ -189,8 +190,21 @@ var (
|
|||||||
argvTy = types.NewPointer(types.NewPointer(types.Typ[types.Int8]))
|
argvTy = types.NewPointer(types.NewPointer(types.Typ[types.Int8]))
|
||||||
)
|
)
|
||||||
|
|
||||||
func isCgoCfunc(f *ssa.Function) bool {
|
func isCgoExternSymbol(f *ssa.Function) bool {
|
||||||
return strings.HasPrefix(f.Name(), "_Cfunc_")
|
name := f.Name()
|
||||||
|
return isCgoCfunc(name) || isCgoCmacro(name)
|
||||||
|
}
|
||||||
|
|
||||||
|
func isCgoCfunc(name string) bool {
|
||||||
|
return strings.HasPrefix(name, "_Cfunc_")
|
||||||
|
}
|
||||||
|
|
||||||
|
func isCgoCmacro(name string) bool {
|
||||||
|
return strings.HasPrefix(name, "_Cmacro_")
|
||||||
|
}
|
||||||
|
|
||||||
|
func isCgoVar(name string) bool {
|
||||||
|
return strings.HasPrefix(name, "__cgo_")
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *context) compileFuncDecl(pkg llssa.Package, f *ssa.Function) (llssa.Function, llssa.PyObjRef, int) {
|
func (p *context) compileFuncDecl(pkg llssa.Package, f *ssa.Function) (llssa.Function, llssa.PyObjRef, int) {
|
||||||
@@ -246,10 +260,11 @@ func (p *context) compileFuncDecl(pkg llssa.Package, f *ssa.Function) (llssa.Fun
|
|||||||
fn.Inline(llssa.NoInline)
|
fn.Inline(llssa.NoInline)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
isCgo := isCgoExternSymbol(f)
|
||||||
if nblk := len(f.Blocks); nblk > 0 {
|
if nblk := len(f.Blocks); nblk > 0 {
|
||||||
p.cgoCalled = false
|
p.cgoCalled = false
|
||||||
p.cgoArgs = nil
|
p.cgoArgs = nil
|
||||||
if isCgoCfunc(f) {
|
if isCgo {
|
||||||
fn.MakeBlocks(1)
|
fn.MakeBlocks(1)
|
||||||
} else {
|
} else {
|
||||||
fn.MakeBlocks(nblk) // to set fn.HasBody() = true
|
fn.MakeBlocks(nblk) // to set fn.HasBody() = true
|
||||||
@@ -278,7 +293,7 @@ func (p *context) compileFuncDecl(pkg llssa.Package, f *ssa.Function) (llssa.Fun
|
|||||||
}
|
}
|
||||||
p.bvals = make(map[ssa.Value]llssa.Expr)
|
p.bvals = make(map[ssa.Value]llssa.Expr)
|
||||||
off := make([]int, len(f.Blocks))
|
off := make([]int, len(f.Blocks))
|
||||||
if isCgoCfunc(f) {
|
if isCgo {
|
||||||
p.cgoArgs = make([]llssa.Expr, len(f.Params))
|
p.cgoArgs = make([]llssa.Expr, len(f.Params))
|
||||||
for i, param := range f.Params {
|
for i, param := range f.Params {
|
||||||
p.cgoArgs[i] = p.compileValue(b, param)
|
p.cgoArgs[i] = p.compileValue(b, param)
|
||||||
@@ -295,7 +310,7 @@ func (p *context) compileFuncDecl(pkg llssa.Package, f *ssa.Function) (llssa.Fun
|
|||||||
doMainInit := (i == 0 && name == "main")
|
doMainInit := (i == 0 && name == "main")
|
||||||
doModInit := (i == 1 && isInit)
|
doModInit := (i == 1 && isInit)
|
||||||
p.compileBlock(b, block, off[i], doMainInit, doModInit)
|
p.compileBlock(b, block, off[i], doMainInit, doModInit)
|
||||||
if isCgoCfunc(f) {
|
if isCgo {
|
||||||
// just process first block for performance
|
// just process first block for performance
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
@@ -401,44 +416,45 @@ func (p *context) compileBlock(b llssa.Builder, block *ssa.BasicBlock, n int, do
|
|||||||
callRuntimeInit(b, pkg)
|
callRuntimeInit(b, pkg)
|
||||||
b.Call(pkg.FuncOf("main.init").Expr)
|
b.Call(pkg.FuncOf("main.init").Expr)
|
||||||
}
|
}
|
||||||
fname := p.goProg.Fset.Position(block.Parent().Pos()).Filename
|
fnName := block.Parent().Name()
|
||||||
if p.cgoFuncs == nil {
|
|
||||||
p.cgoFuncs = make(map[string][]string)
|
|
||||||
}
|
|
||||||
var cgoFuncs []string
|
|
||||||
if funcs, ok := p.cgoFuncs[fname]; ok {
|
|
||||||
cgoFuncs = funcs
|
|
||||||
}
|
|
||||||
cgoReturned := false
|
cgoReturned := false
|
||||||
|
isCgoCfunc := isCgoCfunc(fnName)
|
||||||
|
isCgoCmacro := isCgoCmacro(fnName)
|
||||||
for i, instr := range instrs {
|
for i, instr := range instrs {
|
||||||
if i == 1 && doModInit && p.state == pkgInPatch { // in patch package but no pkgFNoOldInit
|
if i == 1 && doModInit && p.state == pkgInPatch { // in patch package but no pkgFNoOldInit
|
||||||
initFnNameOld := initFnNameOfHasPatch(p.fn.Name())
|
initFnNameOld := initFnNameOfHasPatch(p.fn.Name())
|
||||||
fnOld := pkg.NewFunc(initFnNameOld, llssa.NoArgsNoRet, llssa.InC)
|
fnOld := pkg.NewFunc(initFnNameOld, llssa.NoArgsNoRet, llssa.InC)
|
||||||
b.Call(fnOld.Expr)
|
b.Call(fnOld.Expr)
|
||||||
}
|
}
|
||||||
if isCgoCfunc(block.Parent()) {
|
if isCgoCfunc || isCgoCmacro {
|
||||||
switch instr := instr.(type) {
|
switch instr := instr.(type) {
|
||||||
case *ssa.Alloc:
|
case *ssa.Alloc:
|
||||||
// return value allocation
|
// return value allocation
|
||||||
p.compileInstr(b, instr)
|
p.compileInstr(b, instr)
|
||||||
case *ssa.UnOp:
|
case *ssa.UnOp:
|
||||||
// load cgo function pointer
|
// load cgo function pointer
|
||||||
if instr.Op == token.MUL && strings.HasPrefix(instr.X.Name(), "_cgo_") {
|
varName := instr.X.Name()
|
||||||
cgoFuncs = append(cgoFuncs, instr.X.Name())
|
if instr.Op == token.MUL && strings.HasPrefix(varName, "_cgo_") {
|
||||||
p.cgoFuncs[fname] = cgoFuncs
|
p.cgoSymbols = append(p.cgoSymbols, varName)
|
||||||
p.compileInstr(b, instr)
|
p.compileInstr(b, instr)
|
||||||
}
|
}
|
||||||
case *ssa.Call:
|
case *ssa.Call:
|
||||||
// call c function
|
if isCgoCmacro {
|
||||||
p.compileInstr(b, instr)
|
p.cgoRet = p.compileValue(b, instr.Call.Args[0])
|
||||||
p.cgoCalled = true
|
p.cgoCalled = true
|
||||||
|
} else {
|
||||||
|
// call c function
|
||||||
|
p.compileInstr(b, instr)
|
||||||
|
p.cgoCalled = true
|
||||||
|
}
|
||||||
case *ssa.Return:
|
case *ssa.Return:
|
||||||
// return cgo function result
|
// return cgo function result
|
||||||
if len(instr.Results) > 0 {
|
if isCgoCmacro {
|
||||||
b.Return(p.cgoRet)
|
ty := p.type_(instr.Results[0].Type(), llssa.InGo)
|
||||||
} else {
|
p.cgoRet.Type = p.prog.Pointer(ty)
|
||||||
b.Return(llssa.Nil)
|
p.cgoRet = b.Load(p.cgoRet)
|
||||||
}
|
}
|
||||||
|
b.Return(p.cgoRet)
|
||||||
cgoReturned = true
|
cgoReturned = true
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
@@ -446,18 +462,14 @@ func (p *context) compileBlock(b llssa.Builder, block *ssa.BasicBlock, n int, do
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// is cgo cfunc but not return yet, some funcs has multiple blocks
|
// is cgo cfunc but not return yet, some funcs has multiple blocks
|
||||||
if isCgoCfunc(block.Parent()) && !cgoReturned {
|
if (isCgoCfunc || isCgoCmacro) && !cgoReturned {
|
||||||
if !p.cgoCalled {
|
if !p.cgoCalled {
|
||||||
panic("cgo cfunc not called")
|
panic("cgo cfunc not called")
|
||||||
}
|
}
|
||||||
for _, block := range block.Parent().Blocks {
|
for _, block := range block.Parent().Blocks {
|
||||||
for _, instr := range block.Instrs {
|
for _, instr := range block.Instrs {
|
||||||
if instr, ok := instr.(*ssa.Return); ok {
|
if _, ok := instr.(*ssa.Return); ok {
|
||||||
if len(instr.Results) > 0 {
|
b.Return(p.cgoRet)
|
||||||
b.Return(p.cgoRet)
|
|
||||||
} else {
|
|
||||||
b.Return(llssa.Nil)
|
|
||||||
}
|
|
||||||
goto end
|
goto end
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -902,7 +914,11 @@ func (p *context) compileValue(b llssa.Builder, v ssa.Value) llssa.Expr {
|
|||||||
}
|
}
|
||||||
return pyFn.Expr
|
return pyFn.Expr
|
||||||
case *ssa.Global:
|
case *ssa.Global:
|
||||||
|
varName := v.Name()
|
||||||
val := p.varOf(b, v)
|
val := p.varOf(b, v)
|
||||||
|
if isCgoVar(varName) {
|
||||||
|
p.cgoSymbols = append(p.cgoSymbols, val.Name())
|
||||||
|
}
|
||||||
if debugSymbols {
|
if debugSymbols {
|
||||||
pos := p.fset.Position(v.Pos())
|
pos := p.fset.Position(v.Pos())
|
||||||
b.DIGlobal(val, v.Name(), pos)
|
b.DIGlobal(val, v.Name(), pos)
|
||||||
@@ -971,7 +987,7 @@ func NewPackage(prog llssa.Program, pkg *ssa.Package, files []*ast.File) (ret ll
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NewPackageEx compiles a Go package to LLVM IR package.
|
// NewPackageEx compiles a Go package to LLVM IR package.
|
||||||
func NewPackageEx(prog llssa.Program, patches Patches, pkg *ssa.Package, files []*ast.File) (ret llssa.Package, externs map[string][]string, err error) {
|
func NewPackageEx(prog llssa.Program, patches Patches, pkg *ssa.Package, files []*ast.File) (ret llssa.Package, externs []string, err error) {
|
||||||
pkgProg := pkg.Prog
|
pkgProg := pkg.Prog
|
||||||
pkgTypes := pkg.Pkg
|
pkgTypes := pkg.Pkg
|
||||||
oldTypes := pkgTypes
|
oldTypes := pkgTypes
|
||||||
@@ -1003,6 +1019,8 @@ func NewPackageEx(prog llssa.Program, patches Patches, pkg *ssa.Package, files [
|
|||||||
loaded: map[*types.Package]*pkgInfo{
|
loaded: map[*types.Package]*pkgInfo{
|
||||||
types.Unsafe: {kind: PkgDeclOnly}, // TODO(xsw): PkgNoInit or PkgDeclOnly?
|
types.Unsafe: {kind: PkgDeclOnly}, // TODO(xsw): PkgNoInit or PkgDeclOnly?
|
||||||
},
|
},
|
||||||
|
cgoExports: make(map[string]string),
|
||||||
|
cgoSymbols: make([]string, 0, 128),
|
||||||
}
|
}
|
||||||
ctx.initPyModule()
|
ctx.initPyModule()
|
||||||
ctx.initFiles(pkgPath, files)
|
ctx.initFiles(pkgPath, files)
|
||||||
@@ -1036,7 +1054,13 @@ func NewPackageEx(prog llssa.Program, patches Patches, pkg *ssa.Package, files [
|
|||||||
ctx.initAfter = nil
|
ctx.initAfter = nil
|
||||||
fn()
|
fn()
|
||||||
}
|
}
|
||||||
externs = ctx.cgoFuncs
|
externs = ctx.cgoSymbols
|
||||||
|
for fnName, exportName := range ctx.cgoExports {
|
||||||
|
fn := ret.FuncOf(fnName)
|
||||||
|
if fn != nil {
|
||||||
|
fn.SetName(exportName)
|
||||||
|
}
|
||||||
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
34
cl/import.go
34
cl/import.go
@@ -246,9 +246,10 @@ func (p *context) initLinknameByDoc(doc *ast.CommentGroup, fullName, inPkgName s
|
|||||||
|
|
||||||
func (p *context) initLinkname(line string, f func(inPkgName string) (fullName string, isVar, ok bool)) {
|
func (p *context) initLinkname(line string, f func(inPkgName string) (fullName string, isVar, ok bool)) {
|
||||||
const (
|
const (
|
||||||
linkname = "//go:linkname "
|
linkname = "//go:linkname "
|
||||||
llgolink = "//llgo:link "
|
llgolink = "//llgo:link "
|
||||||
llgolink2 = "// llgo:link "
|
llgolink2 = "// llgo:link "
|
||||||
|
exportName = "//export "
|
||||||
)
|
)
|
||||||
if strings.HasPrefix(line, linkname) {
|
if strings.HasPrefix(line, linkname) {
|
||||||
p.initLink(line, len(linkname), f)
|
p.initLink(line, len(linkname), f)
|
||||||
@@ -256,6 +257,15 @@ func (p *context) initLinkname(line string, f func(inPkgName string) (fullName s
|
|||||||
p.initLink(line, len(llgolink2), f)
|
p.initLink(line, len(llgolink2), f)
|
||||||
} else if strings.HasPrefix(line, llgolink) {
|
} else if strings.HasPrefix(line, llgolink) {
|
||||||
p.initLink(line, len(llgolink), f)
|
p.initLink(line, len(llgolink), f)
|
||||||
|
} else if strings.HasPrefix(line, exportName) {
|
||||||
|
p.initCgoExport(line, len(exportName), f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *context) initCgoExport(line string, prefix int, f func(inPkgName string) (fullName string, isVar, ok bool)) {
|
||||||
|
name := strings.TrimSpace(line[prefix:])
|
||||||
|
if fullName, _, ok := f(name); ok {
|
||||||
|
p.cgoExports[fullName] = name
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -366,6 +376,17 @@ func checkCgo(fnName string) bool {
|
|||||||
(fnName[4] == '_' || strings.HasPrefix(fnName[4:], "Check"))
|
(fnName[4] == '_' || strings.HasPrefix(fnName[4:], "Check"))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var cgoIgnoredNames = map[string]none{
|
||||||
|
"_Cgo_ptr": {},
|
||||||
|
"_Cgo_use": {},
|
||||||
|
"_cgoCheckResult": {},
|
||||||
|
}
|
||||||
|
|
||||||
|
func cgoIgnored(fnName string) bool {
|
||||||
|
_, ok := cgoIgnoredNames[fnName]
|
||||||
|
return ok
|
||||||
|
}
|
||||||
|
|
||||||
const (
|
const (
|
||||||
ignoredFunc = iota
|
ignoredFunc = iota
|
||||||
goFunc = int(llssa.InGo)
|
goFunc = int(llssa.InGo)
|
||||||
@@ -421,7 +442,6 @@ const (
|
|||||||
llgoCgoCMalloc = llgoCgoBase + 0x5
|
llgoCgoCMalloc = llgoCgoBase + 0x5
|
||||||
llgoCgoCheckPointer = llgoCgoBase + 0x6
|
llgoCgoCheckPointer = llgoCgoBase + 0x6
|
||||||
llgoCgoCgocall = llgoCgoBase + 0x7
|
llgoCgoCgocall = llgoCgoBase + 0x7
|
||||||
llgoCgoUse = llgoCgoBase + 0x8
|
|
||||||
|
|
||||||
llgoAtomicOpLast = llgoAtomicOpBase + int(llssa.OpUMin)
|
llgoAtomicOpLast = llgoAtomicOpBase + int(llssa.OpUMin)
|
||||||
)
|
)
|
||||||
@@ -435,10 +455,10 @@ func (p *context) funcName(fn *ssa.Function, ignore bool) (*types.Package, strin
|
|||||||
orgName = funcName(pkg, origin, true)
|
orgName = funcName(pkg, origin, true)
|
||||||
} else {
|
} else {
|
||||||
fname := fn.Name()
|
fname := fn.Name()
|
||||||
if checkCgo(fname) {
|
if checkCgo(fname) && !cgoIgnored(fname) {
|
||||||
return nil, fname, llgoInstr
|
return nil, fname, llgoInstr
|
||||||
}
|
}
|
||||||
if isCgoCfunc(fn) {
|
if isCgoExternSymbol(fn) {
|
||||||
if _, ok := llgoInstrs[fname]; ok {
|
if _, ok := llgoInstrs[fname]; ok {
|
||||||
return nil, fname, llgoInstr
|
return nil, fname, llgoInstr
|
||||||
}
|
}
|
||||||
@@ -450,7 +470,7 @@ func (p *context) funcName(fn *ssa.Function, ignore bool) (*types.Package, strin
|
|||||||
}
|
}
|
||||||
p.ensureLoaded(pkg)
|
p.ensureLoaded(pkg)
|
||||||
orgName = funcName(pkg, fn, false)
|
orgName = funcName(pkg, fn, false)
|
||||||
if ignore && ignoreName(orgName) || checkCgo(fn.Name()) {
|
if ignore && ignoreName(orgName) {
|
||||||
return nil, orgName, ignoredFunc
|
return nil, orgName, ignoredFunc
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -130,11 +130,6 @@ func (p *context) cgoCgocall(b llssa.Builder, args []ssa.Value) (ret llssa.Expr)
|
|||||||
return p.cgoRet
|
return p.cgoRet
|
||||||
}
|
}
|
||||||
|
|
||||||
// func _Cgo_use(v any)
|
|
||||||
func (p *context) cgoUse(b llssa.Builder, args []ssa.Value) {
|
|
||||||
// don't need to do anything
|
|
||||||
}
|
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
// func index(arr *T, idx int) T
|
// func index(arr *T, idx int) T
|
||||||
@@ -324,7 +319,6 @@ var llgoInstrs = map[string]int{
|
|||||||
"_Cfunc__CMalloc": llgoCgoCMalloc,
|
"_Cfunc__CMalloc": llgoCgoCMalloc,
|
||||||
"_cgoCheckPointer": llgoCgoCheckPointer,
|
"_cgoCheckPointer": llgoCgoCheckPointer,
|
||||||
"_cgo_runtime_cgocall": llgoCgoCgocall,
|
"_cgo_runtime_cgocall": llgoCgoCgocall,
|
||||||
"_Cgo_use": llgoCgoUse,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// funcOf returns a function by name and set ftype = goFunc, cFunc, etc.
|
// funcOf returns a function by name and set ftype = goFunc, cFunc, etc.
|
||||||
@@ -468,8 +462,6 @@ func (p *context) call(b llssa.Builder, act llssa.DoAction, call *ssa.CallCommon
|
|||||||
p.cgoCheckPointer(b, args)
|
p.cgoCheckPointer(b, args)
|
||||||
case llgoCgoCgocall:
|
case llgoCgoCgocall:
|
||||||
p.cgoCgocall(b, args)
|
p.cgoCgocall(b, args)
|
||||||
case llgoCgoUse:
|
|
||||||
p.cgoUse(b, args)
|
|
||||||
case llgoAdvance:
|
case llgoAdvance:
|
||||||
ret = p.advance(b, args)
|
ret = p.advance(b, args)
|
||||||
case llgoIndex:
|
case llgoIndex:
|
||||||
|
|||||||
@@ -525,6 +525,7 @@ func buildPkg(ctx *context, aPkg *aPackage, verbose bool) (cgoLdflags []string,
|
|||||||
cl.SetDebug(0)
|
cl.SetDebug(0)
|
||||||
}
|
}
|
||||||
check(err)
|
check(err)
|
||||||
|
aPkg.LPkg = ret
|
||||||
cgoLdflags, err = buildCgo(ctx, aPkg, aPkg.Package.Syntax, externs, verbose)
|
cgoLdflags, err = buildCgo(ctx, aPkg, aPkg.Package.Syntax, externs, verbose)
|
||||||
if needLLFile(ctx.mode) {
|
if needLLFile(ctx.mode) {
|
||||||
pkg.ExportFile += ".ll"
|
pkg.ExportFile += ".ll"
|
||||||
@@ -533,7 +534,6 @@ func buildPkg(ctx *context, aPkg *aPackage, verbose bool) (cgoLdflags []string,
|
|||||||
fmt.Fprintf(os.Stderr, "==> Export %s: %s\n", aPkg.PkgPath, pkg.ExportFile)
|
fmt.Fprintf(os.Stderr, "==> Export %s: %s\n", aPkg.PkgPath, pkg.ExportFile)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
aPkg.LPkg = ret
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -21,6 +21,7 @@ import (
|
|||||||
"fmt"
|
"fmt"
|
||||||
"go/ast"
|
"go/ast"
|
||||||
"go/token"
|
"go/token"
|
||||||
|
"go/types"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
@@ -29,6 +30,7 @@ import (
|
|||||||
|
|
||||||
"github.com/goplus/llgo/internal/buildtags"
|
"github.com/goplus/llgo/internal/buildtags"
|
||||||
"github.com/goplus/llgo/internal/safesplit"
|
"github.com/goplus/llgo/internal/safesplit"
|
||||||
|
llssa "github.com/goplus/llgo/ssa"
|
||||||
)
|
)
|
||||||
|
|
||||||
type cgoDecl struct {
|
type cgoDecl struct {
|
||||||
@@ -51,7 +53,7 @@ static void* _Cmalloc(size_t size) {
|
|||||||
`
|
`
|
||||||
)
|
)
|
||||||
|
|
||||||
func buildCgo(ctx *context, pkg *aPackage, files []*ast.File, externs map[string][]string, verbose bool) (cgoLdflags []string, err error) {
|
func buildCgo(ctx *context, pkg *aPackage, files []*ast.File, externs []string, verbose bool) (cgoLdflags []string, err error) {
|
||||||
cfiles, preambles, cdecls, err := parseCgo_(pkg, files)
|
cfiles, preambles, cdecls, err := parseCgo_(pkg, files)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
@@ -88,20 +90,27 @@ func buildCgo(ctx *context, pkg *aPackage, files []*ast.File, externs map[string
|
|||||||
cgoLdflags = append(cgoLdflags, linkFile)
|
cgoLdflags = append(cgoLdflags, linkFile)
|
||||||
}, verbose)
|
}, verbose)
|
||||||
}
|
}
|
||||||
re := regexp.MustCompile(`^(_cgo_[^_]+_Cfunc_)(.*)$`)
|
re := regexp.MustCompile(`^(_cgo_[^_]+_(Cfunc|Cmacro)_)(.*)$`)
|
||||||
cgoFuncs := make(map[string]string)
|
cgoSymbols := make(map[string]string)
|
||||||
mallocFix := false
|
mallocFix := false
|
||||||
for _, funcs := range externs {
|
for _, symbolName := range externs {
|
||||||
for _, funcName := range funcs {
|
lastPart := symbolName
|
||||||
if m := re.FindStringSubmatch(funcName); len(m) > 0 {
|
lastDot := strings.LastIndex(symbolName, ".")
|
||||||
cgoFuncs[funcName] = m[2]
|
if lastDot != -1 {
|
||||||
// fix missing _cgo_9113e32b6599_Cfunc__Cmalloc
|
lastPart = symbolName[lastDot+1:]
|
||||||
if !mallocFix {
|
}
|
||||||
pkgPrefix := m[1]
|
if strings.HasPrefix(lastPart, "__cgo_") {
|
||||||
mallocName := pkgPrefix + "_Cmalloc"
|
// func ptr var: main.__cgo_func_name
|
||||||
cgoFuncs[mallocName] = "_Cmalloc"
|
cgoSymbols[symbolName] = lastPart
|
||||||
mallocFix = true
|
} else if m := re.FindStringSubmatch(symbolName); len(m) > 0 {
|
||||||
}
|
prefix := m[1] // _cgo_hash_(Cfunc|Cmacro)_
|
||||||
|
name := m[3] // remaining part
|
||||||
|
cgoSymbols[symbolName] = name
|
||||||
|
// fix missing _cgo_9113e32b6599_Cfunc__Cmalloc
|
||||||
|
if !mallocFix && m[2] == "Cfunc" {
|
||||||
|
mallocName := prefix + "_Cmalloc"
|
||||||
|
cgoSymbols[mallocName] = "_Cmalloc"
|
||||||
|
mallocFix = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -113,7 +122,10 @@ func buildCgo(ctx *context, pkg *aPackage, files []*ast.File, externs map[string
|
|||||||
tmpName := tmpFile.Name()
|
tmpName := tmpFile.Name()
|
||||||
defer os.Remove(tmpName)
|
defer os.Remove(tmpName)
|
||||||
code := cgoHeader + "\n\n" + preamble.src
|
code := cgoHeader + "\n\n" + preamble.src
|
||||||
externDecls := genExternDeclsByClang(code, cflags, cgoFuncs)
|
externDecls, err := genExternDeclsByClang(pkg, code, cflags, cgoSymbols)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
if err = os.WriteFile(tmpName, []byte(code+"\n\n"+externDecls), 0644); err != nil {
|
if err = os.WriteFile(tmpName, []byte(code+"\n\n"+externDecls), 0644); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
@@ -134,42 +146,108 @@ type clangASTNode struct {
|
|||||||
Inner []clangASTNode `json:"inner,omitempty"`
|
Inner []clangASTNode `json:"inner,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
func genExternDeclsByClang(src string, cflags []string, cgoFuncs map[string]string) string {
|
func genExternDeclsByClang(pkg *aPackage, src string, cflags []string, cgoSymbols map[string]string) (string, error) {
|
||||||
tmpSrc, err := os.CreateTemp("", "cgo-src-*.c")
|
tmpSrc, err := os.CreateTemp("", "cgo-src-*.c")
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return ""
|
return "", err
|
||||||
}
|
}
|
||||||
defer os.Remove(tmpSrc.Name())
|
defer os.Remove(tmpSrc.Name())
|
||||||
if err := os.WriteFile(tmpSrc.Name(), []byte(src), 0644); err != nil {
|
if err := os.WriteFile(tmpSrc.Name(), []byte(src), 0644); err != nil {
|
||||||
return ""
|
return "", err
|
||||||
}
|
}
|
||||||
args := append([]string{"-Xclang", "-ast-dump=json", "-fsyntax-only"}, cflags...)
|
symbolNames := make(map[string]bool)
|
||||||
args = append(args, tmpSrc.Name())
|
if err := getFuncNames(tmpSrc.Name(), cflags, symbolNames); err != nil {
|
||||||
cmd := exec.Command("clang", args...)
|
return "", err
|
||||||
output, err := cmd.Output()
|
|
||||||
if err != nil {
|
|
||||||
return ""
|
|
||||||
}
|
}
|
||||||
var astRoot clangASTNode
|
macroNames := make(map[string]bool)
|
||||||
if err := json.Unmarshal(output, &astRoot); err != nil {
|
if err := getMacroNames(tmpSrc.Name(), cflags, macroNames); err != nil {
|
||||||
return ""
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
||||||
funcNames := make(map[string]bool)
|
|
||||||
extractFuncNames(&astRoot, funcNames)
|
|
||||||
|
|
||||||
b := strings.Builder{}
|
b := strings.Builder{}
|
||||||
var toRemove []string
|
var toRemove []string
|
||||||
for cgoFunc, funcName := range cgoFuncs {
|
for cgoName, symbolName := range cgoSymbols {
|
||||||
if funcNames[funcName] {
|
if strings.HasPrefix(symbolName, "__cgo_") {
|
||||||
b.WriteString(fmt.Sprintf("void* %s = (void*)%s;\n", cgoFunc, funcName))
|
gofuncName := strings.Replace(cgoName, ".__cgo_", ".", 1)
|
||||||
toRemove = append(toRemove, cgoFunc)
|
gofn := pkg.LPkg.FuncOf(gofuncName)
|
||||||
|
cgoVar := pkg.LPkg.VarOf(cgoName)
|
||||||
|
if gofn != nil {
|
||||||
|
cgoVar.ReplaceAllUsesWith(gofn.Expr)
|
||||||
|
} else {
|
||||||
|
cfuncName := symbolName[len("__cgo_"):]
|
||||||
|
cfn := pkg.LPkg.NewFunc(cfuncName, types.NewSignatureType(nil, nil, nil, nil, nil, false), llssa.InC)
|
||||||
|
cgoVar.ReplaceAllUsesWith(cfn.Expr)
|
||||||
|
}
|
||||||
|
toRemove = append(toRemove, cgoName)
|
||||||
|
} else {
|
||||||
|
usePtr := ""
|
||||||
|
if symbolNames[symbolName] {
|
||||||
|
usePtr = "*"
|
||||||
|
} else if !macroNames[symbolName] {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
/* template:
|
||||||
|
typeof(fputs)* _cgo_1574167f3838_Cfunc_fputs;
|
||||||
|
|
||||||
|
__attribute__((constructor))
|
||||||
|
static void _init__cgo_1574167f3838_Cfunc_fputs() {
|
||||||
|
_cgo_1574167f3838_Cfunc_fputs = fputs;
|
||||||
|
}*/
|
||||||
|
b.WriteString(fmt.Sprintf(`
|
||||||
|
typeof(%s)%s %s;
|
||||||
|
|
||||||
|
__attribute__((constructor))
|
||||||
|
static void _init_%s() {
|
||||||
|
%s = %s;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
symbolName, usePtr, cgoName,
|
||||||
|
cgoName,
|
||||||
|
cgoName, symbolName))
|
||||||
|
toRemove = append(toRemove, cgoName)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for _, funcName := range toRemove {
|
for _, funcName := range toRemove {
|
||||||
delete(cgoFuncs, funcName)
|
delete(cgoSymbols, funcName)
|
||||||
}
|
}
|
||||||
return b.String()
|
return b.String(), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func getMacroNames(file string, cflags []string, macroNames map[string]bool) error {
|
||||||
|
args := append([]string{"-dM", "-E"}, cflags...)
|
||||||
|
args = append(args, file)
|
||||||
|
cmd := exec.Command("clang", args...)
|
||||||
|
output, err := cmd.Output()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for _, line := range strings.Split(string(output), "\n") {
|
||||||
|
if strings.HasPrefix(line, "#define ") {
|
||||||
|
define := strings.TrimPrefix(line, "#define ")
|
||||||
|
parts := strings.SplitN(define, " ", 2)
|
||||||
|
if len(parts) > 1 {
|
||||||
|
macroNames[parts[0]] = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func getFuncNames(file string, cflags []string, symbolNames map[string]bool) error {
|
||||||
|
args := append([]string{"-Xclang", "-ast-dump=json", "-fsyntax-only"}, cflags...)
|
||||||
|
args = append(args, file)
|
||||||
|
cmd := exec.Command("clang", args...)
|
||||||
|
output, err := cmd.Output()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
var astRoot clangASTNode
|
||||||
|
if err := json.Unmarshal(output, &astRoot); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
extractFuncNames(&astRoot, symbolNames)
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func extractFuncNames(node *clangASTNode, funcNames map[string]bool) {
|
func extractFuncNames(node *clangASTNode, funcNames map[string]bool) {
|
||||||
|
|||||||
@@ -117,6 +117,10 @@ func (g Global) InitNil() {
|
|||||||
g.impl.SetInitializer(llvm.ConstNull(g.impl.GlobalValueType()))
|
g.impl.SetInitializer(llvm.ConstNull(g.impl.GlobalValueType()))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (g Global) ReplaceAllUsesWith(v Expr) {
|
||||||
|
g.impl.ReplaceAllUsesWith(v.impl)
|
||||||
|
}
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
// Function represents the parameters, results, and code of a function
|
// Function represents the parameters, results, and code of a function
|
||||||
|
|||||||
@@ -59,6 +59,15 @@ func (v Expr) SetOrdering(ordering AtomicOrdering) Expr {
|
|||||||
return v
|
return v
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (v Expr) SetName(alias string) Expr {
|
||||||
|
v.impl.SetName(alias)
|
||||||
|
return v
|
||||||
|
}
|
||||||
|
|
||||||
|
func (v Expr) Name() string {
|
||||||
|
return v.impl.Name()
|
||||||
|
}
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
type builtinTy struct {
|
type builtinTy struct {
|
||||||
|
|||||||
Reference in New Issue
Block a user