-
-
Notifications
You must be signed in to change notification settings - Fork 97
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
Add multi-threaded options to 3D physics #483
Comments
…ead. UNTESTED I did not test this because PhysicsServer doesn't look thread-safe at all. See #124 And godotengine/godot-proposals#483
I think that this feature would be a huge addon to Godot. PhysicsServers are often used for better performance and multi-threaded access; having more parallel options with 3D Physics Servers will definitely increase their potential. |
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
It is not only an issue with mesh colliders but also 3D gameplay code that requires updated raycast positions from the physics server and/or run in _physics_process() in general. Very common 3D game features like a characters head lookat or inverse kinematic hand and feet placement make the physics server stall the visual server so much that framerates are constantly halfed or worse with multithreading enabled while they run fine in singlethread mode. |
Because of this issue when using godot_voxel in my project with a big terrain I was having consistent stuttering. To fix it I had to patch Godot in such a way that convex polygon shapes are created asynchronously within the BulletPhysicsServer. |
Any Update about this proposal? |
For Godot 4.0, we no longer need to cater to Bullet since only GodotPhysics will be provided b y default. However, we don't have an active physics maintainer anymore, so this feature will take a while to be implemented (and may not be done in time for 4.0). |
Here is an update for the problem I initially had: I am still heavily bottlenecked by shapes having to be created on the main thread (and I'm already limiting ranges / simplifying meshes). I have a job system cooking many meshes really quickly but it ends up most of the time waiting for the main thread. The ideal (and minimal) solution I would need would be the ability to fully create shapes from my own threads, in a safe way. I don't even need physics to simulate on a different thread, yet. To achieve this, the generic API might be affected a little. It depends if we want multithreading to be enforced by the generic API, or if it should be left to implementers. IMO it may depend on the chosen physics engine. I believe Godot Physics could have that feature. Here are more detailed points I wrote earlier on RC:
Overall I think this proposal's title is only part of the problem, and should not be implemented the way Godot has it currently.
|
@Zylann I've seen your contributions, especially in the double precision support. Noce job! |
No, as he went on to work at Rockstar San Diego (and is no longer working on Godot as a result). |
I too have performance issues. If I understood correctly, when you talk about multithreaded physics, the physics will be calculated on a separate thread from the main thread. But physics is still single-threaded. It will be necessary to add the possibility of allocating a certain number of threads where the calculation of the physics will be distributed. This would be useful for game servers. Alternatively, add an option where each world resource will have a dedicated physical thread independent of other worlds so that one instance of Godot can handle multiple games at the same time on a dedicated server. |
I actually edited my proposal recently, check Worlds each running simulation in their own thread is one way of distributing the workload when multiple threads are provided, although that would be an option, because not all games use multiple worlds and yet could need multithreading. For example, simulation in a world can be distributed between interacting islands. It's up to the implementation to support this. |
Anyone know if there's a PR which implements 3D threaded physics for 3.X? |
Not that I know of. Most of the development focus has been on 4.x lately, as it's where most larger-scale projects are being developed. |
Just for clarification, does World3D utilize threads such that each world may be in its own thread? According to the documentation
And then looking at the setting; It's all a bit ambiguous to me. If I had to guess, it sounds like every World3D still shares the same thread even with multi-threaded physics enabled, correct? |
@warent How a server decides to process all those parts that are relevant to it is a matter of each individual server implementation. PhysicsServer, RenderingServer, NavigationServer, AudioServer ... they all do threading differently or not at all. Commonly the servers that use threads do so at a far lower level to speed things up, e.g. each individual item as a group task. Most projects have only 1 World2D/3D in use so having the threading at the world level would have no benefit for those projects. |
There are 3 aspects to this:
The project you are working on:
I am working on a voxel engine, which is also used by other people's projects.
The problem or limitation you are having in your project:
As described in Zylann/godot_voxel#124
In a voxel terrain, many meshes are getting created dynamically at runtime. Creating mesh colliders is 10 times slower than creating visual meshes (mostly due to computing its BVH), and that takes a lot of time. Unlike
VisualServer
andPhysicsServer2D
,PhysicsServer
has no thread model option, so I assume it's basically not thread-safe and runs on the main thread (correct me if I'm wrong). So I currently have no choice but to create my shapes on the main thread, which limits how many I can process per frame before starting to stall the game.Beyond my own case, this also has the broader issue of preventing scenes from being loaded in a thread, since creating shape resources they might contain would be unsafe too (if they actually are, please document!).
Describe the feature / enhancement and how it helps to overcome the problem or limitation:
The way it used to be done:
If I take the same design as
VisualServer
andPhysicsServer2D
from back in Godot 3, here is how it would be a dropdown in ProjectSettings:Single-Unsafe
would be the option for running physics on the main thread, without thread safety (basically how it is now).Single-Safe
would still run physics on the main thread, but with thread-safety in case a thread calls its functionsMultithreaded
would run physics on a separate thread and also be safe to be called from other threads.This would be a first improvement since the physics server would have its own thread time to spend without all the other stuff done in the main one.
If we go for that design,
PhysicsServer
would be wrapped into aPhysicsServerWrapMT
class which pushes calls to its function into a ring buffer, and dequeues calls from that buffer from inside the physics thread (or the main thread in case ofSingle-Safe
).HOWEVER, there is a big flaw in this design, which is important for my use case: when I create a large amount of complex
Shape
resources, it needs to call Bullet/other impl. which in turn has to build its BVH for that shape. So a call to the server sounds unavoidable. If all calls are wrapped the same way, the server will dequeue calls to create shapes, and that can stall the simulation because they will be processed in the same thread that runs the simulation, even if those shapes don't intervene in the simulation yet. And if the result needs to be accessed just after creation, the queue has to be flushed immediately, which defeats the reason to run in a thread in the first place.This problem might exist in other servers already, however that's mostly an assumption I'm making since I don't know their code in detail (especially with Vulkan rewrite). So perhaps not the whole server should be threaded the same. What I'm asking here is that creating resources involving computations which don't intervene in the frame logic should not slow it down.
Maybe resource creation could have another ring buffer and be processed in another thread? (Although I use several high-processing threads already so I'm still wary of creating new threads for too many things, as it may still lead to contention when exceeding CPU limits).
Or, a simple solution would be to not wrap those calls and let them be processed inside the calling thread, which would solve my problem best since I already do the call from a thread. It's only viable if the corresponding server code is officially safe for that kind of contract.
How it should be done:
So that leads to the following:
This alternative could be at least as simple as a
PhysicsServer
API.If this enhancement will not be used often, can it be worked around with a few lines of script?
I already had a workaround so far by doing everything on the main thread. It limits performance dramatically and complexifies the code because I have to scatter away the collider part and maintain a timed queue with the data myself. In many cases you can see terrain parts loading and you have to wait a while for those on the main thread, before you can start moving around and edit.
Is there a reason why this should be core and not an add-on in the asset library?:
This is already core for other servers, the feature is just missing in
PhysicsServer
. But I want to also emphasize on the resource creation case, which is the original reason why I bumped into this.But it can technically be done with different physics engines once GDExtension gets an API to implement physics servers. Point being, it should not be something Godot implements on your behalf (and if so, must be an optional last resort).
The text was updated successfully, but these errors were encountered: