xref: /aosp_15_r20/external/deqp/framework/common/tcuCPUWarmup.cpp (revision 35238bce31c2a825756842865a792f8cf7f89930)
1*35238bceSAndroid Build Coastguard Worker /*-------------------------------------------------------------------------
2*35238bceSAndroid Build Coastguard Worker  * drawElements Quality Program Tester Core
3*35238bceSAndroid Build Coastguard Worker  * ----------------------------------------
4*35238bceSAndroid Build Coastguard Worker  *
5*35238bceSAndroid Build Coastguard Worker  * Copyright 2014 The Android Open Source Project
6*35238bceSAndroid Build Coastguard Worker  *
7*35238bceSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
8*35238bceSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
9*35238bceSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
10*35238bceSAndroid Build Coastguard Worker  *
11*35238bceSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
12*35238bceSAndroid Build Coastguard Worker  *
13*35238bceSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
14*35238bceSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
15*35238bceSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16*35238bceSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
17*35238bceSAndroid Build Coastguard Worker  * limitations under the License.
18*35238bceSAndroid Build Coastguard Worker  *
19*35238bceSAndroid Build Coastguard Worker  *//*!
20*35238bceSAndroid Build Coastguard Worker  * \file
21*35238bceSAndroid Build Coastguard Worker  * \brief CPU warm-up utility, used to counteract CPU throttling.
22*35238bceSAndroid Build Coastguard Worker  *//*--------------------------------------------------------------------*/
23*35238bceSAndroid Build Coastguard Worker 
24*35238bceSAndroid Build Coastguard Worker #include "tcuCPUWarmup.hpp"
25*35238bceSAndroid Build Coastguard Worker #include "deDefs.hpp"
26*35238bceSAndroid Build Coastguard Worker #include "deMath.h"
27*35238bceSAndroid Build Coastguard Worker #include "deClock.h"
28*35238bceSAndroid Build Coastguard Worker 
29*35238bceSAndroid Build Coastguard Worker #include <algorithm>
30*35238bceSAndroid Build Coastguard Worker 
31*35238bceSAndroid Build Coastguard Worker namespace tcu
32*35238bceSAndroid Build Coastguard Worker {
33*35238bceSAndroid Build Coastguard Worker 
34*35238bceSAndroid Build Coastguard Worker namespace warmupCPUInternal
35*35238bceSAndroid Build Coastguard Worker {
36*35238bceSAndroid Build Coastguard Worker 
37*35238bceSAndroid Build Coastguard Worker volatile Unused g_unused;
38*35238bceSAndroid Build Coastguard Worker 
39*35238bceSAndroid Build Coastguard Worker }
40*35238bceSAndroid Build Coastguard Worker 
41*35238bceSAndroid Build Coastguard Worker template <typename T, int Size>
floatMedian(const T (& v)[Size])42*35238bceSAndroid Build Coastguard Worker static inline float floatMedian(const T (&v)[Size])
43*35238bceSAndroid Build Coastguard Worker {
44*35238bceSAndroid Build Coastguard Worker     T temp[Size];
45*35238bceSAndroid Build Coastguard Worker     for (int i = 0; i < Size; i++)
46*35238bceSAndroid Build Coastguard Worker         temp[i] = v[i];
47*35238bceSAndroid Build Coastguard Worker 
48*35238bceSAndroid Build Coastguard Worker     std::sort(DE_ARRAY_BEGIN(temp), DE_ARRAY_END(temp));
49*35238bceSAndroid Build Coastguard Worker 
50*35238bceSAndroid Build Coastguard Worker     return Size % 2 == 0 ? 0.5f * ((float)temp[Size / 2 - 1] + (float)temp[Size / 2]) : (float)temp[Size / 2];
51*35238bceSAndroid Build Coastguard Worker }
52*35238bceSAndroid Build Coastguard Worker 
53*35238bceSAndroid Build Coastguard Worker template <typename T, int Size>
floatRelativeMedianAbsoluteDeviation(const T (& v)[Size])54*35238bceSAndroid Build Coastguard Worker static inline float floatRelativeMedianAbsoluteDeviation(const T (&v)[Size])
55*35238bceSAndroid Build Coastguard Worker {
56*35238bceSAndroid Build Coastguard Worker     const float median = floatMedian(v);
57*35238bceSAndroid Build Coastguard Worker     float absoluteDeviations[Size];
58*35238bceSAndroid Build Coastguard Worker 
59*35238bceSAndroid Build Coastguard Worker     for (int i = 0; i < Size; i++)
60*35238bceSAndroid Build Coastguard Worker         absoluteDeviations[i] = deFloatAbs((float)v[i] - median);
61*35238bceSAndroid Build Coastguard Worker 
62*35238bceSAndroid Build Coastguard Worker     return floatMedian(absoluteDeviations) / median;
63*35238bceSAndroid Build Coastguard Worker }
64*35238bceSAndroid Build Coastguard Worker 
unusedComputation(float initial,int numIterations)65*35238bceSAndroid Build Coastguard Worker static inline float unusedComputation(float initial, int numIterations)
66*35238bceSAndroid Build Coastguard Worker {
67*35238bceSAndroid Build Coastguard Worker     float a = initial;
68*35238bceSAndroid Build Coastguard Worker     int b   = 123;
69*35238bceSAndroid Build Coastguard Worker 
70*35238bceSAndroid Build Coastguard Worker     for (int i = 0; i < numIterations; i++)
71*35238bceSAndroid Build Coastguard Worker     {
72*35238bceSAndroid Build Coastguard Worker         // Arbitrary computations.
73*35238bceSAndroid Build Coastguard Worker         for (int j = 0; j < 4; j++)
74*35238bceSAndroid Build Coastguard Worker         {
75*35238bceSAndroid Build Coastguard Worker             a = deFloatCos(a + (float)b);
76*35238bceSAndroid Build Coastguard Worker             b = (b + 63) % 107 + de::abs((int)(a * 10.0f));
77*35238bceSAndroid Build Coastguard Worker         }
78*35238bceSAndroid Build Coastguard Worker     }
79*35238bceSAndroid Build Coastguard Worker 
80*35238bceSAndroid Build Coastguard Worker     return a + (float)b;
81*35238bceSAndroid Build Coastguard Worker }
82*35238bceSAndroid Build Coastguard Worker 
warmupCPU(void)83*35238bceSAndroid Build Coastguard Worker void warmupCPU(void)
84*35238bceSAndroid Build Coastguard Worker {
85*35238bceSAndroid Build Coastguard Worker     float unused        = *warmupCPUInternal::g_unused.m_v;
86*35238bceSAndroid Build Coastguard Worker     int computationSize = 1;
87*35238bceSAndroid Build Coastguard Worker 
88*35238bceSAndroid Build Coastguard Worker     // Do a rough calibration for computationSize to get unusedComputation's running time above a certain threshold.
89*35238bceSAndroid Build Coastguard Worker     while (computationSize <
90*35238bceSAndroid Build Coastguard Worker            1 << 30) // \note This condition is unlikely to be met. The "real" loop exit is the break below.
91*35238bceSAndroid Build Coastguard Worker     {
92*35238bceSAndroid Build Coastguard Worker         const float singleMeasurementThreshold = 10000.0f;
93*35238bceSAndroid Build Coastguard Worker         const int numMeasurements              = 3;
94*35238bceSAndroid Build Coastguard Worker         int64_t times[numMeasurements];
95*35238bceSAndroid Build Coastguard Worker 
96*35238bceSAndroid Build Coastguard Worker         for (int i = 0; i < numMeasurements; i++)
97*35238bceSAndroid Build Coastguard Worker         {
98*35238bceSAndroid Build Coastguard Worker             const uint64_t startTime = deGetMicroseconds();
99*35238bceSAndroid Build Coastguard Worker             unused                   = unusedComputation(unused, computationSize);
100*35238bceSAndroid Build Coastguard Worker             times[i]                 = (int64_t)(deGetMicroseconds() - startTime);
101*35238bceSAndroid Build Coastguard Worker         }
102*35238bceSAndroid Build Coastguard Worker 
103*35238bceSAndroid Build Coastguard Worker         if (floatMedian(times) >= singleMeasurementThreshold)
104*35238bceSAndroid Build Coastguard Worker             break;
105*35238bceSAndroid Build Coastguard Worker 
106*35238bceSAndroid Build Coastguard Worker         computationSize *= 2;
107*35238bceSAndroid Build Coastguard Worker     }
108*35238bceSAndroid Build Coastguard Worker 
109*35238bceSAndroid Build Coastguard Worker     // Do unusedComputations until running time seems stable enough.
110*35238bceSAndroid Build Coastguard Worker     {
111*35238bceSAndroid Build Coastguard Worker         const int maxNumMeasurements                         = 50;
112*35238bceSAndroid Build Coastguard Worker         const int numConsecutiveMeasurementsRequired         = 5;
113*35238bceSAndroid Build Coastguard Worker         const float relativeMedianAbsoluteDeviationThreshold = 0.05f;
114*35238bceSAndroid Build Coastguard Worker         int64_t latestTimes[numConsecutiveMeasurementsRequired];
115*35238bceSAndroid Build Coastguard Worker 
116*35238bceSAndroid Build Coastguard Worker         for (int measurementNdx = 0;
117*35238bceSAndroid Build Coastguard Worker 
118*35238bceSAndroid Build Coastguard Worker              measurementNdx < maxNumMeasurements &&
119*35238bceSAndroid Build Coastguard Worker              (measurementNdx < numConsecutiveMeasurementsRequired ||
120*35238bceSAndroid Build Coastguard Worker               floatRelativeMedianAbsoluteDeviation(latestTimes) > relativeMedianAbsoluteDeviationThreshold);
121*35238bceSAndroid Build Coastguard Worker 
122*35238bceSAndroid Build Coastguard Worker              measurementNdx++)
123*35238bceSAndroid Build Coastguard Worker         {
124*35238bceSAndroid Build Coastguard Worker             const uint64_t startTime = deGetMicroseconds();
125*35238bceSAndroid Build Coastguard Worker             unused                   = unusedComputation(unused, computationSize);
126*35238bceSAndroid Build Coastguard Worker             latestTimes[measurementNdx % numConsecutiveMeasurementsRequired] =
127*35238bceSAndroid Build Coastguard Worker                 (int64_t)(deGetMicroseconds() - startTime);
128*35238bceSAndroid Build Coastguard Worker         }
129*35238bceSAndroid Build Coastguard Worker     }
130*35238bceSAndroid Build Coastguard Worker 
131*35238bceSAndroid Build Coastguard Worker     *warmupCPUInternal::g_unused.m_v = unused;
132*35238bceSAndroid Build Coastguard Worker }
133*35238bceSAndroid Build Coastguard Worker 
134*35238bceSAndroid Build Coastguard Worker } // namespace tcu
135