1// Copyright (C) 2021 The Android Open Source Project 2// 3// Licensed under the Apache License, Version 2.0 (the "License"); 4// you may not use this file except in compliance with the License. 5// You may obtain a copy of the License at 6// 7// http://www.apache.org/licenses/LICENSE-2.0 8// 9// Unless required by applicable law or agreed to in writing, software 10// distributed under the License is distributed on an "AS IS" BASIS, 11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12// See the License for the specific language governing permissions and 13// limitations under the License. 14 15import {Actions} from '../common/actions'; 16import {AggregateData, isEmptyData} from '../common/aggregation_data'; 17import {ConversionJobStatusUpdate} from '../common/conversion_jobs'; 18import { 19 LogBoundsKey, 20 LogEntriesKey, 21 LogExists, 22 LogExistsKey, 23} from '../common/logs'; 24import {MetricResult} from '../common/metric_data'; 25import {CurrentSearchResults, SearchSummary} from '../common/search_data'; 26 27import { 28 CounterDetails, 29 CpuProfileDetails, 30 FlamegraphDetails, 31 Flow, 32 FtracePanelData, 33 FtraceStat, 34 globals, 35 QuantizedLoad, 36 SliceDetails, 37 ThreadDesc, 38 ThreadStateDetails, 39} from './globals'; 40import {findCurrentSelection} from './keyboard_event_handler'; 41 42export function publishOverviewData( 43 data: {[key: string]: QuantizedLoad|QuantizedLoad[]}) { 44 for (const [key, value] of Object.entries(data)) { 45 if (!globals.overviewStore.has(key)) { 46 globals.overviewStore.set(key, []); 47 } 48 if (value instanceof Array) { 49 globals.overviewStore.get(key)!.push(...value); 50 } else { 51 globals.overviewStore.get(key)!.push(value); 52 } 53 } 54 globals.rafScheduler.scheduleRedraw(); 55} 56 57export function clearOverviewData() { 58 globals.overviewStore.clear(); 59 globals.rafScheduler.scheduleRedraw(); 60} 61 62export function publishTrackData(args: {id: string, data: {}}) { 63 globals.setTrackData(args.id, args.data); 64 if ([LogExistsKey, LogBoundsKey, LogEntriesKey].includes(args.id)) { 65 const data = globals.trackDataStore.get(LogExistsKey) as LogExists; 66 if (data && data.exists) globals.rafScheduler.scheduleFullRedraw(); 67 } else { 68 globals.rafScheduler.scheduleRedraw(); 69 } 70} 71 72export function publishMetricResult(metricResult: MetricResult) { 73 globals.setMetricResult(metricResult); 74 globals.publishRedraw(); 75} 76 77export function publishSelectedFlows(selectedFlows: Flow[]) { 78 globals.selectedFlows = selectedFlows; 79 globals.publishRedraw(); 80} 81 82export function publishCounterDetails(click: CounterDetails) { 83 globals.counterDetails = click; 84 globals.publishRedraw(); 85} 86 87export function publishFlamegraphDetails(click: FlamegraphDetails) { 88 globals.flamegraphDetails = click; 89 globals.publishRedraw(); 90} 91 92export function publishCpuProfileDetails(details: CpuProfileDetails) { 93 globals.cpuProfileDetails = details; 94 globals.publishRedraw(); 95} 96 97export function publishFtraceCounters(counters: FtraceStat[]) { 98 globals.ftraceCounters = counters; 99 globals.publishRedraw(); 100} 101 102export function publishConversionJobStatusUpdate( 103 job: ConversionJobStatusUpdate) { 104 globals.setConversionJobStatus(job.jobName, job.jobStatus); 105 globals.publishRedraw(); 106} 107 108export function publishLoading(numQueuedQueries: number) { 109 globals.numQueuedQueries = numQueuedQueries; 110 // TODO(hjd): Clean up loadingAnimation given that this now causes a full 111 // redraw anyways. Also this should probably just go via the global state. 112 globals.rafScheduler.scheduleFullRedraw(); 113} 114 115export function publishBufferUsage(args: {percentage: number}) { 116 globals.setBufferUsage(args.percentage); 117 globals.publishRedraw(); 118} 119 120export function publishSearch(args: SearchSummary) { 121 globals.searchSummary = args; 122 globals.publishRedraw(); 123} 124 125export function publishSearchResult(args: CurrentSearchResults) { 126 globals.currentSearchResults = args; 127 globals.publishRedraw(); 128} 129 130export function publishRecordingLog(args: {logs: string}) { 131 globals.setRecordingLog(args.logs); 132 globals.publishRedraw(); 133} 134 135export function publishTraceErrors(numErrors: number) { 136 globals.setTraceErrors(numErrors); 137 globals.publishRedraw(); 138} 139 140export function publishMetricError(error: string) { 141 globals.setMetricError(error); 142 globals.logging.logError(error, false); 143 globals.publishRedraw(); 144} 145 146export function publishAggregateData( 147 args: {data: AggregateData, kind: string}) { 148 globals.setAggregateData(args.kind, args.data); 149 if (!isEmptyData(args.data)) { 150 globals.dispatch(Actions.setCurrentTab({tab: args.data.tabName})); 151 } 152 globals.publishRedraw(); 153} 154 155export function publishQueryResult(args: {id: string, data?: {}}) { 156 globals.queryResults.set(args.id, args.data); 157 globals.dispatch(Actions.setCurrentTab({tab: `query_result_${args.id}`})); 158 globals.publishRedraw(); 159} 160 161export function publishThreads(data: ThreadDesc[]) { 162 globals.threads.clear(); 163 data.forEach((thread) => { 164 globals.threads.set(thread.utid, thread); 165 }); 166 globals.publishRedraw(); 167} 168 169export function publishSliceDetails(click: SliceDetails) { 170 globals.sliceDetails = click; 171 const id = click.id; 172 if (id !== undefined && id === globals.state.pendingScrollId) { 173 findCurrentSelection(); 174 globals.dispatch(Actions.setCurrentTab({tab: 'slice'})); 175 globals.dispatch(Actions.clearPendingScrollId({id: undefined})); 176 } 177 globals.publishRedraw(); 178} 179 180export function publishThreadStateDetails(click: ThreadStateDetails) { 181 globals.threadStateDetails = click; 182 globals.publishRedraw(); 183} 184 185export function publishConnectedFlows(connectedFlows: Flow[]) { 186 globals.connectedFlows = connectedFlows; 187 // If a chrome slice is selected and we have any flows in connectedFlows 188 // we will find the flows on the right and left of that slice to set a default 189 // focus. In all other cases the focusedFlowId(Left|Right) will be set to -1. 190 globals.dispatch(Actions.setHighlightedFlowLeftId({flowId: -1})); 191 globals.dispatch(Actions.setHighlightedFlowRightId({flowId: -1})); 192 if (globals.state.currentSelection?.kind === 'CHROME_SLICE') { 193 const sliceId = globals.state.currentSelection.id; 194 for (const flow of globals.connectedFlows) { 195 if (flow.begin.sliceId === sliceId) { 196 globals.dispatch(Actions.setHighlightedFlowRightId({flowId: flow.id})); 197 } 198 if (flow.end.sliceId === sliceId) { 199 globals.dispatch(Actions.setHighlightedFlowLeftId({flowId: flow.id})); 200 } 201 } 202 } 203 204 globals.publishRedraw(); 205} 206 207export function publishFtracePanelData(data: FtracePanelData) { 208 globals.ftracePanelData = data; 209 globals.publishRedraw(); 210} 211