2020-11-14 06:22:10 +08:00
|
|
|
// This file is part of OpenCV project.
|
|
|
|
// It is subject to the license terms in the LICENSE file found in the top-level directory
|
|
|
|
// of this distribution and at http://opencv.org/license.html.
|
|
|
|
|
|
|
|
#include "perf_precomp.hpp"
|
|
|
|
#include <opencv2/dnn/shape_utils.hpp>
|
|
|
|
|
|
|
|
namespace opencv_test {
|
|
|
|
|
|
|
|
struct Conv1DParam_t {
|
|
|
|
int kernel;
|
|
|
|
struct BlobShape { int dims[3]; } shapeIn;
|
|
|
|
int outCN;
|
|
|
|
int groups;
|
|
|
|
int stride;
|
|
|
|
int dilation;
|
|
|
|
int pad[2];
|
|
|
|
const char* padMode;
|
|
|
|
bool hasBias;
|
|
|
|
double declared_flops;
|
|
|
|
};
|
|
|
|
// Details: #12142
|
|
|
|
static const Conv1DParam_t testConvolution1DConfigs[] = {
|
|
|
|
{3, {{1, 6, 10}}, 6, 1, 1, 1, {0, 0}, "VALID", true, 1776.},
|
|
|
|
{3, {{1, 2, 19}}, 2, 2, 2, 1, {1, 1}, "", true, 260.},
|
|
|
|
{3, {{1, 2, 25}}, 2, 2, 1, 1, {2, 2}, "SAME", false, 650.},
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Conv1DParamID
|
|
|
|
{
|
|
|
|
enum {
|
|
|
|
CONV_0 = 0,
|
|
|
|
CONV_LAST = sizeof(testConvolution1DConfigs) / sizeof(testConvolution1DConfigs[0])
|
|
|
|
};
|
|
|
|
int val_;
|
|
|
|
Conv1DParamID(int val = 0) : val_(val) {}
|
|
|
|
operator int() const { return val_; }
|
|
|
|
static ::testing::internal::ParamGenerator<Conv1DParamID> all()
|
|
|
|
{
|
|
|
|
enum { NUM = (int)CONV_LAST };
|
|
|
|
Conv1DParamID v_[NUM]; for (int i = 0; i < NUM; ++i) { v_[i] = Conv1DParamID(i); } // reduce generated code size
|
|
|
|
return ::testing::ValuesIn(v_, v_ + NUM);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
static inline void PrintTo(const Conv1DParamID& v, std::ostream* os)
|
|
|
|
{
|
|
|
|
CV_Assert((int)v >= 0); CV_Assert((int)v < Conv1DParamID::CONV_LAST);
|
|
|
|
const Conv1DParam_t& p = testConvolution1DConfigs[(int)v];
|
|
|
|
|
|
|
|
*os << "GFLOPS=" << cv::format("%.3f", p.declared_flops * 1e-9)
|
|
|
|
<< ", K=[" << p.kernel << "]"
|
|
|
|
<< ", IN={" << p.shapeIn.dims[0] << ", " << p.shapeIn.dims[1] << ", " << p.shapeIn.dims[2] << "}"
|
|
|
|
<< ", OCN=" << p.outCN;
|
|
|
|
if (p.groups > 1)
|
|
|
|
*os << ", G=" << p.groups;
|
|
|
|
if (p.stride != 1)
|
|
|
|
*os << ", S=" << p.stride;
|
|
|
|
if (p.dilation != 1)
|
|
|
|
*os << ", D=" << p.dilation;
|
|
|
|
if (p.pad[0] != 0 && p.pad[1] != 0 )
|
|
|
|
*os << ", P=(" << p.pad[0] << ", " << p.pad[1] << ")";
|
|
|
|
if (!((std::string)p.padMode).empty())
|
|
|
|
*os << ", PM=" << ((std::string)p.padMode);
|
|
|
|
if (p.hasBias)
|
|
|
|
*os << ", BIAS";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
typedef tuple<Conv1DParamID, tuple<Backend, Target> > Conv1DTestParam_t;
|
|
|
|
typedef TestBaseWithParam<Conv1DTestParam_t> Conv1D;
|
|
|
|
|
|
|
|
PERF_TEST_P_(Conv1D, conv1d)
|
|
|
|
{
|
|
|
|
int test_id = (int)get<0>(GetParam());
|
|
|
|
ASSERT_GE(test_id, 0); ASSERT_LT(test_id, Conv1DParamID::CONV_LAST);
|
|
|
|
const Conv1DParam_t& params = testConvolution1DConfigs[test_id];
|
|
|
|
double declared_flops = params.declared_flops;
|
|
|
|
|
|
|
|
DictValue kernel = DictValue::arrayInt(¶ms.kernel, 1);
|
|
|
|
DictValue stride = DictValue::arrayInt(¶ms.stride, 1);
|
|
|
|
DictValue pad = DictValue::arrayInt(¶ms.pad[0], 2);
|
|
|
|
DictValue dilation = DictValue::arrayInt(¶ms.dilation, 1);
|
|
|
|
|
|
|
|
MatShape inputShape = MatShape(params.shapeIn.dims, params.shapeIn.dims + 3);
|
|
|
|
int outChannels = params.outCN;
|
|
|
|
int groups = params.groups;
|
|
|
|
std::string padMode(params.padMode);
|
|
|
|
|
|
|
|
bool hasBias = params.hasBias;
|
|
|
|
Backend backendId = get<0>(get<1>(GetParam()));
|
|
|
|
Target targetId = get<1>(get<1>(GetParam()));
|
|
|
|
|
|
|
|
if (targetId != DNN_TARGET_CPU)
|
|
|
|
throw SkipTestException("Only CPU is supported");
|
|
|
|
|
|
|
|
int inChannels = inputShape[1];
|
|
|
|
|
|
|
|
int sz[] = {outChannels, inChannels / groups, params.kernel};
|
|
|
|
Mat weights(3, &sz[0], CV_32F);
|
|
|
|
randu(weights, -1.0f, 1.0f);
|
|
|
|
|
|
|
|
LayerParams lp;
|
|
|
|
lp.set("kernel_size", kernel);
|
|
|
|
lp.set("pad", pad);
|
|
|
|
if (!padMode.empty())
|
|
|
|
lp.set("pad_mode", padMode);
|
|
|
|
|
|
|
|
lp.set("stride", stride);
|
|
|
|
lp.set("dilation", dilation);
|
|
|
|
lp.set("num_output", outChannels);
|
|
|
|
lp.set("group", groups);
|
|
|
|
lp.set("bias_term", hasBias);
|
|
|
|
lp.type = "Convolution";
|
|
|
|
lp.name = "testLayer";
|
|
|
|
lp.blobs.push_back(weights);
|
|
|
|
|
|
|
|
if (hasBias)
|
|
|
|
{
|
|
|
|
Mat bias(1, outChannels, CV_32F);
|
|
|
|
randu(bias, -1.0f, 1.0f);
|
|
|
|
lp.blobs.push_back(bias);
|
|
|
|
}
|
|
|
|
|
|
|
|
int inpSz[] = {1, inChannels, inputShape[2]};
|
|
|
|
Mat input(3, &inpSz[0], CV_32F);
|
|
|
|
randu(input, -1.0f, 1.0f);
|
|
|
|
|
|
|
|
Net net;
|
|
|
|
net.addLayerToPrev(lp.name, lp.type, lp);
|
|
|
|
|
|
|
|
net.setInput(input);
|
|
|
|
net.setPreferableBackend(backendId);
|
|
|
|
net.setPreferableTarget(targetId);
|
|
|
|
|
|
|
|
// warmup
|
|
|
|
Mat output = net.forward();
|
|
|
|
|
|
|
|
MatShape netInputShape = shape(input);
|
Merge pull request #24411 from alexlyulkov:al/dnn-type-inference
Added int32, int64 support and type inference to dnn #24411
**Added a type inference to dnn similar to the shape inference, added int32 and int64 support.**
- Added getTypes method for layers that calculates layer outputs types and internals types from inputs types (Similar to getMemoryShapes). By default outputs and internals types = input[0] type
- Added type inference pipeline similar to shape inference pipeline. LayersShapes struct (that is used in shape inference pipeline) now contains both shapes and types
- All layers output blobs are now allocated using the calculated types from the type inference.
- Inputs and constants with int32 and int64 types are not automatically converted into float32 now.
- Added int32 and int64 support for all the layers with indexing and for all the layers required in tests.
Added int32 and int64 support for CUDA:
- Added host<->device data moving for int32 and int64
- Added int32 and int64 support for several layers (just slightly modified CUDA C++ templates)
Passed all the accuracy tests on CPU, OCL, OCL_FP16, CUDA, CUDA_FP16. (except RAFT model)
**CURRENT PROBLEMS**:
- ONNX parser always converts int64 constants and layers attributes to int32, so some models with int64 constants doesn't work (e.g. RAFT). The solution is to disable int64->int32 conversion and fix attributes reading in a lot of ONNX layers parsers (https://github.com/opencv/opencv/issues/25102)
- I didn't add type inference and int support to VULCAN, so it doesn't work at all now.
- Some layers don't support int yet, so some unknown models may not work.
**CURRENT WORKAROUNDS**:
- CPU arg_layer indides are implemented in int32 followed by a int32->int64 conversion (the master branch has the same workaround with int32->float conversion)
- CPU and OCL pooling_layer indices are implemented in float followed by a float->int64 conversion
- CPU gather_layer indices are implemented in int32, so int64 indices are converted to int32 (the master branch has the same workaround with float->int32 conversion)
**DISABLED TESTS**:
- RAFT model
**REMOVED TESTS**:
- Greater_input_dtype_int64 (because it doesn't fit ONNX rules, the whole test is just comparing float tensor with int constant)
**TODO IN NEXT PULL REQUESTS**:
- Add int64 support for ONNX parser
- Add int support for more layers
- Add int support for OCL (currently int layers just run on CPU)
- Add int tests
- Add int support for other backends
2024-03-01 22:07:38 +08:00
|
|
|
cv::dnn::MatType netInputType = input.depth();
|
|
|
|
|
|
|
|
bool fp16 = false;
|
|
|
|
#ifdef HAVE_OPENCL
|
|
|
|
fp16 = ocl::Device::getDefault().isExtensionSupported("cl_khr_fp16");
|
|
|
|
#endif
|
|
|
|
if (netInputType == CV_32F && fp16 && targetId == DNN_TARGET_OPENCL_FP16)
|
|
|
|
netInputType = CV_16F;
|
2020-11-14 06:22:10 +08:00
|
|
|
size_t weightsMemory = 0, blobsMemory = 0;
|
Merge pull request #24411 from alexlyulkov:al/dnn-type-inference
Added int32, int64 support and type inference to dnn #24411
**Added a type inference to dnn similar to the shape inference, added int32 and int64 support.**
- Added getTypes method for layers that calculates layer outputs types and internals types from inputs types (Similar to getMemoryShapes). By default outputs and internals types = input[0] type
- Added type inference pipeline similar to shape inference pipeline. LayersShapes struct (that is used in shape inference pipeline) now contains both shapes and types
- All layers output blobs are now allocated using the calculated types from the type inference.
- Inputs and constants with int32 and int64 types are not automatically converted into float32 now.
- Added int32 and int64 support for all the layers with indexing and for all the layers required in tests.
Added int32 and int64 support for CUDA:
- Added host<->device data moving for int32 and int64
- Added int32 and int64 support for several layers (just slightly modified CUDA C++ templates)
Passed all the accuracy tests on CPU, OCL, OCL_FP16, CUDA, CUDA_FP16. (except RAFT model)
**CURRENT PROBLEMS**:
- ONNX parser always converts int64 constants and layers attributes to int32, so some models with int64 constants doesn't work (e.g. RAFT). The solution is to disable int64->int32 conversion and fix attributes reading in a lot of ONNX layers parsers (https://github.com/opencv/opencv/issues/25102)
- I didn't add type inference and int support to VULCAN, so it doesn't work at all now.
- Some layers don't support int yet, so some unknown models may not work.
**CURRENT WORKAROUNDS**:
- CPU arg_layer indides are implemented in int32 followed by a int32->int64 conversion (the master branch has the same workaround with int32->float conversion)
- CPU and OCL pooling_layer indices are implemented in float followed by a float->int64 conversion
- CPU gather_layer indices are implemented in int32, so int64 indices are converted to int32 (the master branch has the same workaround with float->int32 conversion)
**DISABLED TESTS**:
- RAFT model
**REMOVED TESTS**:
- Greater_input_dtype_int64 (because it doesn't fit ONNX rules, the whole test is just comparing float tensor with int constant)
**TODO IN NEXT PULL REQUESTS**:
- Add int64 support for ONNX parser
- Add int support for more layers
- Add int support for OCL (currently int layers just run on CPU)
- Add int tests
- Add int support for other backends
2024-03-01 22:07:38 +08:00
|
|
|
net.getMemoryConsumption(netInputShape, netInputType, weightsMemory, blobsMemory);
|
|
|
|
int64 flops = net.getFLOPS(netInputShape, netInputType);
|
2020-11-14 06:22:10 +08:00
|
|
|
CV_Assert(flops > 0);
|
|
|
|
|
|
|
|
std::cout
|
|
|
|
<< "IN=" << divUp(input.total() * input.elemSize(), 1u<<10) << " Kb " << netInputShape
|
|
|
|
<< " OUT=" << divUp(output.total() * output.elemSize(), 1u<<10) << " Kb " << shape(output)
|
|
|
|
<< " Weights(parameters): " << divUp(weightsMemory, 1u<<10) << " Kb"
|
|
|
|
<< " MFLOPS=" << flops * 1e-6 << std::endl;
|
|
|
|
|
|
|
|
TEST_CYCLE()
|
|
|
|
{
|
|
|
|
Mat res = net.forward();
|
|
|
|
}
|
|
|
|
EXPECT_NEAR(flops, declared_flops, declared_flops * 1e-6);
|
|
|
|
SANITY_CHECK_NOTHING();
|
|
|
|
}
|
|
|
|
|
|
|
|
INSTANTIATE_TEST_CASE_P(/**/, Conv1D, Combine(
|
|
|
|
Conv1DParamID::all(),
|
2023-10-13 21:53:18 +08:00
|
|
|
dnnBackendsAndTargets(/* withInferenceEngine = */false, /* obsolete_withHalide = */false) // defined in ../test/test_common.hpp
|
2020-11-14 06:22:10 +08:00
|
|
|
));
|
|
|
|
|
|
|
|
} // namespace
|