• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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