From 5aa1ec9876d48fce976515b6fda2f98e6840dfd8 Mon Sep 17 00:00:00 2001 From: Alexander Rose <alexander.rose@weirdbyte.de> Date: Sat, 28 May 2022 11:43:13 -0700 Subject: [PATCH] add timing mode --- CHANGELOG.md | 4 +++ src/apps/viewer/app.ts | 2 +- src/apps/viewer/index.html | 5 +++- src/extensions/alpha-orbitals/density.ts | 9 ++++--- src/extensions/alpha-orbitals/orbitals.ts | 10 +++++--- src/mol-canvas3d/canvas3d.ts | 5 +++- src/mol-canvas3d/passes/draw.ts | 3 +++ src/mol-canvas3d/passes/fxaa.ts | 3 +++ src/mol-canvas3d/passes/marking.ts | 3 +++ src/mol-canvas3d/passes/multi-sample.ts | 5 ++++ src/mol-canvas3d/passes/pick.ts | 7 ++++++ src/mol-canvas3d/passes/postprocessing.ts | 3 +++ src/mol-canvas3d/passes/smaa.ts | 5 ++-- src/mol-canvas3d/passes/wboit.ts | 4 ++- .../geometry/texture-mesh/color-smoothing.ts | 7 ++++++ src/mol-gl/compute/grid3d.ts | 25 +++++++++++++------ .../compute/histogram-pyramid/reduction.ts | 5 +++- src/mol-gl/compute/histogram-pyramid/sum.ts | 5 +++- .../compute/marching-cubes/active-voxels.ts | 5 +++- .../compute/marching-cubes/isosurface.ts | 18 +++++-------- src/mol-gl/renderer.ts | 23 +++++++++++++++++ src/mol-math/geometry/gaussian-density/gpu.ts | 11 ++++++-- src/mol-plugin/animation-loop.ts | 14 ++++++++++- .../structure/visual/gaussian-surface-mesh.ts | 17 ++++--------- src/mol-util/debug.ts | 13 ++++++++-- 25 files changed, 158 insertions(+), 53 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index acd20ae27..6075f465b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -6,6 +6,10 @@ Note that since we don't clearly distinguish between a public and private interf ## [Unreleased] +- GPU timing support + - Add ``timing-mode`` Viewer GET param + - Add support for webgl timer queries + - Add timer marks around GPU render & compute operations - Fix ``Scene.clear`` not clearing primitives & volumes arrays (@JonStargaryen) ## [v3.8.2] - 2022-05-22 diff --git a/src/apps/viewer/app.ts b/src/apps/viewer/app.ts index eee9375c9..fa97eec20 100644 --- a/src/apps/viewer/app.ts +++ b/src/apps/viewer/app.ts @@ -47,7 +47,7 @@ import '../../mol-util/polyfill'; import { ObjectKeys } from '../../mol-util/type-helpers'; export { PLUGIN_VERSION as version } from '../../mol-plugin/version'; -export { setDebugMode, setProductionMode } from '../../mol-util/debug'; +export { setDebugMode, setProductionMode, setTimingMode } from '../../mol-util/debug'; const CustomFormats = [ ['g3d', G3dProvider] as const diff --git a/src/apps/viewer/index.html b/src/apps/viewer/index.html index 19dc4a54e..c63978410 100644 --- a/src/apps/viewer/index.html +++ b/src/apps/viewer/index.html @@ -46,7 +46,10 @@ } var debugMode = getParam('debug-mode', '[^&]+').trim() === '1'; - if (debugMode) molstar.setDebugMode(debugMode, debugMode); + if (debugMode) molstar.setDebugMode(debugMode); + + var timingMode = getParam('timing-mode', '[^&]+').trim() === '1'; + if (timingMode) molstar.setTimingMode(timingMode); var hideControls = getParam('hide-controls', '[^&]+').trim() === '1'; var collapseLeftPanel = getParam('collapse-left-panel', '[^&]+').trim() === '1'; diff --git a/src/extensions/alpha-orbitals/density.ts b/src/extensions/alpha-orbitals/density.ts index 73b00ed07..2bade27a8 100644 --- a/src/extensions/alpha-orbitals/density.ts +++ b/src/extensions/alpha-orbitals/density.ts @@ -1,5 +1,5 @@ /** - * Copyright (c) 2020 mol* contributors, licensed under MIT, See LICENSE file for more info. + * Copyright (c) 2020-2022 mol* contributors, licensed under MIT, See LICENSE file for more info. * * @author David Sehnal <david.sehnal@gmail.com> */ @@ -8,6 +8,7 @@ import { sortArray } from '../../mol-data/util'; import { canComputeGrid3dOnGPU } from '../../mol-gl/compute/grid3d'; import { WebGLContext } from '../../mol-gl/webgl/context'; import { Task } from '../../mol-task'; +import { isTimingMode } from '../../mol-util/debug'; import { AlphaOrbital, createGrid, CubeGrid, CubeGridComputationParams, initCubeGrid } from './data-model'; import { gpuComputeAlphaOrbitalsDensityGridValues } from './gpu/compute'; @@ -19,9 +20,9 @@ export function createSphericalCollocationDensityGrid( let matrix: Float32Array; if (canComputeGrid3dOnGPU(webgl)) { - // console.time('gpu'); - matrix = await gpuComputeAlphaOrbitalsDensityGridValues(ctx, webgl!, cubeGrid, orbitals); - // console.timeEnd('gpu'); + if (isTimingMode) webgl.timer.mark('createSphericalCollocationDensityGrid'); + matrix = await gpuComputeAlphaOrbitalsDensityGridValues(ctx, webgl, cubeGrid, orbitals); + if (isTimingMode) webgl.timer.markEnd('createSphericalCollocationDensityGrid'); } else { throw new Error('Missing OES_texture_float WebGL extension.'); } diff --git a/src/extensions/alpha-orbitals/orbitals.ts b/src/extensions/alpha-orbitals/orbitals.ts index 4235852da..c78893fb5 100644 --- a/src/extensions/alpha-orbitals/orbitals.ts +++ b/src/extensions/alpha-orbitals/orbitals.ts @@ -1,5 +1,5 @@ /** - * Copyright (c) 2020 mol* contributors, licensed under MIT, See LICENSE file for more info. + * Copyright (c) 2020-2022 mol* contributors, licensed under MIT, See LICENSE file for more info. * * Inspired by https://github.com/dgasmith/gau2grid. * @@ -10,11 +10,13 @@ import { sortArray } from '../../mol-data/util'; import { canComputeGrid3dOnGPU } from '../../mol-gl/compute/grid3d'; import { WebGLContext } from '../../mol-gl/webgl/context'; import { Task } from '../../mol-task'; +import { isTimingMode } from '../../mol-util/debug'; import { sphericalCollocation } from './collocation'; import { AlphaOrbital, createGrid, CubeGrid, CubeGridComputationParams, initCubeGrid } from './data-model'; import { gpuComputeAlphaOrbitalsGridValues } from './gpu/compute'; // setDebugMode(true); +// setTimingMode(true); export function createSphericalCollocationGrid( params: CubeGridComputationParams, orbital: AlphaOrbital, webgl?: WebGLContext @@ -24,9 +26,9 @@ export function createSphericalCollocationGrid( let matrix: Float32Array; if (canComputeGrid3dOnGPU(webgl)) { - // console.time('gpu'); - matrix = await gpuComputeAlphaOrbitalsGridValues(ctx, webgl!, cubeGrid, orbital); - // console.timeEnd('gpu'); + if (isTimingMode) webgl.timer.mark('createSphericalCollocationGrid'); + matrix = await gpuComputeAlphaOrbitalsGridValues(ctx, webgl, cubeGrid, orbital); + if (isTimingMode) webgl.timer.markEnd('createSphericalCollocationGrid'); } else { // console.time('cpu'); matrix = await sphericalCollocation(cubeGrid, orbital, ctx); diff --git a/src/mol-canvas3d/canvas3d.ts b/src/mol-canvas3d/canvas3d.ts index baeb62636..2af6cf0a4 100644 --- a/src/mol-canvas3d/canvas3d.ts +++ b/src/mol-canvas3d/canvas3d.ts @@ -30,7 +30,7 @@ import { PickData } from './passes/pick'; import { PickHelper } from './passes/pick'; import { ImagePass, ImageProps } from './passes/image'; import { Sphere3D } from '../mol-math/geometry'; -import { isDebugMode } from '../mol-util/debug'; +import { isDebugMode, isTimingMode } from '../mol-util/debug'; import { CameraHelperParams } from './helper/camera-helper'; import { produce } from 'immer'; import { HandleHelperParams } from './helper/handle-helper'; @@ -413,6 +413,7 @@ namespace Canvas3D { cam = stereoCamera; } + if (isTimingMode) webgl.timer.mark('Canvas3D.render'); const ctx = { renderer, camera: cam, scene, helper }; if (MultiSamplePass.isEnabled(p.multiSample)) { const forceOn = !cameraChanged && markingUpdated && !controls.isAnimating; @@ -420,6 +421,8 @@ namespace Canvas3D { } else { passes.draw.render(ctx, p, true); } + if (isTimingMode) webgl.timer.markEnd('Canvas3D.render'); + // if only marking has updated, do not set the flag to dirty pickHelper.dirty = pickHelper.dirty || shouldRender; didRender = true; diff --git a/src/mol-canvas3d/passes/draw.ts b/src/mol-canvas3d/passes/draw.ts index 407e5814b..94131aead 100644 --- a/src/mol-canvas3d/passes/draw.ts +++ b/src/mol-canvas3d/passes/draw.ts @@ -20,6 +20,7 @@ import { WboitPass } from './wboit'; import { AntialiasingPass, PostprocessingPass, PostprocessingProps } from './postprocessing'; import { MarkingPass, MarkingProps } from './marking'; import { CopyRenderable, createCopyRenderable } from '../../mol-gl/compute/util'; +import { isTimingMode } from '../../mol-util/debug'; type Props = { postprocessing: PostprocessingProps @@ -286,6 +287,7 @@ export class DrawPass { } render(ctx: RenderContext, props: Props, toDrawingBuffer: boolean) { + if (isTimingMode) this.webgl.timer.mark('DrawPass.render'); const { renderer, camera, scene, helper } = ctx; renderer.setTransparentBackground(props.transparentBackground); renderer.setDrawingBufferSize(this.colorTarget.getWidth(), this.colorTarget.getHeight()); @@ -297,6 +299,7 @@ export class DrawPass { } else { this._render(renderer, camera, scene, helper, toDrawingBuffer, props); } + if (isTimingMode) this.webgl.timer.markEnd('DrawPass.render'); } getColorTarget(postprocessingProps: PostprocessingProps): RenderTarget { diff --git a/src/mol-canvas3d/passes/fxaa.ts b/src/mol-canvas3d/passes/fxaa.ts index d7aac98f5..ff1a0e878 100644 --- a/src/mol-canvas3d/passes/fxaa.ts +++ b/src/mol-canvas3d/passes/fxaa.ts @@ -18,6 +18,7 @@ import { quad_vert } from '../../mol-gl/shader/quad.vert'; import { fxaa_frag } from '../../mol-gl/shader/fxaa.frag'; import { Viewport } from '../camera/util'; import { RenderTarget } from '../../mol-gl/webgl/render-target'; +import { isTimingMode } from '../../mol-util/debug'; export const FxaaParams = { edgeThresholdMin: PD.Numeric(0.0312, { min: 0.0312, max: 0.0833, step: 0.0001 }, { description: 'Trims the algorithm from processing darks.' }), @@ -83,6 +84,7 @@ export class FxaaPass { } render(viewport: Viewport, target: RenderTarget | undefined) { + if (isTimingMode) this.webgl.timer.mark('FxaaPass.render'); if (target) { target.bind(); } else { @@ -90,6 +92,7 @@ export class FxaaPass { } this.updateState(viewport); this.renderable.render(); + if (isTimingMode) this.webgl.timer.markEnd('FxaaPass.render'); } } diff --git a/src/mol-canvas3d/passes/marking.ts b/src/mol-canvas3d/passes/marking.ts index 3e74a623e..73cde519f 100644 --- a/src/mol-canvas3d/passes/marking.ts +++ b/src/mol-canvas3d/passes/marking.ts @@ -20,6 +20,7 @@ import { Viewport } from '../camera/util'; import { RenderTarget } from '../../mol-gl/webgl/render-target'; import { Color } from '../../mol-util/color'; import { edge_frag } from '../../mol-gl/shader/marking/edge.frag'; +import { isTimingMode } from '../../mol-util/debug'; export const MarkingParams = { enabled: PD.Boolean(true), @@ -117,6 +118,7 @@ export class MarkingPass { } render(viewport: Viewport, target: RenderTarget | undefined) { + if (isTimingMode) this.webgl.timer.mark('MarkingPass.render'); this.edgesTarget.bind(); this.setEdgeState(viewport); this.edge.render(); @@ -128,6 +130,7 @@ export class MarkingPass { } this.setOverlayState(viewport); this.overlay.render(); + if (isTimingMode) this.webgl.timer.markEnd('MarkingPass.render'); } } diff --git a/src/mol-canvas3d/passes/multi-sample.ts b/src/mol-canvas3d/passes/multi-sample.ts index 6f7ca4358..82c861372 100644 --- a/src/mol-canvas3d/passes/multi-sample.ts +++ b/src/mol-canvas3d/passes/multi-sample.ts @@ -25,6 +25,7 @@ import { StereoCamera } from '../camera/stereo'; import { quad_vert } from '../../mol-gl/shader/quad.vert'; import { compose_frag } from '../../mol-gl/shader/compose.frag'; import { MarkingProps } from './marking'; +import { isTimingMode } from '../../mol-util/debug'; const ComposeSchema = { ...QuadSchema, @@ -126,6 +127,7 @@ export class MultiSamplePass { const { camera } = ctx; const { compose, composeTarget, drawPass, webgl } = this; const { gl, state } = webgl; + if (isTimingMode) webgl.timer.mark('MultiSamplePass.renderMultiSample'); // based on the Multisample Anti-Aliasing Render Pass // contributed to three.js by bhouston / http://clara.io/ @@ -198,12 +200,14 @@ export class MultiSamplePass { camera.viewOffset.enabled = false; camera.update(); + if (isTimingMode) webgl.timer.markEnd('MultiSamplePass.renderMultiSample'); } private renderTemporalMultiSample(sampleIndex: number, ctx: RenderContext, props: Props, toDrawingBuffer: boolean) { const { camera } = ctx; const { compose, composeTarget, holdTarget, drawPass, webgl } = this; const { gl, state } = webgl; + if (isTimingMode) webgl.timer.mark('MultiSamplePass.renderTemporalMultiSample'); // based on the Multisample Anti-Aliasing Render Pass // contributed to three.js by bhouston / http://clara.io/ @@ -301,6 +305,7 @@ export class MultiSamplePass { camera.viewOffset.enabled = false; camera.update(); + if (isTimingMode) webgl.timer.markEnd('MultiSamplePass.renderTemporalMultiSample'); return sampleIndex >= offsetList.length ? -2 : sampleIndex; } diff --git a/src/mol-canvas3d/passes/pick.ts b/src/mol-canvas3d/passes/pick.ts index a8d0488ef..05b108bcd 100644 --- a/src/mol-canvas3d/passes/pick.ts +++ b/src/mol-canvas3d/passes/pick.ts @@ -11,6 +11,7 @@ import { WebGLContext } from '../../mol-gl/webgl/context'; import { RenderTarget } from '../../mol-gl/webgl/render-target'; import { Vec3 } from '../../mol-math/linear-algebra'; import { spiral2d } from '../../mol-math/misc'; +import { isTimingMode } from '../../mol-util/debug'; import { unpackRGBToInt, unpackRGBAToDepth } from '../../mol-util/number-packing'; import { Camera, ICamera } from '../camera'; import { StereoCamera } from '../camera/stereo'; @@ -144,6 +145,7 @@ export class PickHelper { } private syncBuffers() { + if (isTimingMode) this.webgl.timer.mark('PickHelper.syncBuffers'); const { pickX, pickY, pickWidth, pickHeight } = this; this.pickPass.objectPickTarget.bind(); @@ -157,6 +159,7 @@ export class PickHelper { this.pickPass.depthPickTarget.bind(); this.webgl.readPixels(pickX, pickY, pickWidth, pickHeight, this.depthBuffer); + if (isTimingMode) this.webgl.timer.markEnd('PickHelper.syncBuffers'); } private getBufferIdx(x: number, y: number): number { @@ -175,6 +178,7 @@ export class PickHelper { } private render(camera: Camera | StereoCamera) { + if (isTimingMode) this.webgl.timer.mark('PickHelper.render'); const { pickX, pickY, pickWidth, pickHeight, halfPickWidth } = this; const { renderer, scene, helper } = this; @@ -194,6 +198,7 @@ export class PickHelper { } this.dirty = false; + if (isTimingMode) this.webgl.timer.markEnd('PickHelper.render'); } private identifyInternal(x: number, y: number, camera: Camera | StereoCamera): PickData | undefined { @@ -214,8 +219,10 @@ export class PickHelper { ) return; if (this.dirty) { + if (isTimingMode) this.webgl.timer.mark('PickHelper.identify'); this.render(camera); this.syncBuffers(); + if (isTimingMode) this.webgl.timer.markEnd('PickHelper.identify'); } const xv = x - viewport.x; diff --git a/src/mol-canvas3d/passes/postprocessing.ts b/src/mol-canvas3d/passes/postprocessing.ts index 53e69da67..fd41b8abe 100644 --- a/src/mol-canvas3d/passes/postprocessing.ts +++ b/src/mol-canvas3d/passes/postprocessing.ts @@ -27,6 +27,7 @@ import { Framebuffer } from '../../mol-gl/webgl/framebuffer'; import { Color } from '../../mol-util/color'; import { FxaaParams, FxaaPass } from './fxaa'; import { SmaaParams, SmaaPass } from './smaa'; +import { isTimingMode } from '../../mol-util/debug'; const OutlinesSchema = { ...QuadSchema, @@ -549,6 +550,7 @@ export class PostprocessingPass { } render(camera: ICamera, toDrawingBuffer: boolean, transparentBackground: boolean, backgroundColor: Color, props: PostprocessingProps) { + if (isTimingMode) this.webgl.timer.mark('PostprocessingPass.render'); this.updateState(camera, transparentBackground, backgroundColor, props); if (props.outline.name === 'on') { @@ -585,6 +587,7 @@ export class PostprocessingPass { gl.clear(gl.COLOR_BUFFER_BIT); this.renderable.render(); + if (isTimingMode) this.webgl.timer.markEnd('PostprocessingPass.render'); } } diff --git a/src/mol-canvas3d/passes/smaa.ts b/src/mol-canvas3d/passes/smaa.ts index 805a44e8f..3002b2ff3 100644 --- a/src/mol-canvas3d/passes/smaa.ts +++ b/src/mol-canvas3d/passes/smaa.ts @@ -22,7 +22,7 @@ import { weights_frag } from '../../mol-gl/shader/smaa/weights.frag'; import { edges_vert } from '../../mol-gl/shader/smaa/edges.vert'; import { edges_frag } from '../../mol-gl/shader/smaa/edges.frag'; import { Viewport } from '../camera/util'; -import { isDebugMode } from '../../mol-util/debug'; +import { isDebugMode, isTimingMode } from '../../mol-util/debug'; export const SmaaParams = { edgeThreshold: PD.Numeric(0.1, { min: 0.05, max: 0.15, step: 0.01 }), @@ -120,6 +120,7 @@ export class SmaaPass { } render(viewport: Viewport, target: RenderTarget | undefined) { + if (isTimingMode) this.webgl.timer.mark('SmaaPass.render'); this.edgesTarget.bind(); this.updateState(viewport); this.edgesRenderable.render(); @@ -135,8 +136,8 @@ export class SmaaPass { } this.updateState(viewport); this.blendRenderable.render(); + if (isTimingMode) this.webgl.timer.markEnd('SmaaPass.render'); } - } // diff --git a/src/mol-canvas3d/passes/wboit.ts b/src/mol-canvas3d/passes/wboit.ts index 89c39d7fb..c78f809a1 100644 --- a/src/mol-canvas3d/passes/wboit.ts +++ b/src/mol-canvas3d/passes/wboit.ts @@ -17,7 +17,7 @@ import { quad_vert } from '../../mol-gl/shader/quad.vert'; import { evaluateWboit_frag } from '../../mol-gl/shader/evaluate-wboit.frag'; import { Framebuffer } from '../../mol-gl/webgl/framebuffer'; import { Vec2 } from '../../mol-math/linear-algebra'; -import { isDebugMode } from '../../mol-util/debug'; +import { isDebugMode, isTimingMode } from '../../mol-util/debug'; const EvaluateWboitSchema = { ...QuadSchema, @@ -71,6 +71,7 @@ export class WboitPass { } render() { + if (isTimingMode) this.webgl.timer.mark('WboitPass.render'); const { state, gl } = this.webgl; state.blendFuncSeparate(gl.SRC_ALPHA, gl.ONE_MINUS_SRC_ALPHA, gl.ONE, gl.ONE_MINUS_SRC_ALPHA); @@ -78,6 +79,7 @@ export class WboitPass { this.renderable.update(); this.renderable.render(); + if (isTimingMode) this.webgl.timer.markEnd('WboitPass.render'); } setSize(width: number, height: number) { diff --git a/src/mol-geo/geometry/texture-mesh/color-smoothing.ts b/src/mol-geo/geometry/texture-mesh/color-smoothing.ts index 516903c02..7d28455cd 100644 --- a/src/mol-geo/geometry/texture-mesh/color-smoothing.ts +++ b/src/mol-geo/geometry/texture-mesh/color-smoothing.ts @@ -20,6 +20,7 @@ import { accumulate_frag } from '../../../mol-gl/shader/compute/color-smoothing/ import { accumulate_vert } from '../../../mol-gl/shader/compute/color-smoothing/accumulate.vert'; import { isWebGL2 } from '../../../mol-gl/webgl/compat'; import { TextureMeshValues } from '../../../mol-gl/renderable/texture-mesh'; +import { isTimingMode } from '../../../mol-util/debug'; export const ColorAccumulateSchema = { drawCount: ValueSpec('number'), @@ -255,6 +256,7 @@ export function calcTextureMeshColorSmoothing(input: ColorSmoothingInput, resolu const { drawBuffers } = webgl.extensions; if (!drawBuffers) throw new Error('need WebGL draw buffers'); + if (isTimingMode) webgl.timer.mark('calcTextureMeshColorSmoothing'); const { gl, resources, state, extensions: { colorBufferHalfFloat, textureHalfFloat } } = webgl; const isInstanceType = input.colorType.endsWith('Instance'); @@ -321,6 +323,7 @@ export function calcTextureMeshColorSmoothing(input: ColorSmoothingInput, resolu const { uCurrentSlice, uCurrentX, uCurrentY } = accumulateRenderable.values; + if (isTimingMode) webgl.timer.mark('ColorAccumulate.render'); setAccumulateDefaults(webgl); gl.viewport(0, 0, width, height); gl.scissor(0, 0, width, height); @@ -349,6 +352,7 @@ export function calcTextureMeshColorSmoothing(input: ColorSmoothingInput, resolu accumulateTexture.detachFramebuffer(framebuffer, 0); countTexture.detachFramebuffer(framebuffer, 1); drawBuffers.drawBuffers([gl.COLOR_ATTACHMENT0, gl.NONE]); + if (isTimingMode) webgl.timer.markEnd('ColorAccumulate.render'); // const accImage = new Float32Array(width * height * 4); // accumulateTexture.attachFramebuffer(framebuffer, 0); @@ -364,6 +368,7 @@ export function calcTextureMeshColorSmoothing(input: ColorSmoothingInput, resolu // normalize + if (isTimingMode) webgl.timer.mark('ColorNormalize.render'); if (!texture) texture = resources.texture('image-uint8', 'rgba', 'ubyte', 'linear'); texture.define(width, height); @@ -376,6 +381,7 @@ export function calcTextureMeshColorSmoothing(input: ColorSmoothingInput, resolu gl.scissor(0, 0, width, height); gl.clear(gl.COLOR_BUFFER_BIT); normalizeRenderable.render(); + if (isTimingMode) webgl.timer.markEnd('ColorNormalize.render'); // const normImage = new Uint8Array(width * height * 4); // texture.attachFramebuffer(framebuffer, 0); @@ -385,6 +391,7 @@ export function calcTextureMeshColorSmoothing(input: ColorSmoothingInput, resolu const gridTransform = Vec4.create(min[0], min[1], min[2], scaleFactor); const type = isInstanceType ? 'volumeInstance' : 'volume'; + if (isTimingMode) webgl.timer.markEnd('calcTextureMeshColorSmoothing'); return { texture, gridDim, gridTexDim: Vec2.create(width, height), gridTransform, type }; } diff --git a/src/mol-gl/compute/grid3d.ts b/src/mol-gl/compute/grid3d.ts index a5676f276..e291661cc 100644 --- a/src/mol-gl/compute/grid3d.ts +++ b/src/mol-gl/compute/grid3d.ts @@ -1,5 +1,5 @@ /** - * Copyright (c) 2020 mol* contributors, licensed under MIT, See LICENSE file for more info. + * Copyright (c) 2020-2022 mol* contributors, licensed under MIT, See LICENSE file for more info. * * @author David Sehnal <david.sehnal@gmail.com> */ @@ -18,8 +18,9 @@ import { createComputeRenderItem } from '../webgl/render-item'; import { createComputeRenderable } from '../renderable'; import { isLittleEndian } from '../../mol-util/is-little-endian'; import { RuntimeContext } from '../../mol-task'; +import { isTimingMode } from '../../mol-util/debug'; -export function canComputeGrid3dOnGPU(webgl?: WebGLContext) { +export function canComputeGrid3dOnGPU(webgl?: WebGLContext): webgl is WebGLContext { return !!webgl?.extensions.textureFloat; } @@ -159,7 +160,8 @@ export function createGrid3dComputeRenderable<S extends RenderableSchema, P, CS> const array = new Uint8Array(uWidth * uWidth * 4); if (spec.cumulative) { - const { gl } = webgl; + const { gl, state } = webgl; + if (isTimingMode) webgl.timer.mark('Grid3dCompute.renderCumulative'); const states = spec.cumulative.states(params); @@ -167,7 +169,7 @@ export function createGrid3dComputeRenderable<S extends RenderableSchema, P, CS> tex[1].define(uWidth, uWidth); resetGl(webgl, uWidth); - gl.clearColor(0, 0, 0, 0); + state.clearColor(0, 0, 0, 0); tex[0].attachFramebuffer(framebuffer, 'color0'); gl.clear(gl.COLOR_BUFFER_BIT); @@ -175,12 +177,13 @@ export function createGrid3dComputeRenderable<S extends RenderableSchema, P, CS> tex[1].attachFramebuffer(framebuffer, 'color0'); gl.clear(gl.COLOR_BUFFER_BIT); - if (spec.cumulative.yieldPeriod) { + if (spec.cumulative.yieldPeriod && !isTimingMode) { await ctx.update({ message: 'Computing...', isIndeterminate: false, current: 0, max: states.length }); } const yieldPeriod = Math.max(1, spec.cumulative.yieldPeriod ?? 1 | 0); + if (isTimingMode) webgl.timer.mark('Grid3dCompute.renderBatch'); for (let i = 0; i < states.length; i++) { ValueCell.update(cells.tCumulativeSum, tex[(i + 1) % 2]); tex[i % 2].attachFramebuffer(framebuffer, 'color0'); @@ -191,23 +194,31 @@ export function createGrid3dComputeRenderable<S extends RenderableSchema, P, CS> if (spec.cumulative.yieldPeriod && i !== states.length - 1) { if (i % yieldPeriod === yieldPeriod - 1) { - webgl.readPixels(0, 0, 1, 1, array); + webgl.waitForGpuCommandsCompleteSync(); + if (isTimingMode) webgl.timer.markEnd('Grid3dCompute.renderBatch'); + if (isTimingMode) webgl.timer.mark('Grid3dCompute.renderBatch'); } - if (ctx.shouldUpdate) { + if (ctx.shouldUpdate && !isTimingMode) { await ctx.update({ current: i + 1 }); } } } + if (isTimingMode) webgl.timer.markEnd('Grid3dCompute.renderBatch'); + if (isTimingMode) webgl.timer.markEnd('Grid3dCompute.renderCumulative'); } else { + if (isTimingMode) webgl.timer.mark('Grid3dCompute.render'); tex[0].define(uWidth, uWidth); tex[0].attachFramebuffer(framebuffer, 'color0'); framebuffer.bind(); resetGl(webgl, uWidth); renderable.update(); renderable.render(); + if (isTimingMode) webgl.timer.markEnd('Grid3dCompute.render'); } + if (isTimingMode) webgl.timer.mark('Grid3dCompute.readPixels'); webgl.readPixels(0, 0, uWidth, uWidth, array); + if (isTimingMode) webgl.timer.markEnd('Grid3dCompute.readPixels'); return new Float32Array(array.buffer, array.byteOffset, nx * ny * nz); }; } diff --git a/src/mol-gl/compute/histogram-pyramid/reduction.ts b/src/mol-gl/compute/histogram-pyramid/reduction.ts index 407e2f9b2..8d162f5e9 100644 --- a/src/mol-gl/compute/histogram-pyramid/reduction.ts +++ b/src/mol-gl/compute/histogram-pyramid/reduction.ts @@ -1,5 +1,5 @@ /** - * Copyright (c) 2019-2021 mol* contributors, licensed under MIT, See LICENSE file for more info. + * Copyright (c) 2019-2022 mol* contributors, licensed under MIT, See LICENSE file for more info. * * @author Alexander Rose <alexander.rose@weirdbyte.de> */ @@ -19,6 +19,7 @@ import { isPowerOfTwo } from '../../../mol-math/misc'; import { quad_vert } from '../../../mol-gl/shader/quad.vert'; import { reduction_frag } from '../../../mol-gl/shader/histogram-pyramid/reduction.frag'; import { isWebGL2 } from '../../webgl/compat'; +import { isTimingMode } from '../../../mol-util/debug'; const HistopyramidReductionSchema = { ...QuadSchema, @@ -120,6 +121,7 @@ export interface HistogramPyramid { } export function createHistogramPyramid(ctx: WebGLContext, inputTexture: Texture, scale: Vec2, gridTexDim: Vec3): HistogramPyramid { + if (isTimingMode) ctx.timer.mark('createHistogramPyramid'); const { gl } = ctx; const w = inputTexture.getWidth(); const h = inputTexture.getHeight(); @@ -193,6 +195,7 @@ export function createHistogramPyramid(ctx: WebGLContext, inputTexture: Texture, } gl.finish(); + if (isTimingMode) ctx.timer.markEnd('createHistogramPyramid'); // printTexture(ctx, pyramidTex, 2) diff --git a/src/mol-gl/compute/histogram-pyramid/sum.ts b/src/mol-gl/compute/histogram-pyramid/sum.ts index 7c0d07a6c..a1cd5919a 100644 --- a/src/mol-gl/compute/histogram-pyramid/sum.ts +++ b/src/mol-gl/compute/histogram-pyramid/sum.ts @@ -1,5 +1,5 @@ /** - * Copyright (c) 2019-2021 mol* contributors, licensed under MIT, See LICENSE file for more info. + * Copyright (c) 2019-2022 mol* contributors, licensed under MIT, See LICENSE file for more info. * * @author Alexander Rose <alexander.rose@weirdbyte.de> */ @@ -16,6 +16,7 @@ import { QuadSchema, QuadValues } from '../util'; import { quad_vert } from '../../../mol-gl/shader/quad.vert'; import { sum_frag } from '../../../mol-gl/shader/histogram-pyramid/sum.frag'; import { isWebGL2 } from '../../webgl/compat'; +import { isTimingMode } from '../../../mol-util/debug'; const HistopyramidSumSchema = { ...QuadSchema, @@ -66,6 +67,7 @@ const sumBytes = new Uint8Array(4); const sumInts = new Int32Array(4); export function getHistopyramidSum(ctx: WebGLContext, pyramidTopTexture: Texture) { + if (isTimingMode) ctx.timer.mark('getHistopyramidSum'); const { gl, resources } = ctx; const renderable = getHistopyramidSumRenderable(ctx, pyramidTopTexture); @@ -93,6 +95,7 @@ export function getHistopyramidSum(ctx: WebGLContext, pyramidTopTexture: Texture ctx.readPixels(0, 0, 1, 1, isWebGL2(gl) ? sumInts : sumBytes); ctx.unbindFramebuffer(); + if (isTimingMode) ctx.timer.markEnd('getHistopyramidSum'); return isWebGL2(gl) ? sumInts[0] diff --git a/src/mol-gl/compute/marching-cubes/active-voxels.ts b/src/mol-gl/compute/marching-cubes/active-voxels.ts index 506020145..b16014c01 100644 --- a/src/mol-gl/compute/marching-cubes/active-voxels.ts +++ b/src/mol-gl/compute/marching-cubes/active-voxels.ts @@ -1,5 +1,5 @@ /** - * Copyright (c) 2019-2021 mol* contributors, licensed under MIT, See LICENSE file for more info. + * Copyright (c) 2019-2022 mol* contributors, licensed under MIT, See LICENSE file for more info. * * @author Alexander Rose <alexander.rose@weirdbyte.de> */ @@ -16,6 +16,7 @@ import { QuadSchema, QuadValues } from '../util'; import { getTriCount } from './tables'; import { quad_vert } from '../../../mol-gl/shader/quad.vert'; import { activeVoxels_frag } from '../../../mol-gl/shader/marching-cubes/active-voxels.frag'; +import { isTimingMode } from '../../../mol-util/debug'; const ActiveVoxelsSchema = { ...QuadSchema, @@ -83,6 +84,7 @@ function setRenderingDefaults(ctx: WebGLContext) { } export function calcActiveVoxels(ctx: WebGLContext, volumeData: Texture, gridDim: Vec3, gridTexDim: Vec3, isoValue: number, gridScale: Vec2) { + if (isTimingMode) ctx.timer.mark('calcActiveVoxels'); const { gl, resources } = ctx; const width = volumeData.getWidth(); const height = volumeData.getHeight(); @@ -115,6 +117,7 @@ export function calcActiveVoxels(ctx: WebGLContext, volumeData: Texture, gridDim // console.log('at', readTexture(ctx, activeVoxelsTex)); gl.finish(); + if (isTimingMode) ctx.timer.markEnd('calcActiveVoxels'); return activeVoxelsTex; } \ No newline at end of file diff --git a/src/mol-gl/compute/marching-cubes/isosurface.ts b/src/mol-gl/compute/marching-cubes/isosurface.ts index d3f11466b..93263e227 100644 --- a/src/mol-gl/compute/marching-cubes/isosurface.ts +++ b/src/mol-gl/compute/marching-cubes/isosurface.ts @@ -1,5 +1,5 @@ /** - * Copyright (c) 2019-2021 mol* contributors, licensed under MIT, See LICENSE file for more info. + * Copyright (c) 2019-2022 mol* contributors, licensed under MIT, See LICENSE file for more info. * * @author Alexander Rose <alexander.rose@weirdbyte.de> */ @@ -19,6 +19,7 @@ import { quad_vert } from '../../../mol-gl/shader/quad.vert'; import { isosurface_frag } from '../../../mol-gl/shader/marching-cubes/isosurface.frag'; import { calcActiveVoxels } from './active-voxels'; import { isWebGL2 } from '../../webgl/compat'; +import { isTimingMode } from '../../../mol-util/debug'; const IsosurfaceSchema = { ...QuadSchema, @@ -122,6 +123,7 @@ export function createIsosurfaceBuffers(ctx: WebGLContext, activeVoxelsBase: Tex const { drawBuffers } = ctx.extensions; if (!drawBuffers) throw new Error('need WebGL draw buffers'); + if (isTimingMode) ctx.timer.mark('createIsosurfaceBuffers'); const { gl, resources, extensions } = ctx; const { pyramidTex, height, levels, scale, count } = histogramPyramid; const width = pyramidTex.getWidth(); @@ -192,6 +194,7 @@ export function createIsosurfaceBuffers(ctx: WebGLContext, activeVoxelsBase: Tex renderable.render(); gl.finish(); + if (isTimingMode) ctx.timer.markEnd('createIsosurfaceBuffers'); return { vertexTexture, groupTexture, normalTexture, vertexCount: count }; } @@ -208,20 +211,11 @@ export function createIsosurfaceBuffers(ctx: WebGLContext, activeVoxelsBase: Tex * Implementation based on http://www.miaumiau.cat/2016/10/stream-compaction-in-webgl/ */ export function extractIsosurface(ctx: WebGLContext, volumeData: Texture, gridDim: Vec3, gridTexDim: Vec3, gridTexScale: Vec2, transform: Mat4, isoValue: number, invert: boolean, packedGroup: boolean, axisOrder: Vec3, vertexTexture?: Texture, groupTexture?: Texture, normalTexture?: Texture) { - // console.time('calcActiveVoxels'); + if (isTimingMode) ctx.timer.mark('extractIsosurface'); const activeVoxelsTex = calcActiveVoxels(ctx, volumeData, gridDim, gridTexDim, isoValue, gridTexScale); - // ctx.waitForGpuCommandsCompleteSync(); - // console.timeEnd('calcActiveVoxels'); - - // console.time('createHistogramPyramid'); const compacted = createHistogramPyramid(ctx, activeVoxelsTex, gridTexScale, gridTexDim); - // ctx.waitForGpuCommandsCompleteSync(); - // console.timeEnd('createHistogramPyramid'); - - // console.time('createIsosurfaceBuffers'); const gv = createIsosurfaceBuffers(ctx, activeVoxelsTex, volumeData, compacted, gridDim, gridTexDim, transform, isoValue, invert, packedGroup, axisOrder, vertexTexture, groupTexture, normalTexture); - // ctx.waitForGpuCommandsCompleteSync(); - // console.timeEnd('createIsosurfaceBuffers'); + if (isTimingMode) ctx.timer.markEnd('extractIsosurface'); return gv; } \ No newline at end of file diff --git a/src/mol-gl/renderer.ts b/src/mol-gl/renderer.ts index abc4ca7c8..93d7e73fd 100644 --- a/src/mol-gl/renderer.ts +++ b/src/mol-gl/renderer.ts @@ -19,6 +19,7 @@ import { degToRad } from '../mol-math/misc'; import { Texture, Textures } from './webgl/texture'; import { arrayMapUpsert } from '../mol-util/array'; import { clamp } from '../mol-math/interpolate'; +import { isTimingMode } from '../mol-util/debug'; export interface RendererStats { programCount: number @@ -360,6 +361,7 @@ namespace Renderer { }; const renderPick = (group: Scene.Group, camera: ICamera, variant: GraphicsRenderVariant, depthTexture: Texture | null, pickType: PickType) => { + if (isTimingMode) ctx.timer.mark('Renderer.renderPick'); state.disable(gl.BLEND); state.enable(gl.DEPTH_TEST); state.depthMask(true); @@ -373,9 +375,11 @@ namespace Renderer { renderObject(renderables[i], variant, Flag.None); } } + if (isTimingMode) ctx.timer.markEnd('Renderer.renderPick'); }; const renderDepth = (group: Scene.Group, camera: ICamera, depthTexture: Texture | null) => { + if (isTimingMode) ctx.timer.mark('Renderer.renderDepth'); state.disable(gl.BLEND); state.enable(gl.DEPTH_TEST); state.depthMask(true); @@ -386,9 +390,11 @@ namespace Renderer { for (let i = 0, il = renderables.length; i < il; ++i) { renderObject(renderables[i], 'depth', Flag.None); } + if (isTimingMode) ctx.timer.markEnd('Renderer.renderDepth'); }; const renderDepthOpaque = (group: Scene.Group, camera: ICamera, depthTexture: Texture | null) => { + if (isTimingMode) ctx.timer.mark('Renderer.renderDepthOpaque'); state.disable(gl.BLEND); state.enable(gl.DEPTH_TEST); state.depthMask(true); @@ -402,9 +408,11 @@ namespace Renderer { renderObject(r, 'depth', Flag.None); } } + if (isTimingMode) ctx.timer.markEnd('Renderer.renderDepthOpaque'); }; const renderDepthTransparent = (group: Scene.Group, camera: ICamera, depthTexture: Texture | null) => { + if (isTimingMode) ctx.timer.mark('Renderer.renderDepthTransparent'); state.disable(gl.BLEND); state.enable(gl.DEPTH_TEST); state.depthMask(true); @@ -418,9 +426,11 @@ namespace Renderer { renderObject(r, 'depth', Flag.None); } } + if (isTimingMode) ctx.timer.markEnd('Renderer.renderDepthTransparent'); }; const renderMarkingDepth = (group: Scene.Group, camera: ICamera, depthTexture: Texture | null) => { + if (isTimingMode) ctx.timer.mark('Renderer.renderMarkingDepth'); state.disable(gl.BLEND); state.enable(gl.DEPTH_TEST); state.depthMask(true); @@ -436,9 +446,11 @@ namespace Renderer { renderObject(renderables[i], 'marking', Flag.None); } } + if (isTimingMode) ctx.timer.markEnd('Renderer.renderMarkingDepth'); }; const renderMarkingMask = (group: Scene.Group, camera: ICamera, depthTexture: Texture | null) => { + if (isTimingMode) ctx.timer.mark('Renderer.renderMarkingMask'); state.disable(gl.BLEND); state.enable(gl.DEPTH_TEST); state.depthMask(true); @@ -454,6 +466,7 @@ namespace Renderer { renderObject(renderables[i], 'marking', Flag.None); } } + if (isTimingMode) ctx.timer.markEnd('Renderer.renderMarkingMask'); }; const renderBlended = (group: Scene.Group, camera: ICamera, depthTexture: Texture | null) => { @@ -462,6 +475,7 @@ namespace Renderer { }; const renderBlendedOpaque = (group: Scene.Group, camera: ICamera, depthTexture: Texture | null) => { + if (isTimingMode) ctx.timer.mark('Renderer.renderBlendedOpaque'); state.disable(gl.BLEND); state.enable(gl.DEPTH_TEST); state.depthMask(true); @@ -477,9 +491,11 @@ namespace Renderer { renderObject(r, 'colorBlended', Flag.BlendedBack); } } + if (isTimingMode) ctx.timer.markEnd('Renderer.renderBlendedOpaque'); }; const renderBlendedTransparent = (group: Scene.Group, camera: ICamera, depthTexture: Texture | null) => { + if (isTimingMode) ctx.timer.mark('Renderer.renderBlendedTransparent'); state.enable(gl.DEPTH_TEST); updateInternal(group, camera, depthTexture, Mask.Transparent, false); @@ -516,9 +532,11 @@ namespace Renderer { } } } + if (isTimingMode) ctx.timer.markEnd('Renderer.renderBlendedTransparent'); }; const renderBlendedVolume = (group: Scene.Group, camera: ICamera, depthTexture: Texture | null) => { + if (isTimingMode) ctx.timer.mark('Renderer.renderBlendedVolume'); state.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA); state.enable(gl.BLEND); @@ -531,9 +549,11 @@ namespace Renderer { renderObject(r, 'colorBlended', Flag.None); } } + if (isTimingMode) ctx.timer.markEnd('Renderer.renderBlendedVolume'); }; const renderWboitOpaque = (group: Scene.Group, camera: ICamera, depthTexture: Texture | null) => { + if (isTimingMode) ctx.timer.mark('Renderer.renderWboitOpaque'); state.disable(gl.BLEND); state.enable(gl.DEPTH_TEST); state.depthMask(true); @@ -551,9 +571,11 @@ namespace Renderer { renderObject(r, 'colorWboit', Flag.None); } } + if (isTimingMode) ctx.timer.markEnd('Renderer.renderWboitOpaque'); }; const renderWboitTransparent = (group: Scene.Group, camera: ICamera, depthTexture: Texture | null) => { + if (isTimingMode) ctx.timer.mark('Renderer.renderWboitTransparent'); updateInternal(group, camera, depthTexture, Mask.Transparent, false); const { renderables } = group; @@ -567,6 +589,7 @@ namespace Renderer { renderObject(r, 'colorWboit', Flag.None); } } + if (isTimingMode) ctx.timer.markEnd('Renderer.renderWboitTransparent'); }; return { diff --git a/src/mol-math/geometry/gaussian-density/gpu.ts b/src/mol-math/geometry/gaussian-density/gpu.ts index 5d3214248..23f7612d6 100644 --- a/src/mol-math/geometry/gaussian-density/gpu.ts +++ b/src/mol-math/geometry/gaussian-density/gpu.ts @@ -21,6 +21,7 @@ import { ValueSpec, AttributeSpec, UniformSpec, TextureSpec, DefineSpec, Values import { gaussianDensity_vert } from '../../../mol-gl/shader/gaussian-density.vert'; import { gaussianDensity_frag } from '../../../mol-gl/shader/gaussian-density.frag'; import { Framebuffer } from '../../../mol-gl/webgl/framebuffer'; +import { isTimingMode } from '../../../mol-util/debug'; const GaussianDensitySchema = { drawCount: ValueSpec('number'), @@ -85,11 +86,17 @@ export function GaussianDensityTexture(webgl: WebGLContext, position: PositionDa } export function GaussianDensityTexture2d(webgl: WebGLContext, position: PositionData, box: Box3D, radius: (index: number) => number, powerOfTwo: boolean, props: GaussianDensityProps, oldTexture?: Texture): GaussianDensityTextureData { - return finalizeGaussianDensityTexture(calcGaussianDensityTexture2d(webgl, position, box, radius, powerOfTwo, props, oldTexture)); + if (isTimingMode) webgl.timer.mark('GaussianDensityTexture2d'); + const data = calcGaussianDensityTexture2d(webgl, position, box, radius, powerOfTwo, props, oldTexture); + if (isTimingMode) webgl.timer.markEnd('GaussianDensityTexture2d'); + return finalizeGaussianDensityTexture(data); } export function GaussianDensityTexture3d(webgl: WebGLContext, position: PositionData, box: Box3D, radius: (index: number) => number, props: GaussianDensityProps, oldTexture?: Texture): GaussianDensityTextureData { - return finalizeGaussianDensityTexture(calcGaussianDensityTexture3d(webgl, position, box, radius, props, oldTexture)); + if (isTimingMode) webgl.timer.mark('GaussianDensityTexture3d'); + const data = calcGaussianDensityTexture3d(webgl, position, box, radius, props, oldTexture); + if (isTimingMode) webgl.timer.markEnd('GaussianDensityTexture3d'); + return finalizeGaussianDensityTexture(data); } function finalizeGaussianDensityTexture({ texture, scale, bbox, gridDim, gridTexDim, gridTexScale, radiusFactor, resolution, maxRadius }: _GaussianDensityTextureData): GaussianDensityTextureData { diff --git a/src/mol-plugin/animation-loop.ts b/src/mol-plugin/animation-loop.ts index c68adf4ec..27c223bf0 100644 --- a/src/mol-plugin/animation-loop.ts +++ b/src/mol-plugin/animation-loop.ts @@ -1,12 +1,15 @@ /** - * Copyright (c) 2020 mol* contributors, licensed under MIT, See LICENSE file for more info. + * Copyright (c) 2020-2022 mol* contributors, licensed under MIT, See LICENSE file for more info. * * @author David Sehnal <david.sehnal@gmail.com> + * @author Alexander Rose <alexander.rose@weirdbyte.de> */ import { PluginContext } from './context'; import { now } from '../mol-util/now'; import { PluginAnimationManager } from '../mol-plugin-state/manager/animation'; +import { isTimingMode } from '../mol-util/debug'; +import { printTimerResults } from '../mol-gl/webgl/timer'; export class PluginAnimationLoop { private currentFrame: any = void 0; @@ -19,6 +22,15 @@ export class PluginAnimationLoop { async tick(t: number, options?: { isSynchronous?: boolean, manualDraw?: boolean, animation?: PluginAnimationManager.AnimationInfo }) { await this.plugin.managers.animation.tick(t, options?.isSynchronous, options?.animation); this.plugin.canvas3d?.tick(t as now.Timestamp, options); + + if (isTimingMode) { + const timerResults = this.plugin.canvas3d?.webgl.timer.resolve(); + if (timerResults) { + for (const result of timerResults) { + printTimerResults([result]); + } + } + } } private frame = () => { diff --git a/src/mol-repr/structure/visual/gaussian-surface-mesh.ts b/src/mol-repr/structure/visual/gaussian-surface-mesh.ts index f5116e435..74c3ecdb4 100644 --- a/src/mol-repr/structure/visual/gaussian-surface-mesh.ts +++ b/src/mol-repr/structure/visual/gaussian-surface-mesh.ts @@ -27,6 +27,7 @@ import { applyMeshColorSmoothing } from '../../../mol-geo/geometry/mesh/color-sm import { applyTextureMeshColorSmoothing } from '../../../mol-geo/geometry/texture-mesh/color-smoothing'; import { ColorSmoothingParams, getColorSmoothingProps } from '../../../mol-geo/geometry/base'; import { Vec3 } from '../../../mol-math/linear-algebra'; +import { isTimingMode } from '../../../mol-util/debug'; const SharedParams = { ...GaussianDensityParams, @@ -213,6 +214,7 @@ const GaussianSurfaceName = 'gaussian-surface'; async function createGaussianSurfaceTextureMesh(ctx: VisualContext, unit: Unit, structure: Structure, theme: Theme, props: GaussianDensityProps, textureMesh?: TextureMesh): Promise<TextureMesh> { if (!ctx.webgl) throw new Error('webgl context required to create gaussian surface texture-mesh'); + if (isTimingMode) ctx.webgl.timer.mark('createGaussianSurfaceTextureMesh'); const { namedTextures, resources, extensions: { colorBufferFloat, textureFloat, colorBufferHalfFloat, textureHalfFloat } } = ctx.webgl; if (!namedTextures[GaussianSurfaceName]) { namedTextures[GaussianSurfaceName] = colorBufferHalfFloat && textureHalfFloat @@ -222,18 +224,13 @@ async function createGaussianSurfaceTextureMesh(ctx: VisualContext, unit: Unit, : resources.texture('image-uint8', 'rgba', 'ubyte', 'linear'); } - // console.time('computeUnitGaussianDensityTexture2d'); const densityTextureData = await computeUnitGaussianDensityTexture2d(structure, unit, theme.size, true, props, ctx.webgl, namedTextures[GaussianSurfaceName]).runInContext(ctx.runtime); - // console.log(densityTextureData); - // console.log('vertexGroupTexture', readTexture(ctx.webgl, densityTextureData.texture)); - // ctx.webgl.waitForGpuCommandsCompleteSync(); - // console.timeEnd('computeUnitGaussianDensityTexture2d'); - const isoLevel = Math.exp(-props.smoothness) / densityTextureData.radiusFactor; const axisOrder = Vec3.create(0, 1, 2); const buffer = textureMesh?.doubleBuffer.get(); const gv = extractIsosurface(ctx.webgl, densityTextureData.texture, densityTextureData.gridDim, densityTextureData.gridTexDim, densityTextureData.gridTexScale, densityTextureData.transform, isoLevel, false, true, axisOrder, buffer?.vertex, buffer?.group, buffer?.normal); + if (isTimingMode) ctx.webgl.timer.markEnd('createGaussianSurfaceTextureMesh'); const boundingSphere = Sphere3D.expand(Sphere3D(), unit.boundary.sphere, densityTextureData.maxRadius); const surface = TextureMesh.create(gv.vertexCount, 1, gv.vertexTexture, gv.groupTexture, gv.normalTexture, boundingSphere, textureMesh); @@ -290,6 +287,7 @@ export function GaussianSurfaceTextureMeshVisual(materialId: number): UnitsVisua async function createStructureGaussianSurfaceTextureMesh(ctx: VisualContext, structure: Structure, theme: Theme, props: GaussianDensityProps, textureMesh?: TextureMesh): Promise<TextureMesh> { if (!ctx.webgl) throw new Error('webgl context required to create structure gaussian surface texture-mesh'); + if (isTimingMode) ctx.webgl.timer.mark('createStructureGaussianSurfaceTextureMesh'); const { namedTextures, resources, extensions: { colorBufferFloat, textureFloat, colorBufferHalfFloat, textureHalfFloat } } = ctx.webgl; if (!namedTextures[GaussianSurfaceName]) { namedTextures[GaussianSurfaceName] = colorBufferHalfFloat && textureHalfFloat @@ -299,18 +297,13 @@ async function createStructureGaussianSurfaceTextureMesh(ctx: VisualContext, str : resources.texture('image-uint8', 'rgba', 'ubyte', 'linear'); } - // console.time('computeUnitGaussianDensityTexture2d'); const densityTextureData = await computeStructureGaussianDensityTexture2d(structure, theme.size, true, props, ctx.webgl, namedTextures[GaussianSurfaceName]).runInContext(ctx.runtime); - // console.log(densityTextureData); - // console.log('vertexGroupTexture', readTexture(ctx.webgl, densityTextureData.texture)); - // ctx.webgl.waitForGpuCommandsCompleteSync(); - // console.timeEnd('computeUnitGaussianDensityTexture2d'); - const isoLevel = Math.exp(-props.smoothness) / densityTextureData.radiusFactor; const axisOrder = Vec3.create(0, 1, 2); const buffer = textureMesh?.doubleBuffer.get(); const gv = extractIsosurface(ctx.webgl, densityTextureData.texture, densityTextureData.gridDim, densityTextureData.gridTexDim, densityTextureData.gridTexScale, densityTextureData.transform, isoLevel, false, true, axisOrder, buffer?.vertex, buffer?.group, buffer?.normal); + if (isTimingMode) ctx.webgl.timer.markEnd('createStructureGaussianSurfaceTextureMesh'); const boundingSphere = Sphere3D.expand(Sphere3D(), structure.boundary.sphere, densityTextureData.maxRadius); const surface = TextureMesh.create(gv.vertexCount, 1, gv.vertexTexture, gv.groupTexture, gv.normalTexture, boundingSphere, textureMesh); diff --git a/src/mol-util/debug.ts b/src/mol-util/debug.ts index a5e7f8ee5..eb688aec3 100644 --- a/src/mol-util/debug.ts +++ b/src/mol-util/debug.ts @@ -30,7 +30,12 @@ let isDebugMode = function getIsDebug() { } }(); -export { isProductionMode, isDebugMode }; +/** + * set to true to gather timings, mostly used in `mol-gl` + */ +let isTimingMode = false; + +export { isProductionMode, isDebugMode, isTimingMode }; export function setProductionMode(value?: boolean) { if (typeof value !== 'undefined') isProductionMode = value; @@ -38,4 +43,8 @@ export function setProductionMode(value?: boolean) { export function setDebugMode(value?: boolean) { if (typeof value !== 'undefined') isDebugMode = value; -} \ No newline at end of file +} + +export function setTimingMode(value?: boolean) { + if (typeof value !== 'undefined') isTimingMode = value; +} -- GitLab