Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[Perf] Re-rendering multiple dicom images at once #3147

Open
shunia opened this issue Oct 16, 2024 · 14 comments
Open

[Perf] Re-rendering multiple dicom images at once #3147

shunia opened this issue Oct 16, 2024 · 14 comments

Comments

@shunia
Copy link

shunia commented Oct 16, 2024

High-level description

I'm building a filmer app that the customer would be able to interact with multiple dicom images at once for adjusting say ww/wc, pan, scale etc, and I was aiming for 60fps for smooth real-time interactions.

I'm using vtkOpenGLRenderWindow under the hood and creating multiple renderWindows and each window holds a dicom image (pixel data) for interactions, which is quite similar to the MultipleRenderWindow example, but with some modifications:

  • use custom managed input event handlers instead of vtk interactors
  • re-render by picking nessesary child renderWindows and call the .traverseAllPasses() api instead of rootWindow.render()
  • split render calls into batches with a size about 40 between frames (with requestAnimationFrame), and ensures each renderWindow will only be re-rendered once within the re-render calls

I'm stuck into a sutuation where in my developing machine I can only achive 60fps when re-rendering mostly 40 images at once with strategies implemented as above, when adjusting ww/wl with:

// the code is definitly much complex than this but the calling sequence is much like this
actor.getProperty().setColorWindow(ww);
actor.getProperty().setColorLevel(wl);
renderWindow.traverseAllPasses();

the hardware may be better in production machine but still the number is far less than what I would want to achive, which I would like to keep 60fps when re-rendering about 100 images at once, because the whole app would load over 2000 images at once (the loading performance and the first rendering performance is not a problem here)

With above senerios, what could I do to improve the performance? Or is there anything I do wrong? Is there ways to batch the re-rendering under the hood in vtkjs to squeeze the performance?

Steps to reproduce

It's somehow a little bit complicate to extract the logic out of my app so for now there's none re-produciable steps.

Detailed Current Behavior

image

Environment

  • vtk.js version: "@kitware/vtk.js": "^30.9.3",
  • Browsers: Chrome Version 129.0.6668.100 (Official Build) (64-bit)
  • OS: Window 10 Enterprise 10.0.19045 Build 19045
@finetjul
Copy link
Member

What exactly does the screenshot represent ?

It's odd that you have to create a new texture each time you re-render. Do you confirm that updatelabelOutlineThicknessTexture() creates vtkOpenGLTexture at each render ?
(fyi @sedghi)

Did you check that MultipleRenderer is any faster ?
If your render windows do not need to share GPU resources, did you try to not make them shared ?

@sedghi
Copy link
Contributor

sedghi commented Oct 16, 2024

when re-rendering mostly 40 images at once

You mean you have 40 textures and 40 render window? I'm not surprised if that does not achieved 60 fps

@shunia
Copy link
Author

shunia commented Oct 17, 2024

Hi @finetjul, @sedghi,

Thank you for your quick responses. I’d like to provide some additional context regarding our current rendering challenges.

Summary of Current Situation:
Our application renders between 200 to 2000 renderWindows simultaneously, each displaying a DICOM image with adjustable parameters like window width/level, scale, pan, etc. Users can select multiple renderWindows to modify images collectively. However, achieving 60 fps with 2000 renderWindows is impractical. To address this, I am currently batching the render calls, processing 40 renderWindows per batch using the traverseAllPasses function for each renderWindow.

Questions and Clarifications:

  1. Regarding updatelabelOutlineThicknessTexture: I have not called this API. What steps should I take to confirm this?
  2. MultipleRenderer Approach: Implementing a MultipleRenderer is not feasible at this stage as our app is based on a MultipleRenderWindow architecture. Reworking the entire implementation is not an option right now.
  3. Render Windows: While I mentioned having 40 renderWindows in the batch process, I need to clarify that I'm managing 200 to 2000 renderWindows simultaneously, rendering only 40 at a time to optimize performance.

I would greatly appreciate your feedback and any suggestions on alternative approaches we might explore. If MultipleRender is truly the only viable solution, I am open to considering it, despite the potential need for extensive codebase modifications. However, I hope we can identify other strategies to test before pursuing that route.

@finetjul
Copy link
Member

