// Copyright (c) 2019 PaddlePaddle Authors. 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. /* * This file defines PaddlePredictor, the api for lite. It supports multiple * hardware including ARM, X86, OpenCL, CUDA and so on. */ #ifndef PADDLE_LITE_API_H_ // NOLINT #define PADDLE_LITE_API_H_ #include #include #include #include #include #include "paddle_place.h" // NOLINT namespace paddle { namespace lite_api { using shape_t = std::vector; using lod_t = std::vector>; enum class LiteModelType { kProtobuf = 0, kNaiveBuffer, UNK }; // Methods for allocating L3Cache on Arm platform enum class L3CacheSetMethod { kDeviceL3Cache = 0, // Use the system L3 Cache size, best performance. kDeviceL2Cache = 1, // Use the system L2 Cache size, trade off performance // with less memory consumption. kAbsolute = 2, // Use the external setting. // kAutoGrow = 3, // Not supported yet, least memory consumption. }; // return true if current device supports OpenCL model LITE_API bool IsOpenCLBackendValid(bool check_fp16_valid = false); // return current opencl device type, // if opencl not enabled or IsOpenCLBackendValid return false, it will return -1 // UNKNOWN:0, QUALCOMM_ADRENO:1, ARM_MALI:2, IMAGINATION_POWERVR:3, OTHERS:4, LITE_API int GetOpenCLDeviceType(); struct LITE_API Tensor { explicit Tensor(void* raw); explicit Tensor(const void* raw); void Resize(const shape_t& shape); /// Readonly data. template const T* data() const; template T* mutable_data(TargetType type = TargetType::kHost) const; void* mutable_metal_data(void* ptr) const; // Share external memory. Note: ensure that the data pointer is in a valid // state // during the prediction process. void ShareExternalMemory(void* data, size_t memory_size, TargetType target); template void CopyFromCpu(const T* data); template void CopyToCpu(T* data) const; /// Shape of the tensor. shape_t shape() const; TargetType target() const; PrecisionType precision() const; void SetPrecision(PrecisionType precision); // LoD of the tensor lod_t lod() const; // Set LoD of the tensor void SetLoD(const lod_t& lod); bool IsInitialized() const; private: void* raw_tensor_; }; /// The PaddlePredictor defines the basic interfaces for different kinds of /// predictors. class LITE_API PaddlePredictor { public: PaddlePredictor() = default; /// Get i-th input. virtual std::unique_ptr GetInput(int i) = 0; /// Get i-th output. virtual std::unique_ptr GetOutput(int i) const = 0; virtual void Run() = 0; virtual std::shared_ptr Clone() = 0; virtual std::shared_ptr Clone( const std::vector& var_names) = 0; virtual std::string GetVersion() const = 0; // Get input names virtual std::vector GetInputNames() = 0; // Get output names virtual std::vector GetOutputNames() = 0; // Get output names virtual std::vector GetParamNames(); /// Release all tmp tensor to compress the size of the memory pool. virtual bool TryShrinkMemory() = 0; // Get Input by name virtual std::unique_ptr GetInputByName(const std::string& name) = 0; /// Get a readonly tensor, return null if no one called `name` exists. virtual std::unique_ptr GetTensor( const std::string& name) const = 0; /// Get a mutable tensor, return null if on one called `name` exists /// internal infereces API, not recommanded. virtual std::unique_ptr GetMutableTensor(const std::string& name); /// Persist the optimized model to disk. This API is only supported by /// CxxConfig, and the persisted model can be reused for MobileConfig. virtual void SaveOptimizedModel( const std::string& model_dir, LiteModelType model_type = LiteModelType::kProtobuf, bool record_info = false); virtual ~PaddlePredictor() = default; protected: int threads_{1}; lite_api::PowerMode mode_{lite_api::LITE_POWER_NO_BIND}; }; /// Base class for all the configs. class LITE_API ConfigBase { std::string model_dir_; int threads_{1}; PowerMode mode_{LITE_POWER_NO_BIND}; // gpu opencl CLTuneMode opencl_tune_mode_{CL_TUNE_NONE}; std::string opencl_bin_path_{""}; std::string opencl_bin_name_{""}; CLPrecisionType opencl_precision_{CL_PRECISION_AUTO}; // Where to cache the npu/xpu/rknpu/apu offline model to the binary files std::string subgraph_model_cache_dir_{""}; // Set the cached npu/xpu/rknpu/apu offline model from the buffers std::map, std::vector>> subgraph_model_cache_buffers_{}; // The selected NNAdapter devices to build and run the model. std::vector nnadapter_device_names_{}; // The NNAdapter context properties for device configuration, model // compilation and execution std::string nnadapter_context_properties_{}; // The directory to find and store the compiled NNAdapter models. std::string nnadapter_model_cache_dir_{""}; // The buffers for loading the compiled NNAdapter models from memory. std::map> nnadapter_model_cache_buffers_{}; // The custom configuration file or buffer for the NNAdapter subgraph // partition, here is an example: // op_type:in_var_name_0,in_var_name1:out_var_name_0,out_var_name1 // op_type::out_var_name_0 // op_type:in_var_name_0 // op_type std::string nnadapter_subgraph_partition_config_path_{}; std::string nnadapter_subgraph_partition_config_buffer_{}; int device_id_{0}; int x86_math_num_threads_ = 1; std::string metal_path_; bool metal_use_mps_{false}; bool metal_use_aggressive_{false}; void* metal_device_{nullptr}; bool metal_use_memory_reuse_{false}; public: explicit ConfigBase(PowerMode mode = LITE_POWER_NO_BIND, int threads = 1); // set Model_dir void set_model_dir(const std::string& x) { model_dir_ = x; } const std::string& model_dir() const { return model_dir_; } // set Thread void set_threads(int threads); int threads() const { return threads_; } // set Power_mode void set_power_mode(PowerMode mode); PowerMode power_mode() const { return mode_; } /// \brief Set path and file name of generated OpenCL compiled kernel binary. /// /// If you use GPU of specific soc, using OpenCL binary will speed up the /// initialization. /// /// \param path Path that OpenCL compiled kernel binay file stores in. Make /// sure the path exist and you have Read&Write permission. /// \param name File name of OpenCL compiled kernel binay. /// \return void void set_opencl_binary_path_name(const std::string& path, const std::string& name); // set GPU opencl tune void set_opencl_tune(CLTuneMode tune_mode = CL_TUNE_NONE, const std::string& path = "", const std::string& name = "", size_t lws_repeats = 4); // set GPU opencl precision void set_opencl_precision(CLPrecisionType p = CL_PRECISION_AUTO); // set subgraph_model_dir void set_subgraph_model_cache_dir(std::string subgraph_model_cache_dir) { subgraph_model_cache_dir_ = subgraph_model_cache_dir; } const std::string& subgraph_model_cache_dir() const { return subgraph_model_cache_dir_; } void set_subgraph_model_cache_buffers(const std::string& key, const std::vector& cfg, const std::vector& bin); const std::map, std::vector>>& subgraph_model_cache_buffers() const { return subgraph_model_cache_buffers_; } // Check if the NNAdapter device is valid. bool check_nnadapter_device_name(const std::string& device_name); // Choose the NNAdapter devices to build and run the model. void set_nnadapter_device_names( const std::vector& device_names) { nnadapter_device_names_ = device_names; } const std::vector& nnadapter_device_names() const { return nnadapter_device_names_; } // Set the context properties by key-value map for NNAdapter device // configuration, model compilation and execution // Such as "HUAWEI_ASCEND_NPU_SELECTED_DEVICE_IDS=0;" void set_nnadapter_context_properties(const std::string& context_properties) { nnadapter_context_properties_ = context_properties; } const std::string& nnadapter_context_properties() const { return nnadapter_context_properties_; } // Enable caching and set the directory to search and store the compiled // NNAdapter models in the file system. void set_nnadapter_model_cache_dir(const std::string& model_cache_dir) { nnadapter_model_cache_dir_ = model_cache_dir; } const std::string& nnadapter_model_cache_dir() const { return nnadapter_model_cache_dir_; } // Set the buffers for loading the compiled NNAdapter models from memory. void set_nnadapter_model_cache_buffers( const std::string& model_cache_token, const std::vector& model_cache_buffer); const std::map>& nnadapter_model_cache_buffers() const { return nnadapter_model_cache_buffers_; } // Enable the custom subgraph partition for NNAdapter by providing the // configuration file or buffer void set_nnadapter_subgraph_partition_config_path( const std::string& subgraph_partition_config_path) { nnadapter_subgraph_partition_config_path_ = subgraph_partition_config_path; } const std::string& nnadapter_subgraph_partition_config_path() const { return nnadapter_subgraph_partition_config_path_; } void set_nnadapter_subgraph_partition_config_buffer( const std::string& subgraph_partition_config_buffer) { nnadapter_subgraph_partition_config_buffer_ = subgraph_partition_config_buffer; } const std::string& nnadapter_subgraph_partition_config_buffer() const { return nnadapter_subgraph_partition_config_buffer_; } // set Device ID void set_device_id(int device_id) { device_id_ = device_id; } int get_device_id() const { return device_id_; } // set x86_math_num_threads void set_x86_math_num_threads(int threads); int x86_math_num_threads() const; void set_metal_lib_path(const std::string& path); void set_metal_use_mps(bool flag); void set_metal_use_aggressive(bool flag); void set_metal_device(void* device); void set_metal_use_memory_reuse(bool flag); std::string metal_lib_path() const { return metal_path_; } bool metal_use_mps() const { return metal_use_mps_; } bool metal_use_aggressive() const { return metal_use_aggressive_; } void* metal_device() const { return metal_device_; } bool metal_use_memory_reuse() const { return metal_use_memory_reuse_; } }; class LITE_API CxxModelBuffer { public: CxxModelBuffer(const char* program_buffer, size_t program_buffer_size, const char* params_buffer, size_t params_buffer_size); CxxModelBuffer(std::string&& program_buffer, std::string&& params_buffer); const std::string& get_program() const; const std::string& get_params() const; bool is_empty() const; CxxModelBuffer() = default; CxxModelBuffer(const CxxModelBuffer&) = delete; private: std::string program_; std::string params_; }; /// CxxConfig is the config for the Full feature predictor. class LITE_API CxxConfig : public ConfigBase { std::vector valid_places_; std::string model_file_; std::string param_file_; std::shared_ptr model_buffer_{nullptr}; std::vector passes_internal_{}; bool quant_model_{false}; // Enable post_quant_dynamic in opt QuantType quant_type_{QuantType::QUANT_INT16}; bool sparse_model_{false}; // Enable sparse_conv_detect_pass in opt float sparse_threshold_{0.6f}; std::map>> preferred_inputs_for_warmup_; #ifdef LITE_WITH_CUDA bool multi_stream_{false}; #endif #ifdef LITE_WITH_MLU lite_api::MLUCoreVersion mlu_core_version_{lite_api::MLUCoreVersion::MLU_270}; int mlu_core_number_{1}; DataLayoutType mlu_input_layout_{DATALAYOUT(kNCHW)}; std::vector mlu_first_conv_mean_{}; std::vector mlu_first_conv_std_{}; #endif public: void set_valid_places(const std::vector& x) { valid_places_ = x; } void set_model_file(const std::string& path) { model_file_ = path; } void set_param_file(const std::string& path) { param_file_ = path; } void set_model_buffer(const char* model_buffer, size_t model_buffer_size, const char* param_buffer, size_t param_buffer_size) { model_buffer_.reset(new CxxModelBuffer( model_buffer, model_buffer_size, param_buffer, param_buffer_size)); } void set_model_buffer(std::shared_ptr model_buffer) { model_buffer_ = model_buffer; } const CxxModelBuffer& get_model_buffer() const; // internal inference to choose passes for model optimizing, // it's designed for internal developer and not recommanded // for comman users. void set_passes_internal( const std::vector& passes_internal = {}) { passes_internal_ = passes_internal; } const std::vector& get_passes_internal() const { return passes_internal_; } const std::vector& valid_places() const { return valid_places_; } std::string model_file() const { return model_file_; } std::string param_file() const { return param_file_; } bool is_model_from_memory() const { return static_cast(model_buffer_); } // note: `model_from_memory` has the same effect as `is_model_from_memory`, // but is_model_from_memory is recommended and `model_from_memory` will be // abandoned in v3.0. bool model_from_memory() const { return static_cast(model_buffer_); } #ifdef LITE_WITH_CUDA void set_multi_stream(bool multi_stream) { multi_stream_ = multi_stream; } bool multi_stream() const { return multi_stream_; } #endif #ifdef LITE_WITH_MLU // set MLU core version, which is used when compiling MLU kernels void set_mlu_core_version(lite_api::MLUCoreVersion core_version); // set MLU core number, which is used when compiling MLU kernels void set_mlu_core_number(int core_number); // whether use MLU's first conv kernel. First conv is a special kernel // provided by MLU, its input is uint8, and also needs two 3-dimentional // vectors which save all inputs' mean and std values // set the 3-dimentional mean vector and 3-dimentional std vector used by // MLU's first conv void set_mlu_firstconv_param(const std::vector& mean, const std::vector& std); // set MLU input layout. User can specify layout of input data to be NHWC, // default is NCHW void set_mlu_input_layout(DataLayoutType layout); lite_api::MLUCoreVersion mlu_core_version() const; int mlu_core_number() const; DataLayoutType mlu_input_layout() const; // std::pair std::pair, std::vector> mlu_firstconv_param() const; #endif // XPU only, set the size of the workspace memory from L3 cache for the // current thread. // **DEPRECATED**, use set_xpu_l3_cache_method() in the future void set_xpu_workspace_l3_size_per_thread(int l3_size = 0x4000000); void set_xpu_l3_cache_method(size_t l3_size, bool locked = false); void set_xpu_gm_workspace_method(size_t gm_size); void set_xpu_conv_autotune(bool autotune = true, const std::string& autotune_file = ""); // XPU only, specify the target device ID for the current thread. // **DEPRECATED**, use xpu_set_device() at the very beginning of each worker // thread void set_xpu_dev_per_thread(int dev_no = 0); // **DEPRECATED**, use set_xpu_multi_encoder_method() in the future void set_xpu_multi_encoder_precision(const std::string& precision = "int16"); void set_xpu_multi_encoder_method(const std::string& precision = "int16", bool adaptive_seqlen = false); // set input tensor for warmup. // It is optional. If you set prefered_inputs, model wil run immediately when // predictor is created template void set_preferred_inputs_for_warmup(const int group_idx, const int tensor_idx, const shape_t& shape, const lod_t& lod = {}, const T fill_value = 0, const void* data = nullptr); const std::map>>& preferred_inputs_for_warmup() const { return preferred_inputs_for_warmup_; } void set_quant_model(bool quant_model) { quant_model_ = quant_model; } bool quant_model() const { return quant_model_; } void set_quant_type(QuantType quant_type) { quant_type_ = quant_type; } QuantType quant_type() const { return quant_type_; } void set_sparse_model(bool sparse_model) { sparse_model_ = sparse_model; } bool sparse_model() const { return sparse_model_; } void set_sparse_threshold(float sparse_threshold) { sparse_threshold_ = sparse_threshold; } float sparse_threshold() const { return sparse_threshold_; } }; /// MobileConfig is the config for the light weight predictor, it will skip /// IR optimization or other unnecessary stages. class LITE_API MobileConfig : public ConfigBase { // whether to load data from memory. Model data will be loaded from memory // buffer if model_from_memory_ is true. bool model_from_memory_{false}; // model data readed from file or memory buffer in combined format. std::string lite_model_file_; // NOTE: This is a deprecated variable and will be removed in latter release. std::string model_buffer_; std::string param_buffer_; public: // set model data in combined format, `set_model_from_file` refers to loading // model from file, set_model_from_buffer refers to loading model from memory // buffer void set_model_from_file(const std::string& x); void set_model_from_buffer(const std::string& x); // return model data in lite_model_file_, which is in combined format. const std::string& lite_model_file() const { return lite_model_file_; } // return model_from_memory_, which indicates whether to load model from // memory buffer. bool is_model_from_memory() const { return model_from_memory_; } // note: `model_from_memory` has the same effect as `is_model_from_memory`, // but is_model_from_memory is recommended and `model_from_memory` will be // abandoned in v3.0. bool model_from_memory() const { return model_from_memory_; } // NOTE: This is a deprecated API and will be removed in latter release. void set_model_buffer(const char* model_buffer, size_t model_buffer_size, const char* param_buffer, size_t param_buffer_size); // NOTE: This is a deprecated API and will be removed in latter release. const std::string& model_buffer() const { return model_buffer_; } // NOTE: This is a deprecated API and will be removed in latter release. const std::string& param_buffer() const { return param_buffer_; } // This is the method for allocating workspace_size according to L3Cache size void SetArmL3CacheSize( L3CacheSetMethod method = L3CacheSetMethod::kDeviceL3Cache, int absolute_val = -1); }; template LITE_API std::shared_ptr CreatePaddlePredictor(const ConfigT&); } // namespace lite_api } // namespace paddle #endif // NOLINT