Threads
How to fix resource performance
Assessment
Without having any markers created, the resource already runs at 0.02-0.03ms, but we are not even doing distance checks yet.
I have created a simple command that just spam creates 500 markers with an offset on each, let's look at the performance now:
Performance is terrible now, and even without rendering any we are above 0.4ms, which is unacceptable.
Let's look at some code now.
Here we can see the "main thread" that works coordinate checks. It loops every thread, and triggers events if you enter or exit on the first time. So how can we improve this?
Remove the native for coord distance calculation
Natives are slow. FiveM has utilities to make checks like those much faster.
local distance = GetDistanceBetweenCoords(coords.x, coords.y, coords.z, v.coords.x, v.coords.y, v.coords.z, true)
gets changed to:
local distance = #(coords - v.coords)
0.28ms! Wow, we have already shaved of 0.15ms, with a change of one line. Use #(a-b) for distance checks, instead of any natives.
Splitting up loops:
This doesn't make us happy enough though. Let's split up the threads into two.
Now we have split this into two threads, you will notice the main thread we looked at before has basically not changed. The only thing we changed is markers
-> drawnMarkers
when we start the for loop.
We add a new variable called drawnMarkers
including all markers that are currently in draw range.
In addition to this, we have added a completely new thread that runs every 500 frames, and repopulates the drawnMarker
table with the now new entries.
Let's now check the performance:
Now we are starting to see massive performance improvements. Outside render range we are down to 0.06ms, inside a marker directly we are at 0.20ms
Now why is the performance so terrible inside a marker? Let's look at ESX' help notification function.
This displays a help text inside which is just "text example" in our markers. What this also does is add a text entry every frame together with displaying the help text. A simple improvement would be deleting the line of AddTextEntry('esxHelpNotification', msg)
and adding the following to the end of the disc-base:hasEnteredMarker
event:
This saves about 0.02ms once again, we want to use the least amount of native calls possible.
Digression: Render distances
As of the config file I have here, the default render distance is 30 units. That is way too much. Reducing this distance to 10 or 15 units cuts down resource time drastically especially when there is a lot of markers in small range. Reducing our example to 10 units brings it to around 0.06ms
Right now we are just spawning markers close to ourselves. If we were to run, let's say, an amazing RP server that has like 400 markers all across the map somewhere, 0.06ms is still too much. How would we tackle this?
My suggestion would be the following:
When registering markers, get the zone this marker is in, and add it to a sub-table of our markers table
Only loop markers of your current zone
Profit
Adjust the disc-base:registerMarker
event at the end to the following:
Our 500ms thread that initially loops over markers now gets adjusted like this:
To unregister a marker correctly, also make the following change:
Now, we will not directly notice a performance improvement, but if we adjust our thread that runs every frame to this, we will see drastic improvements again:
We move the other thread into the same thread that checks for the current marker. It does not need to be its own thread, it does just fine in here. However I would recommend some refactor so it just runs a function in here that does the same as this.
Also, we only run natives like grabbing the ped or the coordinates, or any distance checks if there is at least one element in our drawnMarkers
table. Also, at the end we run a 1 frame wait if that table has more than one element, if not we run a 500 frame wait. This drops the resource time to 0.01ms if there are no markers drawn.
Now, our performance is still about 0.07ms looking at about 11 markers, but that is acceptable. But we can do more.
We want more FPS
If we add another key on the markers we put in the drawnMarker
check, we can save up some more resource:
Running this, we can shorten the main thread loop to this:
Now, we are dropping as far down as 0.05 when rendering 11 markers, which is fantastic. Other than some adjustments to the waits (i.e. dropping it to 1000 from 500, which is certainly viable) we have achieved massive performance improvements in just a short amount of time.
Takeaways
Check if you can split up your threads into multiple threads with longer waits. Filtering out most non-viable markers every 500ms instead of on tick, while keeping the main tick loop is drastic.
Use the least amount of natives possible. Save where you can.
Do not repeat yourself, some natives do not need to run every frame!
See where you can condense threads together if they have the same wait times. You don't need 2 threads that each run every tick, you can just make it one.
Something that disc-base already applies, work with events! I personally do this a lot in my private projects, an example would be:
Instead of checking every x ms if the player is in a vehicle, use the
baseevents
resource that comes with the cfx-server-data and theirenteredVehicle
andexitedVehicle
events.
Refactor your code, generalize it. A lot of code can be reused in so many other parts of your code.
Last updated