-
Notifications
You must be signed in to change notification settings - Fork 4.9k
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
CPU metrics are incorrect on Windows machines with more than 64 cores #40926
CPU metrics are incorrect on Windows machines with more than 64 cores #40926
Comments
Pinging @elastic/elastic-agent-data-plane (Team:Elastic-Agent-Data-Plane) |
fyi @flexitrev |
It is not immediately clear how to solve this, and the solution does not strictly seem like it will be simple, especially without being able to use the Task Manager implementation as a reference. I suspect we will need something like a thread or process per processor group with their processor group affinities set appropriately so that they get scheduled into different processor groups. |
Is this documented anywhere? I don't see it in the My reading of this MS doc suggests our threads can switch between processor groups. Here's python pseudocode to do this. I don't have a >64T system to test with. global last_system_times = {}
def get_cpu_time_deltas():
idle_time_delta, kernel_time_delta, user_time_delta = 0, 0, 0
# Backup original affinity
GetThreadGroupAffinity(GetCurrentThread(), &saved_group_affinity)
# Enumerate each NUMA node
for node in range(GetNumaHighestNodeNumber()):
# Enumerate all the processor groups for this NUMA node
for group_affinity in GetNumaNodeProcessorMask2(node):
# Switch to this processor group
SetThreadGroupAffinity(GetCurrentThread(), &group_affinity)
# Retrieve metrics for this processor group
GetSystemTimes(&idle, &kernel, &user)
# Retrieve old values
if group_affinity.group in last_system_times:
last_idle, last_kernel, last_user = last_system_times[group_affinity.group]
# Compute deltas
idle_time_delta += (idle - last_idle)
kernel_time_delta += (kernel - last_kernel)
user_time_delta += (user - last_user)
# Store latest values
last_system_times[group_affinity.group] = (idle, kernel, user)
# Restore original affinity
SetThreadGroupAffinity(GetCurrentThread(), &saved_group_affinity)
return idle_time_delta, kernel_time_delta, user_time_delta |
@cmacknz @flexitrev @pierrehilbert I ran two instances of the The left side of the dashboard is one with performance counters and right side is latest build from There is a clear correlation between the two sets of data, with the values being identical across both versions. |
I don't think values are identical but instead similars. |
@pierrehilbert I think I know why. When I started two Let me retest it for 30 mins and reduce the interval to |
Yes. |
@pierrehilbert updated results. I ran two binaries at the exact same time (almost) The results are better than before. What do you think? |
I don't see any performance impact that would worry me. This means if performance counters give us better, more reliable data there is no reason we shouldn't use them. |
Agreed, this looks similar at a high level. |
Since this change looks like it is happening in elastic-agent-system-metrics, which has no mechanism to tie things to a stack release, I would like some way for us to switch between both methods to exist. One that ideally we can expose in the configuration of the system/metrics input. This will let us update to the latest version of elastic-agent-system-metrics everywhere (because of a CVE related dependency update for example) without worry of also changing underlying behavior, and it will give us a way to easily switch back to the old way immediately if there is some unexpected problem to work around. Let's avoid the problems we had with turning off the introduction of the degraded state in system metrics, and build a proper feature flag for this from the beginning. |
I implemented a similar thing initially. Something like following:
But your point makes more sense. A flag would be much more preferable. |
I realize I jumped the gun on saying anything about performance here, since the binaries were being run at the same time, and measuring the same thing (thanks @strawgate). What I'd like to make sure is that we are not introducing a performance impact. Can we benchmark the two different approaches? |
Yes. I'll keep you posted on it. |
Reopening until verified once more. |
VihasMakwana/elastic-agent-system-metrics@80aacba
On Windows, Metricbeat measures CPU use via the Windows API call
GetSystemTimes
. Each metrics interval, it fetches the CPU numbers, and compares them to the previous measurement to determine CPU load during that interval. On most systems this includes CPU time "including all threads in all processes, on all processors". However, on systems with more than 64 cores, it returns only the data for the current processor group of up to 64 cores.This has two consequences on high-core machines:
GetSystemTimes
returns data from a different set of cores. If the new processor group has a lower CPU total than the previous one, Metricbeat will report negative numbers for some CPU metrics.The most visible symptom is occasional negative numbers in CPU-related metrics, especially coming in pairs of two adjacent data points.
This seems to apply to ~all Metricbeat versions, and all versions of Windows that support more than 64 CPU cores.
The text was updated successfully, but these errors were encountered: