blob: baa6c435ea6f3b978c89b49b4f06b53dcf250299 [file] [log] [blame]
/*-------------------------------------------------------------------------
* drawElements Quality Program Tester Core
* ----------------------------------------
*
* Copyright 2014 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
*//*!
* \file
* \brief CPU warm-up utility, used to counteract CPU throttling.
*//*--------------------------------------------------------------------*/
#include "tcuCPUWarmup.hpp"
#include "deDefs.hpp"
#include "deMath.h"
#include "deClock.h"
#include <algorithm>
namespace tcu
{
namespace warmupCPUInternal
{
volatile Unused g_unused;
}
template <typename T, int Size>
static inline float floatMedian (const T (&v)[Size])
{
T temp[Size];
for (int i = 0; i < Size; i++)
temp[i] = v[i];
std::sort(DE_ARRAY_BEGIN(temp), DE_ARRAY_END(temp));
return Size % 2 == 0
? 0.5f * ((float)temp[Size/2-1] + (float)temp[Size/2])
: (float)temp[Size/2];
}
template <typename T, int Size>
static inline float floatRelativeMedianAbsoluteDeviation (const T (&v)[Size])
{
const float median = floatMedian(v);
float absoluteDeviations[Size];
for (int i = 0; i < Size; i++)
absoluteDeviations[i] = deFloatAbs((float)v[i] - median);
return floatMedian(absoluteDeviations) / median;
}
static inline float unusedComputation (float initial, int numIterations)
{
float a = initial;
int b = 123;
for (int i = 0; i < numIterations; i++)
{
// Arbitrary computations.
for (int j = 0; j < 4; j++)
{
a = deFloatCos(a + (float)b);
b = (b + 63) % 107 + de::abs((int)(a*10.0f));
}
}
return a + (float)b;
}
void warmupCPU (void)
{
float unused = *warmupCPUInternal::g_unused.m_v;
int computationSize = 1;
// Do a rough calibration for computationSize to get unusedComputation's running time above a certain threshold.
while (computationSize < 1<<30) // \note This condition is unlikely to be met. The "real" loop exit is the break below.
{
const float singleMeasurementThreshold = 10000.0f;
const int numMeasurements = 3;
deInt64 times[numMeasurements];
for (int i = 0; i < numMeasurements; i++)
{
const deUint64 startTime = deGetMicroseconds();
unused = unusedComputation(unused, computationSize);
times[i] = (deInt64)(deGetMicroseconds() - startTime);
}
if (floatMedian(times) >= singleMeasurementThreshold)
break;
computationSize *= 2;
}
// Do unusedComputations until running time seems stable enough.
{
const int maxNumMeasurements = 50;
const int numConsecutiveMeasurementsRequired = 5;
const float relativeMedianAbsoluteDeviationThreshold = 0.05f;
deInt64 latestTimes[numConsecutiveMeasurementsRequired];
for (int measurementNdx = 0;
measurementNdx < maxNumMeasurements &&
(measurementNdx < numConsecutiveMeasurementsRequired ||
floatRelativeMedianAbsoluteDeviation(latestTimes) > relativeMedianAbsoluteDeviationThreshold);
measurementNdx++)
{
const deUint64 startTime = deGetMicroseconds();
unused = unusedComputation(unused, computationSize);
latestTimes[measurementNdx % numConsecutiveMeasurementsRequired] = (deInt64)(deGetMicroseconds() - startTime);
}
}
*warmupCPUInternal::g_unused.m_v = unused;
}
} // tcu