1fb726d48Sopenharmony_ci/* 2fb726d48Sopenharmony_ci * Copyright (C) 2024 Shenzhen Kaihong Digital Industry Development Co., Ltd. 3fb726d48Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License"); 4fb726d48Sopenharmony_ci * you may not use this file except in compliance with the License. 5fb726d48Sopenharmony_ci * You may obtain a copy of the License at 6fb726d48Sopenharmony_ci * 7fb726d48Sopenharmony_ci * http://www.apache.org/licenses/LICENSE-2.0 8fb726d48Sopenharmony_ci * 9fb726d48Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software 10fb726d48Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS, 11fb726d48Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12fb726d48Sopenharmony_ci * See the License for the specific language governing permissions and 13fb726d48Sopenharmony_ci * limitations under the License. 14fb726d48Sopenharmony_ci */ 15fb726d48Sopenharmony_ci 16fb726d48Sopenharmony_ciimport { SpSystemTrace } from '../SpSystemTrace'; 17fb726d48Sopenharmony_ciimport { TraceRow } from '../trace/base/TraceRow'; 18fb726d48Sopenharmony_ciimport { renders } from '../../database/ui-worker/ProcedureWorker'; 19fb726d48Sopenharmony_ciimport { info } from '../../../log/Log'; 20fb726d48Sopenharmony_ciimport { HangStruct } from '../../database/ui-worker/ProcedureWorkerHang'; 21fb726d48Sopenharmony_ciimport { EmptyRender } from '../../database/ui-worker/cpu/ProcedureWorkerCPU'; 22fb726d48Sopenharmony_ciimport { queryHangData } from '../../database/sql/Hang.sql'; 23fb726d48Sopenharmony_ciimport { hangDataSender } from '../../database/data-trafic/HangDataSender'; 24fb726d48Sopenharmony_ciimport { BaseStruct } from '../../bean/BaseStruct'; 25fb726d48Sopenharmony_ciimport { Utils } from '../trace/base/Utils'; 26fb726d48Sopenharmony_ci 27fb726d48Sopenharmony_ciexport type HangType = 'Instant' | 'Circumstantial' | 'Micro' | 'Severe' | ''; 28fb726d48Sopenharmony_ciconst TIME_MS = 1000000; 29fb726d48Sopenharmony_ci/// Hangs聚合泳道 30fb726d48Sopenharmony_ciexport class SpHangChart { 31fb726d48Sopenharmony_ci private trace: SpSystemTrace; 32fb726d48Sopenharmony_ci static funcNameMap: Map<number | string, string> = new Map(); 33fb726d48Sopenharmony_ci 34fb726d48Sopenharmony_ci constructor(trace: SpSystemTrace) { 35fb726d48Sopenharmony_ci this.trace = trace; 36fb726d48Sopenharmony_ci } 37fb726d48Sopenharmony_ci 38fb726d48Sopenharmony_ci static calculateHangType(dur: number): HangType { 39fb726d48Sopenharmony_ci const durMS = dur / TIME_MS; 40fb726d48Sopenharmony_ci if (durMS < 33) { 41fb726d48Sopenharmony_ci return ''; 42fb726d48Sopenharmony_ci } 43fb726d48Sopenharmony_ci else if (durMS < 100) { 44fb726d48Sopenharmony_ci return 'Instant'; 45fb726d48Sopenharmony_ci } 46fb726d48Sopenharmony_ci else if (durMS < 250) { 47fb726d48Sopenharmony_ci return 'Circumstantial'; 48fb726d48Sopenharmony_ci } 49fb726d48Sopenharmony_ci else if (durMS < 500) { 50fb726d48Sopenharmony_ci return 'Micro'; 51fb726d48Sopenharmony_ci } 52fb726d48Sopenharmony_ci else { 53fb726d48Sopenharmony_ci return 'Severe'; 54fb726d48Sopenharmony_ci } 55fb726d48Sopenharmony_ci } 56fb726d48Sopenharmony_ci 57fb726d48Sopenharmony_ci async init(): Promise<void> { 58fb726d48Sopenharmony_ci SpHangChart.funcNameMap = Utils.getInstance().getCallStatckMap(); 59fb726d48Sopenharmony_ci let folder = await this.initFolder(); 60fb726d48Sopenharmony_ci await this.initData(folder); 61fb726d48Sopenharmony_ci } 62fb726d48Sopenharmony_ci 63fb726d48Sopenharmony_ci private hangSupplierFrame( 64fb726d48Sopenharmony_ci traceRow: TraceRow<HangStruct>, 65fb726d48Sopenharmony_ci it: { 66fb726d48Sopenharmony_ci id: number; 67fb726d48Sopenharmony_ci name: string; 68fb726d48Sopenharmony_ci num: number; 69fb726d48Sopenharmony_ci } 70fb726d48Sopenharmony_ci ): void { 71fb726d48Sopenharmony_ci traceRow.supplierFrame = (): Promise<HangStruct[]> => { 72fb726d48Sopenharmony_ci let promiseData = hangDataSender(it.id, traceRow); 73fb726d48Sopenharmony_ci if (promiseData === null) { 74fb726d48Sopenharmony_ci return new Promise<Array<HangStruct>>((resolve) => resolve([])); 75fb726d48Sopenharmony_ci } else { 76fb726d48Sopenharmony_ci return promiseData.then((resultHang: Array<HangStruct>) => 77fb726d48Sopenharmony_ci resultHang.map(hangItem => ({ 78fb726d48Sopenharmony_ci ...hangItem, 79fb726d48Sopenharmony_ci pname: it.name, 80fb726d48Sopenharmony_ci type: SpHangChart.calculateHangType(hangItem.dur!), 81fb726d48Sopenharmony_ci content: SpHangChart.funcNameMap.get(hangItem.id!) 82fb726d48Sopenharmony_ci })) 83fb726d48Sopenharmony_ci ); 84fb726d48Sopenharmony_ci } 85fb726d48Sopenharmony_ci }; 86fb726d48Sopenharmony_ci } 87fb726d48Sopenharmony_ci 88fb726d48Sopenharmony_ci private hangThreadHandler( 89fb726d48Sopenharmony_ci traceRow: TraceRow<HangStruct>, 90fb726d48Sopenharmony_ci it: { 91fb726d48Sopenharmony_ci id: number; 92fb726d48Sopenharmony_ci name: string; 93fb726d48Sopenharmony_ci num: number; 94fb726d48Sopenharmony_ci }, 95fb726d48Sopenharmony_ci hangId: number 96fb726d48Sopenharmony_ci ): void { 97fb726d48Sopenharmony_ci traceRow.onThreadHandler = (useCache): void => { 98fb726d48Sopenharmony_ci let context: CanvasRenderingContext2D; 99fb726d48Sopenharmony_ci if (traceRow.currentContext) { 100fb726d48Sopenharmony_ci context = traceRow.currentContext; 101fb726d48Sopenharmony_ci } else { 102fb726d48Sopenharmony_ci context = traceRow.collect ? this.trace.canvasFavoritePanelCtx! : this.trace.canvasPanelCtx!; 103fb726d48Sopenharmony_ci } 104fb726d48Sopenharmony_ci traceRow.canvasSave(context); 105fb726d48Sopenharmony_ci renders.hang.renderMainThread( 106fb726d48Sopenharmony_ci { 107fb726d48Sopenharmony_ci context: context, 108fb726d48Sopenharmony_ci useCache: useCache, 109fb726d48Sopenharmony_ci type: it.name, 110fb726d48Sopenharmony_ci index: hangId, 111fb726d48Sopenharmony_ci }, 112fb726d48Sopenharmony_ci traceRow 113fb726d48Sopenharmony_ci ); 114fb726d48Sopenharmony_ci traceRow.canvasRestore(context, this.trace); 115fb726d48Sopenharmony_ci }; 116fb726d48Sopenharmony_ci } 117fb726d48Sopenharmony_ci 118fb726d48Sopenharmony_ci /// 初始化聚合泳道信息 119fb726d48Sopenharmony_ci async initData(folder: TraceRow<BaseStruct>): Promise<void> { 120fb726d48Sopenharmony_ci let hangStartTime = new Date().getTime(); 121fb726d48Sopenharmony_ci let hangList = await queryHangData(); 122fb726d48Sopenharmony_ci if (hangList.length === 0) { 123fb726d48Sopenharmony_ci return; 124fb726d48Sopenharmony_ci } 125fb726d48Sopenharmony_ci this.trace.rowsEL?.appendChild(folder); 126fb726d48Sopenharmony_ci for (let i = 0; i < hangList.length; i++) { 127fb726d48Sopenharmony_ci const it: { 128fb726d48Sopenharmony_ci id: number, 129fb726d48Sopenharmony_ci name: string, 130fb726d48Sopenharmony_ci num: number, 131fb726d48Sopenharmony_ci } = hangList[i]; 132fb726d48Sopenharmony_ci let traceRow = TraceRow.skeleton<HangStruct>(); 133fb726d48Sopenharmony_ci traceRow.rowId = `${it.name ?? 'Process'} ${it.id}`; 134fb726d48Sopenharmony_ci traceRow.rowType = TraceRow.ROW_TYPE_HANG; 135fb726d48Sopenharmony_ci traceRow.rowParentId = folder.rowId; 136fb726d48Sopenharmony_ci traceRow.style.height = '40px'; 137fb726d48Sopenharmony_ci traceRow.name = `${it.name ?? 'Process'} ${it.id}`; 138fb726d48Sopenharmony_ci traceRow.rowHidden = !folder.expansion; 139fb726d48Sopenharmony_ci traceRow.setAttribute('children', ''); 140fb726d48Sopenharmony_ci traceRow.favoriteChangeHandler = this.trace.favoriteChangeHandler; 141fb726d48Sopenharmony_ci traceRow.selectChangeHandler = this.trace.selectChangeHandler; 142fb726d48Sopenharmony_ci this.hangSupplierFrame(traceRow, it); 143fb726d48Sopenharmony_ci traceRow.getCacheData = (args: unknown): Promise<Array<unknown>> => hangDataSender(it.id, traceRow, args); 144fb726d48Sopenharmony_ci traceRow.focusHandler = (ev): void => { 145fb726d48Sopenharmony_ci let hangStruct = HangStruct.hoverHangStruct; 146fb726d48Sopenharmony_ci this.trace?.displayTip( 147fb726d48Sopenharmony_ci traceRow, hangStruct, 148fb726d48Sopenharmony_ci `<span>${hangStruct?.type} ${hangStruct?.dur! / TIME_MS}ms</span>` 149fb726d48Sopenharmony_ci ); 150fb726d48Sopenharmony_ci }; 151fb726d48Sopenharmony_ci traceRow.findHoverStruct = (): void => { 152fb726d48Sopenharmony_ci HangStruct.hoverHangStruct = traceRow.getHoverStruct(); 153fb726d48Sopenharmony_ci }; 154fb726d48Sopenharmony_ci this.hangThreadHandler(traceRow, it, i); 155fb726d48Sopenharmony_ci folder.addChildTraceRow(traceRow); 156fb726d48Sopenharmony_ci } 157fb726d48Sopenharmony_ci let durTime = new Date().getTime() - hangStartTime; 158fb726d48Sopenharmony_ci info('The time to load the HangData is: ', durTime); 159fb726d48Sopenharmony_ci } 160fb726d48Sopenharmony_ci 161fb726d48Sopenharmony_ci async initFolder(): Promise<TraceRow<BaseStruct>> { 162fb726d48Sopenharmony_ci let hangFolder = TraceRow.skeleton(); 163fb726d48Sopenharmony_ci hangFolder.rowId = 'Hangs'; 164fb726d48Sopenharmony_ci hangFolder.index = 0; 165fb726d48Sopenharmony_ci hangFolder.rowType = TraceRow.ROW_TYPE_HANG_GROUP; 166fb726d48Sopenharmony_ci hangFolder.rowParentId = ''; 167fb726d48Sopenharmony_ci hangFolder.style.height = '40px'; 168fb726d48Sopenharmony_ci hangFolder.folder = true; 169fb726d48Sopenharmony_ci hangFolder.name = 'Hangs'; 170fb726d48Sopenharmony_ci hangFolder.favoriteChangeHandler = this.trace.favoriteChangeHandler; 171fb726d48Sopenharmony_ci hangFolder.selectChangeHandler = this.trace.selectChangeHandler; // @ts-ignore 172fb726d48Sopenharmony_ci hangFolder.supplier = (): Promise<unknown[]> => new Promise<Array<unknown>>((resolve) => resolve([])); 173fb726d48Sopenharmony_ci hangFolder.onThreadHandler = (useCache): void => { 174fb726d48Sopenharmony_ci hangFolder.canvasSave(this.trace.canvasPanelCtx!); 175fb726d48Sopenharmony_ci if (hangFolder.expansion) { 176fb726d48Sopenharmony_ci // @ts-ignore 177fb726d48Sopenharmony_ci this.trace.canvasPanelCtx?.clearRect(0, 0, hangFolder.frame.width, hangFolder.frame.height); 178fb726d48Sopenharmony_ci } else { 179fb726d48Sopenharmony_ci (renders.empty as EmptyRender).renderMainThread( 180fb726d48Sopenharmony_ci { 181fb726d48Sopenharmony_ci context: this.trace.canvasPanelCtx, 182fb726d48Sopenharmony_ci useCache: useCache, 183fb726d48Sopenharmony_ci type: '', 184fb726d48Sopenharmony_ci }, 185fb726d48Sopenharmony_ci hangFolder 186fb726d48Sopenharmony_ci ); 187fb726d48Sopenharmony_ci } 188fb726d48Sopenharmony_ci hangFolder.canvasRestore(this.trace.canvasPanelCtx!, this.trace); 189fb726d48Sopenharmony_ci }; 190fb726d48Sopenharmony_ci return hangFolder; 191fb726d48Sopenharmony_ci } 192fb726d48Sopenharmony_ci} 193