blob: ccbffc6f136ea931d4c2d35540cd46f8da4b94a4 [file] [log] [blame]
/*
* Copyright (C) 2017 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.
*/
#pragma once
#include "utils/Macros.h"
#include <utils/Timers.h>
#include <array>
#include <functional>
#include <tuple>
namespace android {
namespace uirenderer {
enum JankType {
kMissedVsync = 0,
kHighInputLatency,
kSlowUI,
kSlowSync,
kSlowRT,
kMissedDeadline,
// must be last
NUM_BUCKETS,
};
// For testing
class MockProfileData;
// Try to keep as small as possible, should match ASHMEM_SIZE in
// GraphicsStatsService.java
class ProfileData {
PREVENT_COPY_AND_ASSIGN(ProfileData);
public:
ProfileData() { reset(); }
void reset();
void mergeWith(const ProfileData& other);
void dump(int fd) const;
uint32_t findPercentile(int percentile) const;
uint32_t findGPUPercentile(int percentile) const;
void reportFrame(int64_t duration);
void reportGPUFrame(int64_t duration);
void reportJank() { mJankFrameCount++; }
void reportJankType(JankType type) { mJankTypeCounts[static_cast<int>(type)]++; }
uint32_t totalFrameCount() const { return mTotalFrameCount; }
uint32_t jankFrameCount() const { return mJankFrameCount; }
nsecs_t statsStartTime() const { return mStatStartTime; }
uint32_t jankTypeCount(JankType type) const { return mJankTypeCounts[static_cast<int>(type)]; }
struct HistogramEntry {
uint32_t renderTimeMs;
uint32_t frameCount;
};
void histogramForEach(const std::function<void(HistogramEntry)>& callback) const;
void histogramGPUForEach(const std::function<void(HistogramEntry)>& callback) const;
constexpr static int HistogramSize() {
return std::tuple_size<decltype(ProfileData::mFrameCounts)>::value +
std::tuple_size<decltype(ProfileData::mSlowFrameCounts)>::value;
}
constexpr static int GPUHistogramSize() {
return std::tuple_size<decltype(ProfileData::mGPUFrameCounts)>::value;
}
// Visible for testing
static uint32_t frameTimeForFrameCountIndex(uint32_t index);
static uint32_t frameTimeForSlowFrameCountIndex(uint32_t index);
static uint32_t GPUFrameTimeForFrameCountIndex(uint32_t index);
private:
// Open our guts up to unit tests
friend class MockProfileData;
std::array<uint32_t, NUM_BUCKETS> mJankTypeCounts;
// See comments on kBucket* constants for what this holds
std::array<uint32_t, 57> mFrameCounts;
// Holds a histogram of frame times in 50ms increments from 150ms to 5s
std::array<uint16_t, 97> mSlowFrameCounts;
// Holds a histogram of GPU draw times in 1ms increments. Frames longer than 25ms are placed in
// last bucket.
std::array<uint32_t, 26> mGPUFrameCounts;
uint32_t mTotalFrameCount;
uint32_t mJankFrameCount;
nsecs_t mStatStartTime;
};
// For testing
class MockProfileData : public ProfileData {
public:
std::array<uint32_t, NUM_BUCKETS>& editJankTypeCounts() { return mJankTypeCounts; }
std::array<uint32_t, 57>& editFrameCounts() { return mFrameCounts; }
std::array<uint16_t, 97>& editSlowFrameCounts() { return mSlowFrameCounts; }
uint32_t& editTotalFrameCount() { return mTotalFrameCount; }
uint32_t& editJankFrameCount() { return mJankFrameCount; }
nsecs_t& editStatStartTime() { return mStatStartTime; }
};
} /* namespace uirenderer */
} /* namespace android */