diff --git a/src/plugins/intel_cpu/thirdparty/onednn b/src/plugins/intel_cpu/thirdparty/onednn index f0f8defe2dff50..dc69ce59283efa 160000 --- a/src/plugins/intel_cpu/thirdparty/onednn +++ b/src/plugins/intel_cpu/thirdparty/onednn @@ -1 +1 @@ -Subproject commit f0f8defe2dff5058391f2a66e775e20b5de33b08 +Subproject commit dc69ce59283efad7961af9f9f3ceaf8f4c5b076a diff --git a/src/plugins/intel_cpu/thirdparty/shl b/src/plugins/intel_cpu/thirdparty/shl index 3a7d230ab1ab39..9fb8aac00ca9e3 160000 --- a/src/plugins/intel_cpu/thirdparty/shl +++ b/src/plugins/intel_cpu/thirdparty/shl @@ -1 +1 @@ -Subproject commit 3a7d230ab1ab39b29222ec78cbc3f4e4c3bf7a56 +Subproject commit 9fb8aac00ca9e3fc46887786a0b39d5e7d3cd669 diff --git a/src/plugins/intel_npu/tests/functional/CMakeLists.txt b/src/plugins/intel_npu/tests/functional/CMakeLists.txt index a2c47933c317a0..1d8bf947e63420 100644 --- a/src/plugins/intel_npu/tests/functional/CMakeLists.txt +++ b/src/plugins/intel_npu/tests/functional/CMakeLists.txt @@ -5,6 +5,10 @@ if(ENABLE_LTO) set(CMAKE_INTERPROCEDURAL_OPTIMIZATION_RELEASE ON) endif() +if(SUGGEST_OVERRIDE_SUPPORTED) + set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Wno-suggest-override") +endif() + set(TARGET_NAME ov_npu_func_tests) set(EXCLUDED_FUNC_TESTS_DIR "") set(OPTIONAL_FUNC_TESTS_INCLUDES "") @@ -56,6 +60,8 @@ if(MSVC) target_compile_options(${TARGET_NAME} PRIVATE /Zc:preprocessor) endif() +target_compile_definitions(${TARGET_NAME} PRIVATE CI_BUILD_NUMBER=\"mock_version\") + install( TARGETS ${TARGET_NAME} RUNTIME DESTINATION tests diff --git a/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/fake_plugins.cpp b/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/fake_plugins.cpp new file mode 100644 index 00000000000000..07cdc201aa5c54 --- /dev/null +++ b/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/fake_plugins.cpp @@ -0,0 +1,356 @@ +// Copyright (C) 2024 Intel Corporation +// SPDX-License-Identifier: Apache-2.0 +// + +#include "fake_plugins.hpp" + +#include +#include + +#include "openvino/core/any.hpp" +#include "openvino/core/except.hpp" +#include "openvino/opsets/opset11.hpp" +#include "openvino/pass/constant_folding.hpp" +#include "openvino/pass/manager.hpp" +#include "openvino/pass/serialize.hpp" +#include "openvino/runtime/exec_model_info.hpp" +#include "openvino/runtime/internal_properties.hpp" +#include "openvino/runtime/iplugin.hpp" +#include "openvino/runtime/make_tensor.hpp" +#include "openvino/runtime/properties.hpp" +#include "openvino/util/shared_object.hpp" +#include "transformations/init_node_info.hpp" +#include "transformations/rt_info/fused_names_attribute.hpp" + +using namespace ov::npuw::tests; +namespace { + +bool support_model(const std::shared_ptr& model, const ov::SupportedOpsMap& supported_ops) { + for (const auto& op : model->get_ops()) { + if (supported_ops.find(op->get_friendly_name()) == supported_ops.end()) + return false; + } + return true; +} + +ov::PropertyName RO_property(const std::string& propertyName) { + return ov::PropertyName(propertyName, ov::PropertyMutability::RO); +}; + +ov::PropertyName RW_property(const std::string& propertyName) { + return ov::PropertyName(propertyName, ov::PropertyMutability::RW); +}; + +} // namespace + +// Mock plugins + +FakeCompiledModel::FakeCompiledModel(const std::shared_ptr& model, + const std::shared_ptr& plugin, + const ov::AnyMap& config) + : ov::ICompiledModel(model, plugin), m_config(config), m_model(model) { +} + +// Methods from a base class ov::ICompiledModel +void FakeCompiledModel::export_model(std::ostream& model) const { + OPENVINO_NOT_IMPLEMENTED; +} + +std::shared_ptr FakeCompiledModel::get_runtime_model() const { + auto model = m_model->clone(); + // Add execution information into the model + size_t exec_order = 0; + for (const auto& op : model->get_ordered_ops()) { + auto& info = op->get_rt_info(); + info[ov::exec_model_info::EXECUTION_ORDER] = std::to_string(exec_order++); + info[ov::exec_model_info::IMPL_TYPE] = get_plugin()->get_device_name() + "_ " + op->get_type_info().name; + auto perf_count_enabled = get_property(ov::enable_profiling.name()).as(); + info[ov::exec_model_info::PERF_COUNTER] = perf_count_enabled ? "0" : "not_executed"; + std::string original_names = ov::getFusedNames(op); + if (original_names.empty()) { + original_names = op->get_friendly_name(); + } else if (original_names.find(op->get_friendly_name()) == std::string::npos) { + original_names = op->get_friendly_name() + "," + original_names; + } + info[ov::exec_model_info::ORIGINAL_NAMES] = original_names; + if (op->inputs().size() > 0) + info[ov::exec_model_info::RUNTIME_PRECISION] = op->get_input_element_type(0); + else + info[ov::exec_model_info::RUNTIME_PRECISION] = op->get_output_element_type(0); + + std::stringstream precisions_ss; + for (size_t i = 0; i < op->get_output_size(); i++) { + if (i > 0) + precisions_ss << ","; + precisions_ss << op->get_output_element_type(i); + } + info[ov::exec_model_info::OUTPUT_PRECISIONS] = precisions_ss.str(); + } + return model; +} + +void FakeCompiledModel::set_property(const ov::AnyMap& properties) { + OPENVINO_NOT_IMPLEMENTED; +} + +ov::Any FakeCompiledModel::get_property(const std::string& name) const { + if (name == ov::supported_properties) { + const std::vector supported_properties = {ov::num_streams.name(), + ov::enable_profiling.name()}; + return decltype(ov::supported_properties)::value_type(supported_properties); + } else if (name == ov::num_streams) { + if (m_config.count(ov::internal::exclusive_async_requests.name())) { + auto exclusive_async_requests = m_config.at(ov::internal::exclusive_async_requests.name()).as(); + if (exclusive_async_requests) + return ov::streams::Num(1); + } + return m_config.count(ov::num_streams.name()) ? m_config.at(ov::num_streams.name()) : ov::streams::Num(1); + } else if (name == ov::enable_profiling) { + return m_config.count(ov::enable_profiling.name()) ? m_config.at(ov::enable_profiling.name()) : false; + } else { + OPENVINO_THROW("get property: " + name); + } +} + +const std::shared_ptr& FakeCompiledModel::get_model() const { + return m_model; +} + +ov::SoPtr FakeCompiledModel::get_context() const { + OPENVINO_NOT_IMPLEMENTED; +} + +FakeInferRequest::FakeInferRequest(const std::shared_ptr& compiled_model) + : ov::ISyncInferRequest(compiled_model) { + OPENVINO_ASSERT(compiled_model); + m_model = compiled_model->get_model(); + // Allocate input/output tensors + for (const auto& input : get_inputs()) { + allocate_tensor(input, [this, input, compiled_model](ov::SoPtr& tensor) { + // Can add a check to avoid double work in case of shared tensors + allocate_tensor_impl(tensor, input.get_element_type(), + input.get_partial_shape().is_dynamic() ? ov::Shape{0} : input.get_shape()); + }); + } + for (const auto& output : get_outputs()) { + allocate_tensor(output, [this, output, compiled_model](ov::SoPtr& tensor) { + // Can add a check to avoid double work in case of shared tensors + allocate_tensor_impl(tensor, output.get_element_type(), + output.get_partial_shape().is_dynamic() ? ov::Shape{0} : output.get_shape()); + }); + } +} + +void FakeInferRequest::infer() { + ov::TensorVector input_tensors; + for (const auto& input : get_inputs()) { + input_tensors.emplace_back(ov::make_tensor(get_tensor(input))); + } + ov::TensorVector output_tensors; + for (const auto& output : get_outputs()) { + output_tensors.emplace_back(ov::make_tensor(get_tensor(output))); + } + m_model->evaluate(output_tensors, input_tensors); +} + +std::vector> FakeInferRequest::query_state() const { + OPENVINO_NOT_IMPLEMENTED; +} +std::vector FakeInferRequest::get_profiling_info() const { + OPENVINO_NOT_IMPLEMENTED; +} + +void FakeInferRequest::allocate_tensor_impl(ov::SoPtr& tensor, + const ov::element::Type& element_type, + const ov::Shape& shape) { + if (!tensor || tensor->get_element_type() != element_type) { + tensor = ov::SoPtr(ov::make_tensor(element_type, shape), nullptr); + } else { + tensor->set_shape(shape); + } +} + +std::shared_ptr FakeCompiledModel::create_sync_infer_request() const { + return std::make_shared(std::dynamic_pointer_cast(shared_from_this())); +} + +template +FakePluginBase::FakePluginBase() { + set_device_name(device_name); + m_plugin_name = std::string("openvino_") + std::string(device_name) + std::string("_plugin"); + static const ov::Version version = {CI_BUILD_NUMBER, m_plugin_name.c_str()}; + set_version(version); +} + +template +std::shared_ptr +FakePluginBase::compile_model(const std::shared_ptr& model, + const ov::AnyMap& properties) const { + OPENVINO_ASSERT(model); + if (!support_model(model, query_model(model, properties))) + OPENVINO_THROW("Unsupported model"); + return std::make_shared(model, shared_from_this(), properties); +} + +template +std::shared_ptr +FakePluginBase::compile_model(const std::string& model_path, + const ov::AnyMap& properties) const { + OPENVINO_NOT_IMPLEMENTED; +} + +template +std::shared_ptr FakePluginBase::compile_model( + const std::shared_ptr& model, + const ov::AnyMap& properties, + const ov::SoPtr& context) const { + OPENVINO_NOT_IMPLEMENTED; +} + +template +void FakePluginBase::set_property(const ov::AnyMap& properties) { + for (const auto& it : properties) { + if (it.first == ov::num_streams.name()) + num_streams = it.second.as(); + else if (it.first == ov::enable_profiling.name()) + m_profiling = it.second.as(); + else if (it.first == ov::internal::exclusive_async_requests.name()) + exclusive_async_requests = it.second.as(); + else if (it.first == ov::device::id.name()) + continue; + else + OPENVINO_THROW(get_device_name(), " set config: " + it.first); +}; +} + +template +ov::Any FakePluginBase::get_property(const std::string& name, + const ov::AnyMap& arguments) const { + const static std::vector device_ids = {"0", "1", "2"}; + const static std::vector roProperties{ + RO_property(ov::supported_properties.name()), + RO_property(ov::available_devices.name()), + RO_property(ov::loaded_from_cache.name()), + RO_property(ov::device::uuid.name()), + // RO_property(METRIC_KEY(IMPORT_EXPORT_SUPPORT)), + }; + // the whole config is RW before network is loaded. + const static std::vector rwProperties{ + RW_property(ov::num_streams.name()), + RW_property(ov::enable_profiling.name()), + }; + std::string device_id; + if (arguments.find(ov::device::id.name()) != arguments.end()) { + device_id = arguments.find(ov::device::id.name())->second.as(); + } + if (name == ov::supported_properties) { + std::vector supportedProperties; + supportedProperties.reserve(roProperties.size() + rwProperties.size()); + supportedProperties.insert(supportedProperties.end(), roProperties.begin(), roProperties.end()); + supportedProperties.insert(supportedProperties.end(), rwProperties.begin(), rwProperties.end()); + + return decltype(ov::supported_properties)::value_type(supportedProperties); + } else if (name == ov::internal::supported_properties) { + return decltype(ov::internal::supported_properties)::value_type( + {ov::PropertyName{ov::internal::caching_properties.name(), ov::PropertyMutability::RO}, + ov::PropertyName{ov::internal::exclusive_async_requests.name(), ov::PropertyMutability::RW}}); + } else if (name == ov::internal::exclusive_async_requests) { + return decltype(ov::internal::exclusive_async_requests)::value_type{exclusive_async_requests}; + } else if (name == ov::device::uuid) { + ov::device::UUID uuid; + for (size_t i = 0; i < uuid.MAX_UUID_SIZE; i++) { + if (device_id == device_ids[0]) + uuid.uuid[i] = static_cast(i); + else if (device_id == device_ids[1]) + uuid.uuid[i] = static_cast(i * 2); + else if (device_id == device_ids[2]) + uuid.uuid[i] = static_cast(i * 3); + } + return decltype(ov::device::uuid)::value_type{uuid}; + } else if (name == ov::available_devices) { + return decltype(ov::available_devices)::value_type(device_ids); + } else if (name == ov::device::capabilities) { + std::vector capabilities; + capabilities.push_back(ov::device::capability::EXPORT_IMPORT); + return decltype(ov::device::capabilities)::value_type(capabilities); + } else if (name == "SUPPORTED_CONFIG_KEYS") { // TODO: Remove this key + std::vector configs; + for (const auto& property : rwProperties) { + configs.emplace_back(property); + } + return configs; +// } else if (METRIC_KEY(IMPORT_EXPORT_SUPPORT) == name) { +// return true; + } else if (ov::internal::caching_properties == name) { + std::vector caching_properties = {ov::device::uuid}; + return decltype(ov::internal::caching_properties)::value_type(caching_properties); + } else if (name == "SUPPORTED_METRICS") { // TODO: Remove this key + std::vector configs; + for (const auto& property : roProperties) { + configs.emplace_back(property); + } + return configs; + } else if (name == ov::loaded_from_cache.name()) { + return m_loaded_from_cache; + } else if (name == ov::enable_profiling.name()) { + return decltype(ov::enable_profiling)::value_type{m_profiling}; + } else if (name == ov::streams::num.name()) { + return decltype(ov::streams::num)::value_type{num_streams}; + } + OPENVINO_THROW("Unsupported property: ", name); +} + +template +ov::SoPtr +FakePluginBase::create_context(const ov::AnyMap& remote_properties) const { + OPENVINO_NOT_IMPLEMENTED; +} + +template +ov::SoPtr +FakePluginBase::get_default_context(const ov::AnyMap& remote_properties) const { + OPENVINO_NOT_IMPLEMENTED; +} + +template +std::shared_ptr +FakePluginBase::import_model(std::istream& model, + const ov::AnyMap& properties) const { + OPENVINO_NOT_IMPLEMENTED; +} + +template +std::shared_ptr +FakePluginBase::import_model(std::istream& model, + const ov::SoPtr& context, + const ov::AnyMap& properties) const { + OPENVINO_NOT_IMPLEMENTED; +} + +template +ov::SupportedOpsMap +FakePluginBase::query_model(const std::shared_ptr& model, + const ov::AnyMap& properties) const { + OPENVINO_NOT_IMPLEMENTED; +} + +template<> +FakePluginBase::FakePluginBase() { + m_supported_ops = {"Parameter", "Result", "Add", "Constant", "Reshape"}; + m_dynamism_supported = true; +} + +template<> +FakePluginBase::FakePluginBase() { + m_supported_ops = {"Parameter", "Result", "Add", "Constant", "Subtract", "Reshape"}; + m_dynamism_supported = false; +} + +namespace ov { +namespace npuw { +namespace tests { +template class FakePluginBase; +template class FakePluginBase; +} // namespace tests +} // namespace npuw +} // namespace ov diff --git a/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/fake_plugins.hpp b/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/fake_plugins.hpp new file mode 100644 index 00000000000000..1d118c87b44720 --- /dev/null +++ b/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/fake_plugins.hpp @@ -0,0 +1,106 @@ +// Copyright (C) 2024 Intel Corporation +// SPDX-License-Identifier: Apache-2.0 +// +#pragma once + +#include +#include + +#include + +#include "openvino/runtime/core.hpp" +#include "openvino/runtime/iplugin.hpp" +#include "openvino/util/shared_object.hpp" + +// Code is taken and adopted from: openvino/src/plugins/hetero/tests/functional/hetero_tests.cpp + +namespace ov { +namespace npuw { +namespace tests { +class FakeCompiledModel : public ov::ICompiledModel { +public: + FakeCompiledModel(const std::shared_ptr& model, const std::shared_ptr& plugin, + const ov::AnyMap& config); + + // Methods from a base class ov::ICompiledModel + void export_model(std::ostream& model) const override; + std::shared_ptr get_runtime_model() const override; + void set_property(const ov::AnyMap& properties) override; + ov::Any get_property(const std::string& name) const override; + std::shared_ptr create_sync_infer_request() const override; + const std::shared_ptr& get_model() const; + ov::SoPtr get_context() const; + +private: + ov::AnyMap m_config; + std::shared_ptr m_model; +}; + +class FakeInferRequest : public ov::ISyncInferRequest { +public: + FakeInferRequest(const std::shared_ptr& compiled_model); + ~FakeInferRequest() = default; + + void infer() override; + std::vector> query_state() const override; + std::vector get_profiling_info() const override; + +private: + void allocate_tensor_impl(ov::SoPtr& tensor, const ov::element::Type& element_type, + const ov::Shape& shape); + std::shared_ptr m_model; +}; + +template +class FakePluginBase : public ov::IPlugin { +private: + static constexpr const char* device_name = DeviceType::name; +public: + FakePluginBase(); + + std::shared_ptr compile_model(const std::shared_ptr& model, + const ov::AnyMap& properties) const override; + std::shared_ptr compile_model(const std::string& model_path, + const ov::AnyMap& properties) const override; + std::shared_ptr compile_model(const std::shared_ptr& model, + const ov::AnyMap& properties, + const ov::SoPtr& context) const override; + void set_property(const ov::AnyMap& properties) override; + ov::Any get_property(const std::string& name, const ov::AnyMap& arguments) const override; + ov::SoPtr create_context(const ov::AnyMap& remote_properties) const override; + ov::SoPtr get_default_context(const ov::AnyMap& remote_properties) const override; + std::shared_ptr import_model(std::istream& model, const ov::AnyMap& properties) const override; + std::shared_ptr import_model(std::istream& model, + const ov::SoPtr& context, + const ov::AnyMap& properties) const override; + ov::SupportedOpsMap query_model(const std::shared_ptr& model, + const ov::AnyMap& properties) const override; + +protected: + std::string m_plugin_name; + std::string m_default_device_id = "0"; + std::unordered_set m_supported_ops; + bool m_dynamism_supported = false; + bool m_profiling = false; + bool m_loaded_from_cache{false}; + int32_t num_streams{0}; + bool exclusive_async_requests = false; +}; + +namespace fakes { +struct Npu { + static constexpr const char* name = "FakeNPU"; + static constexpr int num_device_ids = 3; +}; +struct Cpu { + static constexpr const char* name = "FakeCPU"; + static constexpr int num_device_ids = 2; +}; +} // namespace fakes; + +using FakeNpuPlugin = FakePluginBase; +using FakeCpuPlugin = FakePluginBase; + +} // namespace tests +} // namespace npuw +} // namespace ov diff --git a/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/mock_plugins.cpp b/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/mock_plugins.cpp new file mode 100644 index 00000000000000..666776a22d6800 --- /dev/null +++ b/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/mock_plugins.cpp @@ -0,0 +1,299 @@ +// Copyright (C) 2024 Intel Corporation +// SPDX-License-Identifier: Apache-2.0 +// + +#include +#include + +#include +#include + +#include "mock_plugins.hpp" +#include "openvino/core/version.hpp" +#include "openvino/pass/serialize.hpp" +#include "openvino/runtime/make_tensor.hpp" + +#include "npu_private_properties.hpp" +#include "npuw_private_properties.hpp" + +namespace ov { +namespace npuw { +namespace tests { +// Need to also mock async infer request or use infer() call to indicate that start_async() was called. +MockInferRequestBase::MockInferRequestBase(const std::shared_ptr& compiled_model) + : ov::ISyncInferRequest(compiled_model) { + OPENVINO_ASSERT(compiled_model); +} + +void MockInferRequestBase::create_implementation() { + auto inputs = get_compiled_model()->inputs(); + auto outputs = get_compiled_model()->outputs(); + auto in_tensors = create_tensors(inputs); + auto out_tensors = create_tensors(outputs); + + for (std::size_t i = 0; i < inputs.size(); ++i) { + set_tensor(inputs[i], ov::get_tensor_impl(in_tensors[i])); + } + + for (std::size_t i = 0; i < outputs.size(); ++i) { + set_tensor(outputs[i], ov::get_tensor_impl(out_tensors[i])); + } + + + ON_CALL(*this, query_state()).WillByDefault([]() -> std::vector> { + OPENVINO_NOT_IMPLEMENTED; + }); + ON_CALL(*this, get_profiling_info()).WillByDefault([]() -> std::vector { + OPENVINO_NOT_IMPLEMENTED; + }); +} + + +MockCompiledModelBase::MockCompiledModelBase(const std::shared_ptr& model, + std::shared_ptr plugin, + const ov::AnyMap& config, + std::shared_ptr, bool>> infer_req_expectations_ptr) + : ov::ICompiledModel(model, plugin), + m_infer_req_expectations_ptr(infer_req_expectations_ptr), + m_model(model), + m_config(config) { + } + + +void MockCompiledModelBase::create_implementation() { + ON_CALL(*this, inputs()).WillByDefault(testing::ReturnRefOfCopy(m_model->inputs())); + ON_CALL(*this, outputs()).WillByDefault(testing::ReturnRefOfCopy(m_model->outputs())); + ON_CALL(*this, create_infer_request).WillByDefault([this]() { + auto syncRequestImpl = create_sync_infer_request(); + return std::make_shared(syncRequestImpl, get_task_executor(), get_callback_executor()); + }); + ON_CALL(*this, export_model(testing::_)).WillByDefault([](std::ostream& s) { + OPENVINO_NOT_IMPLEMENTED; + }); + ON_CALL(*this, get_runtime_model).WillByDefault(testing::Return(m_model)); + ON_CALL(*this, set_property).WillByDefault([](const ov::AnyMap& properties) { + OPENVINO_NOT_IMPLEMENTED; + }); + ON_CALL(*this, get_property).WillByDefault([this](const std::string& name) -> ov::Any { + if (name == ov::supported_properties) { + const std::vector supported_properties = {ov::num_streams.name(), + ov::enable_profiling.name()}; + return decltype(ov::supported_properties)::value_type(supported_properties); + } else if (name == ov::num_streams) { + if (this->m_config.count(ov::internal::exclusive_async_requests.name())) { + auto exclusive_async_requests = + m_config.at(ov::internal::exclusive_async_requests.name()).as(); + if (exclusive_async_requests) + return ov::streams::Num(1); + } + return this->m_config.count(ov::num_streams.name()) ? m_config.at(ov::num_streams.name()) + : ov::streams::Num(1); + } else if (name == ov::enable_profiling) { + return this->m_config.count(ov::enable_profiling.name()) ? m_config.at(ov::enable_profiling.name()) : false; + } else { + OPENVINO_THROW("get property: " + name); + } + }); + ON_CALL(*this, create_sync_infer_request).WillByDefault([this]() { + std::lock_guard lock(m_mock_creation_mutex); + auto mock_sync_infer_request = std::make_shared( + std::dynamic_pointer_cast(shared_from_this())); + mock_sync_infer_request->create_implementation(); + if (m_infer_req_expectations_ptr) { + auto& expectation = m_infer_req_expectations_ptr->first; + auto& status = m_infer_req_expectations_ptr->second; + expectation(*mock_sync_infer_request); + status = true; // Expectation will be checked + } + return mock_sync_infer_request; + }); +} + + +template +MockPluginBase::MockPluginBase() { + m_plugin_name = std::string("openvino_") + std::string(device_name) + std::string("_plugin"); + const ov::Version version = {CI_BUILD_NUMBER, m_plugin_name.c_str()}; + set_device_name(device_name); + set_version(version); +} + +namespace { + ov::PropertyName RO_property(const std::string& propertyName) { + return ov::PropertyName(propertyName, ov::PropertyMutability::RO); + }; + + ov::PropertyName RW_property(const std::string& propertyName) { + return ov::PropertyName(propertyName, ov::PropertyMutability::RW); + }; +} // anonymous namespace + +template +void MockPluginBase::create_implementation() { + ON_CALL(*this, compile_model(testing::A&>(), testing::_)) + .WillByDefault([this](const std::shared_ptr& model, const ov::AnyMap& properties) { + std::lock_guard lock(m_mock_creation_mutex); + + std::shared_ptr, bool>> infer_req_expectations; + if (m_infer_reqs_to_expectations.count(m_num_compiled_models)) { + infer_req_expectations = m_infer_reqs_to_expectations[m_num_compiled_models]; + } + + auto mock_compiled_model = std::make_shared( + model, shared_from_this(), properties, + infer_req_expectations); + + mock_compiled_model->create_implementation(); + + if (m_models_to_expectations.count(m_num_compiled_models)) { + auto& expectation_and_status = m_models_to_expectations[m_num_compiled_models]; + auto& expectation = expectation_and_status.first; + auto& status = expectation_and_status.second; + expectation(*mock_compiled_model); + status = true; // Expectation will be checked + } + + ++m_num_compiled_models; + + return mock_compiled_model; + }); + ON_CALL(*this, compile_model(testing::A(), testing::_)) + .WillByDefault([](const std::string& model_path, + const ov::AnyMap& properties) -> std::shared_ptr { + OPENVINO_NOT_IMPLEMENTED; + }); + ON_CALL(*this, compile_model(testing::A&>(), testing::_, testing::_)) + .WillByDefault([](const std::shared_ptr& model, const ov::AnyMap& properties, + const ov::SoPtr& context) -> std::shared_ptr { + OPENVINO_NOT_IMPLEMENTED; + }); + ON_CALL(*this, set_property).WillByDefault([this](const ov::AnyMap& properties) { + for (const auto& it : properties) { + if (it.first == ov::num_streams.name()) + num_streams = it.second.as(); + else if (it.first == ov::internal::exclusive_async_requests.name()) + exclusive_async_requests = it.second.as(); + else if (it.first == ov::device::id.name()) + continue; + else + OPENVINO_THROW(get_device_name(), " set config: " + it.first); + } + }); + ON_CALL(*this, get_property).WillByDefault([this](const std::string& name, const ov::AnyMap& arguments) -> ov::Any { + std::vector device_ids; + for (int i = 0; i < DeviceType::num_device_ids; ++i) { + device_ids.push_back(std::to_string(i)); + }; + const static std::vector roProperties{ + RO_property(ov::supported_properties.name()), RO_property(ov::available_devices.name()), + RO_property(ov::device::uuid.name()), RO_property(ov::device::capabilities.name()) + }; + // the whole config is RW before network is loaded. + const static std::vector rwProperties{ + RW_property(ov::num_streams.name()) + }; + + std::string device_id; + if (arguments.find(ov::device::id.name()) != arguments.end()) { + device_id = arguments.find(ov::device::id.name())->second.as(); + } + if (name == ov::supported_properties) { + std::vector supportedProperties; + supportedProperties.reserve(roProperties.size() + rwProperties.size()); + supportedProperties.insert(supportedProperties.end(), roProperties.begin(), roProperties.end()); + supportedProperties.insert(supportedProperties.end(), rwProperties.begin(), rwProperties.end()); + + return decltype(ov::supported_properties)::value_type(supportedProperties); + } else if (name == ov::internal::supported_properties) { + return decltype(ov::internal::supported_properties)::value_type( + {ov::PropertyName{ov::internal::exclusive_async_requests.name(), ov::PropertyMutability::RW}}); + } else if (name == ov::internal::exclusive_async_requests) { + return decltype(ov::internal::exclusive_async_requests)::value_type{exclusive_async_requests}; + } else if (name == ov::device::uuid) { + ov::device::UUID uuid; + for (size_t i = 0; i < uuid.MAX_UUID_SIZE; i++) { + for (int j = 0; j < DeviceType::num_device_ids; ++j) + if (device_id == device_ids[j]) + uuid.uuid[i] = static_cast(i * (j + 1)); + } + return decltype(ov::device::uuid)::value_type{uuid}; + } else if (name == ov::available_devices) { + return decltype(ov::available_devices)::value_type(device_ids); + } else if (name == ov::device::capabilities) { + std::vector capabilities; + capabilities.push_back(ov::device::capability::EXPORT_IMPORT); + return decltype(ov::device::capabilities)::value_type(capabilities); + } else if (name == "SUPPORTED_CONFIG_KEYS") { // TODO: Remove this key + std::vector configs; + for (const auto& property : rwProperties) { + configs.emplace_back(property); + } + return configs; + } else if (name == ov::streams::num.name()) { + return decltype(ov::streams::num)::value_type{num_streams}; + } + OPENVINO_THROW("Unsupported property: ", name); + }); + ON_CALL(*this, create_context) + .WillByDefault([](const ov::AnyMap& remote_properties) -> ov::SoPtr { + OPENVINO_NOT_IMPLEMENTED; + }); + ON_CALL(*this, get_default_context) + .WillByDefault([](const ov::AnyMap& remote_properties) -> ov::SoPtr { + OPENVINO_NOT_IMPLEMENTED; + }); + ON_CALL(*this, import_model(testing::_, testing::_)) + .WillByDefault([](std::istream& model, const ov::AnyMap& properties) + -> std::shared_ptr { + OPENVINO_NOT_IMPLEMENTED; + }); + ON_CALL(*this, import_model(testing::_, testing::_, testing::_)) + .WillByDefault([](std::istream& model, const ov::SoPtr& context, + const ov::AnyMap& properties) -> std::shared_ptr { + OPENVINO_NOT_IMPLEMENTED; + }); + ON_CALL(*this, query_model) + .WillByDefault([](const std::shared_ptr& model, const ov::AnyMap& properties) + -> ov::SupportedOpsMap{ + OPENVINO_NOT_IMPLEMENTED; + }); +} + +template +void MockPluginBase::set_expectations_to_comp_models(int model_idx, std::function expectations) { + m_models_to_expectations[model_idx] = std::pair, bool>(expectations, false); +} + +template +void MockPluginBase::set_expectations_to_infer_reqs(int req_idx, + std::function expectations) { + m_infer_reqs_to_expectations[req_idx] = + std::make_shared, bool>>(expectations, false); +} + +template +MockPluginBase::~MockPluginBase() { + for (auto const& idx_to_expectation_and_status : m_models_to_expectations) { + auto idx = idx_to_expectation_and_status.first; + auto expectation_and_status = idx_to_expectation_and_status.second; + if (!expectation_and_status.second) { + ADD_FAILURE() << DeviceType::name << ": Expectation for model with index " << idx + << " was set, but model with that idx was not compiled"; + } + } + for (auto const& idx_to_expectation_and_status : m_infer_reqs_to_expectations) { + auto idx = idx_to_expectation_and_status.first; + auto expectation_and_status = *idx_to_expectation_and_status.second; + if (!expectation_and_status.second) { + ADD_FAILURE() << DeviceType::name << ": Expectation for request with index " << idx + << " was set, but request with that idx was not created"; + } + } +} + +template class MockPluginBase; +template class MockPluginBase; + +} // namespace tests +} // namespace npuw +} // namespace ov diff --git a/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/mock_plugins.hpp b/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/mock_plugins.hpp new file mode 100644 index 00000000000000..5c02f96ebd581e --- /dev/null +++ b/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/mock_plugins.hpp @@ -0,0 +1,155 @@ +// Copyright (C) 2024 Intel Corporation +// SPDX-License-Identifier: Apache-2.0 +// +#pragma once + +#include +#include + +#include +#include + +#include "openvino/runtime/core.hpp" +#include "openvino/runtime/iplugin.hpp" +#include "openvino/runtime/internal_properties.hpp" +#include "openvino/util/shared_object.hpp" +#include "openvino/runtime/iasync_infer_request.hpp" +#include "openvino/runtime/so_ptr.hpp" +#include "openvino/core/any.hpp" + +// Based on: openvino/src/inference/tests/functional/caching_test.cpp + +namespace ov { +namespace npuw { +namespace tests { + +class MockCompiledModelBase; +using MockCompiledModel = testing::NiceMock; + +// Need to also mock async infer request or use infer() call to indicate that start_async() was called. +class MockInferRequestBase : public ov::ISyncInferRequest { +public: + MockInferRequestBase(const std::shared_ptr& compiled_model); + ~MockInferRequestBase() = default; + + MOCK_METHOD(void, infer, (), (override)); + MOCK_METHOD(std::vector>, query_state, (), (const, override)); + MOCK_METHOD(std::vector, get_profiling_info, (), (const, override)); + + // This must be called *before* the custom ON_CALL() statements. + void create_implementation(); + +private: + ov::TensorVector create_tensors(std::vector> data) { + ov::TensorVector tensors; + for (const auto& el : data) { + const ov::Shape& shape = el.get_shape(); + const ov::element::Type& precision = el.get_element_type(); + tensors.emplace_back(precision, shape); + } + return tensors; + } +}; +using MockInferRequest = testing::NiceMock; + +class MockCompiledModelBase : public ov::ICompiledModel { +public: + MockCompiledModelBase( + const std::shared_ptr& model, std::shared_ptr plugin, + const ov::AnyMap& config, + std::shared_ptr, bool>> infer_req_expectations); + + // Methods from a base class ov::ICompiledModel + MOCK_METHOD(const std::vector>&, outputs, (), (const, override)); + MOCK_METHOD(const std::vector>&, inputs, (), (const, override)); + MOCK_METHOD(std::shared_ptr, create_infer_request, (), (const, override)); + MOCK_METHOD(void, export_model, (std::ostream& model), (const, override)); + MOCK_METHOD(std::shared_ptr, get_runtime_model, (), (const, override)); + MOCK_METHOD(void, set_property, (const ov::AnyMap& properties), (override)); + MOCK_METHOD(ov::Any, get_property, (const std::string& name), (const, override)); + MOCK_METHOD(std::shared_ptr, create_sync_infer_request, (), (const, override)); + + // This must be called *before* the custom ON_CALL() statements. + void create_implementation(); + +private: + std::mutex m_mock_creation_mutex; + std::shared_ptr, bool>> m_infer_req_expectations_ptr; + + std::shared_ptr m_model; + ov::AnyMap m_config; +}; + +template +class MockPluginBase : public ov::IPlugin { +private: + static constexpr const char* device_name = DeviceType::name; +public: + MockPluginBase(); + + // Normal mock method definitions using gMock. + // FIXME: There is an issue with mocking of multiple-arguments functions below with generic MOCK_METHOD macro, + // on Windows. So, old-style MOCK_METHODn macros are used instead. + MOCK_CONST_METHOD2_T(compile_model, + std::shared_ptr(const std::shared_ptr& model, + const ov::AnyMap& properties)); + MOCK_CONST_METHOD2_T(compile_model, + std::shared_ptr(const std::string& model_path, + const ov::AnyMap& properties)); + MOCK_CONST_METHOD3_T(compile_model, + std::shared_ptr(const std::shared_ptr& model, + const ov::AnyMap& properties, + const ov::SoPtr& context)); + MOCK_CONST_METHOD2_T(get_property, ov::Any(const std::string& name, const ov::AnyMap& arguments)); + MOCK_CONST_METHOD2_T(import_model, + std::shared_ptr(std::istream& model, + const ov::AnyMap& properties)); + MOCK_CONST_METHOD3_T(import_model, + std::shared_ptr(std::istream& model, + const ov::SoPtr& context, + const ov::AnyMap& properties)); + MOCK_CONST_METHOD2_T(query_model, + ov::SupportedOpsMap(const std::shared_ptr& model, + const ov::AnyMap& properties)); + + MOCK_METHOD(void, set_property, (const ov::AnyMap& properties), (override)); + MOCK_METHOD(ov::SoPtr, create_context, (const ov::AnyMap& remote_properties), (const, override)); + MOCK_METHOD(ov::SoPtr, get_default_context, (const ov::AnyMap& remote_properties), (const, override)); + + // This must be called *before* the custom ON_CALL() statements. + void create_implementation(); + void set_expectations_to_comp_models(int model_idx, std::function expectations); + void set_expectations_to_infer_reqs(int req_idx, std::function expectations); + + ~MockPluginBase() override; + +private: + std::string m_plugin_name; + std::mutex m_mock_creation_mutex; // Internal gmock object registration is not thread-safe + int m_num_compiled_models{}; + // TODO: Make thread-safe and simplify. + std::map, bool>> m_models_to_expectations; + std::map, bool>>> m_infer_reqs_to_expectations; + + // Properties + int32_t num_streams{0}; + bool exclusive_async_requests = false; +}; + +namespace mocks { +struct Npu { + static constexpr const char* name = "MockNPU"; + static constexpr int num_device_ids = 3; +}; +struct Cpu { + static constexpr const char* name = "MockCPU"; + static constexpr int num_device_ids = 2; +}; +} //namespace mocks + +using MockNpuPlugin = testing::NiceMock>; +using MockCpuPlugin = testing::NiceMock>; + +} // namespace tests +} // namespace npuw +} // namespace ov diff --git a/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/register_in_ov.cpp b/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/register_in_ov.cpp new file mode 100644 index 00000000000000..bc916e652fdce7 --- /dev/null +++ b/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/register_in_ov.cpp @@ -0,0 +1,35 @@ +// Copyright (C) 2024 Intel Corporation +// SPDX-License-Identifier: Apache-2.0 +// + +#include "register_in_ov.hpp" + +#include + +namespace ov { +namespace npuw { +namespace tests { + +std::shared_ptr reg_plugin(ov::Core& core, std::shared_ptr& plugin) { + std::string mock_engine_path = ov::util::make_plugin_library_name(ov::test::utils::getExecutableDirectory(), + std::string("mock_engine") + OV_BUILD_POSTFIX); + std::string plugin_path = ov::util::make_plugin_library_name( + ov::test::utils::getExecutableDirectory(), + std::string("mock_engine_") + plugin->get_device_name() + OV_BUILD_POSTFIX); + if (!std::filesystem::is_regular_file(plugin_path)) { + std::filesystem::copy(mock_engine_path, plugin_path); + } + + auto so = ov::util::load_shared_object(plugin_path.c_str()); + + std::function injectProxyEngine = + make_std_function(so, "InjectPlugin"); + + injectProxyEngine(plugin.get()); + core.register_plugin(plugin_path, plugin->get_device_name()); + + return so; +} +} +} // namespace npuw +} // namespace tests diff --git a/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/register_in_ov.hpp b/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/register_in_ov.hpp new file mode 100644 index 00000000000000..e29443f387db57 --- /dev/null +++ b/src/plugins/intel_npu/tests/functional/behavior/npuw/mocks/register_in_ov.hpp @@ -0,0 +1,30 @@ +// Copyright (C) 2024 Intel Corporation +// SPDX-License-Identifier: Apache-2.0 +// + +#include "common_test_utils/file_utils.hpp" + +#include "openvino/runtime/core.hpp" +#include "openvino/runtime/iplugin.hpp" +#include "openvino/util/shared_object.hpp" + +namespace ov { +namespace npuw { +namespace tests { + +template +std::function make_std_function(const std::shared_ptr so, const std::string& functionName) { + std::function ptr(reinterpret_cast(ov::util::get_symbol(so, functionName.c_str()))); + return ptr; +} + +std::shared_ptr reg_plugin(ov::Core& core, std::shared_ptr& plugin); + +template +std::shared_ptr reg_plugin(ov::Core& core, const std::shared_ptr& plugin) { + auto base_plugin_ptr = std::dynamic_pointer_cast(plugin); + return reg_plugin(core, base_plugin_ptr); +} +} +} // namespace npuw +} // namespace tests diff --git a/src/plugins/intel_npu/tests/functional/behavior/npuw/use_case_tests.cpp b/src/plugins/intel_npu/tests/functional/behavior/npuw/use_case_tests.cpp new file mode 100644 index 00000000000000..d288e490d24e6a --- /dev/null +++ b/src/plugins/intel_npu/tests/functional/behavior/npuw/use_case_tests.cpp @@ -0,0 +1,261 @@ +// Copyright (C) 2024 Intel Corporation +// SPDX-License-Identifier: Apache-2.0 +// + +#include "use_case_tests.hpp" +#include "openvino/opsets/opset11.hpp" +#include "openvino/runtime/exec_model_info.hpp" +#include "openvino/runtime/internal_properties.hpp" +#include "openvino/runtime/make_tensor.hpp" +#include "openvino/runtime/properties.hpp" +#include "openvino/util/file_util.hpp" +#include "openvino/util/shared_object.hpp" +#include "npuw_private_properties.hpp" + +using namespace testing; +using namespace ov::npuw::tests; + +std::shared_ptr NPUWUseCaseTests::create_example_model() { + auto param = std::make_shared(ov::element::i64, ov::PartialShape{1, 3, 2, 2}); + param->set_friendly_name("input"); + auto const_value = ov::opset11::Constant::create(ov::element::i64, ov::Shape{1, 1, 1, 1}, {1}); + const_value->set_friendly_name("const_val"); + auto add = std::make_shared(param, const_value); + add->set_friendly_name("add"); + auto result = std::make_shared(add); + result->set_friendly_name("res"); + return std::make_shared(ov::ResultVector{result}, ov::ParameterVector{param}); +} + +TEST_F(NPUWUseCaseTests, TestInfrastructureIsCorrect) { + // Set expectations first: + EXPECT_CALL(*npu_plugin, get_property).Times(AnyNumber()); + EXPECT_CALL(*npu_plugin, get_property(std::string("AVAILABLE_DEVICES"), _)).Times(1); + EXPECT_CALL(*cpu_plugin, get_property).Times(AnyNumber()); + EXPECT_CALL(*cpu_plugin, get_property(std::string("AVAILABLE_DEVICES"), _)).Times(1); + + // Register mock objects as plugins in OpenVINO: + register_mock_plugins_in_ov(); + + // Do the actual test: + std::vector mock_reference_dev = {{"MockNPU.0"}, {"MockNPU.1"}, {"MockNPU.2"}, + {"MockCPU.0"}, {"MockCPU.1"}}; + auto available_devices = core.get_available_devices(); + for (auto device : available_devices) { + auto it = std::find(mock_reference_dev.begin(), mock_reference_dev.end(), device); + if (it != mock_reference_dev.end()) { + mock_reference_dev.erase(it); + } + } + + // All devices should be found + EXPECT_TRUE(mock_reference_dev.empty()); +} + +TEST_F(NPUWUseCaseTests, CompilationIsSuccessful) { + // Set expectations first: + EXPECT_CALL(*npu_plugin, + compile_model(A&>(), _)).Times(1); + EXPECT_CALL(*cpu_plugin, + compile_model(A&>(), _)).Times(0); + + // Register mock objects as plugins in OpenVINO: + register_mock_plugins_in_ov(); + + // Do the actual test: + auto npuw_properties = + ov::AnyMap{ov::intel_npu::use_npuw(true), + ov::intel_npu::npuw::devices("MockNPU")}; + EXPECT_NO_THROW(core.compile_model(model, "NPU", npuw_properties)); +} + +TEST_F(NPUWUseCaseTests, CompilationIsFailSafe) { + // Set expectations first: + { + InSequence s; + + EXPECT_CALL(*npu_plugin, compile_model(A&>(), _)) + .Times(1) + .WillOnce(Throw(std::runtime_error("Compilation on MockNPU is failed"))); + EXPECT_CALL(*cpu_plugin, compile_model(A&>(), _)).Times(1); + } + + // Register mock objects as plugins in OpenVINO: + register_mock_plugins_in_ov(); + + // Do the actual test: + auto npuw_properties = + ov::AnyMap{ov::intel_npu::use_npuw(true), + ov::intel_npu::npuw::devices("MockNPU,MockCPU")}; + EXPECT_NO_THROW(core.compile_model(model, "NPU", npuw_properties)); +} + +TEST_F(NPUWUseCaseTests, CompilationIsFailed) { + // Set expectations first: + EXPECT_CALL(*npu_plugin, compile_model(A&>(), _)) + .Times(1) + .WillOnce(Throw(std::runtime_error("Compilation on MockNPU is failed"))); + EXPECT_CALL(*cpu_plugin, compile_model(A&>(), _)).Times(0); + + // Register mock objects as plugins in OpenVINO: + register_mock_plugins_in_ov(); + + // Do the actual test: + auto npuw_properties = + ov::AnyMap{ov::intel_npu::use_npuw(true), + ov::intel_npu::npuw::devices("MockNPU")}; + EXPECT_ANY_THROW(core.compile_model(model, "NPU", npuw_properties)); +} + +TEST_F(NPUWUseCaseTests, InferRequestCreationIsSuccessful) { + // Set expectations first: + EXPECT_CALL(*npu_plugin, compile_model(A&>(), _)).Times(1); + EXPECT_CALL(*cpu_plugin, compile_model(A&>(), _)).Times(0); + npu_plugin->set_expectations_to_comp_models(0, [](MockCompiledModel& model) { + EXPECT_CALL(model, create_sync_infer_request()).Times(1); + }); + + // Register mock objects as plugins in OpenVINO: + register_mock_plugins_in_ov(); + + // Do the actual test: + auto npuw_properties = + ov::AnyMap{ov::intel_npu::use_npuw(true), + ov::intel_npu::npuw::devices("MockNPU")}; + auto compiled_model = core.compile_model(model, "NPU", npuw_properties); + EXPECT_NO_THROW(compiled_model.create_infer_request()); +} + +TEST_F(NPUWUseCaseTests, InferRequestCreationIsFailSafe) { + // Set expectations first: + { + InSequence s; + EXPECT_CALL(*npu_plugin, compile_model(A&>(), _)).Times(1); + EXPECT_CALL(*cpu_plugin, compile_model(A&>(), _)).Times(1); + } + npu_plugin->set_expectations_to_comp_models(0, [](MockCompiledModel& model) { + EXPECT_CALL(model, create_sync_infer_request()) + .Times(1) + .WillOnce(Throw(std::runtime_error("Infer request creation on MockNPU is failed"))); + }); + cpu_plugin->set_expectations_to_comp_models(0, [](MockCompiledModel& model) { + EXPECT_CALL(model, create_sync_infer_request()).Times(1); + }); + + // Register mock objects as plugins in OpenVINO: + register_mock_plugins_in_ov(); + + // Do the actual test: + auto npuw_properties = + ov::AnyMap{ov::intel_npu::use_npuw(true), + ov::intel_npu::npuw::devices("MockNPU,MockCPU")}; + auto compiled_model = core.compile_model(model, "NPU", npuw_properties); + EXPECT_NO_THROW(compiled_model.create_infer_request()); +} + +TEST_F(NPUWUseCaseTests, InferRequestCreationIsFailed) { + // Set expectations first: + EXPECT_CALL(*npu_plugin, compile_model(A&>(), _)).Times(1); + EXPECT_CALL(*cpu_plugin, compile_model(A&>(), _)).Times(0); + + npu_plugin->set_expectations_to_comp_models(0, [](MockCompiledModel& model) { + EXPECT_CALL(model, create_sync_infer_request()) + .Times(1) + .WillOnce(Throw(std::runtime_error("Infer request creation on MockNPU is failed"))); + }); + + // Register mock objects as plugins in OpenVINO: + register_mock_plugins_in_ov(); + + // Do the actual test: + auto npuw_properties = + ov::AnyMap{ov::intel_npu::use_npuw(true), + ov::intel_npu::npuw::devices("MockNPU")}; + auto compiled_model = core.compile_model(model, "NPU", npuw_properties); + EXPECT_ANY_THROW(compiled_model.create_infer_request()); +} + +TEST_F(NPUWUseCaseTests, InferIsSuccessful) { + // Set expectations first: + EXPECT_CALL(*npu_plugin, compile_model(A&>(), _)).Times(1); + EXPECT_CALL(*cpu_plugin, compile_model(A&>(), _)).Times(0); + npu_plugin->set_expectations_to_comp_models(0, [](MockCompiledModel& model) { + EXPECT_CALL(model, create_sync_infer_request()).Times(1); + }); + npu_plugin->set_expectations_to_infer_reqs(0, [](MockInferRequest& request) { + EXPECT_CALL(request, infer()).Times(1); + }); + + // Register mock objects as plugins in OpenVINO: + register_mock_plugins_in_ov(); + + // Do the actual test: + auto npuw_properties = + ov::AnyMap{ov::intel_npu::use_npuw(true), + ov::intel_npu::npuw::devices("MockNPU")}; + auto compiled_model = core.compile_model(model, "NPU", npuw_properties); + auto infer_request = compiled_model.create_infer_request(); + EXPECT_NO_THROW(infer_request.infer()); +} + +TEST_F(NPUWUseCaseTests, InferIsFailSafe) { + // Set expectations first: + { + InSequence seq; + EXPECT_CALL(*npu_plugin, compile_model(A&>(), _)).Times(1); + EXPECT_CALL(*cpu_plugin, compile_model(A&>(), _)).Times(1); + } + npu_plugin->set_expectations_to_comp_models(0, [](MockCompiledModel& model) { + EXPECT_CALL(model, create_sync_infer_request()).Times(1); + }); + npu_plugin->set_expectations_to_infer_reqs(0, [](MockInferRequest& request) { + EXPECT_CALL(request, infer()) + .Times(1) + .WillOnce(Throw(std::runtime_error("Infer on MockNPU is failed"))); + }); + cpu_plugin->set_expectations_to_comp_models(0, [](MockCompiledModel& model) { + EXPECT_CALL(model, create_sync_infer_request()).Times(1); + }); + cpu_plugin->set_expectations_to_infer_reqs(0, [](MockInferRequest& request) { + EXPECT_CALL(request, infer()).Times(1); + }); + + // Register mock objects as plugins in OpenVINO: + register_mock_plugins_in_ov(); + + // Do the actual test: + auto npuw_properties = + ov::AnyMap{ov::intel_npu::use_npuw(true), + ov::intel_npu::npuw::devices("MockNPU,MockCPU")}; + auto compiled_model = core.compile_model(model, "NPU", npuw_properties); + auto infer_request = compiled_model.create_infer_request(); + EXPECT_NO_THROW(infer_request.infer()); +} + +TEST_F(NPUWUseCaseTests, InferIsFailed) { + // Set expectations first: + { + InSequence seq; + EXPECT_CALL(*npu_plugin, compile_model(A&>(), _)).Times(1); + EXPECT_CALL(*cpu_plugin, compile_model(A&>(), _)).Times(0); + } + npu_plugin->set_expectations_to_comp_models(0, [](MockCompiledModel& model) { + EXPECT_CALL(model, create_sync_infer_request()).Times(1); + }); + npu_plugin->set_expectations_to_infer_reqs(0, [](MockInferRequest& request) { + EXPECT_CALL(request, infer()) + .Times(1) + .WillOnce(Throw(std::runtime_error("Infer on MockNPU is failed"))); + }); + + // Register mock objects as plugins in OpenVINO: + register_mock_plugins_in_ov(); + + // Do the actual test: + auto npuw_properties = + ov::AnyMap{ov::intel_npu::use_npuw(true), + ov::intel_npu::npuw::devices("MockNPU")}; + auto compiled_model = core.compile_model(model, "NPU", npuw_properties); + auto infer_request = compiled_model.create_infer_request(); + EXPECT_ANY_THROW(infer_request.infer()); +} diff --git a/src/plugins/intel_npu/tests/functional/behavior/npuw/use_case_tests.hpp b/src/plugins/intel_npu/tests/functional/behavior/npuw/use_case_tests.hpp new file mode 100644 index 00000000000000..4243a2a25e1c55 --- /dev/null +++ b/src/plugins/intel_npu/tests/functional/behavior/npuw/use_case_tests.hpp @@ -0,0 +1,51 @@ +// Copyright (C) 2024 Intel Corporation +// SPDX-License-Identifier: Apache-2.0 +// +#pragma once + +#include + +#include + +#include "openvino/runtime/core.hpp" +#include "openvino/runtime/iplugin.hpp" +#include "openvino/runtime/make_tensor.hpp" + +#include "mocks/mock_plugins.hpp" +#include "mocks/register_in_ov.hpp" + +namespace ov { +namespace npuw { +namespace tests { + +class NPUWUseCaseTests : public ::testing::Test { +public: + ov::Core core; + std::shared_ptr npu_plugin; + std::shared_ptr cpu_plugin; + std::shared_ptr model; + + + void SetUp() override { + model = create_example_model(); + npu_plugin = std::make_shared(); + npu_plugin->create_implementation(); + cpu_plugin = std::make_shared(); + cpu_plugin->create_implementation(); + } + + // Make sure it is called after expectations are set! + void register_mock_plugins_in_ov() { + m_shared_objects.push_back(reg_plugin(core, npu_plugin)); + m_shared_objects.push_back(reg_plugin(core, cpu_plugin)); + } + + std::shared_ptr create_example_model(); + +private: + std::vector> m_shared_objects; +}; + +} // namespace tests +} // namespace npuw +} // namespace ov