Path: blob/master/modules/core/src/convert_scale.cpp
16337 views
// This file is part of OpenCV project.1// It is subject to the license terms in the LICENSE file found in the top-level directory2// of this distribution and at http://opencv.org/license.html345#include "precomp.hpp"6#include "opencl_kernels_core.hpp"7#include "convert.hpp"89/****************************************************************************************\10* convertScale[Abs] *11\****************************************************************************************/1213namespace cv14{1516template<typename _Ts, typename _Td> inline void17cvtabs_32f( const _Ts* src, size_t sstep, _Td* dst, size_t dstep,18Size size, float a, float b )19{20#if CV_SIMD21v_float32 va = vx_setall_f32(a), vb = vx_setall_f32(b);22const int VECSZ = v_float32::nlanes*2;23#endif24sstep /= sizeof(src[0]);25dstep /= sizeof(dst[0]);2627for( int i = 0; i < size.height; i++, src += sstep, dst += dstep )28{29int j = 0;30#if CV_SIMD31for( ; j < size.width; j += VECSZ )32{33if( j > size.width - VECSZ )34{35if( j == 0 || src == (_Ts*)dst )36break;37j = size.width - VECSZ;38}39v_float32 v0, v1;40vx_load_pair_as(src + j, v0, v1);41v0 = v_fma(v0, va, vb);42v1 = v_fma(v1, va, vb);43v_store_pair_as(dst + j, v_abs(v0), v_abs(v1));44}45#endif46for( ; j < size.width; j++ )47dst[j] = saturate_cast<_Td>(std::abs(src[j]*a + b));48}49}5051// variant for convrsions 16f <-> ... w/o unrolling52template<typename _Ts, typename _Td> inline void53cvtabs1_32f( const _Ts* src, size_t sstep, _Td* dst, size_t dstep,54Size size, float a, float b )55{56#if CV_SIMD57v_float32 va = vx_setall_f32(a), vb = vx_setall_f32(b);58const int VECSZ = v_float32::nlanes*2;59#endif60sstep /= sizeof(src[0]);61dstep /= sizeof(dst[0]);6263for( int i = 0; i < size.height; i++, src += sstep, dst += dstep )64{65int j = 0;66#if CV_SIMD67for( ; j < size.width; j += VECSZ )68{69if( j > size.width - VECSZ )70{71if( j == 0 || src == (_Ts*)dst )72break;73j = size.width - VECSZ;74}75v_float32 v0;76vx_load_as(src + j, v0);77v0 = v_fma(v0, va, vb);78v_store_as(dst + j, v_abs(v0));79}80#endif81for( ; j < size.width; j++ )82dst[j] = saturate_cast<_Td>(src[j]*a + b);83}84}8586template<typename _Ts, typename _Td> inline void87cvt_32f( const _Ts* src, size_t sstep, _Td* dst, size_t dstep,88Size size, float a, float b )89{90#if CV_SIMD91v_float32 va = vx_setall_f32(a), vb = vx_setall_f32(b);92const int VECSZ = v_float32::nlanes*2;93#endif94sstep /= sizeof(src[0]);95dstep /= sizeof(dst[0]);9697for( int i = 0; i < size.height; i++, src += sstep, dst += dstep )98{99int j = 0;100#if CV_SIMD101for( ; j < size.width; j += VECSZ )102{103if( j > size.width - VECSZ )104{105if( j == 0 || src == (_Ts*)dst )106break;107j = size.width - VECSZ;108}109v_float32 v0, v1;110vx_load_pair_as(src + j, v0, v1);111v0 = v_fma(v0, va, vb);112v1 = v_fma(v1, va, vb);113v_store_pair_as(dst + j, v0, v1);114}115#endif116for( ; j < size.width; j++ )117dst[j] = saturate_cast<_Td>(src[j]*a + b);118}119}120121// variant for convrsions 16f <-> ... w/o unrolling122template<typename _Ts, typename _Td> inline void123cvt1_32f( const _Ts* src, size_t sstep, _Td* dst, size_t dstep,124Size size, float a, float b )125{126#if CV_SIMD127v_float32 va = vx_setall_f32(a), vb = vx_setall_f32(b);128const int VECSZ = v_float32::nlanes;129#endif130sstep /= sizeof(src[0]);131dstep /= sizeof(dst[0]);132133for( int i = 0; i < size.height; i++, src += sstep, dst += dstep )134{135int j = 0;136#if CV_SIMD137for( ; j < size.width; j += VECSZ )138{139if( j > size.width - VECSZ )140{141if( j == 0 || src == (_Ts*)dst )142break;143j = size.width - VECSZ;144}145v_float32 v0;146vx_load_as(src + j, v0);147v0 = v_fma(v0, va, vb);148v_store_as(dst + j, v0);149}150#endif151for( ; j < size.width; j++ )152dst[j] = saturate_cast<_Td>(src[j]*a + b);153}154}155156157template<typename _Ts, typename _Td> inline void158cvt_64f( const _Ts* src, size_t sstep, _Td* dst, size_t dstep,159Size size, double a, double b )160{161#if CV_SIMD_64F162v_float64 va = vx_setall_f64(a), vb = vx_setall_f64(b);163const int VECSZ = v_float64::nlanes*2;164#endif165sstep /= sizeof(src[0]);166dstep /= sizeof(dst[0]);167168for( int i = 0; i < size.height; i++, src += sstep, dst += dstep )169{170int j = 0;171#if CV_SIMD_64F172for( ; j < size.width; j += VECSZ )173{174if( j > size.width - VECSZ )175{176if( j == 0 || src == (_Ts*)dst )177break;178j = size.width - VECSZ;179}180v_float64 v0, v1;181vx_load_pair_as(src + j, v0, v1);182v0 = v_fma(v0, va, vb);183v1 = v_fma(v1, va, vb);184v_store_pair_as(dst + j, v0, v1);185}186#endif187for( ; j < size.width; j++ )188dst[j] = saturate_cast<_Td>(src[j]*a + b);189}190}191192//==================================================================================================193194#define DEF_CVT_SCALE_ABS_FUNC(suffix, cvt, stype, dtype, wtype) \195static void cvtScaleAbs##suffix( const stype* src, size_t sstep, const uchar*, size_t, \196dtype* dst, size_t dstep, Size size, double* scale) \197{ \198cvt(src, sstep, dst, dstep, size, (wtype)scale[0], (wtype)scale[1]); \199}200201202#define DEF_CVT_SCALE_FUNC(suffix, cvt, stype, dtype, wtype) \203static void cvtScale##suffix( const stype* src, size_t sstep, const uchar*, size_t, \204dtype* dst, size_t dstep, Size size, double* scale) \205{ \206cvt(src, sstep, dst, dstep, size, (wtype)scale[0], (wtype)scale[1]); \207}208209DEF_CVT_SCALE_ABS_FUNC(8u, cvtabs_32f, uchar, uchar, float)210DEF_CVT_SCALE_ABS_FUNC(8s8u, cvtabs_32f, schar, uchar, float)211DEF_CVT_SCALE_ABS_FUNC(16u8u, cvtabs_32f, ushort, uchar, float)212DEF_CVT_SCALE_ABS_FUNC(16s8u, cvtabs_32f, short, uchar, float)213DEF_CVT_SCALE_ABS_FUNC(32s8u, cvtabs_32f, int, uchar, float)214DEF_CVT_SCALE_ABS_FUNC(32f8u, cvtabs_32f, float, uchar, float)215DEF_CVT_SCALE_ABS_FUNC(64f8u, cvtabs_32f, double, uchar, float)216217DEF_CVT_SCALE_FUNC(8u, cvt_32f, uchar, uchar, float)218DEF_CVT_SCALE_FUNC(8s8u, cvt_32f, schar, uchar, float)219DEF_CVT_SCALE_FUNC(16u8u, cvt_32f, ushort, uchar, float)220DEF_CVT_SCALE_FUNC(16s8u, cvt_32f, short, uchar, float)221DEF_CVT_SCALE_FUNC(32s8u, cvt_32f, int, uchar, float)222DEF_CVT_SCALE_FUNC(32f8u, cvt_32f, float, uchar, float)223DEF_CVT_SCALE_FUNC(64f8u, cvt_32f, double, uchar, float)224DEF_CVT_SCALE_FUNC(16f8u, cvt_32f, float16_t, uchar, float)225226DEF_CVT_SCALE_FUNC(8u8s, cvt_32f, uchar, schar, float)227DEF_CVT_SCALE_FUNC(8s, cvt_32f, schar, schar, float)228DEF_CVT_SCALE_FUNC(16u8s, cvt_32f, ushort, schar, float)229DEF_CVT_SCALE_FUNC(16s8s, cvt_32f, short, schar, float)230DEF_CVT_SCALE_FUNC(32s8s, cvt_32f, int, schar, float)231DEF_CVT_SCALE_FUNC(32f8s, cvt_32f, float, schar, float)232DEF_CVT_SCALE_FUNC(64f8s, cvt_32f, double, schar, float)233DEF_CVT_SCALE_FUNC(16f8s, cvt_32f, float16_t, schar, float)234235DEF_CVT_SCALE_FUNC(8u16u, cvt_32f, uchar, ushort, float)236DEF_CVT_SCALE_FUNC(8s16u, cvt_32f, schar, ushort, float)237DEF_CVT_SCALE_FUNC(16u, cvt_32f, ushort, ushort, float)238DEF_CVT_SCALE_FUNC(16s16u, cvt_32f, short, ushort, float)239DEF_CVT_SCALE_FUNC(32s16u, cvt_32f, int, ushort, float)240DEF_CVT_SCALE_FUNC(32f16u, cvt_32f, float, ushort, float)241DEF_CVT_SCALE_FUNC(64f16u, cvt_32f, double, ushort, float)242DEF_CVT_SCALE_FUNC(16f16u, cvt1_32f, float16_t, ushort, float)243244DEF_CVT_SCALE_FUNC(8u16s, cvt_32f, uchar, short, float)245DEF_CVT_SCALE_FUNC(8s16s, cvt_32f, schar, short, float)246DEF_CVT_SCALE_FUNC(16u16s, cvt_32f, ushort, short, float)247DEF_CVT_SCALE_FUNC(16s, cvt_32f, short, short, float)248DEF_CVT_SCALE_FUNC(32s16s, cvt_32f, int, short, float)249DEF_CVT_SCALE_FUNC(32f16s, cvt_32f, float, short, float)250DEF_CVT_SCALE_FUNC(64f16s, cvt_32f, double, short, float)251DEF_CVT_SCALE_FUNC(16f16s, cvt1_32f, float16_t, short, float)252253DEF_CVT_SCALE_FUNC(8u32s, cvt_32f, uchar, int, float)254DEF_CVT_SCALE_FUNC(8s32s, cvt_32f, schar, int, float)255DEF_CVT_SCALE_FUNC(16u32s, cvt_32f, ushort, int, float)256DEF_CVT_SCALE_FUNC(16s32s, cvt_32f, short, int, float)257DEF_CVT_SCALE_FUNC(32s, cvt_64f, int, int, double)258DEF_CVT_SCALE_FUNC(32f32s, cvt_32f, float, int, float)259DEF_CVT_SCALE_FUNC(64f32s, cvt_64f, double, int, double)260DEF_CVT_SCALE_FUNC(16f32s, cvt1_32f, float16_t, int, float)261262DEF_CVT_SCALE_FUNC(8u32f, cvt_32f, uchar, float, float)263DEF_CVT_SCALE_FUNC(8s32f, cvt_32f, schar, float, float)264DEF_CVT_SCALE_FUNC(16u32f, cvt_32f, ushort, float, float)265DEF_CVT_SCALE_FUNC(16s32f, cvt_32f, short, float, float)266DEF_CVT_SCALE_FUNC(32s32f, cvt_32f, int, float, float)267DEF_CVT_SCALE_FUNC(32f, cvt_32f, float, float, float)268DEF_CVT_SCALE_FUNC(64f32f, cvt_64f, double, float, double)269DEF_CVT_SCALE_FUNC(16f32f, cvt1_32f, float16_t, float, float)270271DEF_CVT_SCALE_FUNC(8u64f, cvt_64f, uchar, double, double)272DEF_CVT_SCALE_FUNC(8s64f, cvt_64f, schar, double, double)273DEF_CVT_SCALE_FUNC(16u64f, cvt_64f, ushort, double, double)274DEF_CVT_SCALE_FUNC(16s64f, cvt_64f, short, double, double)275DEF_CVT_SCALE_FUNC(32s64f, cvt_64f, int, double, double)276DEF_CVT_SCALE_FUNC(32f64f, cvt_64f, float, double, double)277DEF_CVT_SCALE_FUNC(64f, cvt_64f, double, double, double)278DEF_CVT_SCALE_FUNC(16f64f, cvt_64f, float16_t, double, double)279280DEF_CVT_SCALE_FUNC(8u16f, cvt1_32f, uchar, float16_t, float)281DEF_CVT_SCALE_FUNC(8s16f, cvt1_32f, schar, float16_t, float)282DEF_CVT_SCALE_FUNC(16u16f, cvt1_32f, ushort, float16_t, float)283DEF_CVT_SCALE_FUNC(16s16f, cvt1_32f, short, float16_t, float)284DEF_CVT_SCALE_FUNC(32s16f, cvt1_32f, int, float16_t, float)285DEF_CVT_SCALE_FUNC(32f16f, cvt1_32f, float, float16_t, float)286DEF_CVT_SCALE_FUNC(64f16f, cvt_64f, double, float16_t, double)287DEF_CVT_SCALE_FUNC(16f, cvt1_32f, float16_t, float16_t, float)288289static BinaryFunc getCvtScaleAbsFunc(int depth)290{291static BinaryFunc cvtScaleAbsTab[] =292{293(BinaryFunc)cvtScaleAbs8u, (BinaryFunc)cvtScaleAbs8s8u, (BinaryFunc)cvtScaleAbs16u8u,294(BinaryFunc)cvtScaleAbs16s8u, (BinaryFunc)cvtScaleAbs32s8u, (BinaryFunc)cvtScaleAbs32f8u,295(BinaryFunc)cvtScaleAbs64f8u, 0296};297298return cvtScaleAbsTab[depth];299}300301BinaryFunc getConvertScaleFunc(int sdepth, int ddepth)302{303static BinaryFunc cvtScaleTab[][8] =304{305{306(BinaryFunc)GET_OPTIMIZED(cvtScale8u), (BinaryFunc)GET_OPTIMIZED(cvtScale8s8u), (BinaryFunc)GET_OPTIMIZED(cvtScale16u8u),307(BinaryFunc)GET_OPTIMIZED(cvtScale16s8u), (BinaryFunc)GET_OPTIMIZED(cvtScale32s8u), (BinaryFunc)GET_OPTIMIZED(cvtScale32f8u),308(BinaryFunc)cvtScale64f8u, (BinaryFunc)cvtScale16f8u309},310{311(BinaryFunc)GET_OPTIMIZED(cvtScale8u8s), (BinaryFunc)GET_OPTIMIZED(cvtScale8s), (BinaryFunc)GET_OPTIMIZED(cvtScale16u8s),312(BinaryFunc)GET_OPTIMIZED(cvtScale16s8s), (BinaryFunc)GET_OPTIMIZED(cvtScale32s8s), (BinaryFunc)GET_OPTIMIZED(cvtScale32f8s),313(BinaryFunc)cvtScale64f8s, (BinaryFunc)cvtScale16f8s314},315{316(BinaryFunc)GET_OPTIMIZED(cvtScale8u16u), (BinaryFunc)GET_OPTIMIZED(cvtScale8s16u), (BinaryFunc)GET_OPTIMIZED(cvtScale16u),317(BinaryFunc)GET_OPTIMIZED(cvtScale16s16u), (BinaryFunc)GET_OPTIMIZED(cvtScale32s16u), (BinaryFunc)GET_OPTIMIZED(cvtScale32f16u),318(BinaryFunc)cvtScale64f16u, (BinaryFunc)cvtScale16f16u319},320{321(BinaryFunc)GET_OPTIMIZED(cvtScale8u16s), (BinaryFunc)GET_OPTIMIZED(cvtScale8s16s), (BinaryFunc)GET_OPTIMIZED(cvtScale16u16s),322(BinaryFunc)GET_OPTIMIZED(cvtScale16s), (BinaryFunc)GET_OPTIMIZED(cvtScale32s16s), (BinaryFunc)GET_OPTIMIZED(cvtScale32f16s),323(BinaryFunc)cvtScale64f16s, (BinaryFunc)cvtScale16f16s324},325{326(BinaryFunc)GET_OPTIMIZED(cvtScale8u32s), (BinaryFunc)GET_OPTIMIZED(cvtScale8s32s), (BinaryFunc)GET_OPTIMIZED(cvtScale16u32s),327(BinaryFunc)GET_OPTIMIZED(cvtScale16s32s), (BinaryFunc)GET_OPTIMIZED(cvtScale32s), (BinaryFunc)GET_OPTIMIZED(cvtScale32f32s),328(BinaryFunc)cvtScale64f32s, (BinaryFunc)cvtScale16f32s329},330{331(BinaryFunc)GET_OPTIMIZED(cvtScale8u32f), (BinaryFunc)GET_OPTIMIZED(cvtScale8s32f), (BinaryFunc)GET_OPTIMIZED(cvtScale16u32f),332(BinaryFunc)GET_OPTIMIZED(cvtScale16s32f), (BinaryFunc)GET_OPTIMIZED(cvtScale32s32f), (BinaryFunc)GET_OPTIMIZED(cvtScale32f),333(BinaryFunc)cvtScale64f32f, (BinaryFunc)cvtScale16f32f334},335{336(BinaryFunc)cvtScale8u64f, (BinaryFunc)cvtScale8s64f, (BinaryFunc)cvtScale16u64f,337(BinaryFunc)cvtScale16s64f, (BinaryFunc)cvtScale32s64f, (BinaryFunc)cvtScale32f64f,338(BinaryFunc)cvtScale64f, (BinaryFunc)cvtScale16f64f339},340{341(BinaryFunc)cvtScale8u16f, (BinaryFunc)cvtScale8s16f, (BinaryFunc)cvtScale16u16f,342(BinaryFunc)cvtScale16s16f, (BinaryFunc)cvtScale32s16f, (BinaryFunc)cvtScale32f16f,343(BinaryFunc)cvtScale64f16f, (BinaryFunc)cvtScale16f344},345};346347return cvtScaleTab[CV_MAT_DEPTH(ddepth)][CV_MAT_DEPTH(sdepth)];348}349350#ifdef HAVE_OPENCL351352static bool ocl_convertScaleAbs( InputArray _src, OutputArray _dst, double alpha, double beta )353{354const ocl::Device & d = ocl::Device::getDefault();355356int type = _src.type(), depth = CV_MAT_DEPTH(type), cn = CV_MAT_CN(type);357bool doubleSupport = d.doubleFPConfig() > 0;358if (!doubleSupport && depth == CV_64F)359return false;360361_dst.create(_src.size(), CV_8UC(cn));362int kercn = 1;363if (d.isIntel())364{365static const int vectorWidths[] = {4, 4, 4, 4, 4, 4, 4, -1};366kercn = ocl::checkOptimalVectorWidth( vectorWidths, _src, _dst,367noArray(), noArray(), noArray(),368noArray(), noArray(), noArray(),369noArray(), ocl::OCL_VECTOR_MAX);370}371else372kercn = ocl::predictOptimalVectorWidthMax(_src, _dst);373374int rowsPerWI = d.isIntel() ? 4 : 1;375char cvt[2][50];376int wdepth = std::max(depth, CV_32F);377String build_opt = format("-D OP_CONVERT_SCALE_ABS -D UNARY_OP -D dstT=%s -D DEPTH_dst=%d -D srcT1=%s"378" -D workT=%s -D wdepth=%d -D convertToWT1=%s -D convertToDT=%s"379" -D workT1=%s -D rowsPerWI=%d%s",380ocl::typeToStr(CV_8UC(kercn)), CV_8U,381ocl::typeToStr(CV_MAKE_TYPE(depth, kercn)),382ocl::typeToStr(CV_MAKE_TYPE(wdepth, kercn)), wdepth,383ocl::convertTypeStr(depth, wdepth, kercn, cvt[0]),384ocl::convertTypeStr(wdepth, CV_8U, kercn, cvt[1]),385ocl::typeToStr(wdepth), rowsPerWI,386doubleSupport ? " -D DOUBLE_SUPPORT" : "");387ocl::Kernel k("KF", ocl::core::arithm_oclsrc, build_opt);388if (k.empty())389return false;390391UMat src = _src.getUMat();392UMat dst = _dst.getUMat();393394ocl::KernelArg srcarg = ocl::KernelArg::ReadOnlyNoSize(src),395dstarg = ocl::KernelArg::WriteOnly(dst, cn, kercn);396397if (wdepth == CV_32F)398k.args(srcarg, dstarg, (float)alpha, (float)beta);399else if (wdepth == CV_64F)400k.args(srcarg, dstarg, alpha, beta);401402size_t globalsize[2] = { (size_t)src.cols * cn / kercn, ((size_t)src.rows + rowsPerWI - 1) / rowsPerWI };403return k.run(2, globalsize, NULL, false);404}405406#endif407408} //cv::409410411void cv::convertScaleAbs( InputArray _src, OutputArray _dst, double alpha, double beta )412{413CV_INSTRUMENT_REGION();414415CV_OCL_RUN(_src.dims() <= 2 && _dst.isUMat(),416ocl_convertScaleAbs(_src, _dst, alpha, beta))417418Mat src = _src.getMat();419int cn = src.channels();420double scale[] = {alpha, beta};421_dst.create( src.dims, src.size, CV_8UC(cn) );422Mat dst = _dst.getMat();423BinaryFunc func = getCvtScaleAbsFunc(src.depth());424CV_Assert( func != 0 );425426if( src.dims <= 2 )427{428Size sz = getContinuousSize(src, dst, cn);429func( src.ptr(), src.step, 0, 0, dst.ptr(), dst.step, sz, scale );430}431else432{433const Mat* arrays[] = {&src, &dst, 0};434uchar* ptrs[2] = {};435NAryMatIterator it(arrays, ptrs);436Size sz((int)it.size*cn, 1);437438for( size_t i = 0; i < it.nplanes; i++, ++it )439func( ptrs[0], 0, 0, 0, ptrs[1], 0, sz, scale );440}441}442443//==================================================================================================444445namespace cv {446447#ifdef HAVE_OPENCL448449static bool ocl_normalize( InputArray _src, InputOutputArray _dst, InputArray _mask, int dtype,450double scale, double delta )451{452UMat src = _src.getUMat();453454if( _mask.empty() )455src.convertTo( _dst, dtype, scale, delta );456else if (src.channels() <= 4)457{458const ocl::Device & dev = ocl::Device::getDefault();459460int stype = _src.type(), sdepth = CV_MAT_DEPTH(stype), cn = CV_MAT_CN(stype),461ddepth = CV_MAT_DEPTH(dtype), wdepth = std::max(CV_32F, std::max(sdepth, ddepth)),462rowsPerWI = dev.isIntel() ? 4 : 1;463464float fscale = static_cast<float>(scale), fdelta = static_cast<float>(delta);465bool haveScale = std::fabs(scale - 1) > DBL_EPSILON,466haveZeroScale = !(std::fabs(scale) > DBL_EPSILON),467haveDelta = std::fabs(delta) > DBL_EPSILON,468doubleSupport = dev.doubleFPConfig() > 0;469470if (!haveScale && !haveDelta && stype == dtype)471{472_src.copyTo(_dst, _mask);473return true;474}475if (haveZeroScale)476{477_dst.setTo(Scalar(delta), _mask);478return true;479}480481if ((sdepth == CV_64F || ddepth == CV_64F) && !doubleSupport)482return false;483484char cvt[2][40];485String opts = format("-D srcT=%s -D dstT=%s -D convertToWT=%s -D cn=%d -D rowsPerWI=%d"486" -D convertToDT=%s -D workT=%s%s%s%s -D srcT1=%s -D dstT1=%s",487ocl::typeToStr(stype), ocl::typeToStr(dtype),488ocl::convertTypeStr(sdepth, wdepth, cn, cvt[0]), cn,489rowsPerWI, ocl::convertTypeStr(wdepth, ddepth, cn, cvt[1]),490ocl::typeToStr(CV_MAKE_TYPE(wdepth, cn)),491doubleSupport ? " -D DOUBLE_SUPPORT" : "",492haveScale ? " -D HAVE_SCALE" : "",493haveDelta ? " -D HAVE_DELTA" : "",494ocl::typeToStr(sdepth), ocl::typeToStr(ddepth));495496ocl::Kernel k("normalizek", ocl::core::normalize_oclsrc, opts);497if (k.empty())498return false;499500UMat mask = _mask.getUMat(), dst = _dst.getUMat();501502ocl::KernelArg srcarg = ocl::KernelArg::ReadOnlyNoSize(src),503maskarg = ocl::KernelArg::ReadOnlyNoSize(mask),504dstarg = ocl::KernelArg::ReadWrite(dst);505506if (haveScale)507{508if (haveDelta)509k.args(srcarg, maskarg, dstarg, fscale, fdelta);510else511k.args(srcarg, maskarg, dstarg, fscale);512}513else514{515if (haveDelta)516k.args(srcarg, maskarg, dstarg, fdelta);517else518k.args(srcarg, maskarg, dstarg);519}520521size_t globalsize[2] = { (size_t)src.cols, ((size_t)src.rows + rowsPerWI - 1) / rowsPerWI };522return k.run(2, globalsize, NULL, false);523}524else525{526UMat temp;527src.convertTo( temp, dtype, scale, delta );528temp.copyTo( _dst, _mask );529}530531return true;532}533534#endif535536} // cv::537538void cv::normalize( InputArray _src, InputOutputArray _dst, double a, double b,539int norm_type, int rtype, InputArray _mask )540{541CV_INSTRUMENT_REGION();542543double scale = 1, shift = 0;544int type = _src.type(), depth = CV_MAT_DEPTH(type);545546if( rtype < 0 )547rtype = _dst.fixedType() ? _dst.depth() : depth;548549if( norm_type == CV_MINMAX )550{551double smin = 0, smax = 0;552double dmin = MIN( a, b ), dmax = MAX( a, b );553minMaxIdx( _src, &smin, &smax, 0, 0, _mask );554scale = (dmax - dmin)*(smax - smin > DBL_EPSILON ? 1./(smax - smin) : 0);555if( rtype == CV_32F )556{557scale = (float)scale;558shift = (float)dmin - (float)(smin*scale);559}560else561shift = dmin - smin*scale;562}563else if( norm_type == CV_L2 || norm_type == CV_L1 || norm_type == CV_C )564{565scale = norm( _src, norm_type, _mask );566scale = scale > DBL_EPSILON ? a/scale : 0.;567shift = 0;568}569else570CV_Error( CV_StsBadArg, "Unknown/unsupported norm type" );571572CV_OCL_RUN(_dst.isUMat(),573ocl_normalize(_src, _dst, _mask, rtype, scale, shift))574575Mat src = _src.getMat();576if( _mask.empty() )577src.convertTo( _dst, rtype, scale, shift );578else579{580Mat temp;581src.convertTo( temp, rtype, scale, shift );582temp.copyTo( _dst, _mask );583}584}585586587