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