GAPI (fluid): RGB to YUV optimization (#13174) * GAPI (fluid): RGB to YUV: activate performance tests * GAPI (fluid): speedup 4-8x RGB-to-YUV, 2.5x YUV to RGB with int16 arithmetic * GAPI (fluid): RGB <--> YUV: fixed compiler warning * GAPI (fluid): RGB <--> YUV: additional speedup 2-3x times (10-15x over original) via manual CV_SIMD * GAPI (fluid): RGB <--> YUV: dynamic CV_SIMD dispatching
563 lines
19 KiB
C++
563 lines
19 KiB
C++
// 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.
|
|
//
|
|
// Copyright (C) 2018 Intel Corporation
|
|
|
|
// NB: allow including this *.hpp several times!
|
|
// #pragma once -- don't: this file is NOT once!
|
|
|
|
#if !defined(GAPI_STANDALONE)
|
|
|
|
#include "opencv2/gapi/own/saturate.hpp"
|
|
|
|
#include "opencv2/core.hpp"
|
|
#include "opencv2/core/hal/intrin.hpp"
|
|
|
|
#include <cstdint>
|
|
|
|
#ifdef __GNUC__
|
|
# pragma GCC diagnostic push
|
|
# pragma GCC diagnostic ignored "-Wstrict-overflow"
|
|
#endif
|
|
|
|
using cv::gapi::own::saturate;
|
|
|
|
namespace cv {
|
|
namespace gapi {
|
|
namespace fluid {
|
|
|
|
CV_CPU_OPTIMIZATION_NAMESPACE_BEGIN
|
|
|
|
//----------------------------------
|
|
//
|
|
// Fluid kernels: RGB2Gray, BGR2Gray
|
|
//
|
|
//----------------------------------
|
|
|
|
void run_rgb2gray_impl(uchar out[], const uchar in[], int width,
|
|
float coef_r, float coef_g, float coef_b);
|
|
|
|
//--------------------------------------
|
|
//
|
|
// Fluid kernels: RGB-to-YUV, YUV-to-RGB
|
|
//
|
|
//--------------------------------------
|
|
|
|
void run_rgb2yuv_impl(uchar out[], const uchar in[], int width, const float coef[5]);
|
|
|
|
void run_yuv2rgb_impl(uchar out[], const uchar in[], int width, const float coef[4]);
|
|
|
|
//---------------------
|
|
//
|
|
// Fluid kernels: Sobel
|
|
//
|
|
//---------------------
|
|
|
|
#define RUN_SOBEL_ROW(DST, SRC) \
|
|
void run_sobel_row(DST out[], const SRC *in[], int width, int chan, \
|
|
const float kx[], const float ky[], int border, \
|
|
float scale, float delta, float *buf[], \
|
|
int y, int y0);
|
|
|
|
RUN_SOBEL_ROW(uchar , uchar )
|
|
RUN_SOBEL_ROW(ushort, ushort)
|
|
RUN_SOBEL_ROW( short, uchar )
|
|
RUN_SOBEL_ROW( short, ushort)
|
|
RUN_SOBEL_ROW( short, short)
|
|
RUN_SOBEL_ROW( float, uchar )
|
|
RUN_SOBEL_ROW( float, ushort)
|
|
RUN_SOBEL_ROW( float, short)
|
|
RUN_SOBEL_ROW( float, float)
|
|
|
|
#undef RUN_SOBEL_ROW
|
|
|
|
//----------------------------------------------------------------------
|
|
|
|
#ifndef CV_CPU_OPTIMIZATION_DECLARATIONS_ONLY
|
|
|
|
//----------------------------------
|
|
//
|
|
// Fluid kernels: RGB2Gray, BGR2Gray
|
|
//
|
|
//----------------------------------
|
|
|
|
void run_rgb2gray_impl(uchar out[], const uchar in[], int width,
|
|
float coef_r, float coef_g, float coef_b)
|
|
{
|
|
// assume:
|
|
// - coefficients are less than 1
|
|
// - and their sum equals 1
|
|
|
|
constexpr int unity = 1 << 16; // Q0.0.16 inside ushort:
|
|
ushort rc = static_cast<ushort>(coef_r * unity + 0.5f);
|
|
ushort gc = static_cast<ushort>(coef_g * unity + 0.5f);
|
|
ushort bc = static_cast<ushort>(coef_b * unity + 0.5f);
|
|
|
|
GAPI_Assert(rc + gc + bc <= unity);
|
|
GAPI_Assert(rc + gc + bc >= USHRT_MAX);
|
|
|
|
#if CV_SIMD
|
|
constexpr int nlanes = v_uint8::nlanes;
|
|
if (width >= nlanes)
|
|
{
|
|
for (int w=0; w < width; )
|
|
{
|
|
// process main part of pixels row
|
|
for ( ; w <= width - nlanes; w += nlanes)
|
|
{
|
|
v_uint8 r, g, b;
|
|
v_load_deinterleave(&in[3*w], r, g, b);
|
|
|
|
v_uint16 r0, r1, g0, g1, b0, b1;
|
|
v_expand(r, r0, r1);
|
|
v_expand(g, g0, g1);
|
|
v_expand(b, b0, b1);
|
|
|
|
v_uint16 y0, y1;
|
|
static const ushort half = 1 << 7; // Q0.8.8
|
|
y0 = (v_mul_hi(r0 << 8, vx_setall_u16(rc)) +
|
|
v_mul_hi(g0 << 8, vx_setall_u16(gc)) +
|
|
v_mul_hi(b0 << 8, vx_setall_u16(bc)) +
|
|
vx_setall_u16(half)) >> 8;
|
|
y1 = (v_mul_hi(r1 << 8, vx_setall_u16(rc)) +
|
|
v_mul_hi(g1 << 8, vx_setall_u16(gc)) +
|
|
v_mul_hi(b1 << 8, vx_setall_u16(bc)) +
|
|
vx_setall_u16(half)) >> 8;
|
|
|
|
v_uint8 y;
|
|
y = v_pack(y0, y1);
|
|
v_store(&out[w], y);
|
|
}
|
|
|
|
// process tail (if any)
|
|
if (w < width)
|
|
{
|
|
GAPI_DbgAssert(width - nlanes >= 0);
|
|
w = width - nlanes;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
for (int w=0; w < width; w++)
|
|
{
|
|
uchar r = in[3*w ];
|
|
uchar g = in[3*w + 1];
|
|
uchar b = in[3*w + 2];
|
|
|
|
static const int half = 1 << 15; // Q0.0.16
|
|
ushort y = (r*rc + b*bc + g*gc + half) >> 16;
|
|
out[w] = static_cast<uchar>(y);
|
|
}
|
|
}
|
|
|
|
//--------------------------------------
|
|
//
|
|
// Fluid kernels: RGB-to-YUV, YUV-to-RGB
|
|
//
|
|
//--------------------------------------
|
|
|
|
void run_rgb2yuv_impl(uchar out[], const uchar in[], int width, const float coef[5])
|
|
{
|
|
ushort c0 = static_cast<ushort>(coef[0]*(1 << 16) + 0.5f); // Q0.0.16 un-signed
|
|
ushort c1 = static_cast<ushort>(coef[1]*(1 << 16) + 0.5f);
|
|
ushort c2 = static_cast<ushort>(coef[2]*(1 << 16) + 0.5f);
|
|
short c3 = static_cast<short>(coef[3]*(1 << 12) + 0.5f); // Q1.0.12 signed
|
|
short c4 = static_cast<short>(coef[4]*(1 << 12) + 0.5f);
|
|
|
|
int w = 0;
|
|
|
|
#if CV_SIMD
|
|
static const int nlanes = v_uint8::nlanes;
|
|
for ( ; w <= width - nlanes; w += nlanes)
|
|
{
|
|
v_uint8 r, g, b;
|
|
v_load_deinterleave(&in[3*w], r, g, b);
|
|
|
|
v_uint16 _r0, _r1, _g0, _g1, _b0, _b1;
|
|
v_expand(r, _r0, _r1);
|
|
v_expand(g, _g0, _g1);
|
|
v_expand(b, _b0, _b1);
|
|
|
|
_r0 = _r0 << 7; // Q0.9.7 un-signed
|
|
_r1 = _r1 << 7;
|
|
_g0 = _g0 << 7;
|
|
_g1 = _g1 << 7;
|
|
_b0 = _b0 << 7;
|
|
_b1 = _b1 << 7;
|
|
|
|
v_uint16 _y0, _y1;
|
|
_y0 = v_mul_hi(vx_setall_u16(c0), _r0) // Q0.9.7
|
|
+ v_mul_hi(vx_setall_u16(c1), _g0)
|
|
+ v_mul_hi(vx_setall_u16(c2), _b0);
|
|
_y1 = v_mul_hi(vx_setall_u16(c0), _r1)
|
|
+ v_mul_hi(vx_setall_u16(c1), _g1)
|
|
+ v_mul_hi(vx_setall_u16(c2), _b1);
|
|
|
|
v_int16 r0, r1, b0, b1, y0, y1;
|
|
r0 = v_reinterpret_as_s16(_r0); // Q1.8.7 signed
|
|
r1 = v_reinterpret_as_s16(_r1);
|
|
b0 = v_reinterpret_as_s16(_b0);
|
|
b1 = v_reinterpret_as_s16(_b1);
|
|
y0 = v_reinterpret_as_s16(_y0);
|
|
y1 = v_reinterpret_as_s16(_y1);
|
|
|
|
v_int16 u0, u1, v0, v1;
|
|
u0 = v_mul_hi(vx_setall_s16(c3), b0 - y0); // Q1.12.3
|
|
u1 = v_mul_hi(vx_setall_s16(c3), b1 - y1);
|
|
v0 = v_mul_hi(vx_setall_s16(c4), r0 - y0);
|
|
v1 = v_mul_hi(vx_setall_s16(c4), r1 - y1);
|
|
|
|
v_uint8 y, u, v;
|
|
y = v_pack((_y0 + vx_setall_u16(1 << 6)) >> 7,
|
|
(_y1 + vx_setall_u16(1 << 6)) >> 7);
|
|
u = v_pack_u((u0 + vx_setall_s16(257 << 2)) >> 3, // 257 << 2 = 128.5 * (1 << 3)
|
|
(u1 + vx_setall_s16(257 << 2)) >> 3);
|
|
v = v_pack_u((v0 + vx_setall_s16(257 << 2)) >> 3,
|
|
(v1 + vx_setall_s16(257 << 2)) >> 3);
|
|
|
|
v_store_interleave(&out[3*w], y, u, v);
|
|
}
|
|
#endif
|
|
|
|
for ( ; w < width; w++)
|
|
{
|
|
short r = in[3*w ] << 7; // Q1.8.7 signed
|
|
short g = in[3*w + 1] << 7;
|
|
short b = in[3*w + 2] << 7;
|
|
short y = (c0*r + c1*g + c2*b) >> 16; // Q1.8.7
|
|
short u = c3*(b - y) >> 16; // Q1.12.3
|
|
short v = c4*(r - y) >> 16;
|
|
out[3*w ] = static_cast<uchar>((y + (1 << 6)) >> 7);
|
|
out[3*w + 1] = saturate<uchar>((u + (128 << 3) + (1 << 2)) >> 3);
|
|
out[3*w + 2] = saturate<uchar>((v + (128 << 3) + (1 << 2)) >> 3);
|
|
}
|
|
}
|
|
|
|
void run_yuv2rgb_impl(uchar out[], const uchar in[], int width, const float coef[4])
|
|
{
|
|
short c0 = static_cast<short>(coef[0] * (1 << 12) + 0.5f); // Q1.3.12
|
|
short c1 = static_cast<short>(coef[1] * (1 << 12) + 0.5f);
|
|
short c2 = static_cast<short>(coef[2] * (1 << 12) + 0.5f);
|
|
short c3 = static_cast<short>(coef[3] * (1 << 12) + 0.5f);
|
|
|
|
int w = 0;
|
|
|
|
#if CV_SIMD
|
|
static const int nlanes = v_uint8::nlanes;
|
|
for ( ; w <= width - nlanes; w += nlanes)
|
|
{
|
|
v_uint8 y, u, v;
|
|
v_load_deinterleave(&in[3*w], y, u, v);
|
|
|
|
v_uint16 _y0, _y1, _u0, _u1, _v0, _v1;
|
|
v_expand(y, _y0, _y1);
|
|
v_expand(u, _u0, _u1);
|
|
v_expand(v, _v0, _v1);
|
|
|
|
v_int16 y0, y1, u0, u1, v0, v1;
|
|
y0 = v_reinterpret_as_s16(_y0);
|
|
y1 = v_reinterpret_as_s16(_y1);
|
|
u0 = v_reinterpret_as_s16(_u0);
|
|
u1 = v_reinterpret_as_s16(_u1);
|
|
v0 = v_reinterpret_as_s16(_v0);
|
|
v1 = v_reinterpret_as_s16(_v1);
|
|
|
|
y0 = y0 << 3; // Q1.12.3
|
|
y1 = y1 << 3;
|
|
u0 = (u0 - vx_setall_s16(128)) << 7; // Q1.8.7
|
|
u1 = (u1 - vx_setall_s16(128)) << 7;
|
|
v0 = (v0 - vx_setall_s16(128)) << 7;
|
|
v1 = (v1 - vx_setall_s16(128)) << 7;
|
|
|
|
v_int16 r0, r1, g0, g1, b0, b1;
|
|
r0 = y0 + v_mul_hi(vx_setall_s16(c0), v0); // Q1.12.3
|
|
r1 = y1 + v_mul_hi(vx_setall_s16(c0), v1);
|
|
g0 = y0 + v_mul_hi(vx_setall_s16(c1), u0)
|
|
+ v_mul_hi(vx_setall_s16(c2), v0);
|
|
g1 = y1 + v_mul_hi(vx_setall_s16(c1), u1)
|
|
+ v_mul_hi(vx_setall_s16(c2), v1);
|
|
b0 = y0 + v_mul_hi(vx_setall_s16(c3), u0);
|
|
b1 = y1 + v_mul_hi(vx_setall_s16(c3), u1);
|
|
|
|
v_uint8 r, g, b;
|
|
r = v_pack_u((r0 + vx_setall_s16(1 << 2)) >> 3,
|
|
(r1 + vx_setall_s16(1 << 2)) >> 3);
|
|
g = v_pack_u((g0 + vx_setall_s16(1 << 2)) >> 3,
|
|
(g1 + vx_setall_s16(1 << 2)) >> 3);
|
|
b = v_pack_u((b0 + vx_setall_s16(1 << 2)) >> 3,
|
|
(b1 + vx_setall_s16(1 << 2)) >> 3);
|
|
|
|
v_store_interleave(&out[3*w], r, g, b);
|
|
}
|
|
#endif
|
|
|
|
for ( ; w < width; w++)
|
|
{
|
|
short y = in[3*w ] << 3; // Q1.12.3
|
|
short u = (in[3*w + 1] - 128) << 7; // Q1.8.7
|
|
short v = (in[3*w + 2] - 128) << 7;
|
|
short r = y + ( c0*v >> 16); // Q1.12.3
|
|
short g = y + ((c1*u + c2*v) >> 16);
|
|
short b = y + ((c3*u ) >> 16);
|
|
out[3*w ] = saturate<uchar>((r + (1 << 2)) >> 3);
|
|
out[3*w + 1] = saturate<uchar>((g + (1 << 2)) >> 3);
|
|
out[3*w + 2] = saturate<uchar>((b + (1 << 2)) >> 3);
|
|
}
|
|
}
|
|
|
|
//---------------------
|
|
//
|
|
// Fluid kernels: Sobel
|
|
//
|
|
//---------------------
|
|
|
|
// Sobel 3x3: vertical pass
|
|
template<bool noscale, typename DST>
|
|
static void run_sobel3x3_vert(DST out[], int length, const float ky[],
|
|
float scale, float delta, const int r[], float *buf[])
|
|
{
|
|
float ky0 = ky[0],
|
|
ky1 = ky[1],
|
|
ky2 = ky[2];
|
|
|
|
int r0 = r[0],
|
|
r1 = r[1],
|
|
r2 = r[2];
|
|
|
|
#if CV_SIMD
|
|
// for floating-point output,
|
|
// manual vectoring may be not better than compiler's optimization
|
|
#define EXPLICIT_SIMD_32F 0 // 1=vectorize 32f case explicitly, 0=don't
|
|
#if EXPLICIT_SIMD_32F
|
|
if (std::is_same<DST, float>::value && length >= v_int16::nlanes)
|
|
{
|
|
constexpr static int nlanes = v_float32::nlanes;
|
|
|
|
for (int l=0; l < length; )
|
|
{
|
|
for (; l <= length - nlanes; l += nlanes)
|
|
{
|
|
v_float32 sum = vx_load(&buf[r0][l]) * vx_setall_f32(ky0);
|
|
sum = v_fma(vx_load(&buf[r1][l]), vx_setall_f32(ky1), sum);
|
|
sum = v_fma(vx_load(&buf[r2][l]), vx_setall_f32(ky2), sum);
|
|
|
|
if (!noscale)
|
|
{
|
|
sum = v_fma(sum, vx_setall_f32(scale), vx_setall_f32(delta));
|
|
}
|
|
|
|
v_store(reinterpret_cast<float*>(&out[l]), sum);
|
|
}
|
|
|
|
if (l < length)
|
|
{
|
|
// tail: recalculate last pixels
|
|
GAPI_DbgAssert(length >= nlanes);
|
|
l = length - nlanes;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
if ((std::is_same<DST, short>::value || std::is_same<DST, ushort>::value)
|
|
&& length >= v_int16::nlanes)
|
|
{
|
|
constexpr static int nlanes = v_int16::nlanes;
|
|
|
|
for (int l=0; l < length; )
|
|
{
|
|
for (; l <= length - nlanes; l += nlanes)
|
|
{
|
|
v_float32 sum0 = vx_load(&buf[r0][l]) * vx_setall_f32(ky0);
|
|
sum0 = v_fma(vx_load(&buf[r1][l]), vx_setall_f32(ky1), sum0);
|
|
sum0 = v_fma(vx_load(&buf[r2][l]), vx_setall_f32(ky2), sum0);
|
|
|
|
v_float32 sum1 = vx_load(&buf[r0][l + nlanes/2]) * vx_setall_f32(ky0);
|
|
sum1 = v_fma(vx_load(&buf[r1][l + nlanes/2]), vx_setall_f32(ky1), sum1);
|
|
sum1 = v_fma(vx_load(&buf[r2][l + nlanes/2]), vx_setall_f32(ky2), sum1);
|
|
|
|
if (!noscale)
|
|
{
|
|
sum0 = v_fma(sum0, vx_setall_f32(scale), vx_setall_f32(delta));
|
|
sum1 = v_fma(sum1, vx_setall_f32(scale), vx_setall_f32(delta));
|
|
}
|
|
|
|
v_int32 isum0 = v_round(sum0),
|
|
isum1 = v_round(sum1);
|
|
|
|
if (std::is_same<DST, short>::value)
|
|
{
|
|
// signed short
|
|
v_int16 res = v_pack(isum0, isum1);
|
|
v_store(reinterpret_cast<short*>(&out[l]), res);
|
|
} else
|
|
{
|
|
// unsigned short
|
|
v_uint16 res = v_pack_u(isum0, isum1);
|
|
v_store(reinterpret_cast<ushort*>(&out[l]), res);
|
|
}
|
|
}
|
|
|
|
if (l < length)
|
|
{
|
|
// tail: recalculate last pixels
|
|
GAPI_DbgAssert(length >= nlanes);
|
|
l = length - nlanes;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
if (std::is_same<DST, uchar>::value && length >= v_uint8::nlanes)
|
|
{
|
|
constexpr static int nlanes = v_uint8::nlanes;
|
|
|
|
for (int l=0; l < length; )
|
|
{
|
|
for (; l <= length - nlanes; l += nlanes)
|
|
{
|
|
v_float32 sum0 = vx_load(&buf[r0][l]) * vx_setall_f32(ky0);
|
|
sum0 = v_fma(vx_load(&buf[r1][l]), vx_setall_f32(ky1), sum0);
|
|
sum0 = v_fma(vx_load(&buf[r2][l]), vx_setall_f32(ky2), sum0);
|
|
|
|
v_float32 sum1 = vx_load(&buf[r0][l + nlanes/4]) * vx_setall_f32(ky0);
|
|
sum1 = v_fma(vx_load(&buf[r1][l + nlanes/4]), vx_setall_f32(ky1), sum1);
|
|
sum1 = v_fma(vx_load(&buf[r2][l + nlanes/4]), vx_setall_f32(ky2), sum1);
|
|
|
|
v_float32 sum2 = vx_load(&buf[r0][l + 2*nlanes/4]) * vx_setall_f32(ky0);
|
|
sum2 = v_fma(vx_load(&buf[r1][l + 2*nlanes/4]), vx_setall_f32(ky1), sum2);
|
|
sum2 = v_fma(vx_load(&buf[r2][l + 2*nlanes/4]), vx_setall_f32(ky2), sum2);
|
|
|
|
v_float32 sum3 = vx_load(&buf[r0][l + 3*nlanes/4]) * vx_setall_f32(ky0);
|
|
sum3 = v_fma(vx_load(&buf[r1][l + 3*nlanes/4]), vx_setall_f32(ky1), sum3);
|
|
sum3 = v_fma(vx_load(&buf[r2][l + 3*nlanes/4]), vx_setall_f32(ky2), sum3);
|
|
|
|
if (!noscale)
|
|
{
|
|
sum0 = v_fma(sum0, vx_setall_f32(scale), vx_setall_f32(delta));
|
|
sum1 = v_fma(sum1, vx_setall_f32(scale), vx_setall_f32(delta));
|
|
sum2 = v_fma(sum2, vx_setall_f32(scale), vx_setall_f32(delta));
|
|
sum3 = v_fma(sum3, vx_setall_f32(scale), vx_setall_f32(delta));
|
|
}
|
|
|
|
v_int32 isum0 = v_round(sum0),
|
|
isum1 = v_round(sum1),
|
|
isum2 = v_round(sum2),
|
|
isum3 = v_round(sum3);
|
|
|
|
v_int16 ires0 = v_pack(isum0, isum1),
|
|
ires1 = v_pack(isum2, isum3);
|
|
|
|
v_uint8 res = v_pack_u(ires0, ires1);
|
|
v_store(reinterpret_cast<uchar*>(&out[l]), res);
|
|
}
|
|
|
|
if (l < length)
|
|
{
|
|
// tail: recalculate last pixels
|
|
GAPI_DbgAssert(length >= nlanes);
|
|
l = length - nlanes;
|
|
}
|
|
}
|
|
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
// reference code
|
|
for (int l=0; l < length; l++)
|
|
{
|
|
float sum = buf[r0][l]*ky0 + buf[r1][l]*ky1 + buf[r2][l]*ky2;
|
|
|
|
if (!noscale)
|
|
{
|
|
sum = sum*scale + delta;
|
|
}
|
|
|
|
out[l] = cv::gapi::own::saturate<DST>(sum, rintf);
|
|
}
|
|
}
|
|
|
|
template<typename DST, typename SRC>
|
|
static void run_sobel_impl(DST out[], const SRC *in[], int width, int chan,
|
|
const float kx[], const float ky[], int border,
|
|
float scale, float delta, float *buf[],
|
|
int y, int y0)
|
|
{
|
|
int r[3];
|
|
r[0] = (y - y0) % 3; // buf[r[0]]: previous
|
|
r[1] = (y - y0 + 1) % 3; // this
|
|
r[2] = (y - y0 + 2) % 3; // next row
|
|
|
|
int length = width * chan;
|
|
|
|
// horizontal pass
|
|
|
|
// full horizontal pass is needed only if very 1st row in ROI;
|
|
// for 2nd and further rows, it is enough to convolve only the
|
|
// "next" row - as we can reuse buffers from previous calls to
|
|
// this kernel (note that Fluid processes rows consequently)
|
|
int k0 = (y == y0)? 0: 2;
|
|
|
|
for (int k = k0; k < 3; k++)
|
|
{
|
|
// previous, this , next pixel
|
|
const SRC *s[3] = {in[k] - border*chan , in[k], in[k] + border*chan};
|
|
|
|
// rely on compiler vectoring
|
|
for (int l=0; l < length; l++)
|
|
{
|
|
buf[r[k]][l] = s[0][l]*kx[0] + s[1][l]*kx[1] + s[2][l]*kx[2];
|
|
}
|
|
}
|
|
|
|
// vertical pass
|
|
if (scale == 1 && delta == 0)
|
|
{
|
|
constexpr static bool noscale = true; // omit scaling
|
|
run_sobel3x3_vert<noscale, DST>(out, length, ky, scale, delta, r, buf);
|
|
} else
|
|
{
|
|
constexpr static bool noscale = false; // do scaling
|
|
run_sobel3x3_vert<noscale, DST>(out, length, ky, scale, delta, r, buf);
|
|
}
|
|
}
|
|
|
|
#define RUN_SOBEL_ROW(DST, SRC) \
|
|
void run_sobel_row(DST out[], const SRC *in[], int width, int chan, \
|
|
const float kx[], const float ky[], int border, \
|
|
float scale, float delta, float *buf[], \
|
|
int y, int y0) \
|
|
{ \
|
|
run_sobel_impl(out, in, width, chan, kx, ky, border, scale, delta, buf,y, y0); \
|
|
}
|
|
|
|
RUN_SOBEL_ROW(uchar , uchar )
|
|
RUN_SOBEL_ROW(ushort, ushort)
|
|
RUN_SOBEL_ROW( short, uchar )
|
|
RUN_SOBEL_ROW( short, ushort)
|
|
RUN_SOBEL_ROW( short, short)
|
|
RUN_SOBEL_ROW( float, uchar )
|
|
RUN_SOBEL_ROW( float, ushort)
|
|
RUN_SOBEL_ROW( float, short)
|
|
RUN_SOBEL_ROW( float, float)
|
|
|
|
#undef RUN_SOBEL_ROW
|
|
|
|
#endif // CV_CPU_OPTIMIZATION_DECLARATIONS_ONLY
|
|
|
|
CV_CPU_OPTIMIZATION_NAMESPACE_END
|
|
|
|
} // namespace fluid
|
|
} // namespace gapi
|
|
} // namespace cv
|
|
|
|
#endif // !defined(GAPI_STANDALONE)
|