-
-
Notifications
You must be signed in to change notification settings - Fork 21.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
Fixed Timestep Interpolation (3D) (3.x) #52846
Conversation
cb65f28
to
c1e7fb7
Compare
6579493
to
0eedc19
Compare
This should be getting pretty reasonable now, I could do with some people testing this out. The interpolation defaults to on ( Instructions
Let me know any bugs / strange behaviour. Strictly speaking you'll also want to call |
0eedc19
to
76c75d2
Compare
119d2e4
to
1664ebf
Compare
da836ed
to
ae67e51
Compare
f9a1d79
to
174bec9
Compare
doc/classes/Node.xml
Outdated
<method name="set_physics_interpolated"> | ||
<return type="void" /> | ||
<argument index="0" name="p_interpolated" type="bool" /> | ||
<description> | ||
Enables or disables physics interpolation per node, offering a finer grain of control than turning physics interpolation on and off globally. | ||
[b]Note:[/b] This can be especially useful for [Camera]s, where custom interpolation can sometimes give superior results. | ||
</description> | ||
</method> |
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 and is_physics_interpolated
could be exposed as a property.
Edit: I see you have it commented out, any specific reason?
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 did in fact expose it in earlier versions - when we discussed this @reduz preferred to have this hidden (I think with the aim of not overwhelming users with new concepts, or exposing the minimum possible).
On the other hand, if we get feedback expressing an interest to expose this I think we can easily re-add it.
4b291dc
to
e22c225
Compare
Adds fixed timestep interpolation to the visual server. Switchable on and off with project setting. This version does not add new API for set_transform etc, when nodes have the interpolated flag set they will always use interpolation.
e22c225
to
522bce1
Compare
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.
Great. BTW I've also done preliminary docs for the interpolation and occluder polys, which might be good to link in the beta notes: https://github.com/lawnjelly/Misc/blob/master/FTIDocs/FTI.md I'll be improving these and doing some screenshots etc as I get the chance. |
Thanks! |
Looks good to me too! My apologies for taking so long to review the full thing |
I'm not sure if this should be submitted as a separate issue. Enabling physics interpolation seems to double the speed scale on the second and subsequent emissions of CPUParticles. Halving the speed scale to .5 makes the particles act as they should. The first emission seems to work as intended. I could upload a minimal project if needed, but it is just any CPUParticles. |
https://github.com/godotengine/godot/issues |
There are some significant changes to CPUParticles in the PR. This would be good posted as an issue with an MRP, meanwhile I will try to replicate it. It could be related to something being done / advanced twice, as we have to deal with both previous and current frames when using interpolation. EDIT: Is ok, have found it, was quite simple, see PR below. 👍 |
Hi. v3.5.beta.custom_build (from origin/3.x branch). I'm noticing issues with interpolation of collision shapes. It looks like they doesn't react to |
This is actually expected as far as I can see, nothing is technically going wrong. Collision shapes themselves are not interpolated (they expressly should not be), they are not derived from The wireframe in the editor for the debug collision shapes is purely for debugging, I'm not sure offhand whether it is drawn in worldspace or simply attached to these non-interpolating collision shapes. The wireframe does seem to currently be interpolated (which is probably just the default). It does look a little strange when teleporting I will admit. I can have a look into whether it can be made to match the graphical sphere (if you can create an issue with an MRP), or maybe we can simply turn off the interpolation flag for the wireframe, that might be better for visualization. But it is not a nasty "bug" per se, just an interesting side effect (that's my excuse lol 😁 ). It is also possible I guess that people might hang some other |
I have issues with collision detection when phys interp is on. I have an area containing (0,0,0), and when I'm adding a player to the scene and immediately "teleporting" it to the desired position (outside the area), the Maybe I'm doing something wrong, so I would ask how to add a node to the scene into the desired place? I cannot set it's position before adding to the tree - Godot complains about this. |
This needs an issue and a minimum reproduction project, and I can look into it. 👍 The MRP is necessary by the way because we have had several problems with this kind of thing in the past in the physics (way before interpolation was introduced), @pouleyKetchoupp did some workarounds I believe. This sounds less likely to be directly caused by the interpolation (ideally it should only affect the rendering layer, and not the physics / collision detection), but there may be some interaction going on, and debugging a reproduction project will show what is happening. |
Adds fixed timestep interpolation to the visual server.
Switchable on and off with project setting.
On request from reduz I have changed the
teleport
function name toreset_physics_interpolation
.(While some of the features will be shared with 2D, they are both fairly separable so I will be addressing 2D in a later PR. Once we are happy with the approach in 3.x, I will do similar for 4.x, perhaps after we get some initial testing.)
Details
VisualServer
are now stored in theScenario
(i.e. unique per scene tree).set_scene_tree_physics_interpolation_enabled()
on theSceneTree
, which defaults to the project settingphysics/common/physics_interpolation
(and is always off whenis_editor_hint
is set).physics_interpolated
flag, which can be read and set for individual control withset_physics_interpolated()
.get_global_transform_interpolated()
.CPUParticles
andMultiMeshInstances
now work via a modifiedMultiMesh
that can handle interpolation.Based around proposal : godotengine/godot-proposals#2753
Notes
_physics_process
. Where changes are necessary these are most likely to occur in customCamera
code.Engine
toSceneTree
in response to comments from reduz and jordo. The interpolation lists are also moved to be perScenario
(i.e. SceneTree). This enables the whole system to be flexible for users who are not using the standard SceneTree.Scenario
, theCamera
needs quick access to its scenario when callingcamera_set_transform
. As a result I've added storing thescenario
in theCamera
, in a similar manner to how it is stored inInstance
s.Transform
into a separate classTransformInterpolator
. This was getting repeated in several places in the code (for instances, cameras, and potentially in the scene tree code) and it probably makes sense to put it in one place.slerp
where possible (this is dependent on theBasis
).InterpolatedCamera
. These can be done in a separate PR once we decide on an approach.set_transform
s. This isn't strictly speaking necessary, so try not to get confused by this (it's cheaper than comparing 2 transforms each time, which could be important with a bunch of non-moving objects).get_global_transform_interpolated()
It turns out that in a practical sense, especially for Cameras, it is often necessary to turn off the automatic fixed timestep interpolation and interpolate manually. See below for the problems with @Calinou 's project for an example - mouse looks better without fixed timestep interpolation, either with no interpolation, or using non-fixed timestep based interpolation. This means in
_process
users will want to be able to focus on the displayed (i.e. interpolated) position of an object, rather than the position at the last physics tick.The same issue of needing to know the transform on a frame rather than a physics tick occurs when you need to make something emit at
_process
rather than_physics_process
.For these reasons, the PR includes an implementation for selectively duplicating the interpolation client side, i.e. within the scene tree. The reason for this requirement is that interpolation usually takes place in the VisualServer, and the queue / threading system means that synchronization is difficult to retrieve information from the VisualServer - it may cause a stall.
The solution used involves storing extra optional
InterpolationData
on theSpatial
. This mechanism only is activated the first time the user callsget_global_transform_interpolated()
on a node. This means there is no cost for the vast majority of nodes, both in terms of memory and processing.MultiMeshes
It turned out that a large hurdle was handling MultiMeshes in a reasonable manner. As well as for MultiMeshInstances, these are used by CPUParticles under the hood. MultiMeshes use instancing in GLES3.
As a multimesh is a list of instances, in theory the main work is to maintain a list of previous and current data (transforms, colors and custom data), and interpolate between this. In practice however there are two snags:
The solution I have used here is the simplest conceptually, but it does impose some overheads. Basically for each bulk update of the particle instances (on a physics tick), instead of sending just the current data, it sends the current AND previous data every tick. This is wasteful, but it ensures that the data of the current and previous is always in sync.
This is achieved via a new alternative function:
functions can be used as usual. There is also one other new function of interest:
which can be used to teleport individual instances.
How MultiMesh works internally
MultiMesh was previously implemented purely as classes within each backend, which was a problem, because ideally we want to share interpolation functionality between the two.
The solution was two create a set of wrapper functions in
RasterizerStorage
, which handle interpolation, and in turn call a similar set of underscored functions in the actual backends which are identical to the previous code.In order to store interpolation specific data, the backends are responsible for creating a
MMInterpolator
(multimesh interpolator) object as part of their multimesh, and returning it when the following function is called:From here the Rasterizer storage can now handle storing previous and current data for each MultiMesh, and interpolating before a frame is drawn in a similar way to the Instances and Cameras.
One further gotcha is that
MultiMesh
does not have aScenario
associated with it, and could potentially be used like a resource in several MultiMeshInstances, potentially in differentScenario
s (although unlikely). As the global interpolation flag is now a property of theSceneTree / Scenario
, this makes the situation not quite so ideal, but a flag forinterpolated
is available on eachMultiMesh
and will be set by the corresponding instance that "claims" it.This isn't perfect but it's a bodge around the problem that a
MultiMesh
isn't owned by aScenario
. In the vast majority of cases anyway the global interpolation will either be on or off, and providing aMultiMesh
isn't used in two Scenarios with differentSceneTree
interpolation flags this shouldn't be a problem.CPUParticles (in global (non-local) coordinate mode)
Along with the multimesh changes, there were additional changes to CPU particles to get them to behave correctly with interpolation.
Particle global coordinate space
CPUParticles were previously implemented in a slightly curious way in global mode. Although the particle positions are calculated in global (world) space, instead of specifying the particles in global coordinates, the instance in the VisualServer still had the transform inherited from the parent node. In order to compensate for this the particles are given an extra step to back transform them from global space to local space, by xforming by the affine inverse of the node transform.
This was seen as potentially problematic in terms of interpolation, and also was requiring more CPU than necessary. An obvious simpler solution is to set the instance to an identity transform, and just specify the particles directly in global space.
I have done this in the PR, and it did require some small adjustments but seems to be working correctly. It is possible that there may be more small compensations I have missed, but this seems well worth doing.
Camera code
The most likely area to require changes for existing games is Camera code. While cameras can be used as before except updating only in
_physics_process
and allowing fixed timestep interpolation to interpolate them, best results often involve turning automatic interpolation off and doing this manually.When doing manual interpolation, as with the smoothing addon, it is recommended that in the SceneTree you separate the Camera from the object it is targetting, i.e. it is easier if the Camera is specified in global space and is not a child / grandchild of the object being targetted.
For custom interpolation, I recommend moving the Camera in
_process
, and callingget_global_transform_interpolated()
on the target node, and using this interpolated position to focus on. The other calculations can be done as would normally be done for a camera.Testing
Tested so far in:
Works great in all so far at e.g. 20 ticks per second. In most cases no changes were needed to the games, just switching on interpolation.