blob: 8fd1380ac0bf0fb73096bd8b6227f5094ff5379c [file] [log] [blame]
#include "dnnlowp.h"
#include <cmath>
#include <iostream>
#include <random>
#include <gtest/gtest.h>
#include "caffe2/core/logging.h"
using namespace std;
using namespace dnnlowp;
TEST(Requantization, BatchRequantizationUnitTest) {
// generate input data
default_random_engine eng;
uniform_int_distribution<int32_t> in_max_dis(
10, numeric_limits<int32_t>::max());
uniform_int_distribution<int> zero_point_dis(0, 255);
constexpr int NITER = 1024;
constexpr int LEN = 77;
vector<int32_t> src(LEN);
vector<uint8_t> expected(LEN), actual(LEN);
QuantizationFactory* qfactory = QuantizationFactory::GetDefaultInstance();
for (int i = 0; i < NITER; ++i) {
int32_t in_max = in_max_dis(eng);
uniform_int_distribution<int32_t> in_dis(-in_max, in_max);
for (int j = 0; j < LEN; ++j) {
src[j] = in_dis(eng);
}
// Precise real_multiplier will be (255 / in_max) but intentionally use
// a bigger multiplier to test if saturation is handled correctly.
float real_multiplier = 255 / (1.5 * in_max);
TensorQuantizationParams target_qparams;
target_qparams.zero_point = zero_point_dis(eng);
target_qparams.precision = 8;
RequantizationParams params = qfactory->ChooseRequantizationMultiplier(
real_multiplier, target_qparams);
for (int j = 0; j < LEN; ++j) {
expected[j] = fbgemm::clamp(
target_qparams.zero_point +
std::nearbyint(static_cast<double>(src[j]) * real_multiplier),
8);
}
unsigned long long cycle_begin = __rdtsc();
fbgemm::Requantize(src.data(), actual.data(), LEN, params);
unsigned long long cycle_end = __rdtsc();
double elements_per_cycle = (double)LEN / (cycle_end - cycle_begin);
LOG(INFO) << elements_per_cycle << " elements_per_cycle";
for (int j = 0; j < LEN; ++j) {
EXPECT_EQ((int)expected[j], (int)actual[j])
<< "i " << i << " j " << j << " src " << src[j] << " real_multiplier "
<< real_multiplier << " multiplier " << params.multiplier
<< " right_shift " << params.right_shift << " zero_point "
<< target_qparams.zero_point;
}
}
}
TEST(Requantization, RequantizationUnitTest) {
// Rescaling to a random range [min1, max1] to [min2, max2].
// Make sure the ranges include 0 and inputs don't have input quantization
// error
default_random_engine gen;
QuantizationFactory* qfactory = QuantizationFactory::GetDefaultInstance();
{
// Test 31-bit to 8-bit scaling (the most common one for example used for
// the results of GEMM).
// Dest quantization parameter is pre-determined by actual min/max of the
// values.
// Source scale can vary and zero_offset is 0.
uniform_real_distribution<float> src_scale_exponent_dist(-19, -1);
uniform_real_distribution<float> dst_exponent_dist(0.1, 4);
// Bits used in src_scale plus dst should be <= 23 not to have any
// input quantization error because float has 23 bit precision.
uniform_real_distribution<float> negative_proportion_dist(0, 1);
for (int i = 0; i < 256; ++i) {
TensorQuantizationParams src_qparams;
// scale is 2^-1 ~ 2^-19
src_qparams.scale = powf(2, src_scale_exponent_dist(gen));
src_qparams.zero_point = 0;
src_qparams.precision = 31;
float dst_extend = powf(2, dst_exponent_dist(gen));
float negative_proportion = negative_proportion_dist(gen);
float min = -(dst_extend * negative_proportion);
float max = dst_extend + min;
TensorQuantizationParams dst_qparams =
qfactory->ChooseQuantizationParams(min, max);
// scale = dst_extend / 2^8
// which is between 0.1/2^-8 ~ 2^-4
float real_multiplier = src_qparams.scale / dst_qparams.scale;
RequantizationParams requantization_params =
qfactory->ChooseRequantizationMultiplier(
real_multiplier, dst_qparams);
uniform_real_distribution<float> value_dist(
ceil(min / src_qparams.scale) * src_qparams.scale,
floor(max / src_qparams.scale) * src_qparams.scale);
// round with src_qparams.scale to avoid input quantization error due
// to clipping
float sum_sq = 0, max_err = 0;
constexpr int LEN = 1111;
vector<int32_t> src_q(LEN);
vector<float> src(LEN);
for (int j = 0; j < LEN; ++j) {
float src_orig = value_dist(gen);
src_q[j] = fbgemm::Quantize<int32_t>(
src_orig, 0, src_qparams.scale, 32, true /* signed*/);
src[j] = fbgemm::Dequantize<int32_t>(src_q[j], src_qparams);
// This number shouldn't have any quantization error
EXPECT_EQ(
fbgemm::Quantize<int32_t>(src[j], 0, src_qparams.scale, 32, true),
src_q[j]);
}
vector<uint8_t> dst_q(LEN);
fbgemm::Requantize(
src_q.data(), dst_q.data(), LEN, requantization_params);
for (int j = 0; j < LEN; ++j) {
float dst = fbgemm::Dequantize<uint8_t>(dst_q[j], dst_qparams);
float err = fabsf(dst - src[j]);
sum_sq += err * err;
max_err = std::max(max_err, err);
EXPECT_LE(err, dst_qparams.scale / 1.9);
}
LOG(INFO) << "src_scale " << src_qparams.scale << " dst_extend "
<< dst_extend << " real_multiplier " << real_multiplier
<< " avg_l2_err " << std::sqrt(sum_sq) / 1024 << " max_err "
<< max_err << endl;
// We shouldn't have an error bigger than output quantization error
EXPECT_LE(max_err, dst_qparams.scale / 1.9);
}
}
}