1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2022 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker
8*c8dee2aaSAndroid Build Coastguard Worker #include "src/core/SkOpts.h"
9*c8dee2aaSAndroid Build Coastguard Worker #include "tests/Test.h"
10*c8dee2aaSAndroid Build Coastguard Worker
11*c8dee2aaSAndroid Build Coastguard Worker #include <algorithm>
12*c8dee2aaSAndroid Build Coastguard Worker #include <array>
13*c8dee2aaSAndroid Build Coastguard Worker #include <cstddef>
14*c8dee2aaSAndroid Build Coastguard Worker
15*c8dee2aaSAndroid Build Coastguard Worker #define SK_OPTS_NS RPOptsTest
16*c8dee2aaSAndroid Build Coastguard Worker
17*c8dee2aaSAndroid Build Coastguard Worker #if defined(__clang__)
18*c8dee2aaSAndroid Build Coastguard Worker #pragma clang diagnostic push
19*c8dee2aaSAndroid Build Coastguard Worker #pragma clang diagnostic ignored "-Wunused-function"
20*c8dee2aaSAndroid Build Coastguard Worker #elif defined(__GNUC__)
21*c8dee2aaSAndroid Build Coastguard Worker #pragma GCC diagnostic push
22*c8dee2aaSAndroid Build Coastguard Worker #pragma GCC diagnostic ignored "-Wunused-function"
23*c8dee2aaSAndroid Build Coastguard Worker #pragma GCC diagnostic ignored "-Wunused-variable"
24*c8dee2aaSAndroid Build Coastguard Worker #endif
25*c8dee2aaSAndroid Build Coastguard Worker
26*c8dee2aaSAndroid Build Coastguard Worker #include "src/opts/SkRasterPipeline_opts.h"
27*c8dee2aaSAndroid Build Coastguard Worker
28*c8dee2aaSAndroid Build Coastguard Worker #if defined(__clang__)
29*c8dee2aaSAndroid Build Coastguard Worker #pragma clang diagnostic pop
30*c8dee2aaSAndroid Build Coastguard Worker #elif defined(__GNUC__)
31*c8dee2aaSAndroid Build Coastguard Worker #pragma GCC diagnostic pop
32*c8dee2aaSAndroid Build Coastguard Worker #endif
33*c8dee2aaSAndroid Build Coastguard Worker
34*c8dee2aaSAndroid Build Coastguard Worker constexpr auto F_ = SK_OPTS_NS::F_;
35*c8dee2aaSAndroid Build Coastguard Worker using F = SK_OPTS_NS::F;
36*c8dee2aaSAndroid Build Coastguard Worker using I32 = SK_OPTS_NS::I32;
37*c8dee2aaSAndroid Build Coastguard Worker
38*c8dee2aaSAndroid Build Coastguard Worker template <size_t N>
make_masks(int bits)39*c8dee2aaSAndroid Build Coastguard Worker static std::array<int32_t, N> make_masks(int bits) {
40*c8dee2aaSAndroid Build Coastguard Worker // Make an array of masks that correspond to the bit pattern of `bits`.
41*c8dee2aaSAndroid Build Coastguard Worker std::array<int32_t, N> masks;
42*c8dee2aaSAndroid Build Coastguard Worker for (size_t idx = 0; idx < N; ++idx) {
43*c8dee2aaSAndroid Build Coastguard Worker masks[idx] = (bits & 1) ? ~0 : 0;
44*c8dee2aaSAndroid Build Coastguard Worker bits >>= 1;
45*c8dee2aaSAndroid Build Coastguard Worker }
46*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(!bits);
47*c8dee2aaSAndroid Build Coastguard Worker return masks;
48*c8dee2aaSAndroid Build Coastguard Worker }
49*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(SkRasterPipelineOpts_Any,r)50*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(SkRasterPipelineOpts_Any, r) {
51*c8dee2aaSAndroid Build Coastguard Worker static constexpr size_t N = sizeof(I32) / sizeof(int32_t);
52*c8dee2aaSAndroid Build Coastguard Worker
53*c8dee2aaSAndroid Build Coastguard Worker for (int value = 0; value < (1 << N); ++value) {
54*c8dee2aaSAndroid Build Coastguard Worker // Load masks corresponding to the bit-pattern of `value` into lanes of `i`.
55*c8dee2aaSAndroid Build Coastguard Worker std::array<int32_t, N> masks = make_masks<N>(value);
56*c8dee2aaSAndroid Build Coastguard Worker I32 i = sk_unaligned_load<I32>(masks.data());
57*c8dee2aaSAndroid Build Coastguard Worker
58*c8dee2aaSAndroid Build Coastguard Worker // Verify that the raster pipeline any() matches expectations.
59*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, SK_OPTS_NS::any(i) == std::any_of(masks.begin(), masks.end(),
60*c8dee2aaSAndroid Build Coastguard Worker [](int32_t m) { return m != 0; }));
61*c8dee2aaSAndroid Build Coastguard Worker }
62*c8dee2aaSAndroid Build Coastguard Worker }
63*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(SkRasterPipelineOpts_All,r)64*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(SkRasterPipelineOpts_All, r) {
65*c8dee2aaSAndroid Build Coastguard Worker static constexpr size_t N = sizeof(I32) / sizeof(int32_t);
66*c8dee2aaSAndroid Build Coastguard Worker
67*c8dee2aaSAndroid Build Coastguard Worker for (int value = 0; value < (1 << N); ++value) {
68*c8dee2aaSAndroid Build Coastguard Worker // Load masks corresponding to the bit-pattern of `value` into lanes of `i`.
69*c8dee2aaSAndroid Build Coastguard Worker std::array<int32_t, N> masks = make_masks<N>(value);
70*c8dee2aaSAndroid Build Coastguard Worker I32 i = sk_unaligned_load<I32>(masks.data());
71*c8dee2aaSAndroid Build Coastguard Worker
72*c8dee2aaSAndroid Build Coastguard Worker // Verify that the raster pipeline all() matches expectations.
73*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, SK_OPTS_NS::all(i) == std::all_of(masks.begin(), masks.end(),
74*c8dee2aaSAndroid Build Coastguard Worker [](int32_t m) { return m != 0; }));
75*c8dee2aaSAndroid Build Coastguard Worker }
76*c8dee2aaSAndroid Build Coastguard Worker }
77*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(SkRasterPipelineOpts_Sin,r)78*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(SkRasterPipelineOpts_Sin, r) {
79*c8dee2aaSAndroid Build Coastguard Worker constexpr float Pi = SK_ScalarPI;
80*c8dee2aaSAndroid Build Coastguard Worker constexpr float kTolerance = 0.000875f;
81*c8dee2aaSAndroid Build Coastguard Worker for (float rad = -5*Pi; rad <= 5*Pi; rad += 0.1f) {
82*c8dee2aaSAndroid Build Coastguard Worker F result = SK_OPTS_NS::sin_(F_(rad));
83*c8dee2aaSAndroid Build Coastguard Worker F expected = F_(std::sin(rad));
84*c8dee2aaSAndroid Build Coastguard Worker F delta = SK_OPTS_NS::abs_(expected - result);
85*c8dee2aaSAndroid Build Coastguard Worker
86*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, SK_OPTS_NS::all(delta < kTolerance));
87*c8dee2aaSAndroid Build Coastguard Worker }
88*c8dee2aaSAndroid Build Coastguard Worker }
89*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(SkRasterPipelineOpts_Cos,r)90*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(SkRasterPipelineOpts_Cos, r) {
91*c8dee2aaSAndroid Build Coastguard Worker constexpr float Pi = SK_ScalarPI;
92*c8dee2aaSAndroid Build Coastguard Worker constexpr float kTolerance = 0.000875f;
93*c8dee2aaSAndroid Build Coastguard Worker for (float rad = -5*Pi; rad <= 5*Pi; rad += 0.1f) {
94*c8dee2aaSAndroid Build Coastguard Worker F result = SK_OPTS_NS::cos_(F_(rad));
95*c8dee2aaSAndroid Build Coastguard Worker F expected = F_(std::cos(rad));
96*c8dee2aaSAndroid Build Coastguard Worker F delta = SK_OPTS_NS::abs_(expected - result);
97*c8dee2aaSAndroid Build Coastguard Worker
98*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, SK_OPTS_NS::all(delta < kTolerance));
99*c8dee2aaSAndroid Build Coastguard Worker }
100*c8dee2aaSAndroid Build Coastguard Worker }
101*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(SkRasterPipelineOpts_Tan,r)102*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(SkRasterPipelineOpts_Tan, r) {
103*c8dee2aaSAndroid Build Coastguard Worker // Our tangent diverges more as we get near infinities (x near +- Pi/2),
104*c8dee2aaSAndroid Build Coastguard Worker // so we bring in the domain a little.
105*c8dee2aaSAndroid Build Coastguard Worker constexpr float Pi = SK_ScalarPI;
106*c8dee2aaSAndroid Build Coastguard Worker constexpr float kEpsilon = 0.16f;
107*c8dee2aaSAndroid Build Coastguard Worker constexpr float kTolerance = 0.00175f;
108*c8dee2aaSAndroid Build Coastguard Worker
109*c8dee2aaSAndroid Build Coastguard Worker // Test against various multiples of Pi, to check our periodicity
110*c8dee2aaSAndroid Build Coastguard Worker for (float period : {0.0f, -3*Pi, 3*Pi}) {
111*c8dee2aaSAndroid Build Coastguard Worker for (float rad = -Pi/2 + kEpsilon; rad <= Pi/2 - kEpsilon; rad += 0.01f) {
112*c8dee2aaSAndroid Build Coastguard Worker F result = SK_OPTS_NS::tan_(F_(rad + period));
113*c8dee2aaSAndroid Build Coastguard Worker F expected = F_(std::tan(rad));
114*c8dee2aaSAndroid Build Coastguard Worker F delta = SK_OPTS_NS::abs_(expected - result);
115*c8dee2aaSAndroid Build Coastguard Worker
116*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, SK_OPTS_NS::all(delta < kTolerance));
117*c8dee2aaSAndroid Build Coastguard Worker }
118*c8dee2aaSAndroid Build Coastguard Worker }
119*c8dee2aaSAndroid Build Coastguard Worker }
120*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(SkRasterPipelineOpts_Asin,r)121*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(SkRasterPipelineOpts_Asin, r) {
122*c8dee2aaSAndroid Build Coastguard Worker constexpr float kTolerance = 0.00175f;
123*c8dee2aaSAndroid Build Coastguard Worker for (float x = -1; x <= 1; x += 1.0f/64) {
124*c8dee2aaSAndroid Build Coastguard Worker F result = SK_OPTS_NS::asin_(F_(x));
125*c8dee2aaSAndroid Build Coastguard Worker F expected = F_(asinf(x));
126*c8dee2aaSAndroid Build Coastguard Worker F delta = SK_OPTS_NS::abs_(expected - result);
127*c8dee2aaSAndroid Build Coastguard Worker
128*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, SK_OPTS_NS::all(delta < kTolerance));
129*c8dee2aaSAndroid Build Coastguard Worker }
130*c8dee2aaSAndroid Build Coastguard Worker }
131*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(SkRasterPipelineOpts_Acos,r)132*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(SkRasterPipelineOpts_Acos, r) {
133*c8dee2aaSAndroid Build Coastguard Worker constexpr float kTolerance = 0.00175f;
134*c8dee2aaSAndroid Build Coastguard Worker for (float x = -1; x <= 1; x += 1.0f/64) {
135*c8dee2aaSAndroid Build Coastguard Worker F result = SK_OPTS_NS::acos_(F_(x));
136*c8dee2aaSAndroid Build Coastguard Worker F expected = F_(acosf(x));
137*c8dee2aaSAndroid Build Coastguard Worker F delta = SK_OPTS_NS::abs_(expected - result);
138*c8dee2aaSAndroid Build Coastguard Worker
139*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, SK_OPTS_NS::all(delta < kTolerance));
140*c8dee2aaSAndroid Build Coastguard Worker }
141*c8dee2aaSAndroid Build Coastguard Worker }
142*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(SkRasterPipelineOpts_Atan,r)143*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(SkRasterPipelineOpts_Atan, r) {
144*c8dee2aaSAndroid Build Coastguard Worker constexpr float kTolerance = 0.00175f;
145*c8dee2aaSAndroid Build Coastguard Worker for (float x = -10.0f; x <= 10.0f; x += 0.1f) {
146*c8dee2aaSAndroid Build Coastguard Worker F result = SK_OPTS_NS::atan_(F_(x));
147*c8dee2aaSAndroid Build Coastguard Worker F expected = F_(atanf(x));
148*c8dee2aaSAndroid Build Coastguard Worker F delta = SK_OPTS_NS::abs_(expected - result);
149*c8dee2aaSAndroid Build Coastguard Worker
150*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, SK_OPTS_NS::all(delta < kTolerance));
151*c8dee2aaSAndroid Build Coastguard Worker }
152*c8dee2aaSAndroid Build Coastguard Worker }
153*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(SkRasterPipelineOpts_Atan2,r)154*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(SkRasterPipelineOpts_Atan2, r) {
155*c8dee2aaSAndroid Build Coastguard Worker constexpr float kTolerance = 0.00175f;
156*c8dee2aaSAndroid Build Coastguard Worker for (float y = -3.0f; y <= 3.0f; y += 0.1f) {
157*c8dee2aaSAndroid Build Coastguard Worker for (float x = -3.0f; x <= 3.0f; x += 0.1f) {
158*c8dee2aaSAndroid Build Coastguard Worker F result = SK_OPTS_NS::atan2_(F_(y), F_(x));
159*c8dee2aaSAndroid Build Coastguard Worker F expected = F_(std::atan2(y, x));
160*c8dee2aaSAndroid Build Coastguard Worker F delta = SK_OPTS_NS::abs_(expected - result);
161*c8dee2aaSAndroid Build Coastguard Worker
162*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, SK_OPTS_NS::all(delta < kTolerance));
163*c8dee2aaSAndroid Build Coastguard Worker }
164*c8dee2aaSAndroid Build Coastguard Worker }
165*c8dee2aaSAndroid Build Coastguard Worker }
166*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(SkRasterPipelineOpts_Log2,r)167*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(SkRasterPipelineOpts_Log2, r) {
168*c8dee2aaSAndroid Build Coastguard Worker constexpr float kTolerance = 0.001f;
169*c8dee2aaSAndroid Build Coastguard Worker for (float value : {0.25f, 0.5f, 1.0f, 2.0f, 4.0f, 8.0f}) {
170*c8dee2aaSAndroid Build Coastguard Worker F result = SK_OPTS_NS::approx_log2(F_(value));
171*c8dee2aaSAndroid Build Coastguard Worker F expected = F_(std::log2(value));
172*c8dee2aaSAndroid Build Coastguard Worker F delta = SK_OPTS_NS::abs_(expected - result);
173*c8dee2aaSAndroid Build Coastguard Worker
174*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, SK_OPTS_NS::all(delta < kTolerance));
175*c8dee2aaSAndroid Build Coastguard Worker }
176*c8dee2aaSAndroid Build Coastguard Worker }
177*c8dee2aaSAndroid Build Coastguard Worker
DEF_TEST(SkRasterPipelineOpts_Pow2,r)178*c8dee2aaSAndroid Build Coastguard Worker DEF_TEST(SkRasterPipelineOpts_Pow2, r) {
179*c8dee2aaSAndroid Build Coastguard Worker constexpr float kTolerance = 0.001f;
180*c8dee2aaSAndroid Build Coastguard Worker for (float value : {-80, -5, -2, -1, 0, 1, 2, 3, 5}) {
181*c8dee2aaSAndroid Build Coastguard Worker F result = SK_OPTS_NS::approx_pow2(F_(value));
182*c8dee2aaSAndroid Build Coastguard Worker F expected = F_(std::pow(2.0, value));
183*c8dee2aaSAndroid Build Coastguard Worker F delta = SK_OPTS_NS::abs_(expected - result);
184*c8dee2aaSAndroid Build Coastguard Worker
185*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, SK_OPTS_NS::all(delta < kTolerance));
186*c8dee2aaSAndroid Build Coastguard Worker }
187*c8dee2aaSAndroid Build Coastguard Worker
188*c8dee2aaSAndroid Build Coastguard Worker F result = SK_OPTS_NS::approx_pow2(F_(160));
189*c8dee2aaSAndroid Build Coastguard Worker REPORTER_ASSERT(r, SK_OPTS_NS::all(result == INFINITY));
190*c8dee2aaSAndroid Build Coastguard Worker }
191