llgo/ssa: pyCall; demo: _pydemo/callpy
This commit is contained in:
@@ -4,9 +4,12 @@ import (
|
|||||||
"github.com/goplus/llgo/c"
|
"github.com/goplus/llgo/c"
|
||||||
"github.com/goplus/llgo/py"
|
"github.com/goplus/llgo/py"
|
||||||
"github.com/goplus/llgo/py/math"
|
"github.com/goplus/llgo/py/math"
|
||||||
|
"github.com/goplus/llgo/py/os"
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
x := math.Sqrt(py.Float(2))
|
x := math.Sqrt(py.Float(2))
|
||||||
|
wd := os.Getcwd()
|
||||||
c.Printf(c.Str("sqrt(2) = %f\n"), x.Float64())
|
c.Printf(c.Str("sqrt(2) = %f\n"), x.Float64())
|
||||||
|
c.Printf(c.Str("cwd = %s\n"), wd.CStr())
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,59 @@
|
|||||||
|
; ModuleID = 'main'
|
||||||
|
source_filename = "main"
|
||||||
|
|
||||||
|
@"main.init$guard" = global ptr null
|
||||||
|
@__llgo_argc = global ptr null
|
||||||
|
@__llgo_argv = global ptr null
|
||||||
|
@sqrt = external global ptr
|
||||||
|
@getcwd = external global ptr
|
||||||
|
@0 = private unnamed_addr constant [14 x i8] c"sqrt(2) = %f\0A\00", align 1
|
||||||
|
@1 = private unnamed_addr constant [10 x i8] c"cwd = %s\0A\00", align 1
|
||||||
|
|
||||||
|
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 @"github.com/goplus/llgo/py/math.init"()
|
||||||
|
call void @"github.com/goplus/llgo/py/os.init"()
|
||||||
|
br label %_llgo_2
|
||||||
|
|
||||||
|
_llgo_2: ; preds = %_llgo_1, %_llgo_0
|
||||||
|
ret void
|
||||||
|
}
|
||||||
|
|
||||||
|
define void @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 @PyFloat_FromDouble(double 2.000000e+00)
|
||||||
|
%3 = call ptr @PyObject_CallOneArg(ptr @sqrt, ptr %2)
|
||||||
|
%4 = call ptr @PyObject_CallNoArg(ptr @getcwd)
|
||||||
|
%5 = call double @PyFloat_AsDouble()
|
||||||
|
%6 = call i32 (ptr, ...) @printf(ptr @0, double %5)
|
||||||
|
%7 = call ptr @PyBytes_AsString()
|
||||||
|
%8 = call i32 (ptr, ...) @printf(ptr @1, ptr %7)
|
||||||
|
ret void
|
||||||
|
}
|
||||||
|
|
||||||
|
declare void @"github.com/goplus/llgo/py/math.init"()
|
||||||
|
|
||||||
|
declare void @"github.com/goplus/llgo/py/os.init"()
|
||||||
|
|
||||||
|
declare void @"github.com/goplus/llgo/internal/runtime.init"()
|
||||||
|
|
||||||
|
declare ptr @PyFloat_FromDouble(double)
|
||||||
|
|
||||||
|
declare ptr @PyObject_CallOneArg(ptr, ptr)
|
||||||
|
|
||||||
|
declare ptr @PyObject_CallNoArg(ptr)
|
||||||
|
|
||||||
|
declare double @PyFloat_AsDouble()
|
||||||
|
|
||||||
|
declare i32 @printf(ptr, ...)
|
||||||
|
|
||||||
|
declare ptr @PyBytes_AsString()
|
||||||
|
|||||||
@@ -557,7 +557,8 @@ func (p *context) compileInstrOrValue(b llssa.Builder, iv instrOrValue, asValue
|
|||||||
args := p.compileValues(b, args, kind)
|
args := p.compileValues(b, args, kind)
|
||||||
ret = b.Call(aFn.Expr, args...)
|
ret = b.Call(aFn.Expr, args...)
|
||||||
case pyFunc:
|
case pyFunc:
|
||||||
log.Panicln("pyFunc:", pyFn)
|
args := p.compileValues(b, args, kind)
|
||||||
|
ret = b.Call(pyFn.Expr, args...)
|
||||||
case llgoCstr:
|
case llgoCstr:
|
||||||
ret = cstr(b, args)
|
ret = cstr(b, args)
|
||||||
case llgoAdvance:
|
case llgoAdvance:
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ func testCompile(t *testing.T, src, expected string) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestFromTestpy(t *testing.T) {
|
func TestFromTestpy(t *testing.T) {
|
||||||
cltest.FromDir(t, "callpy", "./_testpy", false)
|
cltest.FromDir(t, "", "./_testpy", false)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestFromTestlibc(t *testing.T) {
|
func TestFromTestlibc(t *testing.T) {
|
||||||
|
|||||||
46
py/bytes.go
Normal file
46
py/bytes.go
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2024 The GoPlus Authors (goplus.org). All rights reserved.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package py
|
||||||
|
|
||||||
|
import (
|
||||||
|
_ "unsafe"
|
||||||
|
|
||||||
|
"github.com/goplus/llgo/c"
|
||||||
|
)
|
||||||
|
|
||||||
|
// https://docs.python.org/3/c-api/bytes.html
|
||||||
|
|
||||||
|
// String returns a new bytes object from a C string.
|
||||||
|
//
|
||||||
|
//go:linkname StringFrom C.PyBytes_FromString
|
||||||
|
func StringFrom(s *c.Char) *Object
|
||||||
|
|
||||||
|
//go:linkname stringFromStringAndSize C.PyBytes_FromStringAndSize
|
||||||
|
func stringFromStringAndSize(s *c.Char, size uintptr) *Object
|
||||||
|
|
||||||
|
// String returns a new bytes object from a Go string.
|
||||||
|
func String(s string) *Object {
|
||||||
|
return stringFromStringAndSize(c.GoStringData(s), uintptr(len(s)))
|
||||||
|
}
|
||||||
|
|
||||||
|
// CStr returns the content of a bytes object as a C string.
|
||||||
|
//
|
||||||
|
// llgo:link (*Object).CStr C.PyBytes_AsString
|
||||||
|
func (o *Object) CStr() *c.Char { return nil }
|
||||||
|
|
||||||
|
// llgo:link (*Object).Strlen C.PyBytes_Size
|
||||||
|
func (o *Object) Strlen() uintptr { return 0 }
|
||||||
@@ -29,4 +29,4 @@ func Float(v float64) *Object
|
|||||||
func FloatFromSring(v *Object) *Object
|
func FloatFromSring(v *Object) *Object
|
||||||
|
|
||||||
// llgo:link (*Object).Float64 C.PyFloat_AsDouble
|
// llgo:link (*Object).Float64 C.PyFloat_AsDouble
|
||||||
func (o *Object) Float64() float64 { panic("unreachable") }
|
func (o *Object) Float64() float64 { return 0 }
|
||||||
|
|||||||
@@ -53,6 +53,6 @@ func Import(name *Object) *Module
|
|||||||
// manipulate a module’s __dict__.
|
// manipulate a module’s __dict__.
|
||||||
//
|
//
|
||||||
// llgo:link (*Module).GetDict C.PyModule_GetDict
|
// llgo:link (*Module).GetDict C.PyModule_GetDict
|
||||||
func (m *Module) GetDict() *Object { panic("unreachable") }
|
func (m *Module) GetDict() *Object { return nil }
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|||||||
32
py/object.go
32
py/object.go
@@ -36,7 +36,7 @@ type Object struct {
|
|||||||
func BuildValue(format *c.Char, __llgo_va_list ...any) *Object
|
func BuildValue(format *c.Char, __llgo_va_list ...any) *Object
|
||||||
|
|
||||||
// llgo:link (*Object).DecRef C.Py_DecRef
|
// llgo:link (*Object).DecRef C.Py_DecRef
|
||||||
func (o *Object) DecRef() { panic("unreachable") }
|
func (o *Object) DecRef() {}
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
@@ -44,10 +44,10 @@ func (o *Object) DecRef() { panic("unreachable") }
|
|||||||
// or nil on failure. This is the equivalent of the Python expression o.attrName.
|
// or nil on failure. This is the equivalent of the Python expression o.attrName.
|
||||||
//
|
//
|
||||||
// llgo:link (*Object).GetAttr C.PyObject_GetAttr
|
// llgo:link (*Object).GetAttr C.PyObject_GetAttr
|
||||||
func (o *Object) GetAttr(attrName *Object) *Object { panic("unreachable") }
|
func (o *Object) GetAttr(attrName *Object) *Object { return nil }
|
||||||
|
|
||||||
// llgo:link (*Object).GetAttrString C.PyObject_GetAttrString
|
// llgo:link (*Object).GetAttrString C.PyObject_GetAttrString
|
||||||
func (o *Object) GetAttrString(attrName *c.Char) *Object { panic("unreachable") }
|
func (o *Object) GetAttrString(attrName *c.Char) *Object { return nil }
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|
||||||
@@ -55,7 +55,7 @@ func (o *Object) GetAttrString(attrName *c.Char) *Object { panic("unreachable")
|
|||||||
// 0 otherwise. This function always succeeds.
|
// 0 otherwise. This function always succeeds.
|
||||||
//
|
//
|
||||||
// llgo:link (*Object).Callable C.PyCallable_Check
|
// llgo:link (*Object).Callable C.PyCallable_Check
|
||||||
func (o *Object) Callable() int { panic("unreachable") }
|
func (o *Object) Callable() c.Int { return 0 }
|
||||||
|
|
||||||
// Call a callable Python object o, with arguments given by the tuple args, and
|
// Call a callable Python object o, with arguments given by the tuple args, and
|
||||||
// named arguments given by the dictionary kwargs.
|
// named arguments given by the dictionary kwargs.
|
||||||
@@ -69,7 +69,7 @@ func (o *Object) Callable() int { panic("unreachable") }
|
|||||||
// This is the equivalent of the Python expression: o(*args, **kwargs).
|
// This is the equivalent of the Python expression: o(*args, **kwargs).
|
||||||
//
|
//
|
||||||
// llgo:link (*Object).Call C.PyObject_Call
|
// llgo:link (*Object).Call C.PyObject_Call
|
||||||
func (o *Object) Call(args, kwargs *Object) *Object { panic("unreachable") }
|
func (o *Object) Call(args, kwargs *Object) *Object { return nil }
|
||||||
|
|
||||||
// Call a callable Python object callable without any arguments. It is the most
|
// Call a callable Python object callable without any arguments. It is the most
|
||||||
// efficient way to call a callable Python object without any argument.
|
// efficient way to call a callable Python object without any argument.
|
||||||
@@ -78,7 +78,7 @@ func (o *Object) Call(args, kwargs *Object) *Object { panic("unreachable") }
|
|||||||
// on failure.
|
// on failure.
|
||||||
//
|
//
|
||||||
// llgo:link (*Object).CallNoArgs C.PyObject_CallNoArgs
|
// llgo:link (*Object).CallNoArgs C.PyObject_CallNoArgs
|
||||||
func (o *Object) CallNoArgs() *Object { panic("unreachable") }
|
func (o *Object) CallNoArgs() *Object { return nil }
|
||||||
|
|
||||||
// Call a callable Python object callable with exactly 1 positional argument arg
|
// Call a callable Python object callable with exactly 1 positional argument arg
|
||||||
// and no keyword arguments.
|
// and no keyword arguments.
|
||||||
@@ -87,7 +87,7 @@ func (o *Object) CallNoArgs() *Object { panic("unreachable") }
|
|||||||
// on failure.
|
// on failure.
|
||||||
//
|
//
|
||||||
// llgo:link (*Object).CallOneArg C.PyObject_CallOneArg
|
// llgo:link (*Object).CallOneArg C.PyObject_CallOneArg
|
||||||
func (o *Object) CallOneArg(arg *Object) *Object { panic("unreachable") }
|
func (o *Object) CallOneArg(arg *Object) *Object { return nil }
|
||||||
|
|
||||||
// Call a callable Python object o, with arguments given by the tuple args. If no
|
// Call a callable Python object o, with arguments given by the tuple args. If no
|
||||||
// arguments are needed, then args can be nil.
|
// arguments are needed, then args can be nil.
|
||||||
@@ -98,7 +98,7 @@ func (o *Object) CallOneArg(arg *Object) *Object { panic("unreachable") }
|
|||||||
// This is the equivalent of the Python expression: o(*args).
|
// This is the equivalent of the Python expression: o(*args).
|
||||||
//
|
//
|
||||||
// llgo:link (*Object).CallObject C.PyObject_CallObject
|
// llgo:link (*Object).CallObject C.PyObject_CallObject
|
||||||
func (o *Object) CallObject(callable, args *Object) *Object { panic("unreachable") }
|
func (o *Object) CallObject(callable, args *Object) *Object { return nil }
|
||||||
|
|
||||||
// Call a callable Python object o, with a variable number of C arguments. The C
|
// Call a callable Python object o, with a variable number of C arguments. The C
|
||||||
// arguments are described using a py.BuildValue style format string. The format
|
// arguments are described using a py.BuildValue style format string. The format
|
||||||
@@ -113,7 +113,7 @@ func (o *Object) CallObject(callable, args *Object) *Object { panic("unreachable
|
|||||||
// faster alternative.
|
// faster alternative.
|
||||||
//
|
//
|
||||||
// llgo:link (*Object).CallFunction C.PyObject_CallFunction
|
// llgo:link (*Object).CallFunction C.PyObject_CallFunction
|
||||||
func (o *Object) CallFunction(format *c.Char, __llgo_va_list ...any) *Object { panic("unreachable") }
|
func (o *Object) CallFunction(format *c.Char, __llgo_va_list ...any) *Object { return nil }
|
||||||
|
|
||||||
// Call a callable Python object o, with a variable number of PyObject* arguments.
|
// Call a callable Python object o, with a variable number of PyObject* arguments.
|
||||||
// The arguments are provided as a variable number of parameters followed by nil.
|
// The arguments are provided as a variable number of parameters followed by nil.
|
||||||
@@ -124,7 +124,7 @@ func (o *Object) CallFunction(format *c.Char, __llgo_va_list ...any) *Object { p
|
|||||||
// This is the equivalent of the Python expression: o(arg1, arg2, ...).
|
// This is the equivalent of the Python expression: o(arg1, arg2, ...).
|
||||||
//
|
//
|
||||||
// llgo:link (*Object).CallFunctionObjArgs C.PyObject_CallFunctionObjArgs
|
// llgo:link (*Object).CallFunctionObjArgs C.PyObject_CallFunctionObjArgs
|
||||||
func (o *Object) CallFunctionObjArgs(__llgo_va_list ...any) *Object { panic("unreachable") }
|
func (o *Object) CallFunctionObjArgs(__llgo_va_list ...any) *Object { return nil }
|
||||||
|
|
||||||
// llgo:link (*Object).CallMethod C.PyObject_CallMethod
|
// llgo:link (*Object).CallMethod C.PyObject_CallMethod
|
||||||
func (o *Object) CallMethod(name *c.Char, format *c.Char, __llgo_va_list ...any) *Object {
|
func (o *Object) CallMethod(name *c.Char, format *c.Char, __llgo_va_list ...any) *Object {
|
||||||
@@ -132,27 +132,27 @@ func (o *Object) CallMethod(name *c.Char, format *c.Char, __llgo_va_list ...any)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// llgo:link (*Object).CallMethodObjArgs C.PyObject_CallMethodObjArgs
|
// llgo:link (*Object).CallMethodObjArgs C.PyObject_CallMethodObjArgs
|
||||||
func (o *Object) CallMethodObjArgs(name *Object, __llgo_va_list ...any) *Object { panic("unreachable") }
|
func (o *Object) CallMethodObjArgs(name *Object, __llgo_va_list ...any) *Object { return nil }
|
||||||
|
|
||||||
// llgo:link (*Object).CallMethodNoArgs C.PyObject_CallMethodNoArgs
|
// llgo:link (*Object).CallMethodNoArgs C.PyObject_CallMethodNoArgs
|
||||||
func (o *Object) CallMethodNoArgs(name *Object) *Object { panic("unreachable") }
|
func (o *Object) CallMethodNoArgs(name *Object) *Object { return nil }
|
||||||
|
|
||||||
// llgo:link (*Object).CallMethodOneArg C.PyObject_CallMethodOneArg
|
// llgo:link (*Object).CallMethodOneArg C.PyObject_CallMethodOneArg
|
||||||
func (o *Object) CallMethodOneArg(name, arg *Object) *Object { panic("unreachable") }
|
func (o *Object) CallMethodOneArg(name, arg *Object) *Object { return nil }
|
||||||
|
|
||||||
// llgo:link (*Object).Vectorcall C.PyObject_Vectorcall
|
// llgo:link (*Object).Vectorcall C.PyObject_Vectorcall
|
||||||
func (o *Object) Vectorcall(args **Object, nargs uintptr, kwnames *Object) *Object {
|
func (o *Object) Vectorcall(args **Object, nargs uintptr, kwnames *Object) *Object {
|
||||||
panic("unreachable")
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// llgo:link (*Object).VectorcallDict C.PyObject_VectorcallDict
|
// llgo:link (*Object).VectorcallDict C.PyObject_VectorcallDict
|
||||||
func (o *Object) VectorcallDict(args **Object, nargs uintptr, kwdict *Object) *Object {
|
func (o *Object) VectorcallDict(args **Object, nargs uintptr, kwdict *Object) *Object {
|
||||||
panic("unreachable")
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// llgo:link (*Object).VectorcallMethod C.PyObject_VectorcallMethod
|
// llgo:link (*Object).VectorcallMethod C.PyObject_VectorcallMethod
|
||||||
func (o *Object) VectorcallMethod(name *Object, args **Object, nargs uintptr, kwnames *Object) *Object {
|
func (o *Object) VectorcallMethod(name *Object, args **Object, nargs uintptr, kwnames *Object) *Object {
|
||||||
panic("unreachable")
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|||||||
30
py/os/os.go
Normal file
30
py/os/os.go
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2024 The GoPlus Authors (goplus.org). All rights reserved.
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package os
|
||||||
|
|
||||||
|
import (
|
||||||
|
_ "unsafe"
|
||||||
|
|
||||||
|
"github.com/goplus/llgo/py"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
LLGoPackage = "py.os"
|
||||||
|
)
|
||||||
|
|
||||||
|
//go:linkname Getcwd py.getcwd
|
||||||
|
func Getcwd() *py.Object
|
||||||
@@ -22,6 +22,10 @@ import (
|
|||||||
"github.com/goplus/llgo/cl/cltest"
|
"github.com/goplus/llgo/cl/cltest"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func TestFromTestpy(t *testing.T) {
|
||||||
|
cltest.FromDir(t, "", "../cl/_testpy", false)
|
||||||
|
}
|
||||||
|
|
||||||
func TestFromTestrt(t *testing.T) {
|
func TestFromTestrt(t *testing.T) {
|
||||||
cltest.FromDir(t, "", "../cl/_testrt", true)
|
cltest.FromDir(t, "", "../cl/_testrt", true)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1176,7 +1176,6 @@ func (b Builder) InlineCall(fn Expr, args ...Expr) (ret Expr) {
|
|||||||
// t4 = t3()
|
// t4 = t3()
|
||||||
// t7 = invoke t5.Println(...t6)
|
// t7 = invoke t5.Println(...t6)
|
||||||
func (b Builder) Call(fn Expr, args ...Expr) (ret Expr) {
|
func (b Builder) Call(fn Expr, args ...Expr) (ret Expr) {
|
||||||
prog := b.Prog
|
|
||||||
if debugInstr {
|
if debugInstr {
|
||||||
var b bytes.Buffer
|
var b bytes.Buffer
|
||||||
name := fn.impl.Name()
|
name := fn.impl.Name()
|
||||||
@@ -1191,11 +1190,16 @@ func (b Builder) Call(fn Expr, args ...Expr) (ret Expr) {
|
|||||||
}
|
}
|
||||||
log.Println(b.String())
|
log.Println(b.String())
|
||||||
}
|
}
|
||||||
|
var kind = fn.kind
|
||||||
|
if kind == vkPyFunc {
|
||||||
|
return b.pyCall(fn, args)
|
||||||
|
}
|
||||||
var ll llvm.Type
|
var ll llvm.Type
|
||||||
var data Expr
|
var data Expr
|
||||||
var sig *types.Signature
|
var sig *types.Signature
|
||||||
|
var prog = b.Prog
|
||||||
var raw = fn.raw.Type
|
var raw = fn.raw.Type
|
||||||
switch fn.kind {
|
switch kind {
|
||||||
case vkClosure:
|
case vkClosure:
|
||||||
data = b.Field(fn, 1)
|
data = b.Field(fn, 1)
|
||||||
fn = b.Field(fn, 0)
|
fn = b.Field(fn, 0)
|
||||||
|
|||||||
@@ -136,7 +136,9 @@ type aProgram struct {
|
|||||||
pyObjPtr Type
|
pyObjPtr Type
|
||||||
pyObjPPtr Type
|
pyObjPPtr Type
|
||||||
|
|
||||||
pyImpTy *types.Signature
|
pyImpTy *types.Signature
|
||||||
|
callNoArg *types.Signature
|
||||||
|
callOneArg *types.Signature
|
||||||
|
|
||||||
needRuntime bool
|
needRuntime bool
|
||||||
needPyInit bool
|
needPyInit bool
|
||||||
@@ -385,7 +387,7 @@ func (p Package) rtFunc(fnName string) Expr {
|
|||||||
return p.NewFunc(name, sig, InGo).Expr
|
return p.NewFunc(name, sig, InGo).Expr
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p Package) cpyFunc(fullName string, sig *types.Signature) Expr {
|
func (p Package) pyFunc(fullName string, sig *types.Signature) Expr {
|
||||||
p.Prog.needPyInit = true
|
p.Prog.needPyInit = true
|
||||||
return p.NewFunc(fullName, sig, InC).Expr
|
return p.NewFunc(fullName, sig, InC).Expr
|
||||||
}
|
}
|
||||||
@@ -483,10 +485,31 @@ func (p Program) tyImportPyModule() *types.Signature {
|
|||||||
return p.pyImpTy
|
return p.pyImpTy
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (p Program) tyCallNoArg() *types.Signature {
|
||||||
|
if p.callNoArg == nil {
|
||||||
|
objPtr := p.PyObjectPtr().raw.Type
|
||||||
|
paramObjPtr := types.NewParam(token.NoPos, nil, "", objPtr)
|
||||||
|
params := types.NewTuple(paramObjPtr)
|
||||||
|
p.callNoArg = types.NewSignatureType(nil, nil, nil, params, params, false)
|
||||||
|
}
|
||||||
|
return p.callNoArg
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p Program) tyCallOneArg() *types.Signature {
|
||||||
|
if p.callOneArg == nil {
|
||||||
|
objPtr := p.PyObjectPtr().raw.Type
|
||||||
|
paramObjPtr := types.NewParam(token.NoPos, nil, "", objPtr)
|
||||||
|
params := types.NewTuple(paramObjPtr, paramObjPtr)
|
||||||
|
results := types.NewTuple(paramObjPtr)
|
||||||
|
p.callOneArg = types.NewSignatureType(nil, nil, nil, params, results, false)
|
||||||
|
}
|
||||||
|
return p.callOneArg
|
||||||
|
}
|
||||||
|
|
||||||
// ImportPyMod imports a Python module.
|
// ImportPyMod imports a Python module.
|
||||||
func (b Builder) ImportPyMod(path string) Expr {
|
func (b Builder) ImportPyMod(path string) Expr {
|
||||||
pkg := b.Func.Pkg
|
pkg := b.Func.Pkg
|
||||||
fnImp := pkg.cpyFunc("PyImport_ImportModule", b.Prog.tyImportPyModule())
|
fnImp := pkg.pyFunc("PyImport_ImportModule", b.Prog.tyImportPyModule())
|
||||||
return b.Call(fnImp, b.CStr(path))
|
return b.Call(fnImp, b.CStr(path))
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -500,4 +523,23 @@ func (p Package) NewPyModVar(name string) Global {
|
|||||||
return g
|
return g
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (b Builder) pyCall(fn Expr, args []Expr) (ret Expr) {
|
||||||
|
prog := b.Prog
|
||||||
|
pkg := b.Func.Pkg
|
||||||
|
sig := fn.raw.Type.(*types.Signature)
|
||||||
|
params := sig.Params()
|
||||||
|
n := params.Len()
|
||||||
|
switch n {
|
||||||
|
case 0:
|
||||||
|
call := pkg.pyFunc("PyObject_CallNoArg", prog.tyCallNoArg())
|
||||||
|
ret = b.Call(call, fn)
|
||||||
|
case 1:
|
||||||
|
call := pkg.pyFunc("PyObject_CallOneArg", prog.tyCallOneArg())
|
||||||
|
ret = b.Call(call, fn, args[0])
|
||||||
|
default:
|
||||||
|
panic("todo")
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// -----------------------------------------------------------------------------
|
// -----------------------------------------------------------------------------
|
||||||
|
|||||||
Reference in New Issue
Block a user