/** * \file imperative/python/src/common.cpp * MegEngine is Licensed under the Apache License, Version 2.0 (the "License") * * Copyright (c) 2014-2021 Megvii Inc. All rights reserved. * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT ARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. */ #include "./common.h" #include #include "./helper.h" #include "./numpy_dtypes.h" #include "megbrain/comp_node.h" #include "megbrain/graph.h" #include "megbrain/imperative/physical_tensor.h" namespace py = pybind11; using namespace mgb; using namespace imperative; namespace { template auto def_TensorND(py::object parent, const char* name) { return py::class_(parent, name) .def_property_readonly( "shape", py::overload_cast<>(&XTensorND::shape, py::const_)) .def_property_readonly( "dtype", py::overload_cast<>(&XTensorND::dtype, py::const_)) .def_property_readonly( "comp_node", py::overload_cast<>(&XTensorND::comp_node, py::const_)) .def("copy_from", &XTensorND::template copy_from) .def("copy_from", &XTensorND::template copy_from) .def("copy_from_fixlayout", py::overload_cast( &XTensorND::template copy_from_fixlayout)) .def("copy_from_fixlayout", py::overload_cast( &XTensorND::template copy_from_fixlayout)); } std::string default_device = "xpux"; } // namespace void set_default_device(const std::string& device) { default_device = device; } std::string get_default_device() { return default_device; } void init_common(py::module m) { auto PyCompNode = py::class_(m, "CompNode") .def(py::init()) .def(py::init( py::overload_cast(&CompNode::load))) .def_property_readonly( "logical_name", [](const CompNode& cn) { return cn.to_string_logical(); }) .def_property_readonly( "physical_name", [](const CompNode& cn) { return cn.to_string_physical(); }) .def_property_readonly( "get_mem_status_bytes", [](const CompNode& cn) { return cn.get_mem_status_bytes(); }) .def_property_readonly( "get_used_memory", [](const CompNode& cn) { return cn.get_used_memory(); }) .def_property_readonly( "get_max_used_memory", [](const CompNode& cn) { return cn.get_max_used_memory(); }) .def_property_readonly( "get_reserved_memory", [](const CompNode& cn) { return cn.get_reserved_memory(); }) .def_property_readonly( "get_max_reserved_memory", [](const CompNode& cn) { return cn.get_max_reserved_memory(); }) .def_static( "reset_max_memory_stats", [](const CompNode& cn) { cn.reset_max_used_memory(); cn.reset_max_reserved_memory(); }) .def("create_event", &CompNode::create_event, py::arg("flags") = 0ul) .def_static("_set_default_device", &set_default_device) .def_static("_get_default_device", &get_default_device) .def("__str__", &CompNode::to_string_logical) .def("__repr__", [](const CompNode& cn) { return mgb::ssprintf( "CompNode(\"%s\" from \"%s\")", cn.to_string_physical().c_str(), cn.to_string_logical().c_str()); }) .def("__hash__", [](CompNode cn) { return mgb::hash(cn); }) .def_static("_sync_all", &CompNode::sync_all) .def(py::self == py::self) .def_static( "_get_device_count", &CompNode::get_device_count, "Get total number of specific devices on this system") .def(py::pickle( [](const CompNode& cn) { return py::str(cn.to_string_logical()); }, [](py::str cn) { return CompNode::load(cn); })); py::class_>(PyCompNode, "Event") .def("record", &CompNode::Event::record) .def("wait", &CompNode::Event::host_wait); py::implicitly_convertible(); def_TensorND(m, "DeviceTensorND") .def("numpy", [](const DeviceTensorND& self) { HostTensorND hv; hv.copy_from(self).sync(); return py::handle( npy::ndarray_from_tensor(hv, npy::ShareType::TRY_SHARE)); }); def_TensorND(m, "HostTensorND") .def(py::init([](py::array data, CompNode cn, DType dtype) { if (!cn.valid()) { throw py::type_error("device must not be None"); } return npy::np2tensor(data.ptr(), npy::Meth::borrow(cn), dtype); })) .def("numpy", [](const HostTensorND& self) { return py::reinterpret_steal( npy::ndarray_from_tensor(self, npy::ShareType::TRY_SHARE)); }); py::class_(m, "OperatorNodeConfig") .def(py::init()) .def_property( "name", [](const OperatorNodeConfig& config) -> py::object { auto name = config.name(); if (name.valid()) { return py::str(name.val()); } else { return py::none(); } }, [](OperatorNodeConfig& config, std::string name) { config.name(std::move(name)); }) .def_property( "dtype", [](const OperatorNodeConfig& config) { return config.output_dtype(); }, [](OperatorNodeConfig& config, DType dtype) { config.output_dtype(dtype); }) .def_property( "comp_node_arr", [](const OperatorNodeConfig& config) -> py::tuple { auto arr = config.comp_node(); std::vector tmp(arr.begin(), arr.end()); return py::cast(tmp); }, [](OperatorNodeConfig& config, std::vector cns) { config.comp_node_arr({cns.begin(), cns.end()}); }) .def_property( "comp_node", [](const OperatorNodeConfig& config) { auto arr = config.comp_node(); if (arr.size() != 1) { throw py::value_error("invalid number of comp_node"); } return arr[0]; }, [](OperatorNodeConfig& config, CompNode cn) { OperatorNodeConfig::CompNodeArray arr{cn}; config.comp_node_arr(arr); }); py::class_(m, "TensorAttr") .def(py::init()) .def(py::init([](const TensorShape& shape, const DType& dtype, const CompNode& comp_node) { return LogicalTensorDesc{TensorLayout{shape, dtype}, comp_node}; })) .def_property( "shape", [](const LogicalTensorDesc& desc) { return static_cast(desc.layout); }, [](LogicalTensorDesc& desc, TensorShape shape) {}) .def_property( "dtype", [](const LogicalTensorDesc& desc) { return desc.layout.dtype; }, [](LogicalTensorDesc& desc, DType dtype) { desc.layout.dtype = dtype; }) .def_readwrite("comp_node", &LogicalTensorDesc::comp_node); py::enum_(m, "DeviceType") .value("UNSPEC", CompNode::DeviceType::UNSPEC) .value("CUDA", CompNode::DeviceType::CUDA) .value("ROCM", CompNode::DeviceType::ROCM) .value("CPU", CompNode::DeviceType::CPU) .value("CAMBRICON", CompNode::DeviceType::CAMBRICON) .value("ATLAS", CompNode::DeviceType::ATLAS) .value("MULTITHREAD", CompNode::DeviceType::MULTITHREAD) .value("MAX_DEVICE_ID", CompNode::DeviceType::MAX_DEVICE_ID); m.def("set_prealloc_config", &CompNode::set_prealloc_config, "specifies how to pre-allocate from raw dev allocator"); m.def("get_cuda_compute_capability", &CompNode::get_compute_capability); m.def("what_is_xpu", [] { return CompNode::Locator::parse("xpux").to_physical().type; }); init_npy_num_bfloat16(m); init_npy_num_intbx(m); init_dtypes(m); }