| Mat Bevilacqua | 8160121 | 2020-07-30 17:26:45 -0700 | [diff] [blame] | 1 | /* | 
|  | 2 | * Copyright (C) 2020 The Android Open Source Project | 
|  | 3 | * | 
|  | 4 | * Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | 5 | * you may not use this file except in compliance with the License. | 
|  | 6 | * You may obtain a copy of the License at | 
|  | 7 | * | 
|  | 8 | *      http://www.apache.org/licenses/LICENSE-2.0 | 
|  | 9 | * | 
|  | 10 | * Unless required by applicable law or agreed to in writing, software | 
|  | 11 | * distributed under the License is distributed on an "AS IS" BASIS, | 
|  | 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | 13 | * See the License for the specific language governing permissions and | 
|  | 14 | * limitations under the License. | 
|  | 15 | */ | 
|  | 16 |  | 
|  | 17 | package com.android.server.powerstats; | 
|  | 18 |  | 
|  | 19 | import java.io.FileInputStream; | 
|  | 20 | import java.io.IOException; | 
|  | 21 |  | 
|  | 22 | /** | 
|  | 23 | * This class implements a utility to parse ODPM data out | 
|  | 24 | * of incident reports contained in bugreports.  The data | 
|  | 25 | * is output to STDOUT in csv format. | 
|  | 26 | */ | 
|  | 27 | public class PowerStatsServiceProtoParser { | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 28 | private static void printEnergyMeterInfo(PowerStatsServiceMeterProto proto) { | 
| Mat Bevilacqua | 8160121 | 2020-07-30 17:26:45 -0700 | [diff] [blame] | 29 | String csvHeader = new String(); | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 30 | for (int i = 0; i < proto.getChannelCount(); i++) { | 
|  | 31 | ChannelProto energyMeterInfo = proto.getChannel(i); | 
|  | 32 | csvHeader += "Index,Timestamp,Duration," + energyMeterInfo.getId() | 
| Mat Bevilacqua | 1e271f5 | 2021-01-27 14:17:49 -0800 | [diff] [blame] | 33 | + "/" + energyMeterInfo.getName() + "/" + energyMeterInfo.getSubsystem() + ","; | 
| Mat Bevilacqua | 8160121 | 2020-07-30 17:26:45 -0700 | [diff] [blame] | 34 | } | 
|  | 35 | System.out.println(csvHeader); | 
|  | 36 | } | 
|  | 37 |  | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 38 | private static void printEnergyMeasurements(PowerStatsServiceMeterProto proto) { | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 39 | int energyMeterInfoCount = proto.getChannelCount(); | 
| Mat Bevilacqua | 8160121 | 2020-07-30 17:26:45 -0700 | [diff] [blame] | 40 |  | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 41 | if (energyMeterInfoCount > 0) { | 
|  | 42 | int energyMeasurementCount = proto.getEnergyMeasurementCount(); | 
|  | 43 | int energyMeasurementSetCount = energyMeasurementCount / energyMeterInfoCount; | 
| Mat Bevilacqua | 8160121 | 2020-07-30 17:26:45 -0700 | [diff] [blame] | 44 |  | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 45 | for (int i = 0; i < energyMeasurementSetCount; i++) { | 
| Mat Bevilacqua | 8160121 | 2020-07-30 17:26:45 -0700 | [diff] [blame] | 46 | String csvRow = new String(); | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 47 | for (int j = 0; j < energyMeterInfoCount; j++) { | 
|  | 48 | EnergyMeasurementProto energyMeasurement = | 
|  | 49 | proto.getEnergyMeasurement(i * energyMeterInfoCount + j); | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 50 | csvRow += energyMeasurement.getId() + "," | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 51 | + energyMeasurement.getTimestampMs() + "," | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 52 | + energyMeasurement.getDurationMs() + "," | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 53 | + energyMeasurement.getEnergyUws() + ","; | 
| Mat Bevilacqua | 8160121 | 2020-07-30 17:26:45 -0700 | [diff] [blame] | 54 | } | 
|  | 55 | System.out.println(csvRow); | 
|  | 56 | } | 
|  | 57 | } else { | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 58 | System.out.println("Error:  energyMeterInfoCount is zero"); | 
|  | 59 | } | 
|  | 60 | } | 
|  | 61 |  | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 62 | private static void printEnergyConsumer(PowerStatsServiceModelProto proto) { | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 63 | String csvHeader = new String(); | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 64 | for (int i = 0; i < proto.getEnergyConsumerCount(); i++) { | 
|  | 65 | EnergyConsumerProto energyConsumer = proto.getEnergyConsumer(i); | 
|  | 66 | csvHeader += "Index,Timestamp," + energyConsumer.getId() + "/" | 
|  | 67 | + energyConsumer.getOrdinal() + "/" | 
|  | 68 | + energyConsumer.getType() + "/" | 
|  | 69 | + energyConsumer.getName() + ","; | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 70 | } | 
|  | 71 | System.out.println(csvHeader); | 
|  | 72 | } | 
|  | 73 |  | 
|  | 74 | private static void printEnergyConsumerResults(PowerStatsServiceModelProto proto) { | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 75 | int energyConsumerCount = proto.getEnergyConsumerCount(); | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 76 |  | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 77 | if (energyConsumerCount > 0) { | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 78 | int energyConsumerResultCount = proto.getEnergyConsumerResultCount(); | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 79 | int energyConsumerResultSetCount = energyConsumerResultCount / energyConsumerCount; | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 80 |  | 
|  | 81 | for (int i = 0; i < energyConsumerResultSetCount; i++) { | 
|  | 82 | String csvRow = new String(); | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 83 | for (int j = 0; j < energyConsumerCount; j++) { | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 84 | EnergyConsumerResultProto energyConsumerResult = | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 85 | proto.getEnergyConsumerResult(i * energyConsumerCount + j); | 
|  | 86 | csvRow += energyConsumerResult.getId() + "," | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 87 | + energyConsumerResult.getTimestampMs() + "," | 
|  | 88 | + energyConsumerResult.getEnergyUws() + ","; | 
| Mat Bevilacqua | 3bf7191 | 2021-01-19 17:48:50 -0800 | [diff] [blame] | 89 | for (int k = 0; k < energyConsumerResult.getAttributionCount(); k++) { | 
|  | 90 | final EnergyConsumerAttributionProto energyConsumerAttribution = | 
|  | 91 | energyConsumerResult.getAttribution(k); | 
|  | 92 | csvRow += energyConsumerAttribution.getUid() + "," | 
|  | 93 | + energyConsumerAttribution.getEnergyUws() + ","; | 
|  | 94 | } | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 95 | } | 
|  | 96 | System.out.println(csvRow); | 
|  | 97 | } | 
|  | 98 | } else { | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 99 | System.out.println("Error:  energyConsumerCount is zero"); | 
| Mat Bevilacqua | 8160121 | 2020-07-30 17:26:45 -0700 | [diff] [blame] | 100 | } | 
|  | 101 | } | 
|  | 102 |  | 
| Mat Bevilacqua | 723784a | 2020-12-22 09:57:01 -0800 | [diff] [blame] | 103 | private static void printPowerEntityInfo(PowerStatsServiceResidencyProto proto) { | 
|  | 104 | String csvHeader = new String(); | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 105 | for (int i = 0; i < proto.getPowerEntityCount(); i++) { | 
|  | 106 | PowerEntityProto powerEntity = proto.getPowerEntity(i); | 
|  | 107 | csvHeader += powerEntity.getId() + "," + powerEntity.getName() + ","; | 
|  | 108 | for (int j = 0; j < powerEntity.getStatesCount(); j++) { | 
|  | 109 | StateProto state = powerEntity.getStates(j); | 
|  | 110 | csvHeader += state.getId() + "," + state.getName() + ","; | 
| Mat Bevilacqua | 723784a | 2020-12-22 09:57:01 -0800 | [diff] [blame] | 111 | } | 
|  | 112 | } | 
|  | 113 | System.out.println(csvHeader); | 
|  | 114 | } | 
|  | 115 |  | 
|  | 116 | private static void printStateResidencyResult(PowerStatsServiceResidencyProto proto) { | 
|  | 117 | for (int i = 0; i < proto.getStateResidencyResultCount(); i++) { | 
|  | 118 | String csvRow = new String(); | 
|  | 119 |  | 
|  | 120 | StateResidencyResultProto stateResidencyResult = proto.getStateResidencyResult(i); | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 121 | csvRow += stateResidencyResult.getId() + ","; | 
| Mat Bevilacqua | 723784a | 2020-12-22 09:57:01 -0800 | [diff] [blame] | 122 |  | 
|  | 123 | for (int j = 0; j < stateResidencyResult.getStateResidencyDataCount(); j++) { | 
|  | 124 | StateResidencyProto stateResidency = stateResidencyResult.getStateResidencyData(j); | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 125 | csvRow += stateResidency.getId() + "," | 
| Mat Bevilacqua | 723784a | 2020-12-22 09:57:01 -0800 | [diff] [blame] | 126 | + stateResidency.getTotalTimeInStateMs() + "," | 
|  | 127 | + stateResidency.getTotalStateEntryCount() + "," | 
|  | 128 | + stateResidency.getLastEntryTimestampMs() + ","; | 
|  | 129 | } | 
|  | 130 | System.out.println(csvRow); | 
|  | 131 | } | 
|  | 132 | } | 
|  | 133 |  | 
| Mat Bevilacqua | 8160121 | 2020-07-30 17:26:45 -0700 | [diff] [blame] | 134 | private static void generateCsvFile(String pathToIncidentReport) { | 
|  | 135 | try { | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 136 | // Print power meter data. | 
|  | 137 | IncidentReportMeterProto irMeterProto = | 
|  | 138 | IncidentReportMeterProto.parseFrom(new FileInputStream(pathToIncidentReport)); | 
| Mat Bevilacqua | 8160121 | 2020-07-30 17:26:45 -0700 | [diff] [blame] | 139 |  | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 140 | if (irMeterProto.hasIncidentReport()) { | 
|  | 141 | PowerStatsServiceMeterProto pssMeterProto = irMeterProto.getIncidentReport(); | 
|  | 142 | printEnergyMeterInfo(pssMeterProto); | 
|  | 143 | printEnergyMeasurements(pssMeterProto); | 
| Mat Bevilacqua | 8160121 | 2020-07-30 17:26:45 -0700 | [diff] [blame] | 144 | } else { | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 145 | System.out.println("Meter incident report not found.  Exiting."); | 
|  | 146 | } | 
|  | 147 |  | 
|  | 148 | // Print power model data. | 
|  | 149 | IncidentReportModelProto irModelProto = | 
|  | 150 | IncidentReportModelProto.parseFrom(new FileInputStream(pathToIncidentReport)); | 
|  | 151 |  | 
|  | 152 | if (irModelProto.hasIncidentReport()) { | 
|  | 153 | PowerStatsServiceModelProto pssModelProto = irModelProto.getIncidentReport(); | 
| Mat Bevilacqua | 6058dc2 | 2021-01-20 17:45:20 -0800 | [diff] [blame] | 154 | printEnergyConsumer(pssModelProto); | 
| Mat Bevilacqua | a375ea0 | 2020-09-21 18:59:06 -0700 | [diff] [blame] | 155 | printEnergyConsumerResults(pssModelProto); | 
|  | 156 | } else { | 
|  | 157 | System.out.println("Model incident report not found.  Exiting."); | 
| Mat Bevilacqua | 8160121 | 2020-07-30 17:26:45 -0700 | [diff] [blame] | 158 | } | 
| Mat Bevilacqua | 723784a | 2020-12-22 09:57:01 -0800 | [diff] [blame] | 159 |  | 
|  | 160 | // Print state residency data. | 
|  | 161 | IncidentReportResidencyProto irResidencyProto = | 
|  | 162 | IncidentReportResidencyProto.parseFrom( | 
|  | 163 | new FileInputStream(pathToIncidentReport)); | 
|  | 164 |  | 
|  | 165 | if (irResidencyProto.hasIncidentReport()) { | 
|  | 166 | PowerStatsServiceResidencyProto pssResidencyProto = | 
|  | 167 | irResidencyProto.getIncidentReport(); | 
|  | 168 | printPowerEntityInfo(pssResidencyProto); | 
|  | 169 | printStateResidencyResult(pssResidencyProto); | 
|  | 170 | } else { | 
|  | 171 | System.out.println("Residency incident report not found.  Exiting."); | 
|  | 172 | } | 
|  | 173 |  | 
| Mat Bevilacqua | 8160121 | 2020-07-30 17:26:45 -0700 | [diff] [blame] | 174 | } catch (IOException e) { | 
|  | 175 | System.out.println("Unable to open incident report file: " + pathToIncidentReport); | 
|  | 176 | System.out.println(e); | 
|  | 177 | } | 
|  | 178 | } | 
|  | 179 |  | 
|  | 180 | /** | 
|  | 181 | * This is the entry point to parse the ODPM data out of incident reports. | 
|  | 182 | * It requires one argument which is the path to the incident_report.proto | 
|  | 183 | * file captured in a bugreport. | 
|  | 184 | * | 
|  | 185 | * @param args Path to incident_report.proto passed in from command line. | 
|  | 186 | */ | 
|  | 187 | public static void main(String[] args) { | 
|  | 188 | if (args.length > 0) { | 
|  | 189 | generateCsvFile(args[0]); | 
|  | 190 | } else { | 
|  | 191 | System.err.println("Usage: PowerStatsServiceProtoParser <incident_report.proto>"); | 
|  | 192 | System.err.println("Missing path to incident_report.proto.  Exiting."); | 
|  | 193 | System.exit(1); | 
|  | 194 | } | 
|  | 195 | } | 
|  | 196 | } |