Skip to content

Commit

Permalink
Add support for shape and data_shape parameters (openvinotoolkit#27314)
Browse files Browse the repository at this point in the history
### Details:
 - Move helper function for reshaping to `npu_tools_utils`
 - Introduce  `shape` and `data_shape` params

### Tickets:
 - E144161

---------

Signed-off-by: Skrebkov, Artemy <[email protected]>
  • Loading branch information
ArtemySkrebkov authored Oct 31, 2024
1 parent f2640a2 commit 272843d
Show file tree
Hide file tree
Showing 5 changed files with 236 additions and 270 deletions.
181 changes: 181 additions & 0 deletions src/plugins/intel_npu/tools/common/include/tools_helpers.hpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,181 @@
// Copyright (C) 2024 Intel Corporation
// SPDX-License-Identifier: Apache-2.0
//

#include <algorithm>
#include <cstdlib>
#include <map>
#include <string>
#include <vector>

#include "openvino/openvino.hpp"

struct InputInfo {
ov::element::Type type;
ov::PartialShape partialShape;
ov::Shape dataShape;
ov::Layout layout;
};
using InputsInfo = std::map<std::string, InputInfo>;

std::string parameterNameToTensorName(std::string& name, std::vector<ov::Output<ov::Node>>& inputs_info) {
auto count_name = std::any_of(inputs_info.begin(), inputs_info.end(), [name](ov::Output<ov::Node>& port) {
return port.get_names().count(name) > 0;
});
if (count_name) {
return name;
} else {
auto inputInfo = std::find_if(inputs_info.begin(), inputs_info.end(), [name](ov::Output<ov::Node>& port) {
return name == port.get_node()->get_friendly_name();
});
if (inputInfo == inputs_info.end()) {
throw std::runtime_error("Provided I/O name \"" + name +
"\" is not found neither in tensor names nor in nodes names.");
}
return inputInfo->get_any_name();
}
}

std::map<std::string, std::vector<std::string>> parseInputParameters(std::string& parameter_string,
std::vector<ov::Output<ov::Node>>& input_info) {
// Parse parameter string like "input0[value0],input1[value1]" or "[value]" (applied to all
// inputs)
std::map<std::string, std::vector<std::string>> return_value;
std::string search_string = parameter_string;
auto start_pos = search_string.find_first_of('[');
auto input_name = search_string.substr(0, start_pos);
while (start_pos != std::string::npos) {
auto end_pos = search_string.find_first_of(']');
if (end_pos == std::string::npos)
break;
input_name = search_string.substr(0, start_pos);
auto input_value = search_string.substr(start_pos + 1, end_pos - start_pos - 1);
if (!input_name.empty()) {
return_value[parameterNameToTensorName(input_name, input_info)].push_back(input_value);
} else {
for (auto& item : input_info) {
return_value[item.get_any_name()].push_back(input_value);
}
}
search_string = search_string.substr(end_pos + 1);
if (search_string.empty() || (search_string.front() != ',' && search_string.front() != '['))
break;
if (search_string.front() == ',') {
if (search_string.length() > 1)
search_string = search_string.substr(1);
else
throw std::logic_error("Can't parse input parameter string, there is nothing after the comma " +
parameter_string);
}
start_pos = search_string.find_first_of('[');
}
if (!search_string.empty())
throw std::logic_error("Can't parse input parameter string: " + parameter_string);
return return_value;
}

void boundDynamicShape(std::shared_ptr<ov::Model>& model) {
for (auto&& item : model->get_parameters()) {
auto shape = item->get_partial_shape();
if (shape.is_static()) {
continue;
}
auto rank = shape.rank();
if (rank.is_dynamic()) {
throw std::logic_error("Rank \"" + rank.to_string() + "\" of the shape \"" + shape.to_string() +
"\" is dynamic which is not supported by NPU");
}
auto layout = item->get_layout();
if (!ov::layout::has_batch(layout)) {
item->set_layout(ov::Layout(layout.to_string().insert(1, "N,")));
layout = item->get_layout();
}
if (shape[ov::layout::batch_idx(layout)].is_dynamic()) {
std::cout << "WARNING: Shape \"" + shape.to_string() + "\"" +
" has dynamic batch size which is not supported by NPU\n"
" Setting batch to 1 forcibly"
<< std::endl;
ov::set_batch(model, 1);
}
shape = item->get_partial_shape();
if (shape.is_dynamic()) {
throw std::logic_error("Model's input shape \"" + shape.to_string() + "\"" +
" is dynamic which is not supported by NPU");
}
}
}

void setModelBatch(std::shared_ptr<ov::Model>& model, uint32_t batch = 1) {
if (batch == 1) {
return;
}
for (auto&& item : model->get_parameters()) {
auto shape = item->get_partial_shape();
auto rank = shape.rank();
if (rank.is_dynamic()) {
throw std::logic_error("Rank \"" + rank.to_string() + "\" of the shape \"" + shape.to_string() +
"\" is dynamic which is not supported by NPU");
}
auto layout = item->get_layout();
if (!ov::layout::has_batch(layout)) {
item->set_layout(ov::Layout(layout.to_string().insert(1, "N,")));
layout = item->get_layout();
}
if (shape[ov::layout::batch_idx(layout)].is_dynamic()) {
throw std::logic_error("ERROR: Shape \"" + shape.to_string() + "\"" +
" has dynamic batch size which is not supported by NPU\n"
"Cannot apply fixed batch: " +
std::to_string(batch) +
". Please remove the parameter from config: \"override_model_batch_size\"");
}
ov::set_batch(model, batch);
}
}

void reshape(ov::OutputVector inputsInfo, InputsInfo& infoMap, std::shared_ptr<ov::Model>& model,
std::string& shapeString, int overrideModelBatchSize, std::string_view device) {
std::vector<InputsInfo> infoMaps;
if (!shapeString.empty()) {
std::map<std::string, std::vector<std::string>> shapesMap = parseInputParameters(shapeString, inputsInfo);

if (overrideModelBatchSize != 1) {
throw std::logic_error(R"(Incompatible params: "shape" and "override_model_batch_size")");
}
for (auto& item : inputsInfo) {
InputInfo info;
auto name = item.get_any_name();

if (!shapesMap.empty()) {
if (shapesMap.count(name)) {
if (shapesMap.at(name).size() > 1) {
// Example: -shape input1[..][..]
throw std::logic_error("shape command line parameter doesn't support multiple "
"shapes for one input.");
}
info.partialShape = shapesMap.at(name)[0];
} else {
info.partialShape = item.get_partial_shape();
}
}
infoMap[name] = std::move(info);
infoMaps.push_back(infoMap);
}
std::map<std::string, ov::PartialShape> newShapes;
for (auto& item : infoMaps) {
for (auto& map : item) {
if (!newShapes.count(map.first)) {
newShapes[map.first] = map.second.partialShape;
}
}
}
model->reshape(newShapes);
} else {
if (device.find("NPU") != std::string::npos ||
// FIXME: SIT on CPU also requires to bound dynamic shapes
device.find("CPU") != std::string::npos || device.find("TEMPLATE") != std::string::npos) {
boundDynamicShape(model);
}

setModelBatch(model, overrideModelBatchSize);
}
}
3 changes: 2 additions & 1 deletion src/plugins/intel_npu/tools/compile_tool/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,8 @@ ov_add_target(ADD_CPPLINT
PRIVATE
openvino::runtime
gflags
Threads::Threads)
Threads::Threads
npu_tools_utils)

