1// Copyright (C) 2018 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 {defer, Deferred} from '../base/deferred'; 16import {assertTrue} from '../base/logging'; 17import {WasmBridgeRequest, WasmBridgeResponse} from '../engine/wasm_bridge'; 18 19import {Engine, LoadingTracker} from './engine'; 20 21const activeWorkers = new Map<string, Worker>(); 22let warmWorker: null|Worker = null; 23 24function createWorker(): Worker { 25 return new Worker('engine_bundle.js'); 26} 27 28// Take the warm engine and start creating a new WASM engine in the background 29// for the next call. 30export function createWasmEngine(id: string): Worker { 31 if (warmWorker === null) { 32 throw new Error('warmupWasmEngine() not called'); 33 } 34 if (activeWorkers.has(id)) { 35 throw new Error(`Duplicate worker ID ${id}`); 36 } 37 const activeWorker = warmWorker; 38 warmWorker = createWorker(); 39 activeWorkers.set(id, activeWorker); 40 return activeWorker; 41} 42 43export function destroyWasmEngine(id: string) { 44 if (!activeWorkers.has(id)) { 45 throw new Error(`Cannot find worker ID ${id}`); 46 } 47 activeWorkers.get(id)!.terminate(); 48 activeWorkers.delete(id); 49} 50 51/** 52 * It's quite slow to compile WASM and (in Chrome) this happens every time 53 * a worker thread attempts to load a WASM module since there is no way to 54 * cache the compiled code currently. To mitigate this we can always keep a 55 * WASM backend 'ready to go' just waiting to be provided with a trace file. 56 * warmupWasmEngineWorker (together with getWasmEngineWorker) 57 * implement this behaviour. 58 */ 59export function warmupWasmEngine(): void { 60 if (warmWorker !== null) { 61 throw new Error('warmupWasmEngine() already called'); 62 } 63 warmWorker = createWorker(); 64} 65 66interface PendingRequest { 67 id: number; 68 respHandler: Deferred<Uint8Array>; 69} 70 71/** 72 * This implementation of Engine uses a WASM backend hosted in a separate 73 * worker thread. 74 */ 75export class WasmEngineProxy extends Engine { 76 readonly id: string; 77 private readonly worker: Worker; 78 private pendingRequests = new Array<PendingRequest>(); 79 private nextRequestId = 0; 80 81 constructor(id: string, worker: Worker, loadingTracker?: LoadingTracker) { 82 super(loadingTracker); 83 this.id = id; 84 this.worker = worker; 85 this.worker.onmessage = this.onMessage.bind(this); 86 } 87 88 async parse(reqData: Uint8Array): Promise<void> { 89 // We don't care about the response data (the method is actually a void). We 90 // just want to linearize and wait for the call to have been completed on 91 // the worker. 92 await this.queueRequest('trace_processor_parse', reqData); 93 } 94 95 async notifyEof(): Promise<void> { 96 // We don't care about the response data (the method is actually a void). We 97 // just want to linearize and wait for the call to have been completed on 98 // the worker. 99 await this.queueRequest('trace_processor_notify_eof', new Uint8Array()); 100 } 101 102 restoreInitialTables(): Promise<void> { 103 // We should never get here, restoreInitialTables() should be called only 104 // when using the HttpRpcEngine. 105 throw new Error('restoreInitialTables() not supported by the WASM engine'); 106 } 107 108 rawQuery(rawQueryArgs: Uint8Array): Promise<Uint8Array> { 109 return this.queueRequest('trace_processor_raw_query', rawQueryArgs); 110 } 111 112 rawComputeMetric(rawComputeMetric: Uint8Array): Promise<Uint8Array> { 113 return this.queueRequest( 114 'trace_processor_compute_metric', rawComputeMetric); 115 } 116 117 async enableMetatrace(): Promise<void> { 118 await this.queueRequest( 119 'trace_processor_enable_metatrace', new Uint8Array()); 120 } 121 122 disableAndReadMetatrace(): Promise<Uint8Array> { 123 return this.queueRequest( 124 'trace_processor_disable_and_read_metatrace', new Uint8Array()); 125 } 126 127 // Enqueues a request to the worker queue via postMessage(). The returned 128 // promised will be resolved once the worker replies to the postMessage() 129 // with the paylad of the response, a proto-encoded object which wraps the 130 // method return value (e.g., RawQueryResult for SQL query results). 131 private queueRequest(methodName: string, reqData: Uint8Array): 132 Deferred<Uint8Array> { 133 const respHandler = defer<Uint8Array>(); 134 const id = this.nextRequestId++; 135 const request: WasmBridgeRequest = {id, methodName, data: reqData}; 136 this.pendingRequests.push({id, respHandler}); 137 this.worker.postMessage(request); 138 return respHandler; 139 } 140 141 onMessage(m: MessageEvent) { 142 const response = m.data as WasmBridgeResponse; 143 assertTrue(this.pendingRequests.length > 0); 144 const request = this.pendingRequests.shift()!; 145 146 // Requests should be executed and ACKed by the worker in the same order 147 // they came in. 148 assertTrue(request.id === response.id); 149 if (response.aborted) { 150 request.respHandler.reject('WASM module crashed'); 151 } else { 152 request.respHandler.resolve(response.data); 153 } 154 } 155} 156