| // Copyright (C) 2022 The Android Open Source Project |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| |
| import {ViewerImpl, ViewerProxy} from '../common/viewer'; |
| import { |
| TrackControllerFactory, |
| trackControllerRegistry, |
| } from '../controller/track_controller'; |
| import {TrackCreator} from '../frontend/track'; |
| import {trackRegistry} from '../frontend/track_registry'; |
| import { |
| Command, |
| EngineProxy, |
| MetricVisualisation, |
| PluginContext, |
| PluginInfo, |
| Store, |
| TracePlugin, |
| TracePluginFactory, |
| TrackInfo, |
| } from '../public'; |
| |
| import {Engine} from './engine'; |
| import {Registry} from './registry'; |
| import {State} from './state'; |
| |
| interface TracePluginContext { |
| plugin: TracePlugin; |
| store: Store<unknown>; |
| engine: EngineProxy; |
| viewer: ViewerProxy; |
| } |
| |
| // Every plugin gets its own PluginContext. This is how we keep track |
| // what each plugin is doing and how we can blame issues on particular |
| // plugins. |
| export class PluginContextImpl implements PluginContext { |
| readonly pluginId: string; |
| private tracePluginFactory?: TracePluginFactory<any>; |
| private _tracePluginCtx?: TracePluginContext; |
| |
| constructor(pluginId: string) { |
| this.pluginId = pluginId; |
| } |
| |
| // ================================================================== |
| // The plugin facing API of PluginContext: |
| registerTrackController(track: TrackControllerFactory): void { |
| trackControllerRegistry.register(track); |
| } |
| |
| registerTrack(track: TrackCreator): void { |
| trackRegistry.register(track); |
| } |
| |
| registerTracePluginFactory<T>(pluginFactory: TracePluginFactory<T>): void { |
| this.tracePluginFactory = pluginFactory; |
| } |
| // ================================================================== |
| |
| // ================================================================== |
| // Internal facing API: |
| findPotentialTracks(): Promise<TrackInfo[]>[] { |
| const tracePlugin = this.tracePlugin; |
| if (tracePlugin && tracePlugin.tracks) { |
| return [tracePlugin.tracks()]; |
| } else { |
| return []; |
| } |
| } |
| |
| onTraceLoad(store: Store<State>, engine: Engine, viewer: ViewerImpl): void { |
| const TracePluginClass = this.tracePluginFactory; |
| if (TracePluginClass) { |
| // Make an engine proxy for this plugin. |
| const engineProxy: EngineProxy = engine.getProxy(this.pluginId); |
| |
| // Make a viewer for this plugin. |
| const viewerProxy: ViewerProxy = viewer.getProxy(this.pluginId); |
| |
| // Extract the initial state and pass to the plugin factory for migration. |
| const initialState = store.state.plugins[this.pluginId]; |
| const migratedState = TracePluginClass.migrate(initialState); |
| |
| // Store the initial state in our root store. |
| store.edit((draft) => { |
| draft.plugins[this.pluginId] = migratedState; |
| }); |
| |
| // Create a proxy store for our plugin to use. |
| const storeProxy = store.createProxy<unknown>(['plugins', this.pluginId]); |
| |
| // Instantiate the plugin. |
| this._tracePluginCtx = { |
| plugin: new TracePluginClass(storeProxy, engineProxy, viewerProxy), |
| store: storeProxy, |
| engine: engineProxy, |
| viewer: viewerProxy, |
| }; |
| } |
| } |
| |
| onTraceClosed() { |
| if (this._tracePluginCtx) { |
| this._tracePluginCtx.plugin.dispose(); |
| this._tracePluginCtx.store.dispose(); |
| this._tracePluginCtx.engine.dispose(); |
| this._tracePluginCtx.viewer.dispose(); |
| this._tracePluginCtx = undefined; |
| } |
| } |
| |
| get tracePlugin(): TracePlugin|undefined { |
| return this._tracePluginCtx?.plugin; |
| } |
| |
| // Unload the plugin. Ideally no plugin code runs after this point. |
| // PluginContext should unregister everything. |
| revoke() { |
| // TODO(hjd): Remove from trackControllerRegistry, trackRegistry, |
| // etc. |
| // TODO(stevegolton): Dispose the trace plugin. |
| } |
| // ================================================================== |
| } |
| |
| // 'Static' registry of all known plugins. |
| export class PluginRegistry extends Registry<PluginInfo> { |
| constructor() { |
| super((info) => info.pluginId); |
| } |
| } |
| |
| export class PluginManager { |
| private registry: PluginRegistry; |
| private contexts: Map<string, PluginContextImpl>; |
| |
| constructor(registry: PluginRegistry) { |
| this.registry = registry; |
| this.contexts = new Map(); |
| } |
| |
| activatePlugin(pluginId: string): void { |
| if (this.isActive(pluginId)) { |
| return; |
| } |
| const pluginInfo = this.registry.get(pluginId); |
| const context = new PluginContextImpl(pluginId); |
| this.contexts.set(pluginId, context); |
| pluginInfo.activate(context); |
| } |
| |
| deactivatePlugin(pluginId: string): void { |
| const context = this.getPluginContext(pluginId); |
| if (context === undefined) { |
| return; |
| } |
| context.revoke(); |
| this.contexts.delete(pluginId); |
| } |
| |
| isActive(pluginId: string): boolean { |
| return this.getPluginContext(pluginId) !== undefined; |
| } |
| |
| getPluginContext(pluginId: string): PluginContextImpl|undefined { |
| return this.contexts.get(pluginId); |
| } |
| |
| findPotentialTracks(): Promise<TrackInfo[]>[] { |
| const promises = []; |
| for (const context of this.contexts.values()) { |
| for (const promise of context.findPotentialTracks()) { |
| promises.push(promise); |
| } |
| } |
| return promises; |
| } |
| |
| onTraceLoad(store: Store<State>, engine: Engine, viewer: ViewerImpl): void { |
| for (const context of this.contexts.values()) { |
| context.onTraceLoad(store, engine, viewer); |
| } |
| } |
| |
| onTraceClose() { |
| for (const context of this.contexts.values()) { |
| context.onTraceClosed(); |
| } |
| } |
| |
| commands(): Command[] { |
| return Array.from(this.contexts.values()).flatMap((ctx) => { |
| const tracePlugin = ctx.tracePlugin; |
| if (tracePlugin && tracePlugin.commands) { |
| return tracePlugin.commands(); |
| } else { |
| return []; |
| } |
| }); |
| } |
| |
| metricVisualisations(): MetricVisualisation[] { |
| return Array.from(this.contexts.values()).flatMap((ctx) => { |
| const tracePlugin = ctx.tracePlugin; |
| if (tracePlugin && tracePlugin.metricVisualisations) { |
| return tracePlugin.metricVisualisations(); |
| } else { |
| return []; |
| } |
| }); |
| } |
| } |
| |
| // TODO(hjd): Sort out the story for global singletons like these: |
| export const pluginRegistry = new PluginRegistry(); |
| export const pluginManager = new PluginManager(pluginRegistry); |