set_target_properties(${TARGET_NAME} PROPERTIES
FOLDER ${CMAKE_CURRENT_SOURCE_DIR}
Expand Down
109 changes: 4 additions & 105 deletions src/plugins/intel_npu/tools/compile_tool/main.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -14,11 +14,12 @@

#include <gflags/gflags.h>

#include "openvino/core/partial_shape.hpp"
#include "openvino/openvino.hpp"
#include <openvino/core/partial_shape.hpp>
#include <openvino/openvino.hpp>

#include "tools_helpers.hpp"


static constexpr char help_message[] = "Optional. Print the usage message.";

static constexpr char model_message[] = "Required. Path to the XML model.";
Expand Down Expand Up @@ -168,64 +169,6 @@ bool isFP32(const ov::element::Type& type) {
return type == ov::element::f32;
}

void boundDynamicShape(std::shared_ptr<ov::Model>& model) {
for (auto&& item : model->get_parameters()) {
auto shape = item->get_partial_shape();
if (shape.is_static()) {
continue;
}
auto rank = shape.rank();
if (rank.is_dynamic()) {
throw std::logic_error("Rank \"" + rank.to_string() + "\" of the shape \"" + shape.to_string() +
"\" is dynamic which is not supported by NPU");
}
auto layout = item->get_layout();
if (!ov::layout::has_batch(layout)) {
item->set_layout(ov::Layout(layout.to_string().insert(1, "N,")));
layout = item->get_layout();
}
if (shape[ov::layout::batch_idx(layout)].is_dynamic()) {
std::cout << "WARNING: Shape \"" + shape.to_string() + "\"" +
" has dynamic batch size which is not supported by NPU\n"
" Setting batch to 1 forcibly"
<< std::endl;
ov::set_batch(model, 1);
}
shape = item->get_partial_shape();
if (shape.is_dynamic()) {
throw std::logic_error("Model's input shape \"" + shape.to_string() + "\"" +
" is dynamic which is not supported by NPU");
}
}
}

void setModelBatch(std::shared_ptr<ov::Model>& model, uint32_t batch = 1) {
if (batch == 1) {
return;
}
for (auto&& item : model->get_parameters()) {
auto shape = item->get_partial_shape();
auto rank = shape.rank();
if (rank.is_dynamic()) {
throw std::logic_error("Rank \"" + rank.to_string() + "\" of the shape \"" + shape.to_string() +
"\" is dynamic which is not supported by NPU");
}
auto layout = item->get_layout();
if (!ov::layout::has_batch(layout)) {
item->set_layout(ov::Layout(layout.to_string().insert(1, "N,")));
layout = item->get_layout();
}
if (shape[ov::layout::batch_idx(layout)].is_dynamic()) {
throw std::logic_error("ERROR: Shape \"" + shape.to_string() + "\"" +
" has dynamic batch size which is not supported by NPU\n"
"Cannot apply fixed batch: " +
std::to_string(batch) +
". Please remove the parameter from config: \"override_model_batch_size\"");
}
ov::set_batch(model, batch);
}
}

void configurePrePostProcessing(std::shared_ptr<ov::Model>& model, const std::string& ip, const std::string& op,
const std::string& iop, const std::string& il, const std::string& ol,
const std::string& iol, const std::string& iml, const std::string& oml,
Expand Down Expand Up @@ -475,50 +418,6 @@ std::string getFileNameFromPath(const std::string& path,

using TimeDiff = std::chrono::milliseconds;

void reshape(ov::OutputVector inputs_info, InputsInfo& info_map, std::shared_ptr<ov::Model>& model) {
std::vector<InputsInfo> info_maps;
if (!FLAGS_shape.empty()) {
std::map<std::string, std::vector<std::string>> shapes_map = parseInputParameters(FLAGS_shape, inputs_info);

if (FLAGS_override_model_batch_size != 1) {
throw std::logic_error("Incompatible params: \"shape\" and \"override_model_batch_size\"");
}
for (auto& item : inputs_info) {
InputInfo info;
auto name = item.get_any_name();

if (!shapes_map.empty()) {
if (shapes_map.count(name)) {
if (shapes_map.at(name).size() > 1) {
// Example: -shape input1[..][..]
throw std::logic_error("shape command line parameter doesn't support multiple "
"shapes for one input.");
}
info.partialShape = shapes_map.at(name)[0];
} else {
info.partialShape = item.get_partial_shape();
}
}
info_map[name] = std::move(info);
info_maps.push_back(info_map);
}
std::map<std::string, ov::PartialShape> newShapes;
for (auto& item : info_maps) {
for (auto& map : item) {
if (!newShapes.count(map.first)) {
newShapes[map.first] = map.second.partialShape;
}
}
}
model->reshape(newShapes);
} else {
if (FLAGS_d.find("NPU") != std::string::npos) {
boundDynamicShape(model);
}

setModelBatch(model, FLAGS_override_model_batch_size);
}
}

int main(int argc, char* argv[]) {
try {
Expand Down Expand Up @@ -552,7 +451,7 @@ int main(int argc, char* argv[]) {
InputsInfo info_map;

std::cout << "Performing reshape" << std::endl;
reshape(std::move(inputs_info), info_map, model);
reshape(std::move(inputs_info), info_map, model, FLAGS_shape, FLAGS_override_model_batch_size, FLAGS_d);

std::cout << "Configuring model pre & post processing" << std::endl;
configurePrePostProcessing(model, FLAGS_ip, FLAGS_op, FLAGS_iop, FLAGS_il, FLAGS_ol, FLAGS_iol, FLAGS_iml,
Expand Down
Loading

0 comments on commit 272843d

Please sign in to comment.