From 615073fe868565a3148ed1b3610cc1252befe5a5 Mon Sep 17 00:00:00 2001 From: Samer Masterson Date: Thu, 4 Apr 2019 21:17:59 +0000 Subject: [PATCH] Fix read access violation in psutil.cpu_count(logical=False) Fix read access violation in psutil.cpu_count(logical=False) by stopping the iteration through the SYSTEM_LOGICAL_PROCESSOR_INFORMATION_EX buffer when we hit the last item. --- psutil/_psutil_windows.c | 14 +++++++++++--- 1 file changed, 11 insertions(+), 3 deletions(-) diff --git a/psutil/_psutil_windows.c b/psutil/_psutil_windows.c index 4dfae2d5f..159e1f745 100644 --- a/psutil/_psutil_windows.c +++ b/psutil/_psutil_windows.c @@ -487,6 +487,7 @@ psutil_cpu_count_phys(PyObject *self, PyObject *args) { DWORD length = 0; DWORD offset = 0; DWORD ncpus = 0; + DWORD prev_processor_info_size = 0; // GetLogicalProcessorInformationEx() is available from Windows 7 // onward. Differently from GetLogicalProcessorInformation() @@ -525,13 +526,20 @@ psutil_cpu_count_phys(PyObject *self, PyObject *args) { } ptr = buffer; - while (ptr->Size > 0 && offset + ptr->Size <= length) { + while (offset < length) { + // Advance ptr by the size of the previous + // SYSTEM_LOGICAL_PROCESSOR_INFORMATION_EX struct. + ptr = (SYSTEM_LOGICAL_PROCESSOR_INFORMATION_EX*)\ + (((char*)ptr) + prev_processor_info_size); + if (ptr->Relationship == RelationProcessorCore) { ncpus += 1; } + + // When offset == length, we've reached the last processor + // info struct in the buffer. offset += ptr->Size; - ptr = (SYSTEM_LOGICAL_PROCESSOR_INFORMATION_EX*)\ - (((char*)ptr) + ptr->Size); + prev_processor_info_size = ptr->Size; } free(buffer);