From 23eb8fddcf87787443cd93dd75bf0a204d300f0c Mon Sep 17 00:00:00 2001 From: Gagandeep Singh Date: Fri, 22 Mar 2024 00:33:30 +0530 Subject: [PATCH 01/12] TEST: Add minimal test for PyTorch C++ API --- integration_tests/pytorch_01.cpp | 7 +++++++ 1 file changed, 7 insertions(+) create mode 100644 integration_tests/pytorch_01.cpp diff --git a/integration_tests/pytorch_01.cpp b/integration_tests/pytorch_01.cpp new file mode 100644 index 0000000..24eba0b --- /dev/null +++ b/integration_tests/pytorch_01.cpp @@ -0,0 +1,7 @@ +#include +#include + +int main() { + torch::Tensor tensor = torch::rand({2, 3}); + std::cout << tensor << std::endl; +} From 7ae591e7c71890f31e7648a6e48e9502dfdd3bf9 Mon Sep 17 00:00:00 2001 From: Gagandeep Singh Date: Fri, 22 Mar 2024 00:34:05 +0530 Subject: [PATCH 02/12] TEST: Add support for pytorch testing in C++ --- integration_tests/CMakeLists.txt | 14 +++++++++++++- 1 file changed, 13 insertions(+), 1 deletion(-) diff --git a/integration_tests/CMakeLists.txt b/integration_tests/CMakeLists.txt index 0ed90ee..50b9373 100644 --- a/integration_tests/CMakeLists.txt +++ b/integration_tests/CMakeLists.txt @@ -110,6 +110,16 @@ macro(RUN_UTIL RUN_FAIL RUN_NAME RUN_FILE_NAME RUN_LABELS RUN_EXTRAFILES RUN_EXT endif() set(WASM_EXEC_FLAGS ${WASM_EXEC_FLAGS} "--experimental-wasi-unstable-preview1") add_test(${name} ${WASM_EXEC_RUNTIME} ${WASM_EXEC_FLAGS} ${CURRENT_BINARY_DIR}/${name}.js) + elseif (LC_BACKEND STREQUAL "pytorch") + # PyTorch C++ API testing + find_package(Torch REQUIRED) + set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} ${TORCH_CXX_FLAGS}") + + add_executable(${name} ${file_name} ${extra_files}) + target_link_libraries(${name} "${TORCH_LIBRARIES}") + target_compile_options(${name} PUBLIC ${gcc_args}) + set_property(TARGET ${name} PROPERTY CXX_STANDARD 17) + add_test(${name} ${CURRENT_BINARY_DIR}/${name}) else () add_executable(${name} ${file_name} ${extra_files}) target_compile_options(${name} PUBLIC ${gcc_args}) @@ -138,7 +148,7 @@ macro(RUN) "${multiValueArgs}" ${ARGN} ) foreach(b ${RUN_LABELS}) - if (NOT (b MATCHES "^(llvm|llvm2|llvm_rtlib|gcc|c|cpp|x86|wasm|gfortran|llvmImplicit|llvmStackArray|fortran|c_nopragma|llvm_nopragma)$")) + if (NOT (b MATCHES "^(llvm|llvm2|llvm_rtlib|gcc|c|cpp|x86|wasm|gfortran|llvmImplicit|llvmStackArray|fortran|c_nopragma|llvm_nopragma|pytorch)$")) message(FATAL_ERROR "Unsupported backend: ${b}") endif() endforeach() @@ -241,3 +251,5 @@ RUN(NAME vector_02.cpp LABELS gcc llvm) RUN(NAME loop_01.cpp LABELS gcc llvm NOFAST) RUN(NAME test_pkg_lnn_01.cpp LABELS gcc llvm NOFAST) + +RUN(NAME pytorch_01.cpp LABELS pytorch) From e88283f3d980c78643d21aa44052829d7c69e52b Mon Sep 17 00:00:00 2001 From: Gagandeep Singh Date: Fri, 22 Mar 2024 00:34:29 +0530 Subject: [PATCH 03/12] TEST: Register pytorch in integration_tests/run_tests.py --- integration_tests/run_tests.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/integration_tests/run_tests.py b/integration_tests/run_tests.py index 9fbeac8..e20207c 100755 --- a/integration_tests/run_tests.py +++ b/integration_tests/run_tests.py @@ -8,7 +8,7 @@ NO_OF_THREADS = 8 # default no of threads is 8 SUPPORTED_BACKENDS = ['llvm', 'llvm2', 'llvm_rtlib', 'c', 'cpp', 'x86', 'wasm', 'gcc', 'llvmImplicit', 'llvmStackArray', 'fortran', - 'c_nopragma', 'llvm_nopragma'] + 'c_nopragma', 'llvm_nopragma', 'pytorch'] BASE_DIR = os.path.dirname(os.path.realpath(__file__)) LC_PATH = f"{BASE_DIR}/../src/bin:$PATH" From 4a334d43e709df64802a3d92ecd973b7ee37c609 Mon Sep 17 00:00:00 2001 From: Gagandeep Singh Date: Fri, 22 Mar 2024 00:34:46 +0530 Subject: [PATCH 04/12] CI: Add testing of pytorch C++ API in CI for Linux --- .github/workflows/CI.yml | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/.github/workflows/CI.yml b/.github/workflows/CI.yml index 752e630..06ce778 100644 --- a/.github/workflows/CI.yml +++ b/.github/workflows/CI.yml @@ -83,3 +83,14 @@ jobs: export CPATH=$CONDA_PREFIX/include:$CPATH ./integration_tests/run_tests.py -b gcc llvm wasm c ./integration_tests/run_tests.py -b gcc llvm wasm c -f + + - name: Test4 (Linux) + shell: bash -l -e {0} + if: contains(matrix.os, 'ubuntu') + run: | + export CPATH=$CONDA_PREFIX/include:$CPATH + conda install --yes pytorch::pytorch + cp -r $CONDA_PREFIX/lib/python3.12/site-packages/torch/include/* $CONDA_PREFIX/include/ + cp -r $CONDA_PREFIX/lib/python3.12/site-packages/torch/lib/* $CONDA_PREFIX/lib/ + cp -r $CONDA_PREFIX/lib/python3.12/site-packages/torch/share/* $CONDA_PREFIX/share/ + ./integration_tests/run_tests.py -b pytorch From 4037908c7e5b4b89a17df661dbdf5c6fcb24bafb Mon Sep 17 00:00:00 2001 From: Gagandeep Singh Date: Fri, 22 Mar 2024 15:35:35 +0530 Subject: [PATCH 05/12] DEV: Use multiple torch::Tensor features in pytorch_01.cpp --- integration_tests/pytorch_01.cpp | 16 ++++++++++++++-- 1 file changed, 14 insertions(+), 2 deletions(-) diff --git a/integration_tests/pytorch_01.cpp b/integration_tests/pytorch_01.cpp index 24eba0b..4b5c96e 100644 --- a/integration_tests/pytorch_01.cpp +++ b/integration_tests/pytorch_01.cpp @@ -1,7 +1,19 @@ #include #include +void check(const torch::Tensor& tensor=torch::empty({1})) { + float array[5] = {4.0, 2.0, 2.0, 12.0, 2.0}; + std::cout << tensor << std::endl; + if( torch::any(torch::abs(tensor - torch::from_blob(array, {5})) > 1e-6).item() ) { + exit(2); + } +} + int main() { - torch::Tensor tensor = torch::rand({2, 3}); - std::cout << tensor << std::endl; + torch::Tensor tensor = torch::ones(5); + tensor[0] = 2.0; + tensor[3] = 6.0; + tensor = 2 * tensor; + check(tensor); + std::cout << tensor << std::endl; } From 0e1722c3fb0dc9260e7e740fbf7b3b36dd51e009 Mon Sep 17 00:00:00 2001 From: Gagandeep Singh Date: Sat, 23 Mar 2024 00:30:22 +0530 Subject: [PATCH 06/12] DEV: Add support for torch::Tensor variables and torch::ones(int scalar) --- src/lc/clang_ast_to_asr.cpp | 101 ++++++++++++++++++++++++++++++++++-- 1 file changed, 97 insertions(+), 4 deletions(-) diff --git a/src/lc/clang_ast_to_asr.cpp b/src/lc/clang_ast_to_asr.cpp index 6212cf8..18cd3f6 100644 --- a/src/lc/clang_ast_to_asr.cpp +++ b/src/lc/clang_ast_to_asr.cpp @@ -44,6 +44,8 @@ enum SpecialFunc { Clear, Data, Reserve, + + TorchOnes }; std::map special_function_map = { @@ -77,6 +79,7 @@ std::map special_function_map = { {"clear", SpecialFunc::Clear}, {"data", SpecialFunc::Data}, {"reserve", SpecialFunc::Reserve}, + {"torch::ones", SpecialFunc::TorchOnes}, }; class OneTimeUseString { @@ -155,6 +158,7 @@ class OneTimeUseASRNode { enum ThirdPartyCPPArrayTypes { XTensorArray, MDSpanArray, + PyTorchArray, }; class ClangASTtoASRVisitor: public clang::RecursiveASTVisitor { @@ -587,19 +591,35 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitorgetTypeClass() == clang::Type::TypeClass::Record ) { const clang::CXXRecordDecl* record_type = clang_type->getAsCXXRecordDecl(); std::string name = record_type->getNameAsString(); + std::string qualified_name = record_type->getQualifiedNameAsString(); if( name == "xtensor_container" || name == "vector" || name == "mdspan" ) { return nullptr; } ASR::symbol_t* type_t = current_scope->resolve_symbol(name); if( !type_t ) { - throw std::runtime_error(name + " not defined."); + if( qualified_name == "at::Tensor" ) { + if( array_type == nullptr || is_third_party_cpp_array == nullptr ) { + throw std::runtime_error("IEC: array_type and is_third_party_cpp_array couldn't be set."); + } + *is_third_party_cpp_array = true; + *array_type = ThirdPartyCPPArrayTypes::PyTorchArray; + type = ASRUtils::TYPE(ASR::make_Array_t(al, l, ASRUtils::TYPE(ASR::make_Real_t(al, l, 4)), + nullptr, 0, ASR::array_physical_typeType::DescriptorArray)); + return type; + } + throw std::runtime_error(qualified_name + " not defined."); } if( clang_type->isUnionType() ) { type = ASRUtils::TYPE(ASR::make_Union_t(al, l, type_t)); } else { type = ASRUtils::TYPE(ASR::make_Struct_t(al, l, type_t)); } - } else if( clang_type->getTypeClass() == clang::Type::TypeClass::SubstTemplateTypeParm ) { + } else if( clang_type->getTypeClass() == clang::Type::TypeClass::Using ) { + const clang::UsingType* using_type = clang_type->getAs(); + return ClangTypeToASRType(using_type->getUnderlyingType(), xshape_result, + array_type, is_third_party_cpp_array); + } else if( clang_type->getTypeClass() == clang::Type::TypeClass::SubstTemplateTypeParm || + clang_type->getTypeClass() == clang::Type::TypeClass::Typedef ) { return nullptr; } else { throw std::runtime_error("clang::QualType not yet supported " + @@ -1310,6 +1330,37 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitor expr_dims; expr_dims.reserve(al, 1); + if( ASR::is_a(*shape_arg) ) { + ASR::dimension_t expr_dim; + expr_dim.loc = Lloc(x); + expr_dim.m_start = ASRUtils::get_constant_zero_with_given_type( + al, ASRUtils::TYPE(ASR::make_Integer_t(al, Lloc(x), 4))); + expr_dim.m_length = shape_arg; + expr_dims.push_back(al, expr_dim); + ASR::ttype_t* type = ASRUtils::TYPE(ASR::make_Array_t(al, Lloc(x), + ASRUtils::TYPE(ASR::make_Real_t(al, Lloc(x), 4)), expr_dims.p, + expr_dims.size(), ASR::array_physical_typeType::FixedSizeArray)); + int num_ones = ASR::down_cast(shape_arg)->m_n; + Vec ones_vec; ones_vec.reserve(al, num_ones); + for( size_t onei = 0; onei < num_ones; onei++ ) { + ones_vec.push_back(al, one); + } + tmp = ASR::make_ArrayConstant_t(al, Lloc(x), ones_vec.p, ones_vec.size(), + type, ASR::arraystorageType::RowMajor); + is_stmt_created = false; + } else if( ASR::is_a(*shape_arg) ) { + throw std::runtime_error("{...} not yet supported in torch::ones"); + } + is_stmt_created = false; } else if( sf == SpecialFunc::All ) { // Handles xt::all() - no arguments // Handle with argument case later. @@ -1874,7 +1925,10 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitorgetType()); + Vec xshape_result; xshape_result.reserve(al, 0); + ThirdPartyCPPArrayTypes array_type; bool is_third_party_array_type = false; + ASR::ttype_t *asr_type = ClangTypeToASRType(x->getType(), &xshape_result, + &array_type, &is_third_party_array_type); ASR::symbol_t *v = ASR::down_cast(ASR::make_Variable_t(al, Lloc(x), current_scope, s2c(al, name), nullptr, 0, ASR::intentType::Local, nullptr, nullptr, ASR::storage_typeType::Default, asr_type, nullptr, ASR::abiType::Source, @@ -1905,6 +1959,38 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitorm_value = ASRUtils::expr_value(init_val); variable_t->m_storage = ASR::storage_typeType::Parameter; } else { + if( is_third_party_array_type ) { + if( array_type == ThirdPartyCPPArrayTypes::PyTorchArray ) { + ASR::dimension_t* dims = nullptr; + size_t n_dims = ASRUtils::extract_dimensions_from_ttype( + ASRUtils::expr_type(init_val), dims); + + Vec empty_dims; empty_dims.reserve(al, n_dims); + for( size_t dimi = 0; dimi < n_dims; dimi++ ) { + ASR::dimension_t empty_dim; + empty_dim.loc = Lloc(x); + empty_dim.m_start = ASRUtils::get_constant_zero_with_given_type( + al, ASRUtils::TYPE(ASR::make_Integer_t(al, Lloc(x), 4))); + empty_dim.m_length = nullptr; + empty_dims.push_back(al, empty_dim); + } + ASR::Variable_t* variable_t = ASR::down_cast(v); + variable_t->m_type = ASRUtils::TYPE(ASR::make_Array_t(al, Lloc(x), + ASRUtils::extract_type(variable_t->m_type), empty_dims.p, empty_dims.size(), + ASR::array_physical_typeType::DescriptorArray)); + variable_t->m_type = ASRUtils::TYPE(ASR::make_Allocatable_t( + al, Lloc(x), variable_t->m_type)); + + Vec alloc_args; alloc_args.reserve(al, 1); + ASR::alloc_arg_t alloc_arg; alloc_arg.loc = Lloc(x); + alloc_arg.m_a = var; + alloc_arg.m_dims = dims; alloc_arg.n_dims = n_dims; + alloc_arg.m_len_expr = nullptr; alloc_arg.m_type = nullptr; + alloc_args.push_back(al, alloc_arg); + current_body->push_back(al, ASRUtils::STMT(ASR::make_Allocate_t( + al, Lloc(x), alloc_args.p, alloc_args.size(), nullptr, nullptr, nullptr))); + } + } add_reshape_if_needed(init_val, var); tmp = ASR::make_Assignment_t(al, Lloc(x), var, init_val, nullptr); is_stmt_created = true; @@ -2465,6 +2551,10 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitorgetNameInfo().getAsString(); + std::string namespace_name = ""; + if( x->getQualifier() ) { + namespace_name = x->getQualifier()->getAsNamespace()->getNameAsString(); + } ASR::symbol_t* sym = resolve_symbol(name); if( name == "operator<<" || name == "cout" || name == "endl" || name == "operator()" || name == "operator+" || name == "operator=" || @@ -2476,7 +2566,8 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitor=" || name == "operator!=" || name == "operator\"\"i" || name == "sin" || - name == "cos" || name == "amin" || name == "operator[]" || name == "sqrt" ) { + name == "cos" || name == "amin" || name == "operator[]" || name == "sqrt" || + name == "ones" ) { if( sym != nullptr && ASR::is_a( *ASRUtils::symbol_get_past_external(sym)) ) { throw std::runtime_error("Special function " + name + " cannot be overshadowed yet."); @@ -2484,6 +2575,8 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitor Date: Sat, 23 Mar 2024 01:03:34 +0530 Subject: [PATCH 07/12] DEV: Add support torch::Tensor::operator[] and binary operations on torch::Tensor --- src/lc/clang_ast_to_asr.cpp | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/src/lc/clang_ast_to_asr.cpp b/src/lc/clang_ast_to_asr.cpp index 18cd3f6..1413b2b 100644 --- a/src/lc/clang_ast_to_asr.cpp +++ b/src/lc/clang_ast_to_asr.cpp @@ -606,6 +606,8 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitor(*ASRUtils::extract_type( - ASRUtils::expr_type(obj))) ) { + ASRUtils::expr_type(obj))) || + ASR::is_a(*obj) ) { TraverseStmt(args[1]); if( !is_stmt_created ) { ASR::expr_t* value = ASRUtils::EXPR(tmp.get()); @@ -2239,10 +2242,10 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitor Date: Sat, 23 Mar 2024 02:20:49 +0530 Subject: [PATCH 08/12] DEV: Add support for declaring functions accepting torch::Tensor --- src/lc/clang_ast_to_asr.cpp | 87 +++++++++++++++++++++++++++++++------ 1 file changed, 74 insertions(+), 13 deletions(-) diff --git a/src/lc/clang_ast_to_asr.cpp b/src/lc/clang_ast_to_asr.cpp index 1413b2b..81fd305 100644 --- a/src/lc/clang_ast_to_asr.cpp +++ b/src/lc/clang_ast_to_asr.cpp @@ -45,7 +45,8 @@ enum SpecialFunc { Data, Reserve, - TorchOnes + TorchOnes, + TorchEmpty, }; std::map special_function_map = { @@ -79,7 +80,9 @@ std::map special_function_map = { {"clear", SpecialFunc::Clear}, {"data", SpecialFunc::Data}, {"reserve", SpecialFunc::Reserve}, + {"torch::ones", SpecialFunc::TorchOnes}, + {"torch::empty", SpecialFunc::TorchEmpty}, }; class OneTimeUseString { @@ -603,7 +606,7 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitorget_unique_name("param"); } - ASR::ttype_t* type = ClangTypeToASRType(x->getType()); + + bool is_third_party_array_type = false; + ThirdPartyCPPArrayTypes array_type; + Vec shape_result; shape_result.reserve(al, 1); + ASR::ttype_t* type = ClangTypeToASRType(x->getType(), &shape_result, + &array_type, &is_third_party_array_type); + if( is_third_party_array_type && + array_type == ThirdPartyCPPArrayTypes::PyTorchArray ) { + if( !x->getDefaultArg() ) { + throw std::runtime_error("torch::Tensor type arguments must have default arguments."); + } + } ASR::intentType intent_type = ASR::intentType::InOut; if( ASR::is_a(*type) ) { intent_type = ASR::intentType::In; @@ -806,20 +820,30 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitor(tmp.get()); + current_scope->add_symbol(name, tmp_sym); + ASR::asr_t* tmp_ = ASR::make_Var_t(al, Lloc(x), tmp_sym); + clang::Expr *init = x->getDefaultArg(); ASR::expr_t* asr_init = nullptr; if (init) { + ASR::expr_t* assignment_target_copy = assignment_target; + assignment_target = ASRUtils::EXPR(tmp_); TraverseStmt(init); - asr_init = ASRUtils::EXPR(tmp.get()); + if( tmp != nullptr && !is_stmt_created ) { + asr_init = ASRUtils::EXPR(tmp.get()); + } } - tmp = ASR::make_Variable_t(al, Lloc(x), current_scope, s2c(al, name), - nullptr, 0, ASR::intentType::InOut, asr_init, nullptr, - ASR::storage_typeType::Default, type, nullptr, ASR::abiType::Source, - ASR::accessType::Public, ASR::presenceType::Required, false); - ASR::symbol_t* tmp_sym = ASR::down_cast(tmp.get()); - current_scope->add_symbol(name, tmp_sym); - tmp = ASR::make_Var_t(al, Lloc(x), tmp_sym); + // TODO: For PyTorch tensor create an intrinsic empty + // and then fill the initialiser value with a call + // to that intrinsic. + + tmp = tmp_; is_stmt_created = false; return true; } @@ -1340,7 +1364,7 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitor expr_dims; expr_dims.reserve(al, 1); if( ASR::is_a(*shape_arg) ) { ASR::dimension_t expr_dim; @@ -1350,7 +1374,7 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitor(shape_arg)->m_n; Vec ones_vec; ones_vec.reserve(al, num_ones); @@ -1497,6 +1521,37 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitor(*args.p[0]) ) { + ASR::ArrayConstant_t* array_constant = ASR::down_cast(args.p[0]); + + Vec empty_dims; empty_dims.reserve(al, array_constant->n_args); + for( size_t idim = 0; idim < array_constant->n_args; idim++ ) { + ASR::dimension_t empty_dim; + empty_dim.loc = Lloc(x); + empty_dim.m_start = ASRUtils::get_constant_zero_with_given_type( + al, ASRUtils::TYPE(ASR::make_Integer_t(al, Lloc(x), 4))); + empty_dim.m_length = nullptr; + empty_dims.push_back(al, empty_dim); + } + ASR::ttype_t* type = ASRUtils::TYPE(ASR::make_Array_t(al, Lloc(x), + ASRUtils::extract_type(ASRUtils::expr_type(assignment_target)), + empty_dims.p, empty_dims.size(), ASR::array_physical_typeType::DescriptorArray)); + type = ASRUtils::TYPE(ASR::make_Allocatable_t(al, Lloc(x), type)); + ASR::down_cast( + ASR::down_cast(assignment_target)->m_v)->m_type = type; + tmp = nullptr; + is_stmt_created = false; + } else { + throw std::runtime_error("Only {...} is allowed for supplying shape to xt::empty."); + } } else if (sf == SpecialFunc::Iota) { tmp = ASR::make_ComplexConstant_t(al, Lloc(x), 0.0, 1.0, ASRUtils::TYPE(ASR::make_Complex_t(al, Lloc(x), 8))); @@ -1932,6 +1987,12 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitorgetType(), &xshape_result, &array_type, &is_third_party_array_type); + if( is_third_party_array_type && + array_type == ThirdPartyCPPArrayTypes::PyTorchArray ) { + if( !x->hasInit() ) { + throw std::runtime_error("torch::Tensor variables must have initialiser value."); + } + } ASR::symbol_t *v = ASR::down_cast(ASR::make_Variable_t(al, Lloc(x), current_scope, s2c(al, name), nullptr, 0, ASR::intentType::Local, nullptr, nullptr, ASR::storage_typeType::Default, asr_type, nullptr, ASR::abiType::Source, From 0bda8b940cb1956a19123b40b7d91fce16a67b7e Mon Sep 17 00:00:00 2001 From: Gagandeep Singh Date: Sat, 23 Mar 2024 03:24:57 +0530 Subject: [PATCH 09/12] DEV: Add support for torch::abs, torch::from_blob, torch::any and .item<...> --- src/lc/clang_ast_to_asr.cpp | 43 ++++++++++++++++++++++++++++++++++++- 1 file changed, 42 insertions(+), 1 deletion(-) diff --git a/src/lc/clang_ast_to_asr.cpp b/src/lc/clang_ast_to_asr.cpp index 81fd305..f984155 100644 --- a/src/lc/clang_ast_to_asr.cpp +++ b/src/lc/clang_ast_to_asr.cpp @@ -47,6 +47,8 @@ enum SpecialFunc { TorchOnes, TorchEmpty, + TorchFromBlob, + TorchTensorItem, }; std::map special_function_map = { @@ -83,6 +85,10 @@ std::map special_function_map = { {"torch::ones", SpecialFunc::TorchOnes}, {"torch::empty", SpecialFunc::TorchEmpty}, + {"torch::from_blob", SpecialFunc::TorchFromBlob}, + {"torch::abs", SpecialFunc::Abs}, + {"torch::any", SpecialFunc::Any}, + {"item", SpecialFunc::TorchTensorItem}, }; class OneTimeUseString { @@ -1159,6 +1165,9 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitorm_v)); } if (special_function_map.find(func_name) == special_function_map.end()) { + if( current_scope->resolve_symbol(func_name) == nullptr ) { + throw std::runtime_error("ICE: " + func_name + " is not handled yet in LC."); + } return false; } SpecialFunc sf = special_function_map[func_name]; @@ -1357,6 +1366,16 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitor(callee); + is_stmt_created = false; } else if (sf == SpecialFunc::TorchOnes) { if( args.size() != 2 ) { // second one is TorchOptions, to be ignored throw std::runtime_error("torch::ones should be called with only one argument."); @@ -1552,6 +1571,28 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitor(*args.p[1]) ) { + ASR::ArrayConstant_t* array_constant = ASR::down_cast(args.p[1]); + + Vec empty_dims; empty_dims.reserve(al, array_constant->n_args); + for( size_t idim = 0; idim < array_constant->n_args; idim++ ) { + if( !ASRUtils::is_value_equal(array_constant->m_args[idim], m_dims[idim].m_length) ) { + throw std::runtime_error("ICE: Could not decipher the equality of the shape " + "and shape of the array provided in torch::from_blob"); + } + } + tmp = reinterpret_cast(args.p[0]); + is_stmt_created = false; + } else { + throw std::runtime_error("Only {...} is allowed for supplying shape to torch::from_blob."); + } } else if (sf == SpecialFunc::Iota) { tmp = ASR::make_ComplexConstant_t(al, Lloc(x), 0.0, 1.0, ASRUtils::TYPE(ASR::make_Complex_t(al, Lloc(x), 8))); @@ -2631,7 +2672,7 @@ class ClangASTtoASRVisitor: public clang::RecursiveASTVisitor=" || name == "operator!=" || name == "operator\"\"i" || name == "sin" || name == "cos" || name == "amin" || name == "operator[]" || name == "sqrt" || - name == "ones" ) { + name == "ones" || name == "from_blob" ) { if( sym != nullptr && ASR::is_a( *ASRUtils::symbol_get_past_external(sym)) ) { throw std::runtime_error("Special function " + name + " cannot be overshadowed yet."); From 0c752db6eaca414e59029fdd771351a551a5e866 Mon Sep 17 00:00:00 2001 From: Gagandeep Singh Date: Sat, 23 Mar 2024 03:26:22 +0530 Subject: [PATCH 10/12] TEST: Improvided pytorch_01.cpp test --- integration_tests/pytorch_01.cpp | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/integration_tests/pytorch_01.cpp b/integration_tests/pytorch_01.cpp index 4b5c96e..786e5cf 100644 --- a/integration_tests/pytorch_01.cpp +++ b/integration_tests/pytorch_01.cpp @@ -3,8 +3,8 @@ void check(const torch::Tensor& tensor=torch::empty({1})) { float array[5] = {4.0, 2.0, 2.0, 12.0, 2.0}; - std::cout << tensor << std::endl; - if( torch::any(torch::abs(tensor - torch::from_blob(array, {5})) > 1e-6).item() ) { + std::cout << tensor << "\n"; + if( torch::any(torch::abs(tensor - torch::from_blob(array, {5})) > 1e-8).item() ) { exit(2); } } @@ -15,5 +15,5 @@ int main() { tensor[3] = 6.0; tensor = 2 * tensor; check(tensor); - std::cout << tensor << std::endl; + std::cout << tensor << "\n"; } From 93fa807f7af6ced817ac1afea989642f94d9a2d9 Mon Sep 17 00:00:00 2001 From: Gagandeep Singh Date: Sat, 23 Mar 2024 03:26:54 +0530 Subject: [PATCH 11/12] TEST: Added llvmPytorch testing --- integration_tests/CMakeLists.txt | 6 +++--- integration_tests/run_tests.py | 2 +- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/integration_tests/CMakeLists.txt b/integration_tests/CMakeLists.txt index 50b9373..c7b8ea8 100644 --- a/integration_tests/CMakeLists.txt +++ b/integration_tests/CMakeLists.txt @@ -84,7 +84,7 @@ macro(RUN_UTIL RUN_FAIL RUN_NAME RUN_FILE_NAME RUN_LABELS RUN_EXTRAFILES RUN_EXT if (ADD_TEST) if ((LC_BACKEND STREQUAL "llvm") OR (LC_BACKEND STREQUAL "cpp") OR (LC_BACKEND STREQUAL "x86") - OR (LC_BACKEND STREQUAL "c") OR (LC_BACKEND STREQUAL "fortran")) + OR (LC_BACKEND STREQUAL "c") OR (LC_BACKEND STREQUAL "fortran") OR (LC_BACKEND STREQUAL "llvmPytorch")) add_custom_command( OUTPUT ${name}.o COMMAND ${LC} -c ${CMAKE_CURRENT_SOURCE_DIR}/${file_name} -o ${name}.o ${extra_args} @@ -148,7 +148,7 @@ macro(RUN) "${multiValueArgs}" ${ARGN} ) foreach(b ${RUN_LABELS}) - if (NOT (b MATCHES "^(llvm|llvm2|llvm_rtlib|gcc|c|cpp|x86|wasm|gfortran|llvmImplicit|llvmStackArray|fortran|c_nopragma|llvm_nopragma|pytorch)$")) + if (NOT (b MATCHES "^(llvm|llvm2|llvm_rtlib|gcc|c|cpp|x86|wasm|gfortran|llvmImplicit|llvmStackArray|fortran|c_nopragma|llvm_nopragma|pytorch|llvmPytorch)$")) message(FATAL_ERROR "Unsupported backend: ${b}") endif() endforeach() @@ -252,4 +252,4 @@ RUN(NAME loop_01.cpp LABELS gcc llvm NOFAST) RUN(NAME test_pkg_lnn_01.cpp LABELS gcc llvm NOFAST) -RUN(NAME pytorch_01.cpp LABELS pytorch) +RUN(NAME pytorch_01.cpp LABELS pytorch llvmPytorch) diff --git a/integration_tests/run_tests.py b/integration_tests/run_tests.py index e20207c..0541bf8 100755 --- a/integration_tests/run_tests.py +++ b/integration_tests/run_tests.py @@ -8,7 +8,7 @@ NO_OF_THREADS = 8 # default no of threads is 8 SUPPORTED_BACKENDS = ['llvm', 'llvm2', 'llvm_rtlib', 'c', 'cpp', 'x86', 'wasm', 'gcc', 'llvmImplicit', 'llvmStackArray', 'fortran', - 'c_nopragma', 'llvm_nopragma', 'pytorch'] + 'c_nopragma', 'llvm_nopragma', 'pytorch', 'llvmPytorch'] BASE_DIR = os.path.dirname(os.path.realpath(__file__)) LC_PATH = f"{BASE_DIR}/../src/bin:$PATH" From 6664dbd82ff9eadc791b83e3cf7937d851650e20 Mon Sep 17 00:00:00 2001 From: Gagandeep Singh Date: Sat, 23 Mar 2024 03:27:09 +0530 Subject: [PATCH 12/12] CI: Test llvmPytorch on CI --- .github/workflows/CI.yml | 2 ++ 1 file changed, 2 insertions(+) diff --git a/.github/workflows/CI.yml b/.github/workflows/CI.yml index 06ce778..065663c 100644 --- a/.github/workflows/CI.yml +++ b/.github/workflows/CI.yml @@ -94,3 +94,5 @@ jobs: cp -r $CONDA_PREFIX/lib/python3.12/site-packages/torch/lib/* $CONDA_PREFIX/lib/ cp -r $CONDA_PREFIX/lib/python3.12/site-packages/torch/share/* $CONDA_PREFIX/share/ ./integration_tests/run_tests.py -b pytorch + export CPATH=$CPATH:$CONDA_PREFIX/include/torch/csrc/api/include + ./integration_tests/run_tests.py -b llvmPytorch