Skip to content

Commit c936df8

Browse files
committed
Fixed error handling
- Centralized error handling to prevent dangling refs - Release only references we know for sure we own
1 parent 56c5bf4 commit c936df8

File tree

1 file changed

+32
-15
lines changed

1 file changed

+32
-15
lines changed

psutil/arch/osx/cpu.c

+32-15
Original file line numberDiff line numberDiff line change
@@ -123,50 +123,58 @@ psutil_cpu_freq(PyObject *self, PyObject *args) {
123123
uint32_t eMin;
124124
uint32_t max;
125125
kern_return_t status;
126+
CFDictionaryRef matching = NULL;
127+
CFTypeRef pCoreRef = NULL;
128+
CFTypeRef eCoreRef = NULL;
129+
io_iterator_t iter = 0;
130+
io_registry_entry_t entry = 0;
126131

127-
CFDictionaryRef matching = IOServiceMatching("AppleARMIODevice");
132+
matching = IOServiceMatching("AppleARMIODevice");
128133
if (matching == 0) {
129134
return PyErr_Format(PyExc_RuntimeError, "IOServiceMatching call failed, 'AppleARMIODevice' not found");
130135
}
131-
io_iterator_t iter;
132136

133137
status = IOServiceGetMatchingServices(kIOMainPortDefault, matching, &iter);
134138
if (status != KERN_SUCCESS) {
135-
return PyErr_Format(PyExc_RuntimeError, "IOServiceGetMatchingServices call failed");
139+
PyErr_Format(PyExc_RuntimeError, "IOServiceGetMatchingServices call failed");
140+
goto error;
136141
}
137142

138-
io_registry_entry_t entry;
139143
while ((entry = IOIteratorNext(iter))) {
140144
io_name_t name;
141145
status = IORegistryEntryGetName(entry, name);
142146
if (status != KERN_SUCCESS) {
147+
if (entry)
148+
IOObjectRelease(entry);
143149
continue;
144150
}
145151
if (strcmp(name, "pmgr") == 0) {
146152
break;
147153
}
154+
IOObjectRelease(entry);
148155
}
149156

150157
if (entry == 0) {
151-
return PyErr_Format(PyExc_RuntimeError, "'pmgr' entry was not found in AppleARMIODevice service");
158+
PyErr_Format(PyExc_RuntimeError, "'pmgr' entry was not found in AppleARMIODevice service");
159+
goto error;
152160
}
153161

154-
CFTypeRef pCoreRef = IORegistryEntryCreateCFProperty(entry, CFSTR("voltage-states5-sram"), kCFAllocatorDefault, 0);
162+
pCoreRef = IORegistryEntryCreateCFProperty(entry, CFSTR("voltage-states5-sram"), kCFAllocatorDefault, 0);
155163
if (!pCoreRef) {
156-
return PyErr_Format(PyExc_RuntimeError, "'voltage-states5-sram' property not found");
164+
PyErr_Format(PyExc_RuntimeError, "'voltage-states5-sram' property not found");
165+
goto error;
157166
}
158167

159-
CFTypeRef eCoreRef = IORegistryEntryCreateCFProperty(entry, CFSTR("voltage-states1-sram"), kCFAllocatorDefault, 0);
168+
eCoreRef = IORegistryEntryCreateCFProperty(entry, CFSTR("voltage-states1-sram"), kCFAllocatorDefault, 0);
160169
if (!eCoreRef) {
161-
return PyErr_Format(PyExc_RuntimeError, "'voltage-states1-sram' property not found");
170+
PyErr_Format(PyExc_RuntimeError, "'voltage-states1-sram' property not found");
171+
goto error;
162172
}
163173

164-
IOObjectRelease(iter);
165-
IOObjectRelease(entry);
166-
167174
size_t pCoreLength = CFDataGetLength(pCoreRef);
168175
if (pCoreLength < 8) {
169-
return PyErr_Format(PyExc_RuntimeError, "expected 'voltage-states5-sram' buffer to have at least size 8");
176+
PyErr_Format(PyExc_RuntimeError, "expected 'voltage-states5-sram' buffer to have at least size 8");
177+
goto error;
170178
}
171179

172180
CFDataGetBytes(pCoreRef, CFRangeMake(0, 4), (UInt8 *) &pMin);
@@ -176,14 +184,23 @@ psutil_cpu_freq(PyObject *self, PyObject *args) {
176184
min = pMin < eMin ? pMin : eMin;
177185
curr = max;
178186

179-
CFRelease(pCoreRef);
180-
CFRelease(eCoreRef);
181187

182188
return Py_BuildValue(
183189
"IKK",
184190
curr / 1000 / 1000,
185191
min / 1000 / 1000,
186192
max / 1000 / 1000);
193+
error:
194+
if (pCoreRef != NULL)
195+
CFRelease(pCoreRef);
196+
if (eCoreRef != NULL)
197+
CFRelease(eCoreRef);
198+
if (iter)
199+
IOObjectRelease(iter);
200+
if (entry)
201+
IOObjectRelease(entry);
202+
return NULL;
203+
187204
}
188205
#else
189206
PyObject *

0 commit comments

Comments
 (0)