If you have that many views in your single page application, it seems that you will get the best performances with having a unique render window with a unique renderer. You would then have many ImageResliceMappers... (or potentially a unique one with a tiled image you created on the cpu)

@shunia
Copy link
Author

shunia commented Oct 18, 2024

...with having a unique render window with a unique renderer...

Could you please elaborate more on this?

...You would then have many ImageResliceMappers...

I did not find any examples on how to use vtkImageResliceMapper, could you please direct me to resources/docs on how to use it?

@finetjul
Copy link
Member

Could you please elaborate more on this?

const fullScreenRenderer = vtkFullScreenRenderWindow.newInstance();
const renderer = fullScreenRenderer.getRenderer();
const renderWindow = fullScreenRenderer.getRenderWindow();

for (let i = 0; i < 1000; ++i) {
  const mapper = vtkImageResliceMapper.newInstance();
  mapper.setSlicePlane(vtkPlane.newInstance({normal: ..., origin: ...));

  const actor = vtkImageSlice.newInstance();
  actor.setMapper(mapper);
  renderer.addActor(actor);
}

I did not find any examples on how to use vtkImageResliceMapper, could you please direct me to resources/docs on how to use it?

https://kitware.github.io/vtk-js/examples/ImageResliceMapper.html

@shunia
Copy link
Author

shunia commented Dec 23, 2024

@finetjul Sorry for the late response, I've been away from this project and just getting back.

For the one renderer solution I believe the point is to reduce the number of canvas to only one, and use multiple mapper to render the images into the unique canvas to gain performance, right? This is clear and definitly would work somehow, the downside would be we need to re-implement a lot of basic dom stuff like layout/event handling/intersections, etc, we need to discuss the possibiltis internally of course.

For now, the rendering performance is acceptable after batching + intersection observers, which means the maximum windows rendered at one screen are less than 500, which is managable.

But at the same time, we hit another wall which is the rendering is blocking the main thread a little bit, especially when initializing all these images into the canvases. I know that it is possible to use OffscreenCanvas or even better Web Worker to transfer the canvas manuplations to the another thread. Is it possible for us to do so? Does the library provide any APIs for this kind of process, or how can we 'hack' into the process to do it?

@finetjul
Copy link
Member

The ManyRenderWindows shows offscreen rendering: https://kitware.github.io/vtk-js/examples/ManyRenderWindows.html

@shunia
Copy link
Author

shunia commented Dec 23, 2024

I did not find any OffscreenCanvas in the said example.

I did find out that cornerstone3D is using OffscreenCanvas to render multiple viewports at one time with a custom render window implementation though: vtkOffscreenMultiRenderWindow

@finetjul
Copy link
Member

The offscreen rendering is done under the hood when you have a parent render window (mainRenderWindow.addRenderWindow(renderWindow);)

@shunia
Copy link
Author

shunia commented Dec 31, 2024

Sorry but I can not find any evidence in the source codes for the usage of OffscreenCanvas.
I've searched direct ref to OffscreenCanvas, but got only result is in ORMTexture.worker.js which is not directly used in rendering.
I've searched the API transferControlToOffscreen which is another way to create a OffscreenCanvas, there is no result at all.

If I'm mistaken here please bare with me.

@finetjul
Copy link
Member

Indeed, it does not rely on OffscreenCanvas. That could be a nice addition though...

@shunia
Copy link
Author

shunia commented Jan 17, 2025

Yes, for my use case, where I need to render a ton of canvases, OffscreenCanvas can be really benificial for UI rendering, since it will significantly reduce the UI blocking on main thread.

A better general solution would be follow how CornerStone usees OffscreenCanvas, by puting multiple renderers inside an offscreen render window, and then use the viewport model to manage the renderers, and then draw the image out to multiple on-screen canvases.

Me personally created an one on one relationship between OffscreenCanvas to on-screen canvas, which I think is easier to manage and understand, but also because I was combining it with the many render window approach, which I don't want things to be more complicate.

@floryst
Copy link
Collaborator

floryst commented Jan 20, 2025

Supporting rendering to an OffscreenCanvas also falls under the umbrella of running the vtk.js rendering pipeline in a webworker, since that's where we'll get the most benefit of using OffscreenCanvas.

Doing that is logically straightforward, but there are details that need ironing out (e.g. the async nature of rendering in a worker), and we haven't done proper investigation into doing this.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

4 participants