test_grayworld.cpp 2.88 KB
Newer Older
1 2 3 4 5 6
#include "test_precomp.hpp"

namespace cvtest {

    using namespace cv;

7
    void ref_autowbGrayworld(InputArray _src, OutputArray _dst, float thresh)
8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
    {
        Mat src = _src.getMat();

        _dst.create(src.size(), src.type());
        Mat dst = _dst.getMat();

        int width  = src.cols,
            height = src.rows,
            N      = width*height,
            N3     = N*3;

        // Calculate sum of pixel values of each channel
        const uchar* src_data = src.ptr<uchar>(0);
        unsigned long sum1 = 0, sum2 = 0, sum3 = 0;
        int i = 0;
23
        unsigned int minRGB, maxRGB, thresh255 = cvRound(thresh * 255);
24
        for ( ; i < N3; i += 3 )
25
        {
26 27
            minRGB = std::min(src_data[i], std::min(src_data[i + 1], src_data[i + 2]));
            maxRGB = std::max(src_data[i], std::max(src_data[i + 1], src_data[i + 2]));
28
            if ( (maxRGB - minRGB) * 255 > thresh255 * maxRGB ) continue;
29
            sum1 += src_data[i];
30 31 32 33 34 35 36 37 38 39 40 41 42
            sum2 += src_data[i + 1];
            sum3 += src_data[i + 2];
        }

        // Find inverse of averages
        double inv1 = sum1 == 0 ? 0.f : (double)N / (double)sum1,
               inv2 = sum2 == 0 ? 0.f : (double)N / (double)sum2,
               inv3 = sum3 == 0 ? 0.f : (double)N / (double)sum3;

        // Find maximum
        double inv_max = std::max(std::max(inv1, inv2), inv3);

        // Scale by maximum
43
        if ( inv_max > 0 )
44 45 46 47 48 49
        {
            inv1 = (double) inv1 / inv_max;
            inv2 = (double) inv2 / inv_max;
            inv3 = (double) inv3 / inv_max;
        }

50
        // Fixed point arithmetic, mul by 2^8 then shift back 8 bits
51 52 53
        int i_inv1 = cvRound(inv1 * (1 << 8)),
            i_inv2 = cvRound(inv2 * (1 << 8)),
            i_inv3 = cvRound(inv3 * (1 << 8));
54

55 56 57
        // Scale input pixel values
        uchar* dst_data = dst.ptr<uchar>(0);
        i = 0;
58
        for ( ; i < N3; i += 3 )
59
        {
60 61 62
            dst_data[i]     = (uchar)((src_data[i]     * i_inv1) >> 8);
            dst_data[i + 1] = (uchar)((src_data[i + 1] * i_inv2) >> 8);
            dst_data[i + 2] = (uchar)((src_data[i + 2] * i_inv3) >> 8);
63 64 65 66 67
        }
    }

    TEST(xphoto_grayworld_white_balance, regression)
    {
Maksim Shabunin's avatar
Maksim Shabunin committed
68
        String dir = cvtest::TS::ptr()->get_data_path() + "cv/xphoto/simple_white_balance/";
69 70 71
        const int nTests = 14;
        const float wb_thresh = 0.5f;
        const float acc_thresh = 2.f;
72

73
        for ( int i = 0; i < nTests; ++i )
74 75 76 77 78 79
        {
            String srcName = dir + format("sources/%02d.png", i + 1);
            Mat src = imread(srcName, IMREAD_COLOR);
            ASSERT_TRUE(!src.empty());

            Mat referenceResult;
80
            ref_autowbGrayworld(src, referenceResult, wb_thresh);
81 82

            Mat currentResult;
83
            xphoto::autowbGrayworld(src, currentResult, wb_thresh);
84

85
            ASSERT_LE(cv::norm(currentResult, referenceResult, NORM_INF), acc_thresh);
86 87 88 89
        }
    }

}