add dynamic shape of BoundingBoxEncode and BoundingBoxDecode.
This commit is contained in:
parent
9b4db1b646
commit
bb4dfa4b69
|
@ -20,20 +20,25 @@
|
||||||
|
|
||||||
namespace mindspore {
|
namespace mindspore {
|
||||||
namespace kernel {
|
namespace kernel {
|
||||||
void BoundingBoxDecodeCpuKernelMod::InitKernel(const CNodePtr &kernel_node) {
|
namespace {
|
||||||
MS_EXCEPTION_IF_NULL(kernel_node);
|
const size_t kInputRank = 2;
|
||||||
kernel_name_ = common::AnfAlgo::GetCNodeName(kernel_node);
|
const size_t kLastDim = 4;
|
||||||
size_t input_num = common::AnfAlgo::GetInputTensorNum(kernel_node);
|
} // namespace
|
||||||
if (input_num != INPUT_NUMS) {
|
|
||||||
MS_LOG(ERROR) << "For '" << kernel_name_ << "', the number of inputs must be 2, but got " << input_num;
|
bool BoundingBoxDecodeCpuKernelMod::Init(const BaseOperatorPtr &base_operator,
|
||||||
}
|
const std::vector<KernelTensorPtr> &inputs,
|
||||||
|
const std::vector<KernelTensorPtr> &outputs) {
|
||||||
|
MS_EXCEPTION_IF_NULL(base_operator);
|
||||||
|
kernel_name_ = base_operator->name();
|
||||||
|
constexpr size_t input_num = 2;
|
||||||
|
CHECK_KERNEL_INPUTS_NUM(inputs.size(), input_num, kernel_name_);
|
||||||
|
|
||||||
const size_t coordinate_size = 4;
|
const size_t coordinate_size = 4;
|
||||||
if (common::AnfAlgo::GetCNodePrimitive(kernel_node)->GetAttr("means")->isa<ValueTuple>() ||
|
auto means = base_operator->GetAttr("means");
|
||||||
common::AnfAlgo::GetCNodePrimitive(kernel_node)->GetAttr("means")->isa<ValueList>()) {
|
if (means->isa<api::ValueSequence>()) {
|
||||||
means_ = common::AnfAlgo::GetNodeAttr<std::vector<float>>(kernel_node, "means");
|
means_ = api::GetValue<std::vector<float>>(means);
|
||||||
} else if (common::AnfAlgo::GetCNodePrimitive(kernel_node)->GetAttr("means")->isa<FloatImm>()) {
|
} else if (means->isa<api::FloatImm>()) {
|
||||||
float mean = common::AnfAlgo::GetNodeAttr<float>(kernel_node, "means");
|
float mean = api::GetValue<float>(means);
|
||||||
for (size_t i = 0; i < coordinate_size; i++) {
|
for (size_t i = 0; i < coordinate_size; i++) {
|
||||||
(void)means_.emplace_back(mean);
|
(void)means_.emplace_back(mean);
|
||||||
}
|
}
|
||||||
|
@ -42,11 +47,11 @@ void BoundingBoxDecodeCpuKernelMod::InitKernel(const CNodePtr &kernel_node) {
|
||||||
<< "', the input 'means' must be a tuple or a list, and dtype must be float, but got is not.";
|
<< "', the input 'means' must be a tuple or a list, and dtype must be float, but got is not.";
|
||||||
}
|
}
|
||||||
|
|
||||||
if (common::AnfAlgo::GetCNodePrimitive(kernel_node)->GetAttr("stds")->isa<ValueTuple>() ||
|
auto stds = base_operator->GetAttr("stds");
|
||||||
common::AnfAlgo::GetCNodePrimitive(kernel_node)->GetAttr("stds")->isa<ValueList>()) {
|
if (stds->isa<api::ValueSequence>()) {
|
||||||
stds_ = common::AnfAlgo::GetNodeAttr<std::vector<float>>(kernel_node, "stds");
|
stds_ = api::GetValue<std::vector<float>>(stds);
|
||||||
} else if (common::AnfAlgo::GetCNodePrimitive(kernel_node)->GetAttr("stds")->isa<FloatImm>()) {
|
} else if (stds->isa<api::FloatImm>()) {
|
||||||
float std = common::AnfAlgo::GetNodeAttr<float>(kernel_node, "stds");
|
float std = api::GetValue<float>(stds);
|
||||||
for (size_t i = 0; i < coordinate_size; i++) {
|
for (size_t i = 0; i < coordinate_size; i++) {
|
||||||
(void)stds_.emplace_back(std);
|
(void)stds_.emplace_back(std);
|
||||||
}
|
}
|
||||||
|
@ -62,17 +67,68 @@ void BoundingBoxDecodeCpuKernelMod::InitKernel(const CNodePtr &kernel_node) {
|
||||||
<< means_.size() << ", and the length of 'stds': " << stds_.size();
|
<< means_.size() << ", and the length of 'stds': " << stds_.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
std::vector<int64_t> max_shape_me = common::AnfAlgo::GetNodeAttr<std::vector<int64_t>>(kernel_node, "max_shape");
|
auto max_shape = base_operator->GetAttr("max_shape");
|
||||||
|
std::vector<int64_t> max_shape_me = api::GetValue<std::vector<int64_t>>(max_shape);
|
||||||
(void)std::transform(max_shape_me.begin(), max_shape_me.end(), std::back_inserter(max_shape_),
|
(void)std::transform(max_shape_me.begin(), max_shape_me.end(), std::back_inserter(max_shape_),
|
||||||
[](const int64_t &value) { return LongToInt(value); });
|
[](const int64_t &value) { return LongToInt(value); });
|
||||||
wh_ratio_clip_ = common::AnfAlgo::GetNodeAttr<float>(kernel_node, "wh_ratio_clip");
|
auto wh_ratio_clip = base_operator->GetAttr("wh_ratio_clip");
|
||||||
|
wh_ratio_clip_ = api::GetValue<float>(wh_ratio_clip);
|
||||||
|
|
||||||
if (max_shape_.size() < MIN_MAX_SHAPE_SIZE) {
|
if (max_shape_.size() < MIN_MAX_SHAPE_SIZE) {
|
||||||
MS_LOG(EXCEPTION) << "For '" << kernel_name_
|
MS_LOG(EXCEPTION) << "For '" << kernel_name_
|
||||||
<< "', the length of 'max_shape' must be at least 2, but got: " << max_shape_.size();
|
<< "', the length of 'max_shape' must be at least 2, but got: " << max_shape_.size();
|
||||||
}
|
}
|
||||||
|
|
||||||
InitTaskFunc(kernel_node);
|
auto kernel_attr = GetKernelAttrFromTensors(inputs, outputs);
|
||||||
|
auto [is_match, index] = MatchKernelAttr(kernel_attr, GetOpSupport());
|
||||||
|
if (!is_match) {
|
||||||
|
MS_LOG(ERROR) << "For '" << kernel_name_ << "' does not support this kernel type: " << kernel_attr;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
kernel_func_ = func_list_[index].second;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
int BoundingBoxDecodeCpuKernelMod::Resize(const BaseOperatorPtr &base_operator,
|
||||||
|
const std::vector<KernelTensorPtr> &inputs,
|
||||||
|
const std::vector<KernelTensorPtr> &outputs,
|
||||||
|
const std::map<uint32_t, tensor::TensorPtr> &inputsOnHost) {
|
||||||
|
if (auto ret = KernelMod::Resize(base_operator, inputs, outputs, inputsOnHost); ret != KRET_OK) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto anchor_box_shape = LongVecToSizeVec(inputs[kIndex0]->GetShapeVector());
|
||||||
|
auto deltas_shape = LongVecToSizeVec(inputs[kIndex1]->GetShapeVector());
|
||||||
|
|
||||||
|
auto it_x = std::find_if(anchor_box_shape.begin(), anchor_box_shape.end(), [](int64_t sh) { return sh <= 0; });
|
||||||
|
if (it_x != anchor_box_shape.end()) {
|
||||||
|
return KRET_UNKNOWN_SHAPE;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t anchor_box_rank = anchor_box_shape.size();
|
||||||
|
size_t deltas_rank = deltas_shape.size();
|
||||||
|
|
||||||
|
if (anchor_box_rank != kInputRank) {
|
||||||
|
MS_LOG(ERROR) << "The rank of anchor box must be 2, but got " << anchor_box_rank;
|
||||||
|
return KRET_RESIZE_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (deltas_rank != kInputRank) {
|
||||||
|
MS_LOG(ERROR) << "The rank of deltas must be 2, but got " << deltas_rank;
|
||||||
|
return KRET_RESIZE_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (anchor_box_shape[1] != kLastDim) {
|
||||||
|
MS_LOG(ERROR) << "The shape of anchor box must be (n, 4), but got the second dimension of " << anchor_box_shape[1];
|
||||||
|
return KRET_RESIZE_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (deltas_shape[1] != kLastDim) {
|
||||||
|
MS_LOG(ERROR) << "The shape of deltas must be (n, 4), but got the second dimension of " << deltas_shape[1];
|
||||||
|
return KRET_RESIZE_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
return KRET_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
|
@ -165,15 +221,6 @@ std::vector<std::pair<KernelAttr, BoundingBoxDecodeCpuKernelMod::BoundingBoxDeco
|
||||||
{KernelAttr().AddInputAttr(kNumberTypeFloat16).AddInputAttr(kNumberTypeFloat16).AddOutputAttr(kNumberTypeFloat16),
|
{KernelAttr().AddInputAttr(kNumberTypeFloat16).AddInputAttr(kNumberTypeFloat16).AddOutputAttr(kNumberTypeFloat16),
|
||||||
&BoundingBoxDecodeCpuKernelMod::LaunchKernel<float16>}};
|
&BoundingBoxDecodeCpuKernelMod::LaunchKernel<float16>}};
|
||||||
|
|
||||||
void BoundingBoxDecodeCpuKernelMod::InitTaskFunc(const CNodePtr &kernel_node) {
|
|
||||||
auto kernel_attr = GetKernelAttrFromNode(kernel_node);
|
|
||||||
auto [is_match, index] = MatchKernelAttr(kernel_attr, GetOpSupport());
|
|
||||||
if (!is_match) {
|
|
||||||
MS_LOG(EXCEPTION) << "BoundingBoxDecode does not support this kernel data type: " << kernel_attr;
|
|
||||||
}
|
|
||||||
kernel_func_ = func_list_[index].second;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<KernelAttr> BoundingBoxDecodeCpuKernelMod::GetOpSupport() {
|
std::vector<KernelAttr> BoundingBoxDecodeCpuKernelMod::GetOpSupport() {
|
||||||
std::vector<KernelAttr> support_list;
|
std::vector<KernelAttr> support_list;
|
||||||
(void)std::transform(func_list_.begin(), func_list_.end(), std::back_inserter(support_list),
|
(void)std::transform(func_list_.begin(), func_list_.end(), std::back_inserter(support_list),
|
||||||
|
|
|
@ -14,12 +14,13 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef MINDSPORE_CCSRC_BACKEND_KERNEL_COMPILER_CPU_BOUNDINGBOX_DECODE_CPU_KERNEL_H_
|
#ifndef MINDSPORE_CCSRC_PLUGIN_DEVICE_CPU_KERNEL_BOUNDINGBOX_DECODE_CPU_KERNEL_H_
|
||||||
#define MINDSPORE_CCSRC_BACKEND_KERNEL_COMPILER_CPU_BOUNDINGBOX_DECODE_CPU_KERNEL_H_
|
#define MINDSPORE_CCSRC_PLUGIN_DEVICE_CPU_KERNEL_BOUNDINGBOX_DECODE_CPU_KERNEL_H_
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
#include <map>
|
||||||
|
|
||||||
#include "plugin/device/cpu/kernel/cpu_kernel.h"
|
#include "plugin/device/cpu/kernel/cpu_kernel.h"
|
||||||
#include "plugin/factory/ms_factory.h"
|
#include "plugin/factory/ms_factory.h"
|
||||||
|
@ -27,23 +28,25 @@
|
||||||
namespace mindspore {
|
namespace mindspore {
|
||||||
namespace kernel {
|
namespace kernel {
|
||||||
constexpr size_t MIN_MAX_SHAPE_SIZE = 2;
|
constexpr size_t MIN_MAX_SHAPE_SIZE = 2;
|
||||||
constexpr size_t INPUT_NUMS = 2;
|
class BoundingBoxDecodeCpuKernelMod : public NativeCpuKernelMod {
|
||||||
class BoundingBoxDecodeCpuKernelMod : public DeprecatedNativeCpuKernelMod {
|
|
||||||
public:
|
public:
|
||||||
BoundingBoxDecodeCpuKernelMod() = default;
|
BoundingBoxDecodeCpuKernelMod() = default;
|
||||||
~BoundingBoxDecodeCpuKernelMod() override = default;
|
~BoundingBoxDecodeCpuKernelMod() override = default;
|
||||||
|
|
||||||
void InitKernel(const CNodePtr &kernel_node) override;
|
bool Init(const BaseOperatorPtr &base_operator, const std::vector<KernelTensorPtr> &inputs,
|
||||||
|
const std::vector<KernelTensorPtr> &outputs) override;
|
||||||
|
|
||||||
bool Launch(const std::vector<AddressPtr> &inputs, const std::vector<AddressPtr> &workspace,
|
bool Launch(const std::vector<AddressPtr> &inputs, const std::vector<AddressPtr> &workspace,
|
||||||
const std::vector<AddressPtr> &outputs) override {
|
const std::vector<AddressPtr> &outputs) override {
|
||||||
return kernel_func_(this, inputs, workspace, outputs);
|
return kernel_func_(this, inputs, workspace, outputs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int Resize(const BaseOperatorPtr &base_operator, const std::vector<KernelTensorPtr> &inputs,
|
||||||
|
const std::vector<KernelTensorPtr> &outputs, const std::map<uint32_t, tensor::TensorPtr> &) override;
|
||||||
|
|
||||||
std::vector<KernelAttr> GetOpSupport() override;
|
std::vector<KernelAttr> GetOpSupport() override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void InitTaskFunc(const CNodePtr &kernel_node);
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
bool LaunchKernel(const std::vector<kernel::AddressPtr> &inputs, const std::vector<kernel::AddressPtr> &workspace,
|
bool LaunchKernel(const std::vector<kernel::AddressPtr> &inputs, const std::vector<kernel::AddressPtr> &workspace,
|
||||||
const std::vector<kernel::AddressPtr> &outputs);
|
const std::vector<kernel::AddressPtr> &outputs);
|
||||||
|
@ -55,10 +58,10 @@ class BoundingBoxDecodeCpuKernelMod : public DeprecatedNativeCpuKernelMod {
|
||||||
|
|
||||||
std::vector<float> means_;
|
std::vector<float> means_;
|
||||||
std::vector<float> stds_;
|
std::vector<float> stds_;
|
||||||
std::vector<int> max_shape_;
|
std::vector<size_t> max_shape_;
|
||||||
float wh_ratio_clip_{0.016};
|
float wh_ratio_clip_{0.016};
|
||||||
};
|
};
|
||||||
} // namespace kernel
|
} // namespace kernel
|
||||||
} // namespace mindspore
|
} // namespace mindspore
|
||||||
|
|
||||||
#endif // MINDSPORE_CCSRC_BACKEND_KERNEL_COMPILER_CPU_BOUNDINGBOX_DECODE_CPU_KERNEL_H_
|
#endif // MINDSPORE_CCSRC_PLUGIN_DEVICE_CPU_KERNEL_BOUNDINGBOX_DECODE_CPU_KERNEL_H_
|
||||||
|
|
|
@ -20,43 +20,25 @@
|
||||||
|
|
||||||
namespace mindspore {
|
namespace mindspore {
|
||||||
namespace kernel {
|
namespace kernel {
|
||||||
std::vector<std::pair<KernelAttr, BoundingBoxEncodeCpuKernelMod::BoundingBoxEncodeFunc>>
|
namespace {
|
||||||
BoundingBoxEncodeCpuKernelMod::func_list_ = {
|
const size_t kInputRank = 2;
|
||||||
{KernelAttr().AddInputAttr(kNumberTypeFloat32).AddInputAttr(kNumberTypeFloat32).AddOutputAttr(kNumberTypeFloat32),
|
const size_t kLastDim = 4;
|
||||||
&BoundingBoxEncodeCpuKernelMod::LaunchKernel<float>},
|
} // namespace
|
||||||
{KernelAttr().AddInputAttr(kNumberTypeFloat16).AddInputAttr(kNumberTypeFloat16).AddOutputAttr(kNumberTypeFloat16),
|
|
||||||
&BoundingBoxEncodeCpuKernelMod::LaunchKernel<float16>}};
|
|
||||||
|
|
||||||
void BoundingBoxEncodeCpuKernelMod::InitTaskFunc(const CNodePtr &kernel_node) {
|
bool BoundingBoxEncodeCpuKernelMod::Init(const BaseOperatorPtr &base_operator,
|
||||||
auto kernel_attr = GetKernelAttrFromNode(kernel_node);
|
const std::vector<KernelTensorPtr> &inputs,
|
||||||
auto [is_match, index] = MatchKernelAttr(kernel_attr, GetOpSupport());
|
const std::vector<KernelTensorPtr> &outputs) {
|
||||||
if (!is_match) {
|
MS_EXCEPTION_IF_NULL(base_operator);
|
||||||
MS_LOG(EXCEPTION) << "BoundingBoxEncode does not support this kernel data type: " << kernel_attr;
|
kernel_name_ = base_operator->name();
|
||||||
}
|
constexpr size_t input_num = 2;
|
||||||
kernel_func_ = func_list_[index].second;
|
CHECK_KERNEL_INPUTS_NUM(inputs.size(), input_num, kernel_name_);
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<KernelAttr> BoundingBoxEncodeCpuKernelMod::GetOpSupport() {
|
|
||||||
std::vector<KernelAttr> support_list;
|
|
||||||
(void)std::transform(func_list_.begin(), func_list_.end(), std::back_inserter(support_list),
|
|
||||||
[](const std::pair<KernelAttr, BoundingBoxEncodeFunc> &pair) { return pair.first; });
|
|
||||||
return support_list;
|
|
||||||
}
|
|
||||||
|
|
||||||
void BoundingBoxEncodeCpuKernelMod::InitKernel(const CNodePtr &kernel_node) {
|
|
||||||
MS_EXCEPTION_IF_NULL(kernel_node);
|
|
||||||
kernel_name_ = common::AnfAlgo::GetCNodeName(kernel_node);
|
|
||||||
size_t input_num = common::AnfAlgo::GetInputTensorNum(kernel_node);
|
|
||||||
if (input_num != INPUT_NUMS) {
|
|
||||||
MS_LOG(ERROR) << "For '" << kernel_name_ << "', the number of inputs must be 2, but got " << input_num;
|
|
||||||
}
|
|
||||||
|
|
||||||
const size_t coordinate_size = 4;
|
const size_t coordinate_size = 4;
|
||||||
if (common::AnfAlgo::GetCNodePrimitive(kernel_node)->GetAttr("means")->isa<ValueTuple>() ||
|
auto means = base_operator->GetAttr("means");
|
||||||
common::AnfAlgo::GetCNodePrimitive(kernel_node)->GetAttr("means")->isa<ValueList>()) {
|
if (means->isa<api::ValueSequence>()) {
|
||||||
means_ = common::AnfAlgo::GetNodeAttr<std::vector<float>>(kernel_node, "means");
|
means_ = api::GetValue<std::vector<float>>(means);
|
||||||
} else if (common::AnfAlgo::GetCNodePrimitive(kernel_node)->GetAttr("means")->isa<FloatImm>()) {
|
} else if (means->isa<api::FloatImm>()) {
|
||||||
float mean = common::AnfAlgo::GetNodeAttr<float>(kernel_node, "means");
|
float mean = api::GetValue<float>(means);
|
||||||
for (size_t i = 0; i < coordinate_size; i++) {
|
for (size_t i = 0; i < coordinate_size; i++) {
|
||||||
(void)means_.emplace_back(mean);
|
(void)means_.emplace_back(mean);
|
||||||
}
|
}
|
||||||
|
@ -65,11 +47,11 @@ void BoundingBoxEncodeCpuKernelMod::InitKernel(const CNodePtr &kernel_node) {
|
||||||
<< "', the input 'means' must be a tuple or a list, and dtype must be float, but got is not.";
|
<< "', the input 'means' must be a tuple or a list, and dtype must be float, but got is not.";
|
||||||
}
|
}
|
||||||
|
|
||||||
if (common::AnfAlgo::GetCNodePrimitive(kernel_node)->GetAttr("stds")->isa<ValueTuple>() ||
|
auto stds = base_operator->GetAttr("stds");
|
||||||
common::AnfAlgo::GetCNodePrimitive(kernel_node)->GetAttr("stds")->isa<ValueList>()) {
|
if (stds->isa<api::ValueSequence>()) {
|
||||||
stds_ = common::AnfAlgo::GetNodeAttr<std::vector<float>>(kernel_node, "stds");
|
stds_ = api::GetValue<std::vector<float>>(stds);
|
||||||
} else if (common::AnfAlgo::GetCNodePrimitive(kernel_node)->GetAttr("stds")->isa<FloatImm>()) {
|
} else if (stds->isa<api::FloatImm>()) {
|
||||||
float std = common::AnfAlgo::GetNodeAttr<float>(kernel_node, "stds");
|
float std = api::GetValue<float>(stds);
|
||||||
for (size_t i = 0; i < coordinate_size; i++) {
|
for (size_t i = 0; i < coordinate_size; i++) {
|
||||||
(void)stds_.emplace_back(std);
|
(void)stds_.emplace_back(std);
|
||||||
}
|
}
|
||||||
|
@ -84,7 +66,15 @@ void BoundingBoxEncodeCpuKernelMod::InitKernel(const CNodePtr &kernel_node) {
|
||||||
"but got the length of 'means': "
|
"but got the length of 'means': "
|
||||||
<< means_.size() << ", and the length of 'stds': " << stds_.size();
|
<< means_.size() << ", and the length of 'stds': " << stds_.size();
|
||||||
}
|
}
|
||||||
InitTaskFunc(kernel_node);
|
|
||||||
|
auto kernel_attr = GetKernelAttrFromTensors(inputs, outputs);
|
||||||
|
auto [is_match, index] = MatchKernelAttr(kernel_attr, GetOpSupport());
|
||||||
|
if (!is_match) {
|
||||||
|
MS_LOG(ERROR) << "For '" << kernel_name_ << "' does not support this kernel type: " << kernel_attr;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
kernel_func_ = func_list_[index].second;
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
|
@ -152,6 +142,62 @@ bool BoundingBoxEncodeCpuKernelMod::LaunchKernel(const std::vector<AddressPtr> &
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int BoundingBoxEncodeCpuKernelMod::Resize(const BaseOperatorPtr &base_operator,
|
||||||
|
const std::vector<KernelTensorPtr> &inputs,
|
||||||
|
const std::vector<KernelTensorPtr> &outputs,
|
||||||
|
const std::map<uint32_t, tensor::TensorPtr> &inputsOnHost) {
|
||||||
|
if (auto ret = KernelMod::Resize(base_operator, inputs, outputs, inputsOnHost); ret != KRET_OK) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto anchor_box_shape = LongVecToSizeVec(inputs[kIndex0]->GetShapeVector());
|
||||||
|
auto groundtruth_box_shape = LongVecToSizeVec(inputs[kIndex1]->GetShapeVector());
|
||||||
|
|
||||||
|
auto it_x = std::find_if(anchor_box_shape.begin(), anchor_box_shape.end(), [](int64_t sh) { return sh <= 0; });
|
||||||
|
if (it_x != anchor_box_shape.end()) {
|
||||||
|
return KRET_UNKNOWN_SHAPE;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto anchor_box_rank = anchor_box_shape.size();
|
||||||
|
auto groundtruth_box_rank = groundtruth_box_shape.size();
|
||||||
|
|
||||||
|
if (anchor_box_rank != kInputRank) {
|
||||||
|
MS_LOG(ERROR) << "The rank of anchor box must be 2, but got " << anchor_box_rank;
|
||||||
|
return KRET_RESIZE_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (groundtruth_box_rank != kInputRank) {
|
||||||
|
MS_LOG(ERROR) << "The rank of groundtruth box must be 2, but got " << groundtruth_box_rank;
|
||||||
|
return KRET_RESIZE_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (anchor_box_shape[1] != kLastDim) {
|
||||||
|
MS_LOG(ERROR) << "The shape of anchor box must be (n, 4), but got the second dimension of " << anchor_box_shape[1];
|
||||||
|
return KRET_RESIZE_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (groundtruth_box_shape[1] != kLastDim) {
|
||||||
|
MS_LOG(ERROR) << "The shape of groundtruth box must be (n, 4), but got the second dimension of "
|
||||||
|
<< groundtruth_box_shape[1];
|
||||||
|
return KRET_RESIZE_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
return KRET_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<std::pair<KernelAttr, BoundingBoxEncodeCpuKernelMod::BoundingBoxEncodeFunc>>
|
||||||
|
BoundingBoxEncodeCpuKernelMod::func_list_ = {
|
||||||
|
{KernelAttr().AddInputAttr(kNumberTypeFloat32).AddInputAttr(kNumberTypeFloat32).AddOutputAttr(kNumberTypeFloat32),
|
||||||
|
&BoundingBoxEncodeCpuKernelMod::LaunchKernel<float>},
|
||||||
|
{KernelAttr().AddInputAttr(kNumberTypeFloat16).AddInputAttr(kNumberTypeFloat16).AddOutputAttr(kNumberTypeFloat16),
|
||||||
|
&BoundingBoxEncodeCpuKernelMod::LaunchKernel<float16>}};
|
||||||
|
|
||||||
|
std::vector<KernelAttr> BoundingBoxEncodeCpuKernelMod::GetOpSupport() {
|
||||||
|
std::vector<KernelAttr> support_list;
|
||||||
|
(void)std::transform(func_list_.begin(), func_list_.end(), std::back_inserter(support_list),
|
||||||
|
[](const std::pair<KernelAttr, BoundingBoxEncodeFunc> &pair) { return pair.first; });
|
||||||
|
return support_list;
|
||||||
|
}
|
||||||
MS_KERNEL_FACTORY_REG(NativeCpuKernelMod, BoundingBoxEncode, BoundingBoxEncodeCpuKernelMod);
|
MS_KERNEL_FACTORY_REG(NativeCpuKernelMod, BoundingBoxEncode, BoundingBoxEncodeCpuKernelMod);
|
||||||
} // namespace kernel
|
} // namespace kernel
|
||||||
} // namespace mindspore
|
} // namespace mindspore
|
||||||
|
|
|
@ -14,10 +14,11 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef MINDSPORE_CCSRC_BACKEND_KERNEL_COMPILER_CPU_BOUNDINGBOX_ENCODE_CPU_KERNEL_H_
|
#ifndef MINDSPORE_CCSRC_PLUGIN_DEVICE_CPU_KERNEL_BOUNDINGBOX_ENCODE_CPU_KERNEL_H_
|
||||||
#define MINDSPORE_CCSRC_BACKEND_KERNEL_COMPILER_CPU_BOUNDINGBOX_ENCODE_CPU_KERNEL_H_
|
#define MINDSPORE_CCSRC_PLUGIN_DEVICE_CPU_KERNEL_BOUNDINGBOX_ENCODE_CPU_KERNEL_H_
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
#include <map>
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
|
||||||
|
@ -27,18 +28,22 @@
|
||||||
namespace mindspore {
|
namespace mindspore {
|
||||||
namespace kernel {
|
namespace kernel {
|
||||||
constexpr size_t INPUT_NUMS = 2;
|
constexpr size_t INPUT_NUMS = 2;
|
||||||
class BoundingBoxEncodeCpuKernelMod : public DeprecatedNativeCpuKernelMod {
|
class BoundingBoxEncodeCpuKernelMod : public NativeCpuKernelMod {
|
||||||
public:
|
public:
|
||||||
BoundingBoxEncodeCpuKernelMod() = default;
|
BoundingBoxEncodeCpuKernelMod() = default;
|
||||||
~BoundingBoxEncodeCpuKernelMod() override = default;
|
~BoundingBoxEncodeCpuKernelMod() override = default;
|
||||||
|
|
||||||
void InitKernel(const CNodePtr &kernel_node) override;
|
bool Init(const BaseOperatorPtr &base_operator, const std::vector<KernelTensorPtr> &inputs,
|
||||||
|
const std::vector<KernelTensorPtr> &outputs) override;
|
||||||
|
|
||||||
bool Launch(const std::vector<AddressPtr> &inputs, const std::vector<AddressPtr> &workspace,
|
bool Launch(const std::vector<AddressPtr> &inputs, const std::vector<AddressPtr> &workspace,
|
||||||
const std::vector<AddressPtr> &outputs) override {
|
const std::vector<AddressPtr> &outputs) override {
|
||||||
return kernel_func_(this, inputs, workspace, outputs);
|
return kernel_func_(this, inputs, workspace, outputs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int Resize(const BaseOperatorPtr &base_operator, const std::vector<KernelTensorPtr> &inputs,
|
||||||
|
const std::vector<KernelTensorPtr> &outputs, const std::map<uint32_t, tensor::TensorPtr> &) override;
|
||||||
|
|
||||||
std::vector<KernelAttr> GetOpSupport() override;
|
std::vector<KernelAttr> GetOpSupport() override;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -51,11 +56,10 @@ class BoundingBoxEncodeCpuKernelMod : public DeprecatedNativeCpuKernelMod {
|
||||||
static std::vector<std::pair<KernelAttr, BoundingBoxEncodeFunc>> func_list_;
|
static std::vector<std::pair<KernelAttr, BoundingBoxEncodeFunc>> func_list_;
|
||||||
BoundingBoxEncodeFunc kernel_func_;
|
BoundingBoxEncodeFunc kernel_func_;
|
||||||
|
|
||||||
void InitTaskFunc(const CNodePtr &kernel_node);
|
|
||||||
std::vector<float> means_;
|
std::vector<float> means_;
|
||||||
std::vector<float> stds_;
|
std::vector<float> stds_;
|
||||||
};
|
};
|
||||||
} // namespace kernel
|
} // namespace kernel
|
||||||
} // namespace mindspore
|
} // namespace mindspore
|
||||||
|
|
||||||
#endif // MINDSPORE_CCSRC_BACKEND_KERNEL_COMPILER_CPU_BOUNDINGBOX_ENCODE_CPU_KERNEL_H_
|
#endif // MINDSPORE_CCSRC_PLUGIN_DEVICE_CPU_KERNEL_BOUNDINGBOX_ENCODE_CPU_KERNEL_H_
|
||||||
|
|
|
@ -18,9 +18,121 @@
|
||||||
|
|
||||||
namespace mindspore {
|
namespace mindspore {
|
||||||
namespace kernel {
|
namespace kernel {
|
||||||
MS_REG_GPU_KERNEL_ONE(
|
bool BoundingBoxDecodeGpuKernelMod::Init(const BaseOperatorPtr &base_operator,
|
||||||
BoundingBoxDecode,
|
const std::vector<KernelTensorPtr> &inputs,
|
||||||
KernelAttr().AddInputAttr(kNumberTypeFloat32).AddInputAttr(kNumberTypeFloat32).AddOutputAttr(kNumberTypeFloat32),
|
const std::vector<KernelTensorPtr> &outputs) {
|
||||||
BoundingBoxDecodeGpuKernelMod, float)
|
MS_EXCEPTION_IF_NULL(base_operator);
|
||||||
|
kernel_name_ = base_operator->GetPrim()->name();
|
||||||
|
constexpr size_t input_num = 2;
|
||||||
|
CHECK_KERNEL_INPUTS_NUM(inputs.size(), input_num, kernel_name_);
|
||||||
|
|
||||||
|
const size_t coordinate_size = 4;
|
||||||
|
auto means = base_operator->GetAttr("means");
|
||||||
|
if (means->isa<api::ValueSequence>()) {
|
||||||
|
means_ = api::GetValue<std::vector<float>>(means);
|
||||||
|
} else if (means->isa<api::FloatImm>()) {
|
||||||
|
float mean = api::GetValue<float>(means);
|
||||||
|
for (size_t i = 0; i < coordinate_size; i++) {
|
||||||
|
(void)means_.emplace_back(mean);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
MS_LOG(EXCEPTION) << "For '" << kernel_name_
|
||||||
|
<< "', the input 'means' must be a tuple or a list, and dtype must be float, but got is not.";
|
||||||
|
}
|
||||||
|
|
||||||
|
auto stds = base_operator->GetAttr("stds");
|
||||||
|
if (stds->isa<api::ValueSequence>()) {
|
||||||
|
stds_ = api::GetValue<std::vector<float>>(stds);
|
||||||
|
} else if (stds->isa<api::FloatImm>()) {
|
||||||
|
float std = api::GetValue<float>(stds);
|
||||||
|
for (size_t i = 0; i < coordinate_size; i++) {
|
||||||
|
(void)stds_.emplace_back(std);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
MS_LOG(EXCEPTION) << "For '" << kernel_name_
|
||||||
|
<< "', the input 'stds' must be a tuple or a list, and dtype must be float, but got is not.";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (means_.size() < coordinate_size || stds_.size() < coordinate_size) {
|
||||||
|
MS_LOG(EXCEPTION) << "For '" << kernel_name_
|
||||||
|
<< "', the length of input 'means' and 'stds' must be at least 4, "
|
||||||
|
"but got the length of 'means': "
|
||||||
|
<< means_.size() << ", and the length of 'stds': " << stds_.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
auto max_shape = base_operator->GetAttr("max_shape");
|
||||||
|
std::vector<int64_t> max_shape_me = api::GetValue<std::vector<int64_t>>(max_shape);
|
||||||
|
(void)std::transform(max_shape_me.begin(), max_shape_me.end(), std::back_inserter(max_shape_),
|
||||||
|
[](const int64_t &value) { return LongToInt(value); });
|
||||||
|
auto wh_ratio_clip = base_operator->GetAttr("wh_ratio_clip");
|
||||||
|
wh_ratio_clip_ = api::GetValue<float>(wh_ratio_clip);
|
||||||
|
|
||||||
|
if (max_shape_.size() < kMinMaxShapeSize) {
|
||||||
|
MS_LOG(EXCEPTION) << "For '" << kernel_name_
|
||||||
|
<< "', the length of 'max_shape' must be at least 2, but got: " << max_shape_.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
auto kernel_attr = GetKernelAttrFromTensors(inputs, outputs);
|
||||||
|
auto [is_match, index] = MatchKernelAttr(kernel_attr, GetOpSupport());
|
||||||
|
if (!is_match) {
|
||||||
|
MS_LOG(ERROR) << "For '" << kernel_name_ << "' does not support this kernel type: " << kernel_attr;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
kernel_func_ = func_list_[index].second;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
int BoundingBoxDecodeGpuKernelMod::Resize(const BaseOperatorPtr &base_operator,
|
||||||
|
const std::vector<KernelTensorPtr> &inputs,
|
||||||
|
const std::vector<KernelTensorPtr> &outputs,
|
||||||
|
const std::map<uint32_t, tensor::TensorPtr> &inputsOnHost) {
|
||||||
|
if (auto ret = KernelMod::Resize(base_operator, inputs, outputs, inputsOnHost); ret != KRET_OK) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
return KRET_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
bool BoundingBoxDecodeGpuKernelMod::LaunchKernel(const std::vector<AddressPtr> &inputs,
|
||||||
|
const std::vector<AddressPtr> &workspace,
|
||||||
|
const std::vector<AddressPtr> &outputs, void *stream_ptr) {
|
||||||
|
T *rois_addr = GetDeviceAddress<T>(inputs, 0);
|
||||||
|
T *deltas_addr = GetDeviceAddress<T>(inputs, 1);
|
||||||
|
T *bboxes_addr = GetDeviceAddress<T>(outputs, 0);
|
||||||
|
|
||||||
|
if (inputs[0]->size != inputs[1]->size) {
|
||||||
|
MS_LOG(ERROR) << "For '" << kernel_name_ << "', rois box size must equal with deltas box size: " << inputs[1]->size
|
||||||
|
<< ", but got " << inputs[0]->size;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const size_t coordinate = 4;
|
||||||
|
const size_t block_size = inputs[0]->size / sizeof(T);
|
||||||
|
if ((block_size % coordinate) != 0) {
|
||||||
|
MS_LOG(ERROR) << "For '" << kernel_name_ << ", the size of the box should be a multiple of 4.";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
BoundingBoxDecode(block_size / coordinate, rois_addr, deltas_addr, bboxes_addr, means_[0], means_[1], means_[2],
|
||||||
|
means_[3], stds_[0], stds_[1], stds_[2], stds_[3], max_shape_[0], max_shape_[1], wh_ratio_clip_,
|
||||||
|
reinterpret_cast<cudaStream_t>(stream_ptr));
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<std::pair<KernelAttr, BoundingBoxDecodeGpuKernelMod::BoundingBoxDecodeLaunchFunc>>
|
||||||
|
BoundingBoxDecodeGpuKernelMod::func_list_ = {
|
||||||
|
{KernelAttr().AddInputAttr(kNumberTypeFloat32).AddInputAttr(kNumberTypeFloat32).AddOutputAttr(kNumberTypeFloat32),
|
||||||
|
&BoundingBoxDecodeGpuKernelMod::LaunchKernel<float>}};
|
||||||
|
|
||||||
|
std::vector<KernelAttr> BoundingBoxDecodeGpuKernelMod::GetOpSupport() {
|
||||||
|
std::vector<KernelAttr> support_list;
|
||||||
|
(void)std::transform(
|
||||||
|
func_list_.begin(), func_list_.end(), std::back_inserter(support_list),
|
||||||
|
[](const std::pair<KernelAttr, BoundingBoxDecodeGpuKernelMod::BoundingBoxDecodeLaunchFunc> &pair) {
|
||||||
|
return pair.first;
|
||||||
|
});
|
||||||
|
return support_list;
|
||||||
|
}
|
||||||
|
|
||||||
|
MS_KERNEL_FACTORY_REG(NativeGpuKernelMod, BoundingBoxDecode, BoundingBoxDecodeGpuKernelMod);
|
||||||
} // namespace kernel
|
} // namespace kernel
|
||||||
} // namespace mindspore
|
} // namespace mindspore
|
||||||
|
|
|
@ -14,10 +14,12 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef MINDSPORE_CCSRC_KERNEL_GPU_OTHER_BOUNDINGBOX_DECODE_GPU_KERNEL_H
|
#ifndef MINDSPORE_CCSRC_PLUGIN_DEVICE_GPU_KERNEL_OTHER_BOUNDINGBOX_DECODE_GPU_KERNEL_H
|
||||||
#define MINDSPORE_CCSRC_KERNEL_GPU_OTHER_BOUNDINGBOX_DECODE_GPU_KERNEL_H
|
#define MINDSPORE_CCSRC_PLUGIN_DEVICE_GPU_KERNEL_OTHER_BOUNDINGBOX_DECODE_GPU_KERNEL_H
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
#include <utility>
|
||||||
|
#include <map>
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include "plugin/device/gpu/kernel/cuda_impl/cuda_ops/boundingbox_decode_impl.cuh"
|
#include "plugin/device/gpu/kernel/cuda_impl/cuda_ops/boundingbox_decode_impl.cuh"
|
||||||
|
@ -26,128 +28,37 @@
|
||||||
|
|
||||||
namespace mindspore {
|
namespace mindspore {
|
||||||
namespace kernel {
|
namespace kernel {
|
||||||
template <typename T>
|
constexpr size_t kMinMaxShapeSize = 2;
|
||||||
class BoundingBoxDecodeGpuKernelMod : public DeprecatedNativeGpuKernelMod {
|
class BoundingBoxDecodeGpuKernelMod : public NativeGpuKernelMod {
|
||||||
public:
|
public:
|
||||||
BoundingBoxDecodeGpuKernelMod()
|
BoundingBoxDecodeGpuKernelMod() : rois_size_(0), deltas_size_(0), bboxes_size_(0), wh_ratio_clip_(0.016) {}
|
||||||
: rois_size_(0), deltas_size_(0), bboxes_size_(0), wh_ratio_clip_(0.016), is_null_input_(false) {}
|
|
||||||
~BoundingBoxDecodeGpuKernelMod() override = default;
|
~BoundingBoxDecodeGpuKernelMod() override = default;
|
||||||
|
|
||||||
bool Launch(const std::vector<AddressPtr> &inputs, const std::vector<AddressPtr> &workspace,
|
bool Launch(const std::vector<AddressPtr> &inputs, const std::vector<AddressPtr> &workspace,
|
||||||
const std::vector<AddressPtr> &outputs, void *stream_ptr) override {
|
const std::vector<AddressPtr> &outputs, void *stream_ptr) override {
|
||||||
if (is_null_input_) {
|
return kernel_func_(this, inputs, workspace, outputs, stream_ptr);
|
||||||
return true;
|
|
||||||
}
|
|
||||||
T *rois_addr = GetDeviceAddress<T>(inputs, 0);
|
|
||||||
T *deltas_addr = GetDeviceAddress<T>(inputs, 1);
|
|
||||||
T *bboxes_addr = GetDeviceAddress<T>(outputs, 0);
|
|
||||||
|
|
||||||
if (inputs[0]->size != inputs[1]->size) {
|
|
||||||
MS_LOG(ERROR) << "For '" << kernel_name_
|
|
||||||
<< "', rois box size must equal with deltas box size: " << inputs[1]->size << ", but got "
|
|
||||||
<< inputs[0]->size;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
const size_t coordinate = 4;
|
|
||||||
const size_t block_size = inputs[0]->size / sizeof(T);
|
|
||||||
if ((block_size % coordinate) != 0) {
|
|
||||||
MS_LOG(ERROR) << "For '" << kernel_name_ << ", the size of the box should be a multiple of 4.";
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
BoundingBoxDecode(block_size / coordinate, rois_addr, deltas_addr, bboxes_addr, means_[0], means_[1], means_[2],
|
|
||||||
means_[3], stds_[0], stds_[1], stds_[2], stds_[3], max_shape_[0], max_shape_[1], wh_ratio_clip_,
|
|
||||||
reinterpret_cast<cudaStream_t>(stream_ptr));
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Init(const CNodePtr &kernel_node) override {
|
bool Init(const BaseOperatorPtr &base_operator, const std::vector<KernelTensorPtr> &inputs,
|
||||||
kernel_name_ = common::AnfAlgo::GetCNodeName(kernel_node);
|
const std::vector<KernelTensorPtr> &outputs) override;
|
||||||
kernel_node_ = kernel_node;
|
|
||||||
MS_EXCEPTION_IF_NULL(kernel_node);
|
|
||||||
size_t input_num = common::AnfAlgo::GetInputTensorNum(kernel_node);
|
|
||||||
if (input_num != 2) {
|
|
||||||
MS_LOG(EXCEPTION) << "For '" << kernel_name_ << "', the number of inputs should be 2, but got " << input_num;
|
|
||||||
}
|
|
||||||
rois_size_ = sizeof(T);
|
|
||||||
deltas_size_ = sizeof(T);
|
|
||||||
bboxes_size_ = sizeof(T);
|
|
||||||
|
|
||||||
auto logits_shape = common::AnfAlgo::GetPrevNodeOutputInferShape(kernel_node, 0);
|
int Resize(const BaseOperatorPtr &base_operator, const std::vector<KernelTensorPtr> &inputs,
|
||||||
auto labels_shape = common::AnfAlgo::GetPrevNodeOutputInferShape(kernel_node, 1);
|
const std::vector<KernelTensorPtr> &outputs,
|
||||||
auto output_shape = common::AnfAlgo::GetOutputInferShape(kernel_node, 0);
|
const std::map<uint32_t, tensor::TensorPtr> &inputsOnHost) override;
|
||||||
is_null_input_ = CHECK_SHAPE_NULL(logits_shape, kernel_name_, "anchor_box") ||
|
|
||||||
CHECK_SHAPE_NULL(labels_shape, kernel_name_, "deltas") ||
|
|
||||||
CHECK_SHAPE_NULL(output_shape, kernel_name_, "output");
|
|
||||||
if (is_null_input_) {
|
|
||||||
InitSizeLists();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
for (size_t i = 0; i < logits_shape.size(); i++) {
|
|
||||||
rois_size_ *= logits_shape[i];
|
|
||||||
}
|
|
||||||
rois_size_ *= SizeOf(logits_shape);
|
|
||||||
deltas_size_ *= SizeOf(labels_shape);
|
|
||||||
bboxes_size_ *= SizeOf(output_shape);
|
|
||||||
|
|
||||||
InitSizeLists();
|
|
||||||
|
|
||||||
const size_t coordinate_size = 4;
|
|
||||||
auto prim = common::AnfAlgo::GetCNodePrimitive(kernel_node);
|
|
||||||
MS_EXCEPTION_IF_NULL(prim);
|
|
||||||
auto means = prim->GetAttr("means");
|
|
||||||
MS_EXCEPTION_IF_NULL(means);
|
|
||||||
if (means->isa<ValueTuple>() || means->isa<ValueList>()) {
|
|
||||||
means_ = GetAttr<std::vector<float>>(kernel_node, "means");
|
|
||||||
} else if (means->isa<FloatImm>()) {
|
|
||||||
float mean = GetAttr<float>(kernel_node, "means");
|
|
||||||
for (size_t i = 0; i < coordinate_size; i++) {
|
|
||||||
means_.emplace_back(mean);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
MS_LOG(EXCEPTION) << "For '" << kernel_name_ << "', attribute means type is invalid.";
|
|
||||||
}
|
|
||||||
|
|
||||||
auto stds = prim->GetAttr("stds");
|
|
||||||
MS_EXCEPTION_IF_NULL(stds);
|
|
||||||
if (stds->isa<ValueTuple>() || stds->isa<ValueList>()) {
|
|
||||||
stds_ = GetAttr<std::vector<float>>(kernel_node, "stds");
|
|
||||||
} else if (stds->isa<FloatImm>()) {
|
|
||||||
float std = GetAttr<float>(kernel_node, "stds");
|
|
||||||
for (size_t i = 0; i < coordinate_size; i++) {
|
|
||||||
stds_.emplace_back(std);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
MS_LOG(EXCEPTION) << "For '" << kernel_name_ << "', attribute stds type is invalid.";
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<int64_t> max_shape_me = GetAttr<std::vector<int64_t>>(kernel_node, "max_shape");
|
|
||||||
(void)std::transform(max_shape_me.begin(), max_shape_me.end(), std::back_inserter(max_shape_),
|
|
||||||
[](const int64_t &value) { return static_cast<int>(value); });
|
|
||||||
wh_ratio_clip_ = GetAttr<float>(kernel_node, "wh_ratio_clip");
|
|
||||||
|
|
||||||
if (means_.size() < coordinate_size || stds_.size() < coordinate_size) {
|
|
||||||
MS_LOG(EXCEPTION) << "For '" << kernel_name_ << "', the both size of means or stds cannot be less than 4, but got"
|
|
||||||
<< " the size of means: " << means_.size() << ", the size of stds: " << stds_.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (max_shape_.size() < 2) {
|
|
||||||
MS_LOG(EXCEPTION) << "For '" << kernel_name_ << "', the size of max_shape cannot be less than 2, but got "
|
|
||||||
<< max_shape_.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void InitSizeLists() override {
|
std::vector<KernelAttr> GetOpSupport() override;
|
||||||
input_size_list_.push_back(rois_size_);
|
template <typename T>
|
||||||
input_size_list_.push_back(deltas_size_);
|
bool LaunchKernel(const std::vector<AddressPtr> &inputs, const std::vector<AddressPtr> &workspace,
|
||||||
output_size_list_.push_back(bboxes_size_);
|
const std::vector<AddressPtr> &outputs, void *stream_ptr);
|
||||||
}
|
using BoundingBoxDecodeLaunchFunc =
|
||||||
|
std::function<bool(BoundingBoxDecodeGpuKernelMod *, const std::vector<kernel::AddressPtr> &,
|
||||||
|
const std::vector<kernel::AddressPtr> &, const std::vector<kernel::AddressPtr> &, void *)>;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
std::string kernel_name_{};
|
||||||
|
BoundingBoxDecodeLaunchFunc kernel_func_;
|
||||||
|
static std::vector<std::pair<KernelAttr, BoundingBoxDecodeLaunchFunc>> func_list_;
|
||||||
size_t rois_size_;
|
size_t rois_size_;
|
||||||
size_t deltas_size_;
|
size_t deltas_size_;
|
||||||
size_t bboxes_size_;
|
size_t bboxes_size_;
|
||||||
|
@ -155,9 +66,8 @@ class BoundingBoxDecodeGpuKernelMod : public DeprecatedNativeGpuKernelMod {
|
||||||
std::vector<float> stds_;
|
std::vector<float> stds_;
|
||||||
std::vector<int> max_shape_;
|
std::vector<int> max_shape_;
|
||||||
float wh_ratio_clip_;
|
float wh_ratio_clip_;
|
||||||
bool is_null_input_;
|
|
||||||
};
|
};
|
||||||
} // namespace kernel
|
} // namespace kernel
|
||||||
} // namespace mindspore
|
} // namespace mindspore
|
||||||
|
|
||||||
#endif // MINDSPORE_CCSRC_KERNEL_GPU_OTHER_BOUNDINGBOX_DECODE_GPU_KERNEL_H
|
#endif // MINDSPORE_CCSRC_PLUGIN_DEVICE_GPU_KERNEL_OTHER_BOUNDINGBOX_DECODE_GPU_KERNEL_H
|
||||||
|
|
|
@ -18,9 +18,111 @@
|
||||||
|
|
||||||
namespace mindspore {
|
namespace mindspore {
|
||||||
namespace kernel {
|
namespace kernel {
|
||||||
MS_REG_GPU_KERNEL_ONE(
|
bool BoundingBoxEncodeGpuKernelMod::Init(const BaseOperatorPtr &base_operator,
|
||||||
BoundingBoxEncode,
|
const std::vector<KernelTensorPtr> &inputs,
|
||||||
KernelAttr().AddInputAttr(kNumberTypeFloat32).AddInputAttr(kNumberTypeFloat32).AddOutputAttr(kNumberTypeFloat32),
|
const std::vector<KernelTensorPtr> &outputs) {
|
||||||
BoundingBoxEncodeGpuKernelMod, float)
|
MS_EXCEPTION_IF_NULL(base_operator);
|
||||||
|
kernel_name_ = base_operator->GetPrim()->name();
|
||||||
|
constexpr size_t input_num = 2;
|
||||||
|
CHECK_KERNEL_INPUTS_NUM(inputs.size(), input_num, kernel_name_);
|
||||||
|
|
||||||
|
const size_t coordinate_size = 4;
|
||||||
|
auto means = base_operator->GetAttr("means");
|
||||||
|
if (means->isa<api::ValueSequence>()) {
|
||||||
|
means_ = api::GetValue<std::vector<float>>(means);
|
||||||
|
} else if (means->isa<api::FloatImm>()) {
|
||||||
|
float mean = api::GetValue<float>(means);
|
||||||
|
for (size_t i = 0; i < coordinate_size; i++) {
|
||||||
|
(void)means_.emplace_back(mean);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
MS_LOG(EXCEPTION) << "For '" << kernel_name_
|
||||||
|
<< "', the input 'means' must be a tuple or a list, and dtype must be float, but got is not.";
|
||||||
|
}
|
||||||
|
|
||||||
|
auto stds = base_operator->GetAttr("stds");
|
||||||
|
if (stds->isa<api::ValueSequence>()) {
|
||||||
|
stds_ = api::GetValue<std::vector<float>>(stds);
|
||||||
|
} else if (stds->isa<api::FloatImm>()) {
|
||||||
|
float std = api::GetValue<float>(stds);
|
||||||
|
for (size_t i = 0; i < coordinate_size; i++) {
|
||||||
|
(void)stds_.emplace_back(std);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
MS_LOG(EXCEPTION) << "For '" << kernel_name_
|
||||||
|
<< "', the input 'stds' must be a tuple or a list, and dtype must be float, but got is not.";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (means_.size() < coordinate_size || stds_.size() < coordinate_size) {
|
||||||
|
MS_LOG(EXCEPTION) << "For '" << kernel_name_
|
||||||
|
<< "', the length of input 'means' and 'stds' must be at least 4, "
|
||||||
|
"but got the length of 'means': "
|
||||||
|
<< means_.size() << ", and the length of 'stds': " << stds_.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
auto kernel_attr = GetKernelAttrFromTensors(inputs, outputs);
|
||||||
|
auto [is_match, index] = MatchKernelAttr(kernel_attr, GetOpSupport());
|
||||||
|
if (!is_match) {
|
||||||
|
MS_LOG(ERROR) << "For '" << kernel_name_ << "' does not support this kernel type: " << kernel_attr;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
kernel_func_ = func_list_[index].second;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
int BoundingBoxEncodeGpuKernelMod::Resize(const BaseOperatorPtr &base_operator,
|
||||||
|
const std::vector<KernelTensorPtr> &inputs,
|
||||||
|
const std::vector<KernelTensorPtr> &outputs,
|
||||||
|
const std::map<uint32_t, tensor::TensorPtr> &inputsOnHost) {
|
||||||
|
if (auto ret = KernelMod::Resize(base_operator, inputs, outputs, inputsOnHost); ret != KRET_OK) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
return KRET_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
bool BoundingBoxEncodeGpuKernelMod::LaunchKernel(const std::vector<AddressPtr> &inputs,
|
||||||
|
const std::vector<AddressPtr> &workspace,
|
||||||
|
const std::vector<AddressPtr> &outputs, void *stream_ptr) {
|
||||||
|
T *anchor_addr = GetDeviceAddress<T>(inputs, 0);
|
||||||
|
T *groundtruth_addr = GetDeviceAddress<T>(inputs, 1);
|
||||||
|
T *deltas_addr = GetDeviceAddress<T>(outputs, 0);
|
||||||
|
|
||||||
|
if (inputs[0]->size != inputs[1]->size) {
|
||||||
|
MS_LOG(ERROR) << "For '" << kernel_name_
|
||||||
|
<< "', anchor box size must equal with groundtruth box size: " << inputs[1]->size << ", but got "
|
||||||
|
<< inputs[0]->size;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const size_t coordinate = 4;
|
||||||
|
const size_t block_size = inputs[0]->size / sizeof(T);
|
||||||
|
if ((block_size % coordinate) != 0) {
|
||||||
|
MS_LOG(ERROR) << "For '" << kernel_name_ << ", the size of the box should be a multiple of 4.";
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
BoundingBoxEncode(block_size / coordinate, anchor_addr, groundtruth_addr, deltas_addr, means_[0], means_[1],
|
||||||
|
means_[2], means_[3], stds_[0], stds_[1], stds_[2], stds_[3],
|
||||||
|
reinterpret_cast<cudaStream_t>(stream_ptr));
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<std::pair<KernelAttr, BoundingBoxEncodeGpuKernelMod::BoundingBoxEncodeLaunchFunc>>
|
||||||
|
BoundingBoxEncodeGpuKernelMod::func_list_ = {
|
||||||
|
{KernelAttr().AddInputAttr(kNumberTypeFloat32).AddInputAttr(kNumberTypeFloat32).AddOutputAttr(kNumberTypeFloat32),
|
||||||
|
&BoundingBoxEncodeGpuKernelMod::LaunchKernel<float>}};
|
||||||
|
|
||||||
|
std::vector<KernelAttr> BoundingBoxEncodeGpuKernelMod::GetOpSupport() {
|
||||||
|
std::vector<KernelAttr> support_list;
|
||||||
|
(void)std::transform(
|
||||||
|
func_list_.begin(), func_list_.end(), std::back_inserter(support_list),
|
||||||
|
[](const std::pair<KernelAttr, BoundingBoxEncodeGpuKernelMod::BoundingBoxEncodeLaunchFunc> &pair) {
|
||||||
|
return pair.first;
|
||||||
|
});
|
||||||
|
return support_list;
|
||||||
|
}
|
||||||
|
|
||||||
|
MS_KERNEL_FACTORY_REG(NativeGpuKernelMod, BoundingBoxEncode, BoundingBoxEncodeGpuKernelMod);
|
||||||
} // namespace kernel
|
} // namespace kernel
|
||||||
} // namespace mindspore
|
} // namespace mindspore
|
||||||
|
|
|
@ -14,131 +14,55 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef MINDSPORE_CCSRC_KERNEL_GPU_OTHER_BOUNDINGBOX_ENCODE_GPU_KERNEL_H
|
#ifndef MINDSPORE_CCSRC_PLUGIN_DEVICE_GPU_KERNEL_OTHER_BOUNDINGBOX_ENCODE_GPU_KERNEL_H
|
||||||
#define MINDSPORE_CCSRC_KERNEL_GPU_OTHER_BOUNDINGBOX_ENCODE_GPU_KERNEL_H
|
#define MINDSPORE_CCSRC_PLUGIN_DEVICE_GPU_KERNEL_OTHER_BOUNDINGBOX_ENCODE_GPU_KERNEL_H
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
#include <map>
|
||||||
|
#include <utility>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <algorithm>
|
||||||
#include "plugin/device/gpu/kernel/cuda_impl/cuda_ops/boundingbox_encode_impl.cuh"
|
#include "plugin/device/gpu/kernel/cuda_impl/cuda_ops/boundingbox_encode_impl.cuh"
|
||||||
#include "plugin/device/gpu/kernel/gpu_kernel.h"
|
#include "plugin/device/gpu/kernel/gpu_kernel.h"
|
||||||
#include "plugin/device/gpu/kernel/gpu_kernel_factory.h"
|
#include "plugin/device/gpu/kernel/gpu_kernel_factory.h"
|
||||||
|
|
||||||
namespace mindspore {
|
namespace mindspore {
|
||||||
namespace kernel {
|
namespace kernel {
|
||||||
template <typename T>
|
class BoundingBoxEncodeGpuKernelMod : public NativeGpuKernelMod {
|
||||||
class BoundingBoxEncodeGpuKernelMod : public DeprecatedNativeGpuKernelMod {
|
|
||||||
public:
|
public:
|
||||||
BoundingBoxEncodeGpuKernelMod() : anchor_size_(0), groundtruth_size_(0), deltas_size_(0), is_null_input_(false) {}
|
BoundingBoxEncodeGpuKernelMod() : anchor_size_(0), groundtruth_size_(0), deltas_size_(0) {}
|
||||||
~BoundingBoxEncodeGpuKernelMod() override = default;
|
~BoundingBoxEncodeGpuKernelMod() override = default;
|
||||||
|
|
||||||
bool Launch(const std::vector<AddressPtr> &inputs, const std::vector<AddressPtr> &workspace,
|
bool Launch(const std::vector<AddressPtr> &inputs, const std::vector<AddressPtr> &workspace,
|
||||||
const std::vector<AddressPtr> &outputs, void *stream_ptr) override {
|
const std::vector<AddressPtr> &outputs, void *stream_ptr) override {
|
||||||
if (is_null_input_) {
|
return kernel_func_(this, inputs, workspace, outputs, stream_ptr);
|
||||||
return true;
|
|
||||||
}
|
|
||||||
T *anchor_addr = GetDeviceAddress<T>(inputs, 0);
|
|
||||||
T *groundtruth_addr = GetDeviceAddress<T>(inputs, 1);
|
|
||||||
T *deltas_addr = GetDeviceAddress<T>(outputs, 0);
|
|
||||||
|
|
||||||
if (inputs[0]->size != inputs[1]->size) {
|
|
||||||
MS_LOG(ERROR) << "For '" << kernel_name_
|
|
||||||
<< "', anchor box size must equal with groundtruth box size: " << inputs[1]->size << ", but got "
|
|
||||||
<< inputs[0]->size;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
const size_t coordinate = 4;
|
|
||||||
const size_t block_size = inputs[0]->size / sizeof(T);
|
|
||||||
if ((block_size % coordinate) != 0) {
|
|
||||||
MS_LOG(ERROR) << "For '" << kernel_name_ << ", the size of the box should be a multiple of 4.";
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
BoundingBoxEncode(block_size / coordinate, anchor_addr, groundtruth_addr, deltas_addr, means_[0], means_[1],
|
|
||||||
means_[2], means_[3], stds_[0], stds_[1], stds_[2], stds_[3],
|
|
||||||
reinterpret_cast<cudaStream_t>(stream_ptr));
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Init(const CNodePtr &kernel_node) override {
|
bool Init(const BaseOperatorPtr &base_operator, const std::vector<KernelTensorPtr> &inputs,
|
||||||
kernel_name_ = common::AnfAlgo::GetCNodeName(kernel_node);
|
const std::vector<KernelTensorPtr> &outputs) override;
|
||||||
kernel_node_ = kernel_node;
|
|
||||||
MS_EXCEPTION_IF_NULL(kernel_node);
|
|
||||||
size_t input_num = common::AnfAlgo::GetInputTensorNum(kernel_node);
|
|
||||||
if (input_num != 2) {
|
|
||||||
MS_LOG(EXCEPTION) << "For '" << kernel_name_ << "', the number of inputs should be 2, but got " << input_num;
|
|
||||||
}
|
|
||||||
anchor_size_ = sizeof(T);
|
|
||||||
groundtruth_size_ = sizeof(T);
|
|
||||||
deltas_size_ = sizeof(T);
|
|
||||||
|
|
||||||
auto logits_shape = common::AnfAlgo::GetPrevNodeOutputInferShape(kernel_node, 0);
|
int Resize(const BaseOperatorPtr &base_operator, const std::vector<KernelTensorPtr> &inputs,
|
||||||
auto labels_shape = common::AnfAlgo::GetPrevNodeOutputInferShape(kernel_node, 1);
|
const std::vector<KernelTensorPtr> &outputs,
|
||||||
auto output_shape = common::AnfAlgo::GetOutputInferShape(kernel_node, 0);
|
const std::map<uint32_t, tensor::TensorPtr> &inputsOnHost) override;
|
||||||
is_null_input_ = CHECK_SHAPE_NULL(logits_shape, kernel_name_, "anchor_box") ||
|
|
||||||
CHECK_SHAPE_NULL(labels_shape, kernel_name_, "groundtruth_box") ||
|
|
||||||
CHECK_SHAPE_NULL(output_shape, kernel_name_, "output");
|
|
||||||
if (is_null_input_) {
|
|
||||||
InitSizeLists();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
anchor_size_ *= SizeOf(logits_shape);
|
|
||||||
groundtruth_size_ *= SizeOf(labels_shape);
|
|
||||||
deltas_size_ *= SizeOf(output_shape);
|
|
||||||
|
|
||||||
InitSizeLists();
|
|
||||||
|
|
||||||
const size_t coordinate_size = 4;
|
|
||||||
auto prim = common::AnfAlgo::GetCNodePrimitive(kernel_node);
|
|
||||||
MS_EXCEPTION_IF_NULL(prim);
|
|
||||||
auto means = prim->GetAttr("means");
|
|
||||||
MS_EXCEPTION_IF_NULL(means);
|
|
||||||
if (means->isa<ValueTuple>() || means->isa<ValueList>()) {
|
|
||||||
means_ = GetAttr<std::vector<float>>(kernel_node, "means");
|
|
||||||
} else if (means->isa<FloatImm>()) {
|
|
||||||
float mean = GetAttr<float>(kernel_node, "means");
|
|
||||||
for (size_t i = 0; i < coordinate_size; i++) {
|
|
||||||
means_.emplace_back(mean);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
MS_LOG(EXCEPTION) << "For '" << kernel_name_ << "', attribute means type is invalid.";
|
|
||||||
}
|
|
||||||
auto stds = prim->GetAttr("stds");
|
|
||||||
MS_EXCEPTION_IF_NULL(stds);
|
|
||||||
if (stds->isa<ValueTuple>() || stds->isa<ValueList>()) {
|
|
||||||
stds_ = GetAttr<std::vector<float>>(kernel_node, "stds");
|
|
||||||
} else if (stds->isa<FloatImm>()) {
|
|
||||||
float std = GetAttr<float>(kernel_node, "stds");
|
|
||||||
for (size_t i = 0; i < coordinate_size; i++) {
|
|
||||||
stds_.emplace_back(std);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
MS_LOG(EXCEPTION) << "For '" << kernel_name_ << "', attribute stds type is invalid.";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (means_.size() < coordinate_size || stds_.size() < coordinate_size) {
|
|
||||||
MS_LOG(EXCEPTION) << "For '" << kernel_name_ << "', the both size of means or stds cannot be less than 4, but got"
|
|
||||||
<< " the size of means: " << means_.size() << ", the size of stds: " << stds_.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void InitSizeLists() override {
|
std::vector<KernelAttr> GetOpSupport() override;
|
||||||
input_size_list_.push_back(anchor_size_);
|
template <typename T>
|
||||||
input_size_list_.push_back(groundtruth_size_);
|
bool LaunchKernel(const std::vector<AddressPtr> &inputs, const std::vector<AddressPtr> &workspace,
|
||||||
output_size_list_.push_back(deltas_size_);
|
const std::vector<AddressPtr> &outputs, void *stream_ptr);
|
||||||
}
|
using BoundingBoxEncodeLaunchFunc =
|
||||||
|
std::function<bool(BoundingBoxEncodeGpuKernelMod *, const std::vector<kernel::AddressPtr> &,
|
||||||
|
const std::vector<kernel::AddressPtr> &, const std::vector<kernel::AddressPtr> &, void *)>;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
std::string kernel_name_{};
|
||||||
|
BoundingBoxEncodeLaunchFunc kernel_func_;
|
||||||
|
static std::vector<std::pair<KernelAttr, BoundingBoxEncodeLaunchFunc>> func_list_;
|
||||||
size_t anchor_size_;
|
size_t anchor_size_;
|
||||||
size_t groundtruth_size_;
|
size_t groundtruth_size_;
|
||||||
size_t deltas_size_;
|
size_t deltas_size_;
|
||||||
std::vector<float> means_;
|
std::vector<float> means_;
|
||||||
std::vector<float> stds_;
|
std::vector<float> stds_;
|
||||||
bool is_null_input_;
|
|
||||||
};
|
};
|
||||||
} // namespace kernel
|
} // namespace kernel
|
||||||
} // namespace mindspore
|
} // namespace mindspore
|
||||||
|
|
|
@ -31,7 +31,6 @@ abstract::ShapePtr BoundingBoxDecodeInferShape(const PrimitivePtr &primitive,
|
||||||
const std::vector<AbstractBasePtr> &input_args) {
|
const std::vector<AbstractBasePtr> &input_args) {
|
||||||
MS_EXCEPTION_IF_NULL(primitive);
|
MS_EXCEPTION_IF_NULL(primitive);
|
||||||
auto prim_name = primitive->name();
|
auto prim_name = primitive->name();
|
||||||
|
|
||||||
auto anchor_box = input_args[0]->BuildShape();
|
auto anchor_box = input_args[0]->BuildShape();
|
||||||
auto deltas = input_args[1]->BuildShape();
|
auto deltas = input_args[1]->BuildShape();
|
||||||
|
|
||||||
|
|
|
@ -0,0 +1,116 @@
|
||||||
|
/**
|
||||||
|
* Copyright 2022 Huawei Technologies Co., Ltd
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "ops/bounding_box_encode.h"
|
||||||
|
|
||||||
|
#include <algorithm>
|
||||||
|
#include <set>
|
||||||
|
|
||||||
|
#include "abstract/ops/primitive_infer_map.h"
|
||||||
|
#include "ops/op_utils.h"
|
||||||
|
#include "utils/check_convert_utils.h"
|
||||||
|
#include "mindapi/src/helper.h"
|
||||||
|
|
||||||
|
namespace mindspore {
|
||||||
|
namespace ops {
|
||||||
|
namespace {
|
||||||
|
abstract::ShapePtr BoundingBoxEncodeInferShape(const PrimitivePtr &primitive,
|
||||||
|
const std::vector<AbstractBasePtr> &input_args) {
|
||||||
|
MS_EXCEPTION_IF_NULL(primitive);
|
||||||
|
auto prim_name = primitive->name();
|
||||||
|
|
||||||
|
auto anchor_box = input_args[0]->BuildShape();
|
||||||
|
auto groundtruth_box = input_args[1]->BuildShape();
|
||||||
|
|
||||||
|
MS_EXCEPTION_IF_NULL(anchor_box);
|
||||||
|
MS_EXCEPTION_IF_NULL(groundtruth_box);
|
||||||
|
|
||||||
|
const int64_t input_num = 2;
|
||||||
|
(void)CheckAndConvertUtils::CheckInteger("arg size", SizeToLong(input_args.size()), kEqual, input_num, prim_name);
|
||||||
|
|
||||||
|
auto anchor_box_shape = CheckAndConvertUtils::ConvertShapePtrToShapeMap(input_args[0]->BuildShape())[kShape];
|
||||||
|
auto groundtruth_box_shape = CheckAndConvertUtils::ConvertShapePtrToShapeMap(input_args[1]->BuildShape())[kShape];
|
||||||
|
|
||||||
|
const int64_t kShapeSize = 2;
|
||||||
|
(void)CheckAndConvertUtils::CheckInteger("anchor box rank", SizeToLong(anchor_box_shape.size()), kEqual, kShapeSize,
|
||||||
|
prim_name);
|
||||||
|
(void)CheckAndConvertUtils::CheckInteger("groundtruth box rank", SizeToLong(groundtruth_box_shape.size()), kEqual,
|
||||||
|
kShapeSize, prim_name);
|
||||||
|
|
||||||
|
if (anchor_box_shape[0] != groundtruth_box_shape[0]) {
|
||||||
|
MS_EXCEPTION(ValueError)
|
||||||
|
<< "For '" << prim_name
|
||||||
|
<< "', 'anchor_box' and 'groundtruth_box' must have the same first dimension. But got anchor_box_shape[0]: "
|
||||||
|
<< anchor_box_shape[0] << ", groundtruth_box_shape[0]: " << groundtruth_box_shape[0] << ".";
|
||||||
|
}
|
||||||
|
|
||||||
|
const int64_t last_dimension = 4;
|
||||||
|
if (anchor_box_shape[1] != last_dimension) {
|
||||||
|
MS_EXCEPTION(ValueError) << "For '" << prim_name
|
||||||
|
<< "', 'anchor_box' last dimension must be 4, but got: " << anchor_box_shape[1] << ".";
|
||||||
|
}
|
||||||
|
if (groundtruth_box_shape[1] != last_dimension) {
|
||||||
|
MS_EXCEPTION(ValueError) << "For '" << prim_name
|
||||||
|
<< "', 'groundtruth_box' last dimension must be 4, but got: " << groundtruth_box_shape[1]
|
||||||
|
<< ".";
|
||||||
|
}
|
||||||
|
|
||||||
|
auto x_shape = anchor_box->cast<abstract::ShapePtr>();
|
||||||
|
MS_EXCEPTION_IF_NULL(x_shape);
|
||||||
|
return x_shape;
|
||||||
|
}
|
||||||
|
|
||||||
|
TypePtr BoundingBoxEncodeInferType(const PrimitivePtr &primitive, const std::vector<AbstractBasePtr> &input_args) {
|
||||||
|
MS_EXCEPTION_IF_NULL(primitive);
|
||||||
|
auto prim_name = primitive->name();
|
||||||
|
|
||||||
|
for (const auto &item : input_args) {
|
||||||
|
MS_EXCEPTION_IF_NULL(item);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::set<TypePtr> valid_x_type;
|
||||||
|
(void)valid_x_type.emplace(kFloat16);
|
||||||
|
(void)valid_x_type.emplace(kFloat32);
|
||||||
|
|
||||||
|
for (size_t i = 0; i < input_args.size(); i++) {
|
||||||
|
auto x_type = input_args[i]->BuildType();
|
||||||
|
MS_EXCEPTION_IF_NULL(x_type);
|
||||||
|
(void)CheckAndConvertUtils::CheckTensorTypeValid("x_dtype", x_type, valid_x_type, prim_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::map<std::string, TypePtr> types;
|
||||||
|
(void)types.emplace("anchor_box", input_args[0]->BuildType());
|
||||||
|
(void)types.emplace("groundtruth_box", input_args[1]->BuildType());
|
||||||
|
|
||||||
|
return CheckAndConvertUtils::CheckTensorTypeSame(types, common_valid_types, prim_name);
|
||||||
|
}
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
MIND_API_OPERATOR_IMPL(BoundingBoxEncode, BaseOperator);
|
||||||
|
AbstractBasePtr BoundingBoxEncodeInfer(const abstract::AnalysisEnginePtr &, const PrimitivePtr &primitive,
|
||||||
|
const std::vector<AbstractBasePtr> &input_args) {
|
||||||
|
MS_EXCEPTION_IF_NULL(primitive);
|
||||||
|
|
||||||
|
const int64_t kInputNum = 2;
|
||||||
|
CheckAndConvertUtils::CheckInputArgs(input_args, kEqual, kInputNum, primitive->name());
|
||||||
|
|
||||||
|
auto infer_type = BoundingBoxEncodeInferType(primitive, input_args);
|
||||||
|
auto infer_shape = BoundingBoxEncodeInferShape(primitive, input_args);
|
||||||
|
return abstract::MakeAbstract(infer_shape, infer_type);
|
||||||
|
}
|
||||||
|
REGISTER_PRIMITIVE_EVAL_IMPL(BoundingBoxEncode, prim::kPrimBoundingBoxEncode, BoundingBoxEncodeInfer, nullptr, true);
|
||||||
|
} // namespace ops
|
||||||
|
} // namespace mindspore
|
|
@ -0,0 +1,44 @@
|
||||||
|
/**
|
||||||
|
* Copyright 2022 Huawei Technologies Co., Ltd
|
||||||
|
*
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef MINDSPORE_CORE_OPS_BOUNDING_BOX_ENCODE_H_
|
||||||
|
#define MINDSPORE_CORE_OPS_BOUNDING_BOX_ENCODE_H_
|
||||||
|
|
||||||
|
#include <map>
|
||||||
|
#include <vector>
|
||||||
|
#include <string>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
#include "ops/base_operator.h"
|
||||||
|
#include "mindapi/base/types.h"
|
||||||
|
|
||||||
|
namespace mindspore {
|
||||||
|
namespace ops {
|
||||||
|
constexpr auto kNameBoundingBoxEncode = "BoundingBoxEncode";
|
||||||
|
class MIND_API BoundingBoxEncode : public BaseOperator {
|
||||||
|
public:
|
||||||
|
MIND_API_BASE_MEMBER(BoundingBoxEncode);
|
||||||
|
BoundingBoxEncode() : BaseOperator(kNameBoundingBoxEncode) {
|
||||||
|
InitIOName({"anchor_box", "groundtruth_box"}, {"output"});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
abstract::AbstractBasePtr BoundingBoxEncodeInfer(const abstract::AnalysisEnginePtr &, const PrimitivePtr &primitive,
|
||||||
|
const std::vector<abstract::AbstractBasePtr> &input_args);
|
||||||
|
} // namespace ops
|
||||||
|
} // namespace mindspore
|
||||||
|
|
||||||
|
#endif // MINDSPORE_CORE_OPS_BOUNDING_BOX_ENCODE_H_
|
|
@ -914,6 +914,7 @@ GVAR_DEF(PrimitivePtr, kPrimSparseApplyRMSProp, std::make_shared<Primitive>("Spa
|
||||||
GVAR_DEF(PrimitivePtr, kPrimApplyKerasMomentum, std::make_shared<Primitive>("ApplyKerasMomentum"));
|
GVAR_DEF(PrimitivePtr, kPrimApplyKerasMomentum, std::make_shared<Primitive>("ApplyKerasMomentum"));
|
||||||
GVAR_DEF(PrimitivePtr, kPrimLARSUpdate, std::make_shared<Primitive>("LARSUpdate"));
|
GVAR_DEF(PrimitivePtr, kPrimLARSUpdate, std::make_shared<Primitive>("LARSUpdate"));
|
||||||
GVAR_DEF(PrimitivePtr, kPrimBoundingBoxDecode, std::make_shared<Primitive>("BoundingBoxDecode"));
|
GVAR_DEF(PrimitivePtr, kPrimBoundingBoxDecode, std::make_shared<Primitive>("BoundingBoxDecode"));
|
||||||
|
GVAR_DEF(PrimitivePtr, kPrimBoundingBoxEncode, std::make_shared<Primitive>("BoundingBoxEncode"));
|
||||||
GVAR_DEF(PrimitivePtr, kPrimROIAlign, std::make_shared<Primitive>("ROIAlign"));
|
GVAR_DEF(PrimitivePtr, kPrimROIAlign, std::make_shared<Primitive>("ROIAlign"));
|
||||||
GVAR_DEF(PrimitivePtr, kPrimApplyAddSign, std::make_shared<Primitive>("ApplyAddSign"));
|
GVAR_DEF(PrimitivePtr, kPrimApplyAddSign, std::make_shared<Primitive>("ApplyAddSign"));
|
||||||
GVAR_DEF(PrimitivePtr, kPrimApplyAdagrad, std::make_shared<Primitive>("ApplyAdagrad"));
|
GVAR_DEF(PrimitivePtr, kPrimApplyAdagrad, std::make_shared<Primitive>("ApplyAdagrad"));
|
||||||
|
|
|
@ -1,2 +1,2 @@
|
||||||
Note: This is the mindspore Lite inference framework size threshold. Offline review is required before modify this value!!!
|
Note: This is the mindspore Lite inference framework size threshold. Offline review is required before modify this value!!!
|
||||||
1083704
|
1085704
|
||||||
|
|
|
@ -153,20 +153,6 @@ class BoundingBoxEncode(PrimitiveWithInfer):
|
||||||
validator.check_equal_int(len(means), 4, "means len", self.name)
|
validator.check_equal_int(len(means), 4, "means len", self.name)
|
||||||
validator.check_equal_int(len(stds), 4, "stds len", self.name)
|
validator.check_equal_int(len(stds), 4, "stds len", self.name)
|
||||||
|
|
||||||
def infer_shape(self, anchor_box, groundtruth_box):
|
|
||||||
validator.check('anchor_box shape[0]', anchor_box[0], 'groundtruth_box shape[0]', groundtruth_box[0], Rel.EQ,
|
|
||||||
self.name)
|
|
||||||
validator.check("anchor_box rank", len(anchor_box), "", 2, Rel.EQ, self.name)
|
|
||||||
validator.check("groundtruth_box rank", len(groundtruth_box), "", 2, Rel.EQ, self.name)
|
|
||||||
validator.check_equal_int(anchor_box[1], 4, 'anchor_box shape[1]', self.name)
|
|
||||||
validator.check_equal_int(groundtruth_box[1], 4, 'groundtruth_box shape[1]', self.name)
|
|
||||||
return anchor_box
|
|
||||||
|
|
||||||
def infer_dtype(self, anchor_box, groundtruth_box):
|
|
||||||
args = {"anchor_box": anchor_box, "groundtruth_box": groundtruth_box}
|
|
||||||
validator.check_tensors_dtypes_same_and_valid(args, mstype.number_type, self.name)
|
|
||||||
return anchor_box
|
|
||||||
|
|
||||||
|
|
||||||
class BartlettWindow(Primitive):
|
class BartlettWindow(Primitive):
|
||||||
r"""
|
r"""
|
||||||
|
@ -662,6 +648,7 @@ class CheckBprop(PrimitiveWithInfer):
|
||||||
self.prim_to_check = prim_to_check
|
self.prim_to_check = prim_to_check
|
||||||
|
|
||||||
def infer_shape(self, xshapes, yshapes):
|
def infer_shape(self, xshapes, yshapes):
|
||||||
|
"""infer shape"""
|
||||||
tips = f"user defined method 'bprop'"
|
tips = f"user defined method 'bprop'"
|
||||||
validator.check_value_type('grads', xshapes, (tuple,), tips)
|
validator.check_value_type('grads', xshapes, (tuple,), tips)
|
||||||
validator.check_value_type('params', yshapes, (tuple,), tips)
|
validator.check_value_type('params', yshapes, (tuple,), tips)
|
||||||
|
@ -682,6 +669,7 @@ class CheckBprop(PrimitiveWithInfer):
|
||||||
return xshapes
|
return xshapes
|
||||||
|
|
||||||
def infer_dtype(self, xdtypes, ydtypes):
|
def infer_dtype(self, xdtypes, ydtypes):
|
||||||
|
"""infer dtype"""
|
||||||
tips = f"user defined method 'bprop'"
|
tips = f"user defined method 'bprop'"
|
||||||
validator.check_value_type('grads', xdtypes, (tuple,), tips)
|
validator.check_value_type('grads', xdtypes, (tuple,), tips)
|
||||||
validator.check_value_type('params', ydtypes, (tuple,), tips)
|
validator.check_value_type('params', ydtypes, (tuple,), tips)
|
||||||
|
|
|
@ -92,5 +92,31 @@ def test_bounding_box_decode_functional_modes():
|
||||||
test_bounding_box_decode_functional()
|
test_bounding_box_decode_functional()
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
@pytest.mark.level0
|
||||||
test_bounding_box_decode_functional_modes()
|
@pytest.mark.platform_x86_cpu
|
||||||
|
@pytest.mark.env_onecard
|
||||||
|
def test_dynamic_shape_boundingbox_decode():
|
||||||
|
"""
|
||||||
|
Feature: Test dynamic shape of BoundingBoxDecode operator
|
||||||
|
Description: dynamic input
|
||||||
|
Expectation: success.
|
||||||
|
"""
|
||||||
|
anchor = np.array([[4, 1, 2, 1], [2, 2, 2, 3]], np.float32)
|
||||||
|
deltas = np.array([[3, 1, 2, 2], [1, 2, 1, 4]], np.float32)
|
||||||
|
means = (0.1, 0.1, 0.2, 0.2)
|
||||||
|
stds = (2.0, 2.0, 3.0, 3.0)
|
||||||
|
anchor_box = Tensor(anchor, mindspore.float32)
|
||||||
|
deltas_box = Tensor(deltas, mindspore.float32)
|
||||||
|
expect_deltas = np.array([[28.6500, 0.0000, 0.0000, 33.8500],
|
||||||
|
[0.0000, 0.0000, 15.8663, 72.7000]], np.float32)
|
||||||
|
|
||||||
|
error = np.ones(shape=[2, 4]) * 1.0e-4
|
||||||
|
|
||||||
|
context.set_context(mode=context.GRAPH_MODE, device_target='CPU')
|
||||||
|
boundingbox_decode = NetBoundingBoxDecode(means, stds)
|
||||||
|
input_dyn = Tensor(shape=[None, 4], dtype=anchor_box.dtype)
|
||||||
|
deltas_box_dyn = Tensor(shape=[None, 4], dtype=deltas_box.dtype)
|
||||||
|
boundingbox_decode.set_inputs(input_dyn, deltas_box_dyn)
|
||||||
|
output = boundingbox_decode(anchor_box, deltas_box)
|
||||||
|
diff = output.asnumpy() - expect_deltas
|
||||||
|
assert np.all(abs(diff) < error)
|
||||||
|
|
|
@ -112,5 +112,29 @@ def test_bounding_box_encode_functional_modes():
|
||||||
test_bounding_box_encode_functional()
|
test_bounding_box_encode_functional()
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
@pytest.mark.level0
|
||||||
test_bounding_box_encode_functional_modes()
|
@pytest.mark.platform_x86_cpu
|
||||||
|
@pytest.mark.env_onecard
|
||||||
|
def test_dynamic_shape_boundingbox_encode():
|
||||||
|
"""
|
||||||
|
Feature: Test dynamic shape of BoundingBoxEncode operator
|
||||||
|
Description: dynamic input
|
||||||
|
Expectation: success.
|
||||||
|
"""
|
||||||
|
anchor = np.array([[4, 1, 6, 9], [2, 5, 5, 9]]).astype(np.float32)
|
||||||
|
gt = np.array([[3, 2, 7, 7], [1, 5, 5, 8]]).astype(np.float32)
|
||||||
|
means = (0.1, 0.1, 0.2, 0.2)
|
||||||
|
stds = (2.0, 2.0, 3.0, 3.0)
|
||||||
|
anchor_box = Tensor(anchor, mindspore.float32)
|
||||||
|
groundtruth_box = Tensor(gt, mindspore.float32)
|
||||||
|
expect_deltas = bbox2delta(anchor, gt, means, stds)
|
||||||
|
|
||||||
|
error = np.ones(shape=[2, 4]) * 1.0e-6
|
||||||
|
context.set_context(mode=context.GRAPH_MODE, device_target='CPU')
|
||||||
|
boundingbox_encode = NetBoundingBoxEncode(means, stds)
|
||||||
|
input_dyn = Tensor(shape=[None, 4], dtype=anchor_box.dtype)
|
||||||
|
groundtruth_box_dyn = Tensor(shape=[None, 4], dtype=groundtruth_box.dtype)
|
||||||
|
boundingbox_encode.set_inputs(input_dyn, groundtruth_box_dyn)
|
||||||
|
output = boundingbox_encode(anchor_box, groundtruth_box)
|
||||||
|
diff = output.asnumpy() - expect_deltas
|
||||||
|
assert np.all(abs(diff) < error)
|
||||||
|
|
|
@ -90,3 +90,33 @@ def test_bounding_box_decode_functional_modes():
|
||||||
test_bounding_box_decode_functional()
|
test_bounding_box_decode_functional()
|
||||||
context.set_context(mode=context.PYNATIVE_MODE, device_target="GPU")
|
context.set_context(mode=context.PYNATIVE_MODE, device_target="GPU")
|
||||||
test_bounding_box_decode_functional()
|
test_bounding_box_decode_functional()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.level0
|
||||||
|
@pytest.mark.platform_x86_gpu_training
|
||||||
|
@pytest.mark.env_onecard
|
||||||
|
def test_dynamic_shape_boundingbox_decode():
|
||||||
|
"""
|
||||||
|
Feature: Test dynamic shape of BoundingBoxDecode operator
|
||||||
|
Description: dynamic input
|
||||||
|
Expectation: success.
|
||||||
|
"""
|
||||||
|
anchor = np.array([[4, 1, 2, 1], [2, 2, 2, 3]], np.float32)
|
||||||
|
deltas = np.array([[3, 1, 2, 2], [1, 2, 1, 4]], np.float32)
|
||||||
|
means = (0.1, 0.1, 0.2, 0.2)
|
||||||
|
stds = (2.0, 2.0, 3.0, 3.0)
|
||||||
|
anchor_box = Tensor(anchor, mindspore.float32)
|
||||||
|
deltas_box = Tensor(deltas, mindspore.float32)
|
||||||
|
expect_deltas = np.array([[28.6500, 0.0000, 0.0000, 33.8500],
|
||||||
|
[0.0000, 0.0000, 15.8663, 72.7000]], np.float32)
|
||||||
|
|
||||||
|
error = np.ones(shape=[2, 4]) * 1.0e-4
|
||||||
|
|
||||||
|
context.set_context(mode=context.GRAPH_MODE, device_target='GPU')
|
||||||
|
boundingbox_decode = NetBoundingBoxDecode(means, stds)
|
||||||
|
input_dyn = Tensor(shape=[None, 4], dtype=anchor_box.dtype)
|
||||||
|
deltas_box_dyn = Tensor(shape=[None, 4], dtype=deltas_box.dtype)
|
||||||
|
boundingbox_decode.set_inputs(input_dyn, deltas_box_dyn)
|
||||||
|
output = boundingbox_decode(anchor_box, deltas_box)
|
||||||
|
diff = output.asnumpy() - expect_deltas
|
||||||
|
assert np.all(abs(diff) < error)
|
||||||
|
|
|
@ -110,3 +110,31 @@ def test_bounding_box_encode_functional_modes():
|
||||||
test_bounding_box_encode_functional()
|
test_bounding_box_encode_functional()
|
||||||
context.set_context(mode=context.PYNATIVE_MODE, device_target="GPU")
|
context.set_context(mode=context.PYNATIVE_MODE, device_target="GPU")
|
||||||
test_bounding_box_encode_functional()
|
test_bounding_box_encode_functional()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.level0
|
||||||
|
@pytest.mark.platform_x86_gpu_training
|
||||||
|
@pytest.mark.env_onecard
|
||||||
|
def test_dynamic_shape_boundingbox_encode():
|
||||||
|
"""
|
||||||
|
Feature: Test dynamic shape of BoundingBoxEncode operator
|
||||||
|
Description: dynamic input
|
||||||
|
Expectation: success.
|
||||||
|
"""
|
||||||
|
anchor = np.array([[4, 1, 6, 9], [2, 5, 5, 9]]).astype(np.float32)
|
||||||
|
gt = np.array([[3, 2, 7, 7], [1, 5, 5, 8]]).astype(np.float32)
|
||||||
|
means = (0.1, 0.1, 0.2, 0.2)
|
||||||
|
stds = (2.0, 2.0, 3.0, 3.0)
|
||||||
|
anchor_box = Tensor(anchor, mindspore.float32)
|
||||||
|
groundtruth_box = Tensor(gt, mindspore.float32)
|
||||||
|
expect_deltas = bbox2delta(anchor, gt, means, stds)
|
||||||
|
|
||||||
|
error = np.ones(shape=[2, 4]) * 1.0e-6
|
||||||
|
context.set_context(mode=context.GRAPH_MODE, device_target='GPU')
|
||||||
|
boundingbox_encode = NetBoundingBoxEncode(means, stds)
|
||||||
|
input_dyn = Tensor(shape=[None, 4], dtype=anchor_box.dtype)
|
||||||
|
groundtruth_box_dyn = Tensor(shape=[None, 4], dtype=groundtruth_box.dtype)
|
||||||
|
boundingbox_encode.set_inputs(input_dyn, groundtruth_box_dyn)
|
||||||
|
output = boundingbox_encode(anchor_box, groundtruth_box)
|
||||||
|
diff = output.asnumpy() - expect_deltas
|
||||||
|
assert np.all(abs(diff) < error)
|
||||||
|
|
Loading…
Reference in New Issue