1*d57664e9SAndroid Build Coastguard Worker /* 2*d57664e9SAndroid Build Coastguard Worker * Copyright (C) 2017 The Android Open Source Project 3*d57664e9SAndroid Build Coastguard Worker * 4*d57664e9SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*d57664e9SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*d57664e9SAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*d57664e9SAndroid Build Coastguard Worker * 8*d57664e9SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*d57664e9SAndroid Build Coastguard Worker * 10*d57664e9SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*d57664e9SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*d57664e9SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*d57664e9SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*d57664e9SAndroid Build Coastguard Worker * limitations under the License. 15*d57664e9SAndroid Build Coastguard Worker */ 16*d57664e9SAndroid Build Coastguard Worker 17*d57664e9SAndroid Build Coastguard Worker #pragma once 18*d57664e9SAndroid Build Coastguard Worker 19*d57664e9SAndroid Build Coastguard Worker #include "Properties.h" 20*d57664e9SAndroid Build Coastguard Worker #include "utils/Macros.h" 21*d57664e9SAndroid Build Coastguard Worker 22*d57664e9SAndroid Build Coastguard Worker #include <utils/Timers.h> 23*d57664e9SAndroid Build Coastguard Worker 24*d57664e9SAndroid Build Coastguard Worker #include <array> 25*d57664e9SAndroid Build Coastguard Worker #include <functional> 26*d57664e9SAndroid Build Coastguard Worker #include <tuple> 27*d57664e9SAndroid Build Coastguard Worker 28*d57664e9SAndroid Build Coastguard Worker namespace android { 29*d57664e9SAndroid Build Coastguard Worker namespace uirenderer { 30*d57664e9SAndroid Build Coastguard Worker 31*d57664e9SAndroid Build Coastguard Worker enum JankType { 32*d57664e9SAndroid Build Coastguard Worker kMissedVsync = 0, 33*d57664e9SAndroid Build Coastguard Worker kHighInputLatency, 34*d57664e9SAndroid Build Coastguard Worker kSlowUI, 35*d57664e9SAndroid Build Coastguard Worker kSlowSync, 36*d57664e9SAndroid Build Coastguard Worker kSlowRT, 37*d57664e9SAndroid Build Coastguard Worker kMissedDeadline, 38*d57664e9SAndroid Build Coastguard Worker kMissedDeadlineLegacy, 39*d57664e9SAndroid Build Coastguard Worker 40*d57664e9SAndroid Build Coastguard Worker // must be last 41*d57664e9SAndroid Build Coastguard Worker NUM_BUCKETS, 42*d57664e9SAndroid Build Coastguard Worker }; 43*d57664e9SAndroid Build Coastguard Worker 44*d57664e9SAndroid Build Coastguard Worker // For testing 45*d57664e9SAndroid Build Coastguard Worker class MockProfileData; 46*d57664e9SAndroid Build Coastguard Worker 47*d57664e9SAndroid Build Coastguard Worker // Try to keep as small as possible, should match ASHMEM_SIZE in 48*d57664e9SAndroid Build Coastguard Worker // GraphicsStatsService.java 49*d57664e9SAndroid Build Coastguard Worker class ProfileData { 50*d57664e9SAndroid Build Coastguard Worker PREVENT_COPY_AND_ASSIGN(ProfileData); 51*d57664e9SAndroid Build Coastguard Worker 52*d57664e9SAndroid Build Coastguard Worker public: ProfileData()53*d57664e9SAndroid Build Coastguard Worker ProfileData() { reset(); } 54*d57664e9SAndroid Build Coastguard Worker 55*d57664e9SAndroid Build Coastguard Worker void reset(); 56*d57664e9SAndroid Build Coastguard Worker void mergeWith(const ProfileData& other); 57*d57664e9SAndroid Build Coastguard Worker void dump(int fd) const; 58*d57664e9SAndroid Build Coastguard Worker uint32_t findPercentile(int percentile) const; 59*d57664e9SAndroid Build Coastguard Worker uint32_t findGPUPercentile(int percentile) const; 60*d57664e9SAndroid Build Coastguard Worker 61*d57664e9SAndroid Build Coastguard Worker void reportFrame(int64_t duration); 62*d57664e9SAndroid Build Coastguard Worker void reportGPUFrame(int64_t duration); reportJank()63*d57664e9SAndroid Build Coastguard Worker void reportJank() { mJankFrameCount++; } reportJankLegacy()64*d57664e9SAndroid Build Coastguard Worker void reportJankLegacy() { mJankLegacyFrameCount++; } reportJankType(JankType type)65*d57664e9SAndroid Build Coastguard Worker void reportJankType(JankType type) { mJankTypeCounts[static_cast<int>(type)]++; } 66*d57664e9SAndroid Build Coastguard Worker totalFrameCount()67*d57664e9SAndroid Build Coastguard Worker uint32_t totalFrameCount() const { return mTotalFrameCount; } jankFrameCount()68*d57664e9SAndroid Build Coastguard Worker uint32_t jankFrameCount() const { return mJankFrameCount; } jankLegacyFrameCount()69*d57664e9SAndroid Build Coastguard Worker uint32_t jankLegacyFrameCount() const { return mJankLegacyFrameCount; } statsStartTime()70*d57664e9SAndroid Build Coastguard Worker nsecs_t statsStartTime() const { return mStatStartTime; } jankTypeCount(JankType type)71*d57664e9SAndroid Build Coastguard Worker uint32_t jankTypeCount(JankType type) const { return mJankTypeCounts[static_cast<int>(type)]; } pipelineType()72*d57664e9SAndroid Build Coastguard Worker RenderPipelineType pipelineType() const { return mPipelineType; } 73*d57664e9SAndroid Build Coastguard Worker 74*d57664e9SAndroid Build Coastguard Worker struct HistogramEntry { 75*d57664e9SAndroid Build Coastguard Worker uint32_t renderTimeMs; 76*d57664e9SAndroid Build Coastguard Worker uint32_t frameCount; 77*d57664e9SAndroid Build Coastguard Worker }; 78*d57664e9SAndroid Build Coastguard Worker void histogramForEach(const std::function<void(HistogramEntry)>& callback) const; 79*d57664e9SAndroid Build Coastguard Worker void histogramGPUForEach(const std::function<void(HistogramEntry)>& callback) const; 80*d57664e9SAndroid Build Coastguard Worker HistogramSize()81*d57664e9SAndroid Build Coastguard Worker constexpr static int HistogramSize() { 82*d57664e9SAndroid Build Coastguard Worker return std::tuple_size<decltype(ProfileData::mFrameCounts)>::value + 83*d57664e9SAndroid Build Coastguard Worker std::tuple_size<decltype(ProfileData::mSlowFrameCounts)>::value; 84*d57664e9SAndroid Build Coastguard Worker } 85*d57664e9SAndroid Build Coastguard Worker GPUHistogramSize()86*d57664e9SAndroid Build Coastguard Worker constexpr static int GPUHistogramSize() { 87*d57664e9SAndroid Build Coastguard Worker return std::tuple_size<decltype(ProfileData::mGPUFrameCounts)>::value; 88*d57664e9SAndroid Build Coastguard Worker } 89*d57664e9SAndroid Build Coastguard Worker 90*d57664e9SAndroid Build Coastguard Worker // Visible for testing 91*d57664e9SAndroid Build Coastguard Worker static uint32_t frameTimeForFrameCountIndex(uint32_t index); 92*d57664e9SAndroid Build Coastguard Worker static uint32_t frameTimeForSlowFrameCountIndex(uint32_t index); 93*d57664e9SAndroid Build Coastguard Worker static uint32_t GPUFrameTimeForFrameCountIndex(uint32_t index); 94*d57664e9SAndroid Build Coastguard Worker 95*d57664e9SAndroid Build Coastguard Worker private: 96*d57664e9SAndroid Build Coastguard Worker // Open our guts up to unit tests 97*d57664e9SAndroid Build Coastguard Worker friend class MockProfileData; 98*d57664e9SAndroid Build Coastguard Worker 99*d57664e9SAndroid Build Coastguard Worker std::array<uint32_t, NUM_BUCKETS> mJankTypeCounts; 100*d57664e9SAndroid Build Coastguard Worker // See comments on kBucket* constants for what this holds 101*d57664e9SAndroid Build Coastguard Worker std::array<uint32_t, 57> mFrameCounts; 102*d57664e9SAndroid Build Coastguard Worker // Holds a histogram of frame times in 50ms increments from 150ms to 5s 103*d57664e9SAndroid Build Coastguard Worker std::array<uint16_t, 97> mSlowFrameCounts; 104*d57664e9SAndroid Build Coastguard Worker // Holds a histogram of GPU draw times in 1ms increments. Frames longer than 25ms are placed in 105*d57664e9SAndroid Build Coastguard Worker // last bucket. 106*d57664e9SAndroid Build Coastguard Worker std::array<uint32_t, 26> mGPUFrameCounts; 107*d57664e9SAndroid Build Coastguard Worker 108*d57664e9SAndroid Build Coastguard Worker uint32_t mTotalFrameCount; 109*d57664e9SAndroid Build Coastguard Worker uint32_t mJankFrameCount; 110*d57664e9SAndroid Build Coastguard Worker uint32_t mJankLegacyFrameCount; 111*d57664e9SAndroid Build Coastguard Worker nsecs_t mStatStartTime; 112*d57664e9SAndroid Build Coastguard Worker 113*d57664e9SAndroid Build Coastguard Worker // true if HWUI renders with Vulkan pipeline 114*d57664e9SAndroid Build Coastguard Worker RenderPipelineType mPipelineType; 115*d57664e9SAndroid Build Coastguard Worker }; 116*d57664e9SAndroid Build Coastguard Worker 117*d57664e9SAndroid Build Coastguard Worker // For testing 118*d57664e9SAndroid Build Coastguard Worker class MockProfileData : public ProfileData { 119*d57664e9SAndroid Build Coastguard Worker public: editJankTypeCounts()120*d57664e9SAndroid Build Coastguard Worker std::array<uint32_t, NUM_BUCKETS>& editJankTypeCounts() { return mJankTypeCounts; } editFrameCounts()121*d57664e9SAndroid Build Coastguard Worker std::array<uint32_t, 57>& editFrameCounts() { return mFrameCounts; } editSlowFrameCounts()122*d57664e9SAndroid Build Coastguard Worker std::array<uint16_t, 97>& editSlowFrameCounts() { return mSlowFrameCounts; } editTotalFrameCount()123*d57664e9SAndroid Build Coastguard Worker uint32_t& editTotalFrameCount() { return mTotalFrameCount; } editJankFrameCount()124*d57664e9SAndroid Build Coastguard Worker uint32_t& editJankFrameCount() { return mJankFrameCount; } editStatStartTime()125*d57664e9SAndroid Build Coastguard Worker nsecs_t& editStatStartTime() { return mStatStartTime; } 126*d57664e9SAndroid Build Coastguard Worker }; 127*d57664e9SAndroid Build Coastguard Worker 128*d57664e9SAndroid Build Coastguard Worker } /* namespace uirenderer */ 129*d57664e9SAndroid Build Coastguard Worker } /* namespace android */ 130