Last Updated on 13/02/2025
Unreal has Async Physic Tick which helps you to simulate physics with consistent framerate. This is great for replicating physics simulations and for better and accurate physics. Let’s see how you can implement and use Async Tick properly.
Setting it Up
Actor Component & Scene Component
Async Physic Tick is in Actor Component, Scene Component inherints from it so you can use same thing in both.
Found here (ActorComponent.h):
data:image/s3,"s3://crabby-images/b8cf7/b8cf7fc7668439d85abfdad36d69839da88d7f47" alt=""
You can do this:
public:
virtual void AsyncPhysicsTickComponent(float DeltaTime, float SimTime) override;
void UMySceneComponent::AsyncPhysicsTickComponent(float DeltaTime, float SimTime)
{
PhysicsTickWorker(DeltaTime); //Function where you keep all your physics tasks
Super::AsyncPhysicsTickComponent(DeltaTime, SimTime); //Add this if you want it to work in Blueprints, this should be at the bottom, after your c++ code
ReceiveAsyncPhysicsTick(DeltaTime, SimTime); //This might get execute in Super, i ran my code without it and it was fine
}
You need to enable Async Tick in constructor, theres this thing but it’s in private:
data:image/s3,"s3://crabby-images/61c1b/61c1b304cf683eeba029a80e2f8bc2677a97e9f9" alt=""
data:image/s3,"s3://crabby-images/23dca/23dcac0d7dc33ec1383c8eefd6b58dbb77ce628e" alt=""
It goes here:
data:image/s3,"s3://crabby-images/bc3c4/bc3c49a8c16dd6abeda9987f03576e165031d096" alt=""
You can use this function:
I used it in construction and it was fine.
data:image/s3,"s3://crabby-images/c381a/c381af142916c6931ba31f85d08e434515247c21" alt=""
UMySceneComponent::UMySceneComponent(const FObjectInitializer& ObjectInitializer)
: Super(ObjectInitializer)
{
SetAsyncPhysicsTickEnabled(true);
}
Just don’t put SetAsyncPhysicsTickEnabled in begin play:
It wont trigger in multiplayer, if it doesn’t have owner.
data:image/s3,"s3://crabby-images/be4a8/be4a8ed5cd2964a45385c724882fe2c9c695fe60" alt=""
Actor
Almost same thing in Actor:
public:
virtual void AsyncPhysicsTickActor(float DeltaTime, float SimTime) override;
void AMyActor::AsyncPhysicsTickActor(float DeltaTime, float SimTime)
{
PhysicsTickWorker(DeltaTime);
Super::AsyncPhysicsTickActor(DeltaTime, SimTime);
ReceiveAsyncPhysicsTick(DeltaTime, SimTime);
}
Constructor:
AMyActor::AMyActor()
{
// Set this pawn to call Tick() every frame. You can turn this off to improve performance if you don't need it.
PrimaryActorTick.bCanEverTick = true;
PrimaryActorTick.TickGroup = TG_PrePhysics;
bAsyncPhysicsTickEnabled = true; //You can access this in here
}
In Actor bAsyncPhysicsTickEnabled is little different:
data:image/s3,"s3://crabby-images/46a24/46a243635cd8a596b4a6ac8a255cf0633ba72179" alt=""
In actor blueprint it’s this thing:
data:image/s3,"s3://crabby-images/48568/485682c32775a7999662f25db1ba6426937d0fe0" alt=""
Enable it in here:
data:image/s3,"s3://crabby-images/31b01/31b018047cc2ca8ece3b3e9b1e5ab002128b2593" alt=""
Now we just need to tinker with Async Tick Settings in Edit -> Project Settings.
Enable Tick Physics Async and set the time step.
data:image/s3,"s3://crabby-images/72d16/72d16b8b66cd91cfa8ba63f5fbde32425c17dafb" alt=""
You can calculate delta time like this (in the box), 1 / 120 which means 120 frames or ticks in second.
data:image/s3,"s3://crabby-images/916b5/916b5ae61965ab894470fa2ba582e90c7202abe8" alt=""
AddForce
AddForce applies Delta Time from the main thread so using this in Async tick is the worst thing you can do.
Small physics test that shows why not use AddForce in Async tick. This test was done in 5.5.
This was simple setup, just cube flying straight up. No Gravity.
Final velocity and Travel Distance are the most important. Delta Time was not applied to Add Force, it was applied to Add Impulse.
data:image/s3,"s3://crabby-images/8fa5e/8fa5e0f8c13392658425fe3bc0ec431a9c6e5eee" alt=""
Also there’s no difference between Add Force At Location or Add Force both will give same result. Accel Change means it just includes mass into force.
data:image/s3,"s3://crabby-images/0f65e/0f65e1a5299000194d4d452946b7e3ad8f1047fc" alt=""
AddImpulse
AddImpulse is what you want to use in Async tick. You need to apply Delta Time of async tick into your force calculation before adding it into AddImpulse.
Remember that now you are applying acceleration.
So something like this, will work correctly:
data:image/s3,"s3://crabby-images/f4b8d/f4b8dda8348d71d993eda4fb617aedc8ba9fa312" alt=""
Some of the functions only work in main thread, like get component location or rotation, this can result in stuttering because values are updated in other tick. It’s best to build your own functions which gets the values from the physics rigidbody.
There can be a small stutter on movement which can become visible on high fps (high refresh rate on async tick as well) – you need to use some kind of interpolation plan if you are planning to run async tick on high framerate – type in “smooth” or “framerate” in project settings and you can see some settings related to this problem. You can set Smooth Framerate to activate when fps is in specific range so keep that in mind.
Check out the Mover Plugin, there’s good info about physics & async tick in there:
Syncing Ticks
One important thing you should consider is syncing ticks. Adjusting the application window or changing windowed modes or full screen mode makes the Async tick stall, normal tick includes the time of how long the application was stuck.
Basic example in full screen. No adjusting the app or anything. No server. Async is set to 120 fps.
You can see that in both ticks the bar is filling up the same rate.
All videos are sped up by x4.
Spamming F11 key.
Delta is increasing. Ticks are now off sync.
This is after we implement the fix.
The Fix
First we should count the frames. And then the Delta.
data:image/s3,"s3://crabby-images/25e66/25e6654b2741d3f6d529ac112e6d5a1c65712479" alt=""
You should think about the time range of counting frames, if application is running 24/7 it can become over 10 million (120 fps x 60 seconds x 60 minutes x 24 hours = 10 368 000 frames). It’s still small in single precision range but you just shouldn’t let it get into millions. So think about resetting the value.
You also need to sync server tick counter to joining clients.
In server & client scenario:
You can simply send rpc command to reset the point counter and delta (you need to reset both same time).
data:image/s3,"s3://crabby-images/7e7bb/7e7bbdab5b5e825cc98c22cccb3abe727d0d5bff" alt=""
Before:
This is the place where we can apply delta value.
data:image/s3,"s3://crabby-images/2e9a9/2e9a93a8dea383b34d8c292224475d8c57d6dd93" alt=""
After:
There should be some kind of range where we apply Delta, async tick still needs to run in it’s own rate otherwise it would be just become normal tick.
So if Delta is too large we apply it and set it to zero.
data:image/s3,"s3://crabby-images/87446/87446ec56ddf3fd4a803b7f3bff734ddb46cdac7" alt=""
In single player game this really doesn’t matter and it even might be better that way, because it just means that your car, rocket or whatever you are simulating is just taking a pause. But if it’s a race against the clock the clock is moving but the car is not, because clock is using normal tick, so keep that in mind.
Adding the missed frames back to physic object is complicated because you have to sync velocity, angular velocity, location and rotation. So don’t just increase velocity and pray, your car is still in wrong location. If you are using Unreal’s Chaos Vehicle classes it should already have these things brain stormed or not. This is mostly just for building your own simulation classes.
Check out these Unreal Classes for more:
UChaosVehicleMovementComponent | Unreal Engine 5.5 Documentation | Epic Developer Community
UChaosVehicleSimulation | Unreal Engine 5.5 Documentation | Epic Developer Community
UChaosWheeledVehicleMovementComponent | Unreal Engine 5.5 Documentation | Epic Developer Community
UCharacterTrajectoryComponent | Unreal Engine 5.5 Documentation | Epic Developer Community
Classes | Unreal Engine 5.5 Documentation | Epic Developer Community
Coming Up! Rocket Example
Simulating a rocket with async tick and some crazy stuff!