-
-
Notifications
You must be signed in to change notification settings - Fork 35.5k
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
WebGLRenderer: Add compileAsync()
.
#19752
Conversation
src/renderers/WebGLRenderer.js
Outdated
|
||
} | ||
|
||
let foundTarget = scene === target; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There's some effort made here to allow cases where the target is already attached to the scene (or IS the scene), but I'm not sure how useful that is? Could simplify things a tiny bit if that case was deemed unnecessary.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
FYI, I have removed the second scene parameter and exchanged it with camera
so compile()
and compileAsync()
both have the same signature and perform layer testing for lights.
src/renderers/WebGLRenderer.js
Outdated
|
||
// if some materials are still not ready, wait a bit and check again | ||
|
||
setTimeout( checkMaterialsReady, 10 ); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This should probably ideally be a rAF, but I'm not sure what the best pattern is for syncing up with any existing render loop is and I wanted to keep this PR simple initially for the sake of discussion.
src/renderers/WebGLRenderer.js
Outdated
|
||
} | ||
|
||
if ( extensions.get( 'KHR_parallel_shader_compile' ) !== null ) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This check could easily go away and be replaced by an arbitrary call to checkMaterialsReady()
every time. The downside to that is that systems which don't support KHR_parallel_shader_compile
will immediately report that every program is ready and thus try to render with it right away, thus erasing some of the potential benefits. I figured that if someone was calling this method at all they would be OK with some delay, so we might as well give the compiler a frame's worth of breathing room before we notified the callback. Wouldn't eliminate stalls on those browsers, but it would still reduce them somewhat for developers using this pattern.
// program as ready immediately. It may cause a stall when it's first used. | ||
let programReady = !parameters.rendererExtensionParallelShaderCompile; | ||
|
||
this.isReady = function () { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This method, however, is new.
Great PR @toji ! Unfortunate that we can't make use of parallel compiling for the regular workflow without major architectural changes, but at least having the option to pre-compiling is already very nice. I would love to have this and the related PR reviewed and merged sooner rather than later. IMO we should also emphasize this option in as many examples as possible, the net advantage of using parallel compiling is huge! |
Review ping? Looks like I need to rebase but I'd appreciate some feedback on the direction before putting too much more effort into it. |
The performance problem on Chrome I mentioned in #16662 seems being resolved. https://twitter.com/gfxprogrammer/status/1301246609129222144 I think it's good time to go for KHR_parallel_shader_compile support. |
(@mrdoob: This is the patch we were talking about last week in the office.) I've rebased this and made sure it works against the newest Three, and refactored the signature for the function a bit. Now it's called 'compileAsync', takes the object to compile first, the scene it will be added to as an optional second parameter, and returns a promise when the object can be added to the optional scene (or rendered as the scene, if no explicit scene was given) without waiting on compilation. Example: renderer.compileAsync(newObject, scene).then(() => {
scene.add(newObject);
}); Plus I can happily confirm that testing after rebasing shows that it still offers significant benefits! And for whatever reason, it's even easier to see the difference in Chrome's dev tools now. Consider this profile of the webgl_loader_gltf_transmission example, before and after: You can see that, in total, the time it takes to get to the first render of the object after load is almost exactly the same, with the biggest difference being that the largest blocking task gets broken up into a couple of smaller chunks with a nice gap in the middle that allows a few more frames to be drawn while the shaders compile. This is great for keeping the page interactive during loads! (The remaining big blocking calls are mostly |
Yes, Yes, Yes! We need this @mrdoob! |
src/renderers/WebGLRenderer.js
Outdated
// Wait for all the materials in the new object to indicate that they're | ||
// ready to be used before resolving the promise. | ||
|
||
return new Promise( ( resolve ) => { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Kind of minor but... is it possible to use async/await instead?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I don't think it is in this case. The problem is that this kicks off a polling loop with setTimeout()
, which of course is callback based. If we want to treat it as a promise we have to explicitly wrap it in a Promise()
constructor.
This does still allow developers to call the function as await renderer.compileAsync(....);
though! And I've updated the two examples this was used in to demonstrate exactly that, because it's a much nicer pattern.
Ops, I broke |
Rebased to fix the |
src/renderers/WebGLRenderer.js
Outdated
|
||
// Only initialize materials in the new scene, not the targetScene. | ||
|
||
scene.traverse( function ( object ) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could we add scene overrideMaterial
support? It seemed to work as expected when I did the naive thing:
const _compileMaterial = function ( material, object ) {
if ( Array.isArray( material ) ) {
for ( let i = 0; i < material.length; i ++ ) {
const material2 = material[ i ];
getProgram( material2, targetScene, object );
compiling.add( material2 );
}
} else {
getProgram( material, targetScene, object );
compiling.add( material );
}
};
if (scene.overrideMaterial) {
scene.traverse( function ( object ) {
const material = object.material;
if ( material ) {
_compileMaterial(scene.overrideMaterial, object);
}
});
} else {
scene.traverse( function ( object ) {
const material = object.material;
if ( material ) {
_compileMaterial(material, object);
}
} );
}
This would be useful to have! |
As an user, shader compilation that blocks the main thread is always a problem in real-world projects (websites with various animations usually). The guys from Needle implemented it already in their engine too: |
The compileAsync method adds a way to allow apps to wait on shader compilation before adding objects to a scene.
📦 Bundle sizeFull ESM build, minified and gzipped.
🌳 Bundle size after tree-shakingMinimal build including a renderer, camera, empty scene, and dependencies.
|
Rebased now that #19745 has landed! Thanks @Mugen87! Brief testing shows that there is still the expected wait for compile to happen, but it looks like there's also been some shifting around of how |
Fix typo.
I'll try to debug both examples using |
Sorry! Clicked the wrong button! Thanks for taking a look, |
Code style improvements.
Keep signature of `compileAsync()` similar to `compile()`.
compileAsync()
.
During testing I feel a conceptual peculiarity with our Pre-compiling only works if the render state during the pre-compilation is equal to the actual render state in the scene. E.g. if you call three.js/src/renderers/webgl/WebGLPrograms.js Lines 316 to 328 in 30b6736
So it's best to use |
This will align well with WebGPU where compilation is automatically async, and provides promises on completion so rAF games are not needed either. I was already looking at implementing compile() for the WebGPURenderer, and adding it to the fallback WebGL path shouldn't be complicated either. Compilation isn't forced in the current WebGPURenderer.render() so calling render() doesn't not necessarily render what you expected or indeed anything. As an aside, forcing the webGPURender.render() to wait for all shaders to compile is rather messy, because often the frame buffer texture has expired by the time compilation has finished and you get console errors and no output, unless you use GPURenderBundles. |
Looking at the semantics of I have the feeling more users would benefit from |
@Mugen87 I think Edit: Specially considering that it'll match the behavior of |
I have filed a PR to explore the suggested concept: #26964 |
I can see now how the camera is useful (that seems like a fairly recent addition to
I think this model significantly reduces the usefulness of the method. Seems like I didn't do a very good job of communicating my intent here, sorry for that. If you only cared about the first display of the entire scene then you could omit the optional second scene and it would function as you describe. But my hope with the second scene was that you could use it to say "I plan on adding this object tree to that one. Tell me when I can do so without blocking due to shader compiles." It used the render state from both scenes combined so that if you were loading an individual mesh you could still get all the necessary lighting and other state information. As a concrete example, on my page https://xrdinosaurs.com I would load the entire environment but the dinosaurs were loaded by the user clicking a button. Because it is typically viewed in VR, avoiding loading jank is critical. Every time the user clicked the button to load a new dinosaur, I would load the glTF mesh, then call It's not clear how I would use the API with the updated signature effectively in this scenario. I can't add the dinosaur mesh to the environment and then call The current signature works if the only thing I'm concerned about it initial display of the entire scene, but it seems to me like that's more of a concern for the examples than many larger apps. Similarly, an
|
Okay, I understand it make sense to add |
It shouldn't be expensive since no rendering happens. But yes, the copy/paste operations to reproduce the same render state would be inconvenient. |
TL;DR: Provide a function like
renderer.compile()
that notifies a callback when the given object can be added to the scene with minimal blocking on shader program compiling/linking.This proposal builds on #19745, would fix #16321, and is an alternative to #16662.
The longest blocking operations that tend to happen when an object is first added to a scene are textures uploads and shader compiles. While texture uploads can be dramatically reduced by either using ImageBitmaps (see #19518) or compressed textures, shader compiles are already asynchronous and the best practice for reducing the time that they block is to give them time to finish prior to first use. If the
KHR_parallel_shader_compile
extension is available we can do even better by polling to discover when the compile is finished prior to it's first use.#19745 takes a step in the right direction by deferring any potentially blocking shader calls until the first time the shader is actually used for rendering, but Three's current architecture is such that most shader programs are compiled/linked immediately before their first use, which forces the worst-case scenario and causes the first query from the program (for example, a
getProgramInfoLog()
orgetUniformLocation()
call) to block while the compile finishes. You can see an example here recorded from the GLTF Loader example. Notice that thegetProgramInfoLog()
call takes 49msThis PR suggests adding a new method to the
WebGLRenderer
which functions similarly to the existingcompile()
method with a couple key differences: It supports passing objects that are not yet part of the scene, and takes a callback which will be called when the shaders for the passed objects have finished compiling and can be added to the scene with minimal blocking.This would functionally look something like this:
Using this pattern and the code in this PR I get the same first load shown above looking like this:
Notice that the
getProgramInfoLog()
call that previously took 49ms has shrunk so much that you can't even see it without zooming in. It's now under 2ms. You just got back ~3 frames of stalled renderer! (Especially important for XR uses.)(Also notice in the wider view of the timeline that the PMREM generator block, which is the first large column, is significantly faster too. That's actually a side effect of #19745, but ultimately it's following the same principle.)
To preemptively address a couple of questions that I feel will come out of this:
Why not update the existing
compile()
function to support this?This is a very viable option! But in order to maintain backwards compatibility you're either grappling with some very non-intuitive argument ordering or forcing people using the call to attach the new object to the scene, call
compile()
, then immediately detach it again and wait for the callback to once again re-attach it. 😝 Otherwise they're doing very similar things.Why doesn't this method take a camera like
compile()
?Turns out that's not really necessary for what this method and
compile()
are doing, which is simply preloading the material shaders. The camera is being passed to the light setup, which IS very necessary for the shaders, but it's only being used to compute the view matrix which only really affects specular computation and doesn't have any bearing on the shader compilation. FWIW I thinkcompile()
should drop thecamera
argument too.What about adding a
parallelCompile
property on the material, like in #16662?This was my first inclination as well, but it suffers from several drawbacks enforced due to Three's architecture. For one, it breaks the model of objects being visible as soon as they're added to the scene. They just "show up" at some point and the developer has no idea when. That's fine for some uses, not for others. Also, crucially, unless great lengths are taken to prevent it any changes to the scene's lighting setup will cause another parallel compile, during which the objects will flicker out of existence for a few frames. This is a far more objectionable artifact. Finally, some apps (like the glTF loader example) only draw when the user is interacting with the scene (like rotating the camera). Without an active animation loop the
parallelCompile
approach doesn't have a way of indicating that the object is ready for drawing and thus you get stuck with an empty scene until you click and drag, which is clearly unwanted behavior.By making this an explicit call, the developer has exact control over what objects they are willing to wait on and what should happen when they're ready. It doesn't "magically" make existing apps better, but does offer an optimization path for devs that care about it.