Home | History | Annotate | Download | only in base
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 // MSVC++ requires this to be set before any other includes to get M_PI.
      6 #define _USE_MATH_DEFINES
      7 #include <cmath>
      8 
      9 #include "base/cpu.h"
     10 #include "base/memory/aligned_memory.h"
     11 #include "base/memory/scoped_ptr.h"
     12 #include "base/strings/string_number_conversions.h"
     13 #include "base/strings/stringize_macros.h"
     14 #include "media/base/vector_math.h"
     15 #include "media/base/vector_math_testing.h"
     16 #include "testing/gtest/include/gtest/gtest.h"
     17 
     18 using std::fill;
     19 
     20 namespace media {
     21 
     22 // Default test values.
     23 static const float kScale = 0.5;
     24 static const float kInputFillValue = 1.0;
     25 static const float kOutputFillValue = 3.0;
     26 static const int kVectorSize = 8192;
     27 
     28 class VectorMathTest : public testing::Test {
     29  public:
     30 
     31   VectorMathTest() {
     32     // Initialize input and output vectors.
     33     input_vector_.reset(static_cast<float*>(base::AlignedAlloc(
     34         sizeof(float) * kVectorSize, vector_math::kRequiredAlignment)));
     35     output_vector_.reset(static_cast<float*>(base::AlignedAlloc(
     36         sizeof(float) * kVectorSize, vector_math::kRequiredAlignment)));
     37   }
     38 
     39   void FillTestVectors(float input, float output) {
     40     // Setup input and output vectors.
     41     fill(input_vector_.get(), input_vector_.get() + kVectorSize, input);
     42     fill(output_vector_.get(), output_vector_.get() + kVectorSize, output);
     43   }
     44 
     45   void VerifyOutput(float value) {
     46     for (int i = 0; i < kVectorSize; ++i)
     47       ASSERT_FLOAT_EQ(output_vector_.get()[i], value);
     48   }
     49 
     50  protected:
     51   scoped_ptr_malloc<float, base::ScopedPtrAlignedFree> input_vector_;
     52   scoped_ptr_malloc<float, base::ScopedPtrAlignedFree> output_vector_;
     53 
     54   DISALLOW_COPY_AND_ASSIGN(VectorMathTest);
     55 };
     56 
     57 // Ensure each optimized vector_math::FMAC() method returns the same value.
     58 TEST_F(VectorMathTest, FMAC) {
     59   static const float kResult = kInputFillValue * kScale + kOutputFillValue;
     60 
     61   {
     62     SCOPED_TRACE("FMAC");
     63     FillTestVectors(kInputFillValue, kOutputFillValue);
     64     vector_math::FMAC(
     65         input_vector_.get(), kScale, kVectorSize, output_vector_.get());
     66     VerifyOutput(kResult);
     67   }
     68 
     69   {
     70     SCOPED_TRACE("FMAC_C");
     71     FillTestVectors(kInputFillValue, kOutputFillValue);
     72     vector_math::FMAC_C(
     73         input_vector_.get(), kScale, kVectorSize, output_vector_.get());
     74     VerifyOutput(kResult);
     75   }
     76 
     77 #if defined(ARCH_CPU_X86_FAMILY)
     78   {
     79     ASSERT_TRUE(base::CPU().has_sse());
     80     SCOPED_TRACE("FMAC_SSE");
     81     FillTestVectors(kInputFillValue, kOutputFillValue);
     82     vector_math::FMAC_SSE(
     83         input_vector_.get(), kScale, kVectorSize, output_vector_.get());
     84     VerifyOutput(kResult);
     85   }
     86 #endif
     87 
     88 #if defined(ARCH_CPU_ARM_FAMILY) && defined(USE_NEON)
     89   {
     90     SCOPED_TRACE("FMAC_NEON");
     91     FillTestVectors(kInputFillValue, kOutputFillValue);
     92     vector_math::FMAC_NEON(
     93         input_vector_.get(), kScale, kVectorSize, output_vector_.get());
     94     VerifyOutput(kResult);
     95   }
     96 #endif
     97 }
     98 
     99 // Ensure each optimized vector_math::FMUL() method returns the same value.
    100 TEST_F(VectorMathTest, FMUL) {
    101   static const float kResult = kInputFillValue * kScale;
    102 
    103   {
    104     SCOPED_TRACE("FMUL");
    105     FillTestVectors(kInputFillValue, kOutputFillValue);
    106     vector_math::FMUL(
    107         input_vector_.get(), kScale, kVectorSize, output_vector_.get());
    108     VerifyOutput(kResult);
    109   }
    110 
    111   {
    112     SCOPED_TRACE("FMUL_C");
    113     FillTestVectors(kInputFillValue, kOutputFillValue);
    114     vector_math::FMUL_C(
    115         input_vector_.get(), kScale, kVectorSize, output_vector_.get());
    116     VerifyOutput(kResult);
    117   }
    118 
    119 #if defined(ARCH_CPU_X86_FAMILY)
    120   {
    121     ASSERT_TRUE(base::CPU().has_sse());
    122     SCOPED_TRACE("FMUL_SSE");
    123     FillTestVectors(kInputFillValue, kOutputFillValue);
    124     vector_math::FMUL_SSE(
    125         input_vector_.get(), kScale, kVectorSize, output_vector_.get());
    126     VerifyOutput(kResult);
    127   }
    128 #endif
    129 
    130 #if defined(ARCH_CPU_ARM_FAMILY) && defined(USE_NEON)
    131   {
    132     SCOPED_TRACE("FMUL_NEON");
    133     FillTestVectors(kInputFillValue, kOutputFillValue);
    134     vector_math::FMUL_NEON(
    135         input_vector_.get(), kScale, kVectorSize, output_vector_.get());
    136     VerifyOutput(kResult);
    137   }
    138 #endif
    139 }
    140 
    141 namespace {
    142 
    143 class EWMATestScenario {
    144  public:
    145   EWMATestScenario(float initial_value, const float src[], int len,
    146                    float smoothing_factor)
    147       : initial_value_(initial_value),
    148         data_(static_cast<float*>(
    149             len == 0 ? NULL :
    150             base::AlignedAlloc(len * sizeof(float),
    151                                vector_math::kRequiredAlignment))),
    152         data_len_(len),
    153         smoothing_factor_(smoothing_factor),
    154         expected_final_avg_(initial_value),
    155         expected_max_(0.0f) {
    156     if (data_len_ > 0)
    157       memcpy(data_.get(), src, len * sizeof(float));
    158   }
    159 
    160   // Copy constructor and assignment operator for ::testing::Values(...).
    161   EWMATestScenario(const EWMATestScenario& other) { *this = other; }
    162   EWMATestScenario& operator=(const EWMATestScenario& other) {
    163     this->initial_value_ = other.initial_value_;
    164     this->smoothing_factor_ = other.smoothing_factor_;
    165     if (other.data_len_ == 0) {
    166       this->data_.reset();
    167     } else {
    168       this->data_.reset(static_cast<float*>(
    169         base::AlignedAlloc(other.data_len_ * sizeof(float),
    170                            vector_math::kRequiredAlignment)));
    171       memcpy(this->data_.get(), other.data_.get(),
    172              other.data_len_ * sizeof(float));
    173     }
    174     this->data_len_ = other.data_len_;
    175     this->expected_final_avg_ = other.expected_final_avg_;
    176     this->expected_max_ = other.expected_max_;
    177     return *this;
    178   }
    179 
    180   EWMATestScenario ScaledBy(float scale) const {
    181     EWMATestScenario result(*this);
    182     float* p = result.data_.get();
    183     float* const p_end = p + result.data_len_;
    184     for (; p < p_end; ++p)
    185       *p *= scale;
    186     return result;
    187   }
    188 
    189   EWMATestScenario WithImpulse(float value, int offset) const {
    190     EWMATestScenario result(*this);
    191     result.data_.get()[offset] = value;
    192     return result;
    193   }
    194 
    195   EWMATestScenario HasExpectedResult(float final_avg_value,
    196                                      float max_value) const {
    197     EWMATestScenario result(*this);
    198     result.expected_final_avg_ = final_avg_value;
    199     result.expected_max_ = max_value;
    200     return result;
    201   }
    202 
    203   void RunTest() const {
    204     {
    205       SCOPED_TRACE("EWMAAndMaxPower");
    206       const std::pair<float, float>& result = vector_math::EWMAAndMaxPower(
    207           initial_value_, data_.get(), data_len_, smoothing_factor_);
    208       EXPECT_NEAR(expected_final_avg_, result.first, 0.0000001f);
    209       EXPECT_NEAR(expected_max_, result.second, 0.0000001f);
    210     }
    211 
    212     {
    213       SCOPED_TRACE("EWMAAndMaxPower_C");
    214       const std::pair<float, float>& result = vector_math::EWMAAndMaxPower_C(
    215           initial_value_, data_.get(), data_len_, smoothing_factor_);
    216       EXPECT_NEAR(expected_final_avg_, result.first, 0.0000001f);
    217       EXPECT_NEAR(expected_max_, result.second, 0.0000001f);
    218     }
    219 
    220 #if defined(ARCH_CPU_X86_FAMILY)
    221     {
    222       ASSERT_TRUE(base::CPU().has_sse());
    223       SCOPED_TRACE("EWMAAndMaxPower_SSE");
    224       const std::pair<float, float>& result = vector_math::EWMAAndMaxPower_SSE(
    225           initial_value_, data_.get(), data_len_, smoothing_factor_);
    226       EXPECT_NEAR(expected_final_avg_, result.first, 0.0000001f);
    227       EXPECT_NEAR(expected_max_, result.second, 0.0000001f);
    228     }
    229 #endif
    230 
    231 #if defined(ARCH_CPU_ARM_FAMILY) && defined(USE_NEON)
    232     {
    233       SCOPED_TRACE("EWMAAndMaxPower_NEON");
    234       const std::pair<float, float>& result = vector_math::EWMAAndMaxPower_NEON(
    235           initial_value_, data_.get(), data_len_, smoothing_factor_);
    236       EXPECT_NEAR(expected_final_avg_, result.first, 0.0000001f);
    237       EXPECT_NEAR(expected_max_, result.second, 0.0000001f);
    238     }
    239 #endif
    240   }
    241 
    242  private:
    243   float initial_value_;
    244   scoped_ptr_malloc<float, base::ScopedPtrAlignedFree> data_;
    245   int data_len_;
    246   float smoothing_factor_;
    247   float expected_final_avg_;
    248   float expected_max_;
    249 };
    250 
    251 }  // namespace
    252 
    253 typedef testing::TestWithParam<EWMATestScenario> VectorMathEWMAAndMaxPowerTest;
    254 
    255 TEST_P(VectorMathEWMAAndMaxPowerTest, Correctness) {
    256   GetParam().RunTest();
    257 }
    258 
    259 static const float kZeros[] = {  // 32 zeros
    260   0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
    261   0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0
    262 };
    263 
    264 static const float kOnes[] = {  // 32 ones
    265   1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1,
    266   1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1
    267 };
    268 
    269 static const float kCheckerboard[] = {  // 32 alternating 0, 1
    270   0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1,
    271   0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1
    272 };
    273 
    274 static const float kInverseCheckerboard[] = {  // 32 alternating 1, 0
    275   1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0,
    276   1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0, 1, 0
    277 };
    278 
    279 INSTANTIATE_TEST_CASE_P(
    280     Scenarios, VectorMathEWMAAndMaxPowerTest,
    281     ::testing::Values(
    282          // Zero-length input: Result should equal initial value.
    283          EWMATestScenario(0.0f, NULL, 0, 0.0f).HasExpectedResult(0.0f, 0.0f),
    284          EWMATestScenario(1.0f, NULL, 0, 0.0f).HasExpectedResult(1.0f, 0.0f),
    285 
    286          // Smoothing factor of zero: Samples have no effect on result.
    287          EWMATestScenario(0.0f, kOnes, 32, 0.0f).HasExpectedResult(0.0f, 1.0f),
    288          EWMATestScenario(1.0f, kZeros, 32, 0.0f).HasExpectedResult(1.0f, 0.0f),
    289 
    290          // Smothing factor of one: Result = last sample squared.
    291          EWMATestScenario(0.0f, kCheckerboard, 32, 1.0f)
    292              .ScaledBy(2.0f)
    293              .HasExpectedResult(4.0f, 4.0f),
    294          EWMATestScenario(1.0f, kInverseCheckerboard, 32, 1.0f)
    295              .ScaledBy(2.0f)
    296              .HasExpectedResult(0.0f, 4.0f),
    297 
    298          // Smoothing factor of 1/4, muted signal.
    299          EWMATestScenario(1.0f, kZeros, 1, 0.25f)
    300              .HasExpectedResult(powf(0.75, 1.0f), 0.0f),
    301          EWMATestScenario(1.0f, kZeros, 2, 0.25f)
    302              .HasExpectedResult(powf(0.75, 2.0f), 0.0f),
    303          EWMATestScenario(1.0f, kZeros, 3, 0.25f)
    304              .HasExpectedResult(powf(0.75, 3.0f), 0.0f),
    305          EWMATestScenario(1.0f, kZeros, 12, 0.25f)
    306              .HasExpectedResult(powf(0.75, 12.0f), 0.0f),
    307          EWMATestScenario(1.0f, kZeros, 13, 0.25f)
    308              .HasExpectedResult(powf(0.75, 13.0f), 0.0f),
    309          EWMATestScenario(1.0f, kZeros, 14, 0.25f)
    310              .HasExpectedResult(powf(0.75, 14.0f), 0.0f),
    311          EWMATestScenario(1.0f, kZeros, 15, 0.25f)
    312              .HasExpectedResult(powf(0.75, 15.0f), 0.0f),
    313 
    314          // Smoothing factor of 1/4, constant full-amplitude signal.
    315          EWMATestScenario(0.0f, kOnes, 1, 0.25f).HasExpectedResult(0.25f, 1.0f),
    316          EWMATestScenario(0.0f, kOnes, 2, 0.25f)
    317              .HasExpectedResult(0.4375f, 1.0f),
    318          EWMATestScenario(0.0f, kOnes, 3, 0.25f)
    319              .HasExpectedResult(0.578125f, 1.0f),
    320          EWMATestScenario(0.0f, kOnes, 12, 0.25f)
    321              .HasExpectedResult(0.96832365f, 1.0f),
    322          EWMATestScenario(0.0f, kOnes, 13, 0.25f)
    323              .HasExpectedResult(0.97624274f, 1.0f),
    324          EWMATestScenario(0.0f, kOnes, 14, 0.25f)
    325              .HasExpectedResult(0.98218205f, 1.0f),
    326          EWMATestScenario(0.0f, kOnes, 15, 0.25f)
    327              .HasExpectedResult(0.98663654f, 1.0f),
    328 
    329          // Smoothing factor of 1/4, checkerboard signal.
    330          EWMATestScenario(0.0f, kCheckerboard, 1, 0.25f)
    331              .HasExpectedResult(0.0f, 0.0f),
    332          EWMATestScenario(0.0f, kCheckerboard, 2, 0.25f)
    333              .HasExpectedResult(0.25f, 1.0f),
    334          EWMATestScenario(0.0f, kCheckerboard, 3, 0.25f)
    335              .HasExpectedResult(0.1875f, 1.0f),
    336          EWMATestScenario(0.0f, kCheckerboard, 12, 0.25f)
    337              .HasExpectedResult(0.55332780f, 1.0f),
    338          EWMATestScenario(0.0f, kCheckerboard, 13, 0.25f)
    339              .HasExpectedResult(0.41499585f, 1.0f),
    340          EWMATestScenario(0.0f, kCheckerboard, 14, 0.25f)
    341              .HasExpectedResult(0.56124689f, 1.0f),
    342          EWMATestScenario(0.0f, kCheckerboard, 15, 0.25f)
    343              .HasExpectedResult(0.42093517f, 1.0f),
    344 
    345          // Smoothing factor of 1/4, inverse checkerboard signal.
    346          EWMATestScenario(0.0f, kInverseCheckerboard, 1, 0.25f)
    347              .HasExpectedResult(0.25f, 1.0f),
    348          EWMATestScenario(0.0f, kInverseCheckerboard, 2, 0.25f)
    349              .HasExpectedResult(0.1875f, 1.0f),
    350          EWMATestScenario(0.0f, kInverseCheckerboard, 3, 0.25f)
    351              .HasExpectedResult(0.390625f, 1.0f),
    352          EWMATestScenario(0.0f, kInverseCheckerboard, 12, 0.25f)
    353              .HasExpectedResult(0.41499585f, 1.0f),
    354          EWMATestScenario(0.0f, kInverseCheckerboard, 13, 0.25f)
    355              .HasExpectedResult(0.56124689f, 1.0f),
    356          EWMATestScenario(0.0f, kInverseCheckerboard, 14, 0.25f)
    357              .HasExpectedResult(0.42093517f, 1.0f),
    358          EWMATestScenario(0.0f, kInverseCheckerboard, 15, 0.25f)
    359              .HasExpectedResult(0.56570137f, 1.0f),
    360 
    361          // Smoothing factor of 1/4, impluse signal.
    362          EWMATestScenario(0.0f, kZeros, 3, 0.25f)
    363              .WithImpulse(2.0f, 0)
    364              .HasExpectedResult(0.562500f, 4.0f),
    365          EWMATestScenario(0.0f, kZeros, 3, 0.25f)
    366              .WithImpulse(2.0f, 1)
    367              .HasExpectedResult(0.75f, 4.0f),
    368          EWMATestScenario(0.0f, kZeros, 3, 0.25f)
    369              .WithImpulse(2.0f, 2)
    370              .HasExpectedResult(1.0f, 4.0f),
    371          EWMATestScenario(0.0f, kZeros, 32, 0.25f)
    372              .WithImpulse(2.0f, 0)
    373              .HasExpectedResult(0.00013394f, 4.0f),
    374          EWMATestScenario(0.0f, kZeros, 32, 0.25f)
    375              .WithImpulse(2.0f, 1)
    376              .HasExpectedResult(0.00017858f, 4.0f),
    377          EWMATestScenario(0.0f, kZeros, 32, 0.25f)
    378              .WithImpulse(2.0f, 2)
    379              .HasExpectedResult(0.00023811f, 4.0f),
    380          EWMATestScenario(0.0f, kZeros, 32, 0.25f)
    381              .WithImpulse(2.0f, 3)
    382              .HasExpectedResult(0.00031748f, 4.0f)
    383     ));
    384 
    385 }  // namespace media
    386