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

Change physics joint tests to freeze bodies as kinematic #1177

Open
wants to merge 1 commit into
base: master
Choose a base branch
from

Conversation

mihe
Copy link
Contributor

@mihe mihe commented Mar 19, 2025

The joint tests found in 3d/physics_tests/test/functional/test_joints.gd are currently behaving a bit strange when you use Jolt Physics as the 3D physics engine, while having all the test options set to their default value:

Static_WithSleeping.mp4

The reason for this seems to be related to the fact that the upper body (parent_body) in this test is by default set to use freeze = true, which by default uses freeze_mode = RigidBody3D.FREEZE_MODE_STATIC. This effectively turns the body into a StaticBody3D, which as the name suggests isn't really meant to be moved, and thus have no linear or angular velocities associated with it, which will cause problems when connected to joints, as solving for the underlying joint constraints (in either physics engine) involves the bodies' velocities.

The fact that the lower body just sits there seems to be related to the lower body going to sleep, which we can confirm by disabling sleeping:

Static_WithoutSleeping.mp4

Static bodies in Jolt are always considered to be sleeping, so it makes sense to me why moving the upper body wouldn't wake up the lower body, despite being connected by a constraint. I don't quite understand why this isn't the case for Godot Physics as well though.

Note however that something looks/feels a bit off ("sluggish") with how the lower body behaves even when just disabling sleeping, which as mentioned above is related to there being no velocities associated with the static body.

If we instead change freeze_mode to RigidBody3D.FREEZE_MODE_KINEMATIC, we get this, which works even when leaving sleeping enabled:

Kinematic.mp4

This can be reproduced with Godot Physics as well to some degree, although the discrepancy between the two body modes is less pronounced there, which I think is due to the fact that Jolt by default only uses 2 position iterations in its solver, whereas I think Godot Physics solves for both velocity and position in its 16 default solver iterations.

Upping the solver iterations to something like 16 for Jolt makes things behave a bit better, even when using FREEZE_MODE_STATIC, but this would be a hack in this case.

This PR instead fixes all of this by simply switching freeze_mode to FREEZE_MODE_KINEMATIC in these tests, as it should be.

Copy link
Member

@rburing rburing left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Indeed this freeze mode makes more sense.

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

Successfully merging this pull request may close these issues.

3 participants