diff --git a/integration_tests/CMakeLists.txt b/integration_tests/CMakeLists.txt index 52ccd6d074..e35e0d2397 100644 --- a/integration_tests/CMakeLists.txt +++ b/integration_tests/CMakeLists.txt @@ -660,7 +660,7 @@ RUN(NAME bindc_05 LABELS llvm c EXTRAFILES bindc_05b.c) RUN(NAME bindc_06 LABELS llvm c EXTRAFILES bindc_06b.c) -RUN(NAME bindpy_01 LABELS cpython c_py EXTRA_ARGS --enable-cpython NOFAST COPY_TO_BIN bindpy_01_module.py) +RUN(NAME bindpy_01 LABELS cpython llvm_py c_py EXTRA_ARGS --enable-cpython NOFAST COPY_TO_BIN bindpy_01_module.py) RUN(NAME bindpy_02 LABELS cpython c_py EXTRA_ARGS --link-numpy COPY_TO_BIN bindpy_02_module.py) RUN(NAME bindpy_03 LABELS cpython c_py EXTRA_ARGS --link-numpy NOFAST COPY_TO_BIN bindpy_03_module.py) RUN(NAME bindpy_04 LABELS cpython c_py EXTRA_ARGS --link-numpy NOFAST COPY_TO_BIN bindpy_04_module.py) diff --git a/integration_tests/bindpy_05.py b/integration_tests/bindpy_05.py index c648e5f610..8cb4422cd0 100644 --- a/integration_tests/bindpy_05.py +++ b/integration_tests/bindpy_05.py @@ -41,7 +41,7 @@ def PyObject_CallObject(a: CPtr, b: CPtr) -> CPtr: pass @ccall(header="Python.h") -def PyLong_AsLongLong(a: CPtr) -> i32: +def PyLong_AsLongLong(a: CPtr) -> i64: pass def my_f(): @@ -62,9 +62,9 @@ def my_f(): _Py_DecRef(pArgs) assert bool(pValue), "Call to my_f failed\n" - ans: i32 = PyLong_AsLongLong(pValue) + ans: i64 = PyLong_AsLongLong(pValue) print("Ans is", ans) - assert ans == 5 + assert ans == i64(5) def main0(): diff --git a/src/bin/lpython.cpp b/src/bin/lpython.cpp index 5b08a1c9ed..e5d2a9041f 100644 --- a/src/bin/lpython.cpp +++ b/src/bin/lpython.cpp @@ -321,6 +321,7 @@ int emit_c(const std::string &infile, pass_manager.use_default_passes(true); compiler_options.po.always_run = true; compiler_options.po.run_fun = "f"; + compiler_options.po.c_skip_bindpy_pass = true; pass_manager.apply_passes(al, asr, compiler_options.po, diagnostics); @@ -370,6 +371,7 @@ int emit_c_to_file(const std::string &infile, const std::string &outfile, compiler_options.po.run_fun = "f"; compiler_options.po.always_run = true; + compiler_options.po.c_skip_bindpy_pass = true; pass_manager.use_default_passes(true); pass_manager.apply_passes(al, asr, compiler_options.po, diagnostics); @@ -1130,7 +1132,7 @@ int compile_python_using_llvm( LCompilers::LPython::DynamicLibrary cpython_lib; LCompilers::LPython::DynamicLibrary symengine_lib; - if (compiler_options.enable_cpython) { + if (compiler_options.po.enable_cpython) { LCompilers::LPython::open_cpython_library(cpython_lib); } if (compiler_options.enable_symengine) { @@ -1149,7 +1151,7 @@ int compile_python_using_llvm( e.execfn("__module___main_____main__global_stmts"); } - if (compiler_options.enable_cpython) { + if (compiler_options.po.enable_cpython) { LCompilers::LPython::close_cpython_library(cpython_lib); } if (compiler_options.enable_symengine) { @@ -1533,7 +1535,7 @@ int link_executable(const std::vector &infiles, cmd += " -L$CONDA_PREFIX/lib -Wl,-rpath -Wl,$CONDA_PREFIX/lib -lsymengine"; } - if (compiler_options.enable_cpython) { + if (compiler_options.po.enable_cpython) { std::string py_version = "3.10"; std::string py_flags = R"(-I $CONDA_PREFIX/include/python)" + py_version + R"( -L$CONDA_PREFIX/lib -Wl,-rpath -Wl,$CONDA_PREFIX/lib -lpython)" + py_version + R"()"; if (compiler_options.link_numpy) { @@ -1592,7 +1594,7 @@ int link_executable(const std::vector &infiles, if (compiler_options.enable_symengine) { cmd += " -L$CONDA_PREFIX/lib -Wl,-rpath -Wl,$CONDA_PREFIX/lib -lsymengine"; } - if (compiler_options.enable_cpython) { + if (compiler_options.po.enable_cpython) { std::string py_version = "3.10"; std::string py_flags = R"(-I $CONDA_PREFIX/include/python)" + py_version + R"( -L$CONDA_PREFIX/lib -Wl,-rpath -Wl,$CONDA_PREFIX/lib -lpython)" + py_version + R"()"; if (compiler_options.link_numpy) { @@ -1919,7 +1921,7 @@ int main(int argc, char *argv[]) app.add_flag("--dump-all-passes", compiler_options.po.dump_all_passes, "Apply all the passes and dump the ASR into a file"); app.add_flag("--dump-all-passes-fortran", compiler_options.po.dump_fortran, "Apply all passes and dump the ASR after each pass into fortran file"); app.add_flag("--cumulative", compiler_options.po.pass_cumulative, "Apply all the passes cumulatively till the given pass"); - app.add_flag("--enable-cpython", compiler_options.enable_cpython, "Enable CPython runtime"); + app.add_flag("--enable-cpython", compiler_options.po.enable_cpython, "Enable CPython runtime"); app.add_flag("--enable-symengine", compiler_options.enable_symengine, "Enable Symengine runtime"); app.add_flag("--link-numpy", compiler_options.link_numpy, "Enable NumPy runtime (implies --enable-cpython)"); app.add_flag("--separate-compilation", separate_compilation, "Generates unique names for all the symbols"); @@ -1981,7 +1983,7 @@ int main(int argc, char *argv[]) } if (compiler_options.link_numpy) { - compiler_options.enable_cpython = true; + compiler_options.po.enable_cpython = true; } if (arg_version) { diff --git a/src/libasr/CMakeLists.txt b/src/libasr/CMakeLists.txt index aaf9db92c8..0d69d7cfaa 100644 --- a/src/libasr/CMakeLists.txt +++ b/src/libasr/CMakeLists.txt @@ -37,6 +37,7 @@ set(SRC pass/for_all.cpp pass/while_else.cpp pass/global_stmts.cpp + pass/python_bind.cpp pass/select_case.cpp pass/init_expr.cpp pass/implied_do_loops.cpp diff --git a/src/libasr/asr_utils.cpp b/src/libasr/asr_utils.cpp index bcce3db313..7243ec4bb8 100644 --- a/src/libasr/asr_utils.cpp +++ b/src/libasr/asr_utils.cpp @@ -1707,6 +1707,97 @@ void append_error(diag::Diagnostics& diag, const std::string& msg, //Initialize pointer to zero so that it can be initialized in first call to get_instance ASRUtils::LabelGenerator* ASRUtils::LabelGenerator::label_generator = nullptr; +ASR::expr_t *type_enum_to_asr_expr(Allocator &al, enum TTYPE_T t, const Location &l, std::string n, + SymbolTable *current_scope, ASR::intentType intent) { + ASR::ttype_t *type = nullptr; + + Str s; + s.from_str(al, n); + + switch (t) { + case VOID: + return nullptr; + case I1: + type = ASRUtils::TYPE(ASR::make_Logical_t(al, l, 4)); + break; + case I8: + type = ASRUtils::TYPE(ASR::make_Integer_t(al, l, 1)); + break; + case I32: + type = ASRUtils::TYPE(ASR::make_Integer_t(al, l, 4)); + break; + case I64: + type = ASRUtils::TYPE(ASR::make_Integer_t(al, l, 8)); + break; + case U8: + type = ASRUtils::TYPE(ASR::make_UnsignedInteger_t(al, l, 1)); + break; + case U32: + type = ASRUtils::TYPE(ASR::make_UnsignedInteger_t(al, l, 4)); + break; + case U64: + type = ASRUtils::TYPE(ASR::make_UnsignedInteger_t(al, l, 8)); + break; + case F32: + type = ASRUtils::TYPE(ASR::make_Real_t(al, l, 4)); + break; + case F64: + type = ASRUtils::TYPE(ASR::make_Real_t(al, l, 8)); + break; + case STR: + type = ASRUtils::TYPE(ASR::make_Character_t(al, l, 1, -2, nullptr)); + break; + case PTR: + type = ASRUtils::TYPE(ASR::make_CPtr_t(al, l)); + break; + case PTR_TO_PTR: + type = ASRUtils::TYPE(ASR::make_Pointer_t(al, l, ASRUtils::TYPE(ASR::make_CPtr_t(al, l)))); + break; + } + LCOMPILERS_ASSERT(type); + ASR::symbol_t *v = ASR::down_cast(ASR::make_Variable_t(al, l, current_scope, s.c_str(al), nullptr, + 0, intent, nullptr, nullptr, ASR::storage_typeType::Default, + type, nullptr, ASR::abiType::BindC, ASR::Public, + ASR::presenceType::Required, true)); + current_scope->add_symbol(n, v); + return ASRUtils::EXPR(ASR::make_Var_t(al, l, v)); +} + +void declare_function(Allocator &al, ASRFunc fn, const Location &l, SymbolTable *parent_scope, + std::string header_name) { + Str s; + char *c_header = nullptr; + if (header_name != "") { + s.from_str(al, header_name); + c_header = s.c_str(al); + } + s.from_str(al, fn.m_name); + Vec args; + args.reserve(al, fn.args.size()); + SymbolTable *current_scope = al.make_new(parent_scope); + int c = 0; + for (auto j: fn.args) { + args.push_back(al, type_enum_to_asr_expr(al, j, l, fn.m_name + std::to_string(++c), current_scope, + ASRUtils::intent_in)); + } + ASR::expr_t *retval = type_enum_to_asr_expr(al, fn.retvar, l, "_lpython_return_variable", current_scope, + ASRUtils::intent_return_var); + char *fn_name = s.c_str(al); + ASR::asr_t *f = ASRUtils::make_Function_t_util(al, l, current_scope, fn_name, nullptr, 0, args.p, args.n, + nullptr, 0, retval, ASR::abiType::BindC, ASR::accessType::Public, + ASR::deftypeType::Interface, nullptr, false, false, false, false, false, nullptr, 0, + false, false, false, c_header); + + parent_scope->add_symbol(fn.m_name, ASR::down_cast(f)); +} + +void declare_functions(Allocator &al, std::vector fns, const Location &l, SymbolTable *parent_scope, + std::string header_name) { + for (auto i: fns) { + declare_function(al, i, l, parent_scope, header_name); + } +} + } // namespace ASRUtils diff --git a/src/libasr/asr_utils.h b/src/libasr/asr_utils.h index 89f3adb0a5..42d64c4c1c 100644 --- a/src/libasr/asr_utils.h +++ b/src/libasr/asr_utils.h @@ -5415,6 +5415,41 @@ static inline bool is_argument_of_type_CPtr(ASR::expr_t *var) { return is_argument; } +enum TTYPE_T { + VOID, + I1, + I8, + STR, + I32, + I64, + U8, + U32, + U64, + F32, + F64, + PTR, + PTR_TO_PTR, +}; + +typedef struct { + std::string m_name; + std::vector args; + enum TTYPE_T retvar; +} ASRFunc; + +// Create a variable with name `n` and type `t` and add the symbol into `currect_scope` +// Returns the variable +ASR::expr_t *type_enum_to_asr_expr(Allocator &al, enum TTYPE_T t, const Location &l, std::string n, + SymbolTable *current_scope, ASR::intentType intent); + +// created a BindC Interface function decleration in the `parent_scope` +void declare_function(Allocator &al, ASRFunc fn, const Location &l, SymbolTable *parent_scope, + std::string header_name=""); + +// created a BindC Interface functions decleration in the `parent_scope` +void declare_functions(Allocator &al, std::vector fns, const Location &l, SymbolTable *parent_scope, + std::string header_name=""); + } // namespace ASRUtils } // namespace LCompilers diff --git a/src/libasr/codegen/asr_to_c.cpp b/src/libasr/codegen/asr_to_c.cpp index 381e3c7902..ca0178ac89 100644 --- a/src/libasr/codegen/asr_to_c.cpp +++ b/src/libasr/codegen/asr_to_c.cpp @@ -820,7 +820,7 @@ R"( } std::string body; - if (compiler_options.enable_cpython) { + if (compiler_options.po.enable_cpython) { headers.insert("Python.h"); body += R"( Py_Initialize(); @@ -851,7 +851,7 @@ R"( // Initialise Numpy body += src; } - if (compiler_options.enable_cpython) { + if (compiler_options.po.enable_cpython) { body += R"( if (Py_FinalizeEx() < 0) { fprintf(stderr,"BindPython: Unknown Error\n"); diff --git a/src/libasr/pass/pass_manager.h b/src/libasr/pass/pass_manager.h index c76c4a4126..6c3a11cd84 100644 --- a/src/libasr/pass/pass_manager.h +++ b/src/libasr/pass/pass_manager.h @@ -54,6 +54,7 @@ #include #include #include +#include #include #include #include @@ -79,6 +80,7 @@ namespace LCompilers { {"do_loops", &pass_replace_do_loops}, {"while_else", &pass_while_else}, {"global_stmts", &pass_wrap_global_stmts}, + {"python_bind", &pass_python_bind}, {"implied_do_loops", &pass_replace_implied_do_loops}, {"array_op", &pass_replace_array_op}, {"symbolic", &pass_replace_symbolic}, @@ -206,6 +208,7 @@ namespace LCompilers { PassManager(): apply_default_passes{false}, c_skip_pass{false} { _passes = { + "python_bind", "nested_vars", "global_stmts", "transform_optional_argument_functions", diff --git a/src/libasr/pass/python_bind.cpp b/src/libasr/pass/python_bind.cpp new file mode 100644 index 0000000000..5a668a739e --- /dev/null +++ b/src/libasr/pass/python_bind.cpp @@ -0,0 +1,455 @@ +#include +#include +#include +#include +#include +#include +#include + + +namespace LCompilers { +ASR::expr_t *cpython_to_native(Allocator &al, ASR::expr_t *exp, ASR::ttype_t *type, const ASR::Function_t &f, + SymbolTable &parent_scope) { + ASR::ttype_t *i1_type = ASRUtils::TYPE(ASR::make_Integer_t(al, f.base.base.loc, 1)); + ASR::ttype_t *i1ptr_type = ASRUtils::TYPE(ASR::make_Pointer_t(al, f.base.base.loc, i1_type)); + ASR::ttype_t *i4_type = ASRUtils::TYPE(ASR::make_Integer_t(al, f.base.base.loc, 4)); + ASR::ttype_t *i8_type = ASRUtils::TYPE(ASR::make_Integer_t(al, f.base.base.loc, 8)); + ASR::ttype_t *u8_type = ASRUtils::TYPE(ASR::make_UnsignedInteger_t(al, f.base.base.loc, 8)); + ASR::ttype_t *f8_type = ASRUtils::TYPE(ASR::make_Real_t(al, f.base.base.loc, 8)); + ASR::ttype_t *ptr_t = ASRUtils::TYPE(ASR::make_CPtr_t(al, f.base.base.loc)); + + ASR::expr_t *conv_result = nullptr; + if (type->type == ASR::ttypeType::Integer) { + ASR::symbol_t *sym_PyLong_AsLongLong = parent_scope.resolve_symbol("PyLong_AsLongLong"); + Vec args_PyLong_AsLongLong; + args_PyLong_AsLongLong.reserve(al, 1); + args_PyLong_AsLongLong.push_back(al, {f.base.base.loc, exp}); + conv_result = ASRUtils::EXPR(ASR::make_Cast_t(al, f.base.base.loc, + ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, + sym_PyLong_AsLongLong, nullptr, args_PyLong_AsLongLong.p, args_PyLong_AsLongLong.n, + i8_type, nullptr, nullptr)), + ASR::IntegerToInteger, type, nullptr)); + } else if (type->type == ASR::ttypeType::UnsignedInteger) { + ASR::symbol_t *sym_PyLong_AsUnsignedLongLong = parent_scope.resolve_symbol("PyLong_AsUnsignedLongLong"); + Vec args_PyLong_AsUnsignedLongLong; + args_PyLong_AsUnsignedLongLong.reserve(al, 1); + args_PyLong_AsUnsignedLongLong.push_back(al, {f.base.base.loc, exp}); + conv_result = ASRUtils::EXPR(ASR::make_Cast_t(al, f.base.base.loc, + ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, + sym_PyLong_AsUnsignedLongLong, nullptr, args_PyLong_AsUnsignedLongLong.p, + args_PyLong_AsUnsignedLongLong.n, u8_type, nullptr, nullptr)), + ASR::UnsignedIntegerToUnsignedInteger, type, nullptr)); + } else if (type->type == ASR::ttypeType::Real) { + ASR::symbol_t *sym_PyFloat_AsDouble = parent_scope.resolve_symbol("PyFloat_AsDouble"); + Vec args_PyFloat_AsDouble; + args_PyFloat_AsDouble.reserve(al, 1); + args_PyFloat_AsDouble.push_back(al, {f.base.base.loc, exp}); + conv_result = ASRUtils::EXPR(ASR::make_Cast_t(al, f.base.base.loc, + ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, + sym_PyFloat_AsDouble, nullptr, args_PyFloat_AsDouble.p, args_PyFloat_AsDouble.n, + f8_type, nullptr, nullptr)), + ASR::RealToReal, type, nullptr)); + } else if (type->type == ASR::ttypeType::Logical) { + ASR::symbol_t *sym_PyObject_IsTrue = parent_scope.resolve_symbol("PyObject_IsTrue"); + Vec args_PyObject_IsTrue; + args_PyObject_IsTrue.reserve(al, 1); + args_PyObject_IsTrue.push_back(al, {f.base.base.loc, exp}); + conv_result = ASRUtils::EXPR(ASR::make_Cast_t(al, f.base.base.loc, + ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, + sym_PyObject_IsTrue, nullptr, args_PyObject_IsTrue.p, args_PyObject_IsTrue.n, + i4_type, nullptr, nullptr)), + ASR::IntegerToLogical, type, nullptr)); + } else if (type->type == ASR::ttypeType::Character) { + ASR::symbol_t *sym_PyUnicode_AsUTF8AndSize = parent_scope.resolve_symbol("PyUnicode_AsUTF8AndSize"); + Vec args_PyUnicode_AsUTF8AndSize; + args_PyUnicode_AsUTF8AndSize.reserve(al, 1); + args_PyUnicode_AsUTF8AndSize.push_back(al, {f.base.base.loc, exp}); + args_PyUnicode_AsUTF8AndSize.push_back(al, + {f.base.base.loc, ASRUtils::EXPR(ASR::make_PointerNullConstant_t(al, f.base.base.loc, ptr_t))}); + conv_result = ASRUtils::EXPR(ASR::make_Cast_t(al, f.base.base.loc, + ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, + sym_PyUnicode_AsUTF8AndSize, nullptr, args_PyUnicode_AsUTF8AndSize.p, + args_PyUnicode_AsUTF8AndSize.n, i1ptr_type, nullptr, nullptr)), + ASR::RealToReal, type, nullptr)); + } else { + throw LCompilersException( + "Returning from CPython with " + ASRUtils::get_type_code(type) + " type not supported"); + } + + LCOMPILERS_ASSERT(conv_result); + return conv_result; +} + +ASR::expr_t *native_to_cpython(Allocator &al, ASR::expr_t *exp, const ASR::Function_t &f, + SymbolTable &parent_scope) { + ASR::ttype_t *i4_type = ASRUtils::TYPE(ASR::make_Integer_t(al, f.base.base.loc, 4)); + ASR::ttype_t *i8_type = ASRUtils::TYPE(ASR::make_Integer_t(al, f.base.base.loc, 8)); + ASR::ttype_t *u8_type = ASRUtils::TYPE(ASR::make_UnsignedInteger_t(al, f.base.base.loc, 8)); + ASR::ttype_t *f8_type = ASRUtils::TYPE(ASR::make_Real_t(al, f.base.base.loc, 8)); + ASR::ttype_t *ptr_t = ASRUtils::TYPE(ASR::make_CPtr_t(al, f.base.base.loc)); + + ASR::expr_t *conv_result = nullptr; + ASR::ttype_t *type = ASRUtils::expr_type(exp); + if (type->type == ASR::ttypeType::Integer) { + ASR::symbol_t *sym_PyLong_FromLongLong = parent_scope.resolve_symbol("PyLong_FromLongLong"); + Vec args_PyLong_FromLongLong; + args_PyLong_FromLongLong.reserve(al, 1); + args_PyLong_FromLongLong.push_back(al, + {f.base.base.loc, ASRUtils::EXPR(ASR::make_Cast_t(al, f.base.base.loc, exp, + ASR::cast_kindType::IntegerToInteger, i8_type, nullptr))}); + conv_result = ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, + sym_PyLong_FromLongLong, nullptr, args_PyLong_FromLongLong.p, args_PyLong_FromLongLong.n, + ptr_t, nullptr, nullptr)); + } else if (type->type == ASR::ttypeType::UnsignedInteger) { + ASR::symbol_t *sym_PyLong_FromUnsignedLongLong = parent_scope.resolve_symbol("PyLong_FromUnsignedLongLong"); + Vec args_PyLong_FromUnsignedLongLong; + args_PyLong_FromUnsignedLongLong.reserve(al, 1); + args_PyLong_FromUnsignedLongLong.push_back(al, + {f.base.base.loc, ASRUtils::EXPR(ASR::make_Cast_t(al, f.base.base.loc, exp, + ASR::cast_kindType::UnsignedIntegerToUnsignedInteger, u8_type, nullptr))}); + conv_result = ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, + sym_PyLong_FromUnsignedLongLong, nullptr, args_PyLong_FromUnsignedLongLong.p, + args_PyLong_FromUnsignedLongLong.n, ptr_t, nullptr, nullptr)); + } else if (type->type == ASR::ttypeType::Logical) { + ASR::symbol_t *sym_PyBool_FromLong = parent_scope.resolve_symbol("PyBool_FromLong"); + Vec args_PyBool_FromLong; + args_PyBool_FromLong.reserve(al, 1); + args_PyBool_FromLong.push_back(al, + {f.base.base.loc, ASRUtils::EXPR(ASR::make_Cast_t(al, f.base.base.loc, exp, + ASR::cast_kindType::LogicalToInteger, i4_type, nullptr))}); + conv_result = ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, sym_PyBool_FromLong, + nullptr, args_PyBool_FromLong.p, args_PyBool_FromLong.n, ptr_t, nullptr, nullptr)); + } else if (type->type == ASR::ttypeType::Real) { + ASR::symbol_t *sym_PyFloat_FromDouble = parent_scope.resolve_symbol("PyFloat_FromDouble"); + Vec args_PyFloat_FromDouble; + args_PyFloat_FromDouble.reserve(al, 1); + args_PyFloat_FromDouble.push_back(al, + {f.base.base.loc, ASRUtils::EXPR(ASR::make_Cast_t(al, f.base.base.loc, exp, + ASR::cast_kindType::RealToReal, f8_type, nullptr))}); + conv_result = ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, sym_PyFloat_FromDouble, + nullptr, args_PyFloat_FromDouble.p, args_PyFloat_FromDouble.n, ptr_t, nullptr, nullptr)); + } else if (type->type == ASR::ttypeType::Character) { + ASR::symbol_t *sym_PyUnicode_FromString = parent_scope.resolve_symbol("PyUnicode_FromString"); + Vec args_PyUnicode_FromString; + args_PyUnicode_FromString.reserve(al, 1); + args_PyUnicode_FromString.push_back(al, {f.base.base.loc, exp}); + conv_result = ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, + sym_PyUnicode_FromString, nullptr, args_PyUnicode_FromString.p, args_PyUnicode_FromString.n, + ptr_t, nullptr, nullptr)); + } else { + throw LCompilersException( + "Calling CPython with " + ASRUtils::get_type_code(ASRUtils::expr_type(exp)) + " type not supported"); + } + + LCOMPILERS_ASSERT(conv_result); + return conv_result; +} + +void generate_body(Allocator &al, ASR::Function_t &f, SymbolTable &parent_scope) { + Vec body; + body.reserve(al, 1); + Str s; + + /* + if (!Py_IsInitialized()) { + Py_Initialize(); + void *pA = Py_DecodeLocale("", NULL); + PySys_SetArgv(1, &pA); + } + */ + ASR::ttype_t *i4_type = ASRUtils::TYPE(ASR::make_Integer_t(al, f.base.base.loc, 4)); + ASR::ttype_t *ptr_t = ASRUtils::TYPE(ASR::make_CPtr_t(al, f.base.base.loc)); + + ASR::symbol_t *sym_Py_IsInitialized = parent_scope.resolve_symbol("Py_IsInitialized"); + ASR::symbol_t *sym_Py_Initialize = parent_scope.resolve_symbol("Py_Initialize"); + LCOMPILERS_ASSERT(sym_Py_IsInitialized) + ASR::asr_t *call_Py_IsInitialized = ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, + sym_Py_IsInitialized, nullptr, nullptr, 0, i4_type, nullptr, nullptr); + ASR::asr_t * if_cond = ASR::make_IntegerCompare_t(al, f.base.base.loc, ASRUtils::EXPR(call_Py_IsInitialized), + ASR::cmpopType::Eq, ASRUtils::EXPR(ASR::make_IntegerConstant_t(al, + f.base.base.loc, 0, i4_type)), i4_type, nullptr); + Vec if_body; + if_body.reserve(al, 2); + ASR::asr_t *call_Py_Initialize = ASRUtils::make_SubroutineCall_t_util(al, f.base.base.loc, sym_Py_Initialize, + nullptr, nullptr, 0, nullptr, nullptr, false, false); + if_body.push_back(al, ASRUtils::STMT(call_Py_Initialize)); + + ASR::symbol_t *sym_Py_DecodeLocale = parent_scope.resolve_symbol("Py_DecodeLocale"); + Vec args_Py_DecodeLocale; + s.from_str(al, ""); + args_Py_DecodeLocale.reserve(al, 1); + ASR::ttype_t *str_type = ASRUtils::TYPE(ASR::make_Character_t(al, f.base.base.loc, 1, s.size(), nullptr)); + args_Py_DecodeLocale.push_back(al, {f.base.base.loc, ASRUtils::EXPR(ASR::make_StringConstant_t(al, + f.base.base.loc, s.c_str(al), str_type))}); + args_Py_DecodeLocale.push_back(al, {f.base.base.loc, ASRUtils::EXPR(ASR::make_PointerNullConstant_t(al, + f.base.base.loc, ptr_t))}); + s.from_str(al, "pA"); + ASR::asr_t *pA = ASR::make_Variable_t(al, f.base.base.loc, f.m_symtab, s.c_str(al), nullptr, 0, + ASRUtils::intent_local, nullptr, nullptr, ASR::storage_typeType::Default, ptr_t, nullptr, + ASR::abiType::Source, ASR::Public, ASR::presenceType::Required, false); + ASR::expr_t *pA_ref = ASRUtils::EXPR(ASR::make_Var_t(al, f.base.base.loc, ASR::down_cast(pA))); + f.m_symtab->add_symbol(std::string("pA"), ASR::down_cast(pA)); + if_body.push_back(al, ASRUtils::STMT( + ASR::make_Assignment_t(al, f.base.base.loc, pA_ref, + ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, sym_Py_DecodeLocale, nullptr, + args_Py_DecodeLocale.p, args_Py_DecodeLocale.n, ptr_t, nullptr, nullptr)), + nullptr))); + + ASR::symbol_t *sym_PySys_SetArgv = parent_scope.resolve_symbol("PySys_SetArgv"); + Vec args_PySys_SetArgv; + s.from_str(al, ""); + args_PySys_SetArgv.reserve(al, 1); + args_PySys_SetArgv.push_back(al, {f.base.base.loc, ASRUtils::EXPR(ASR::make_IntegerConstant_t(al, f.base.base.loc, + 1, i4_type))}); + args_PySys_SetArgv.push_back(al, {f.base.base.loc, ASRUtils::EXPR(ASR::make_GetPointer_t(al, f.base.base.loc, + pA_ref, ptr_t, nullptr))}); + if_body.push_back(al, ASRUtils::STMT((ASRUtils::make_SubroutineCall_t_util(al, f.base.base.loc, sym_PySys_SetArgv, + nullptr, args_PySys_SetArgv.p, args_PySys_SetArgv.n, nullptr, nullptr, + false, false)))); + + body.push_back(al, ASRUtils::STMT(ASR::make_If_t(al, f.base.base.loc, ASRUtils::EXPR(if_cond), if_body.p, if_body.n, + nullptr, 0))); + + /* + void *pName = PyUnicode_FromString(module_name); + void *pModule = PyImport_Import(pName); + void *pFunc = PyObject_GetAttrString(pModule, func_name); + */ + + ASR::symbol_t *sym_PyUnicode_FromString = parent_scope.resolve_symbol("PyUnicode_FromString"); + Vec args_PyUnicode_FromString; + s.from_str(al, f.m_module_file); + args_PyUnicode_FromString.reserve(al, 1); + str_type = ASRUtils::TYPE(ASR::make_Character_t(al, f.base.base.loc, 1, s.size(), nullptr)); + args_PyUnicode_FromString.push_back(al, {f.base.base.loc, ASRUtils::EXPR(ASR::make_StringConstant_t(al, + f.base.base.loc, s.c_str(al), str_type))}); + s.from_str(al, "pName"); + ASR::asr_t *pName = ASR::make_Variable_t(al, f.base.base.loc, f.m_symtab, s.c_str(al), nullptr, 0, + ASRUtils::intent_local, nullptr, nullptr, ASR::storage_typeType::Default, ptr_t, nullptr, + ASR::abiType::Source, ASR::Public, ASR::presenceType::Required, false); + ASR::expr_t *pName_ref = ASRUtils::EXPR(ASR::make_Var_t(al, f.base.base.loc, ASR::down_cast(pName))); + f.m_symtab->add_symbol(std::string("pName"), ASR::down_cast(pName)); + body.push_back(al, ASRUtils::STMT( + ASR::make_Assignment_t(al, f.base.base.loc, pName_ref, + ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, sym_PyUnicode_FromString, nullptr, + args_PyUnicode_FromString.p, args_PyUnicode_FromString.n, ptr_t, nullptr, nullptr)), + nullptr))); + + ASR::symbol_t *sym_PyImport_Import = parent_scope.resolve_symbol("PyImport_Import"); + Vec args_PyImport_Import; + args_PyImport_Import.reserve(al, 1); + args_PyImport_Import.push_back(al, {f.base.base.loc, pName_ref}); + s.from_str(al, "pModule"); + ASR::asr_t *pModule = ASR::make_Variable_t(al, f.base.base.loc, f.m_symtab, s.c_str(al), nullptr, 0, + ASRUtils::intent_local, nullptr, nullptr, ASR::storage_typeType::Default, ptr_t, nullptr, + ASR::abiType::Source, ASR::Public, ASR::presenceType::Required, false); + ASR::expr_t *pModule_ref = ASRUtils::EXPR(ASR::make_Var_t(al, f.base.base.loc, + ASR::down_cast(pModule))); + f.m_symtab->add_symbol(std::string("pModule"), ASR::down_cast(pModule)); + body.push_back(al, ASRUtils::STMT( + ASR::make_Assignment_t(al, f.base.base.loc, pModule_ref, + ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, sym_PyImport_Import, nullptr, + args_PyImport_Import.p, args_PyImport_Import.n, ptr_t, nullptr, nullptr)), + nullptr))); + + ASR::symbol_t *sym_PyObject_GetAttrString = parent_scope.resolve_symbol("PyObject_GetAttrString"); + Vec args_PyObject_GetAttrString; + s.from_str(al, f.m_module_file); + args_PyObject_GetAttrString.reserve(al, 2); + args_PyObject_GetAttrString.push_back(al, {f.base.base.loc, pModule_ref}); + s.from_str(al, f.m_name); + str_type = ASRUtils::TYPE(ASR::make_Character_t(al, f.base.base.loc, 1, s.size(), nullptr)); + args_PyObject_GetAttrString.push_back(al, {f.base.base.loc, ASRUtils::EXPR(ASR::make_StringConstant_t(al, + f.base.base.loc, s.c_str(al), str_type))}); + s.from_str(al, "pFunc"); + ASR::asr_t *pFunc = ASR::make_Variable_t(al, f.base.base.loc, f.m_symtab, s.c_str(al), nullptr, 0, + ASRUtils::intent_local, nullptr, nullptr, ASR::storage_typeType::Default, ptr_t, nullptr, + ASR::abiType::Source, ASR::Public, ASR::presenceType::Required, false); + ASR::expr_t *pFunc_ref = ASRUtils::EXPR(ASR::make_Var_t(al, f.base.base.loc, ASR::down_cast(pFunc))); + f.m_symtab->add_symbol(std::string("pFunc"), ASR::down_cast(pFunc)); + body.push_back(al, ASRUtils::STMT( + ASR::make_Assignment_t(al, f.base.base.loc, pFunc_ref, + ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, sym_PyObject_GetAttrString, nullptr, + args_PyObject_GetAttrString.p, args_PyObject_GetAttrString.n, ptr_t, nullptr, + nullptr)), nullptr))); + + // creating CPython tuple for arguments list + ASR::symbol_t *sym_PyTuple_New = parent_scope.resolve_symbol("PyTuple_New"); + Vec args_PyTuple_New; + args_PyTuple_New.reserve(al, 1); + args_PyTuple_New.push_back(al, {f.base.base.loc, ASRUtils::EXPR(ASR::make_IntegerConstant_t(al, f.base.base.loc, + f.n_args, i4_type))}); + s.from_str(al, "pArgs"); + ASR::asr_t *pArgs = ASR::make_Variable_t(al, f.base.base.loc, f.m_symtab, s.c_str(al), nullptr, 0, + ASRUtils::intent_local, nullptr, nullptr, ASR::storage_typeType::Default, ptr_t, nullptr, + ASR::abiType::Source, ASR::Public, ASR::presenceType::Required, false); + ASR::expr_t *pArgs_ref = ASRUtils::EXPR(ASR::make_Var_t(al, f.base.base.loc, ASR::down_cast(pArgs))); + f.m_symtab->add_symbol(std::string("pArgs"), ASR::down_cast(pArgs)); + body.push_back(al, ASRUtils::STMT( + ASR::make_Assignment_t(al, f.base.base.loc, pArgs_ref, + ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, sym_PyTuple_New, nullptr, + args_PyTuple_New.p, args_PyTuple_New.n, ptr_t, nullptr, nullptr)), nullptr))); + + // Converting arguments to CPython types + ASR::symbol_t *sym_PyTuple_SetItem = parent_scope.resolve_symbol("PyTuple_SetItem"); + for (size_t i = 0; i < f.n_args; i++) { + Vec args_PyTuple_SetItem; + args_PyTuple_SetItem.reserve(al, 3); + args_PyTuple_SetItem.push_back(al, {f.base.base.loc, pArgs_ref}); + args_PyTuple_SetItem.push_back(al, {f.base.base.loc, ASRUtils::EXPR(ASR::make_IntegerConstant_t(al, + f.base.base.loc, i, i4_type))}); + args_PyTuple_SetItem.push_back(al, {f.base.base.loc, native_to_cpython(al, f.m_args[i], f, parent_scope)}); + std::string p = "pA" + std::to_string(i); + s.from_str(al, p); + ASR::asr_t *pA = ASR::make_Variable_t(al, f.base.base.loc, f.m_symtab, s.c_str(al), nullptr, 0, + ASRUtils::intent_local, nullptr, nullptr, ASR::storage_typeType::Default, i4_type, nullptr, + ASR::abiType::Source, ASR::Public, ASR::presenceType::Required, false); + ASR::expr_t *pA_ref = ASRUtils::EXPR(ASR::make_Var_t(al, f.base.base.loc, ASR::down_cast(pA))); + f.m_symtab->add_symbol(p, ASR::down_cast(pA)); + body.push_back(al, + ASRUtils::STMT(ASR::make_Assignment_t(al, f.base.base.loc, pA_ref, + ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, sym_PyTuple_SetItem, nullptr, + args_PyTuple_SetItem.p, args_PyTuple_SetItem.n, i4_type, nullptr, nullptr)), nullptr))); + } + + // calling CPython Function + ASR::symbol_t *sym_PyObject_CallObject = parent_scope.resolve_symbol("PyObject_CallObject"); + Vec args_PyObject_CallObject; + args_PyObject_CallObject.reserve(al, 2); + args_PyObject_CallObject.push_back(al, {f.base.base.loc, pFunc_ref}); + args_PyObject_CallObject.push_back(al, {f.base.base.loc, pArgs_ref}); + s.from_str(al, "pReturn"); + ASR::asr_t *pReturn = ASR::make_Variable_t(al, f.base.base.loc, f.m_symtab, s.c_str(al), nullptr, 0, + ASRUtils::intent_local, nullptr, nullptr, ASR::storage_typeType::Default, ptr_t, nullptr, + ASR::abiType::Source, ASR::Public, ASR::presenceType::Required, false); + ASR::expr_t *pReturn_ref = ASRUtils::EXPR(ASR::make_Var_t(al, f.base.base.loc, + ASR::down_cast(pReturn))); + f.m_symtab->add_symbol(std::string("pReturn"), ASR::down_cast(pReturn)); + body.push_back(al, ASRUtils::STMT( + ASR::make_Assignment_t(al, f.base.base.loc, pReturn_ref, + ASRUtils::EXPR(ASRUtils::make_FunctionCall_t_util(al, f.base.base.loc, sym_PyObject_CallObject, nullptr, + args_PyObject_CallObject.p, args_PyObject_CallObject.n, ptr_t, nullptr, nullptr)), + nullptr))); + + // Converting CPython result to native type + ASR::ttype_t *ret_type = ASRUtils::get_FunctionType(f)->m_return_var_type; + if (ret_type) { + std::string return_var_name = "_lpython_return_variable"; + ASR::asr_t *ret_var = (ASR::asr_t*)f.m_symtab->get_symbol(return_var_name); + LCOMPILERS_ASSERT(ret_var); + ASR::expr_t *ret_var_ref = ASRUtils::EXPR(ASR::make_Var_t(al, f.base.base.loc, + ASR::down_cast(ret_var))); + ASR::expr_t *ret_conv_result = cpython_to_native(al, pReturn_ref, ret_type, f, parent_scope); + body.push_back(al, ASRUtils::STMT(ASR::make_Assignment_t(al, f.base.base.loc, ret_var_ref, ret_conv_result, + nullptr))); + } + + /* + Py_DecRef(pName); + Py_DecRef(pArgs); + Py_DecRef(pReturn); + */ + ASR::symbol_t *sym_Py_DecRef = parent_scope.resolve_symbol("Py_DecRef"); + + Vec args_Py_DecRef; + args_Py_DecRef.reserve(al, 1); + args_Py_DecRef.push_back(al, {f.base.base.loc, pName_ref}); + body.push_back(al, ASRUtils::STMT(ASRUtils::make_SubroutineCall_t_util(al, f.base.base.loc, sym_Py_DecRef, nullptr, + args_Py_DecRef.p, args_Py_DecRef.n, nullptr, nullptr, false, false))); + + Vec args_Py_DecRef2; + args_Py_DecRef2.reserve(al, 1); + args_Py_DecRef2.push_back(al, {f.base.base.loc, pArgs_ref}); + body.push_back(al, ASRUtils::STMT(ASRUtils::make_SubroutineCall_t_util(al, f.base.base.loc, sym_Py_DecRef, nullptr, + args_Py_DecRef2.p, args_Py_DecRef2.n, nullptr, nullptr, false, false))); + + Vec args_Py_DecRef3; + args_Py_DecRef3.reserve(al, 1); + args_Py_DecRef3.push_back(al, {f.base.base.loc, pReturn_ref}); + body.push_back(al, ASRUtils::STMT(ASRUtils::make_SubroutineCall_t_util(al, f.base.base.loc, sym_Py_DecRef, nullptr, + args_Py_DecRef3.p, args_Py_DecRef3.n, nullptr, nullptr, false, false))); + + // reassignment + f.m_body = body.p; + f.n_body = body.n; + ASRUtils::get_FunctionType(f)->m_abi = ASR::abiType::Source; + ASRUtils::get_FunctionType(f)->m_deftype = ASR::deftypeType::Implementation; +} + +void pass_python_bind(Allocator &al, ASR::TranslationUnit_t &unit, const PassOptions &pass_options) { + if (pass_options.c_skip_bindpy_pass) { + // FIXME: C backend supports arrays, it is used in bindpy_02 to bindpy_04 tests. + // This pass currently does not handle any aggregate types. + // Once we include support for aggregate types, we should remove this check, and + // remove the `c_backend` variable from PassOptions. + // We will also need to remove the special handling of BindPython ABI in C + // backend as it would be handled by this ASR pass. + return; + } + + if (!pass_options.enable_cpython) { + // python_bind pass is skipped if CPython is not enabled + return; + } + + std::vector fns; + fns.push_back({"Py_Initialize", {}, ASRUtils::VOID}); + fns.push_back({"Py_IsInitialized", {}, ASRUtils::I32}); + // fns.push_back({"PyRun_SimpleString", {STR}, ASRUtils::I32}); + fns.push_back({"Py_DecodeLocale", {ASRUtils::STR, ASRUtils::PTR}, ASRUtils::PTR}); + fns.push_back({"PySys_SetArgv", {ASRUtils::I32, ASRUtils::PTR_TO_PTR}, ASRUtils::VOID}); + fns.push_back({"Py_FinalizeEx", {}, ASRUtils::I32}); + fns.push_back({"PyUnicode_FromString", {ASRUtils::STR}, ASRUtils::PTR}); + fns.push_back({"PyUnicode_AsUTF8AndSize", {ASRUtils::PTR, ASRUtils::PTR}, ASRUtils::STR}); + fns.push_back({"PyImport_Import", {ASRUtils::PTR}, ASRUtils::PTR}); + fns.push_back({"Py_DecRef", {ASRUtils::PTR}, ASRUtils::VOID}); + fns.push_back({"Py_IncRef", {ASRUtils::PTR}, ASRUtils::VOID}); + fns.push_back({"PyObject_GetAttrString", {ASRUtils::PTR, ASRUtils::STR}, ASRUtils::PTR}); + fns.push_back({"PyTuple_New", {ASRUtils::I32}, ASRUtils::PTR}); + fns.push_back({"PyTuple_SetItem", {ASRUtils::PTR, ASRUtils::I32, ASRUtils::PTR}, ASRUtils::I32}); + fns.push_back({"PyObject_CallObject", {ASRUtils::PTR, ASRUtils::PTR}, ASRUtils::PTR}); + fns.push_back({"PyLong_AsLongLong", {ASRUtils::PTR}, ASRUtils::I64}); + fns.push_back({"PyLong_AsUnsignedLongLong", {ASRUtils::PTR}, ASRUtils::U64}); + fns.push_back({"PyLong_FromLongLong", {ASRUtils::I64}, ASRUtils::PTR}); + fns.push_back({"PyLong_FromUnsignedLongLong", {ASRUtils::U64}, ASRUtils::PTR}); + fns.push_back({"PyFloat_FromDouble", {ASRUtils::F64}, ASRUtils::PTR}); + fns.push_back({"PyFloat_AsDouble", {ASRUtils::PTR}, ASRUtils::F64}); + fns.push_back({"PyBool_FromLong", {ASRUtils::I32}, ASRUtils::PTR}); + fns.push_back({"PyObject_IsTrue", {ASRUtils::PTR}, ASRUtils::I32}); + + Location *l = al.make_new(); + l->first = 0; + l->last = 0; + + ASRUtils::declare_functions(al, fns, *l, unit.m_symtab, "Python.h"); + + for (auto &item : unit.m_symtab->get_scope()) { + if (ASR::is_a(*item.second)) { + ASR::Function_t *f = ASR::down_cast(item.second); + if (ASRUtils::get_FunctionType(f)->m_abi == ASR::abiType::BindPython) { + if (f->n_body == 0 && f->m_module_file) { + generate_body(al, *f, *unit.m_symtab); + } + } + } + else if (ASR::is_a(*item.second)) { + ASR::Module_t *module = ASR::down_cast(item.second); + for (auto &module_item: module->m_symtab->get_scope() ) { + if (ASR::is_a(*module_item.second)) { + ASR::Function_t *f = ASR::down_cast(module_item.second); + if (ASRUtils::get_FunctionType(f)->m_abi == ASR::abiType::BindPython) { + if (f->n_body == 0 && f->m_module_file) { + generate_body(al, *f, *module->m_symtab); + } + } + } + } + } + + } + + PassUtils::UpdateDependenciesVisitor u(al); + u.visit_TranslationUnit(unit); +} + +} // namespace LCompilers diff --git a/src/libasr/pass/python_bind.h b/src/libasr/pass/python_bind.h new file mode 100644 index 0000000000..9f4c792184 --- /dev/null +++ b/src/libasr/pass/python_bind.h @@ -0,0 +1,14 @@ +#ifndef LIBASR_PASS_PYTHON_BIND_H +#define LIBASR_PASS_PYTHON_BIND_H + +#include +#include + +namespace LCompilers { + + void pass_python_bind(Allocator &al, ASR::TranslationUnit_t &unit, + const PassOptions &pass_options); + +} // namespace LCompilers + +#endif // LIBASR_PASS_PYTHON_BIND_H diff --git a/src/libasr/utils.h b/src/libasr/utils.h index 9ff8b0f963..97417b2bf4 100644 --- a/src/libasr/utils.h +++ b/src/libasr/utils.h @@ -59,6 +59,8 @@ struct PassOptions { bool tree = false; bool with_intrinsic_mods = false; bool c_mangling = false; + bool enable_cpython = false; + bool c_skip_bindpy_pass = false; }; struct CompilerOptions { @@ -98,7 +100,6 @@ struct CompilerOptions { std::string arg_o = ""; bool emit_debug_info = false; bool emit_debug_line_column = false; - bool enable_cpython = false; bool enable_symengine = false; bool link_numpy = false; bool run = false;