Skip to content

Commit 19082b3

Browse files
committed
fix: memory leak and string corruption in ARM Mac temperature sensors
This commit addresses several issues in the temperature sensor implementation for ARM Mac: 1. Memory leak caused by unreleased system resources 2. Increasing port counts in Activity Monitor 3. String corruption where the first character becomes "\x00" Changes: 1. Resource Management: - Added proper resource cleanup using `defer` statements - Refactored code to share the HID system client instead of creating new ones - Fixed memory leaks by ensuring all CoreFoundation objects are properly released 2. String Handling: - Fixed buffer allocation for string conversion - Properly handle null terminators in C string to Go string conversion - Added correct string length calculations 3. Code Structure: - Reduced resource allocation by sharing system client between functions - Enhanced code readability with better comments
1 parent 90efec0 commit 19082b3

File tree

1 file changed

+17
-15
lines changed

1 file changed

+17
-15
lines changed

sensors/sensors_darwin_arm64.go

+17-15
Original file line numberDiff line numberDiff line change
@@ -41,11 +41,16 @@ func TemperaturesWithContext(ctx context.Context) ([]TemperatureStat, error) {
4141
}
4242

4343
ta.matching(0xff00, 5)
44-
thermalNames := ta.getProductNames()
45-
thermalValues := ta.getThermalValues()
44+
defer ta.cfRelease(uintptr(ta.sensors))
45+
46+
// Create HID system client
47+
system := ta.ioHIDEventSystemClientCreate(common.KCFAllocatorDefault)
48+
defer ta.cfRelease(uintptr(system))
49+
50+
thermalNames := ta.getProductNames(system)
51+
thermalValues := ta.getThermalValues(system)
4652
result := dumpNameValues(thermalNames, thermalValues)
4753

48-
ta.cfRelease(uintptr(ta.sensors))
4954
return result, nil
5055
}
5156

@@ -84,59 +89,57 @@ type temperatureArm struct {
8489
sensors unsafe.Pointer
8590
}
8691

87-
func (ta *temperatureArm) getProductNames() []string {
92+
func (ta *temperatureArm) getProductNames(system unsafe.Pointer) []string {
8893
ioHIDServiceClientCopyProperty := common.GetFunc[common.IOHIDServiceClientCopyPropertyFunc](ta.ioKit, common.IOHIDServiceClientCopyPropertySym)
89-
9094
cfStringGetLength := common.GetFunc[common.CFStringGetLengthFunc](ta.cf, common.CFStringGetLengthSym)
9195
cfStringGetCString := common.GetFunc[common.CFStringGetCStringFunc](ta.cf, common.CFStringGetCStringSym)
9296

9397
var names []string
94-
system := ta.ioHIDEventSystemClientCreate(common.KCFAllocatorDefault)
9598

9699
ta.ioHIDEventSystemClientSetMatching(uintptr(system), uintptr(ta.sensors))
97100
matchingsrvs := ta.ioHIDEventSystemClientCopyServices(uintptr(system))
98101

99102
if matchingsrvs == nil {
100103
return nil
101104
}
105+
defer ta.cfRelease(uintptr(matchingsrvs))
102106

103107
count := ta.cfArrayGetCount(uintptr(matchingsrvs))
104108

105109
var i int32
106110
str := ta.cfStr("Product")
111+
defer ta.cfRelease(uintptr(str))
112+
107113
for i = 0; i < count; i++ {
108114
sc := ta.cfArrayGetValueAtIndex(uintptr(matchingsrvs), i)
109115
name := ioHIDServiceClientCopyProperty(uintptr(sc), uintptr(str))
110116

111117
if name != nil {
112-
length := cfStringGetLength(uintptr(name)) + 1 // null terminator
113-
buf := make([]byte, length-1)
118+
length := cfStringGetLength(uintptr(name)) + 1 // include null terminator
119+
buf := make([]byte, length) // allocate buffer with full length
114120
cfStringGetCString(uintptr(name), &buf[0], length, common.KCFStringEncodingUTF8)
115121

116-
names = append(names, string(buf))
122+
names = append(names, string(buf[:length-1])) // remove null terminator
117123
ta.cfRelease(uintptr(name))
118124
} else {
119125
names = append(names, "noname")
120126
}
121127
}
122128

123-
ta.cfRelease(uintptr(matchingsrvs))
124-
ta.cfRelease(uintptr(str))
125129
return names
126130
}
127131

128-
func (ta *temperatureArm) getThermalValues() []float64 {
132+
func (ta *temperatureArm) getThermalValues(system unsafe.Pointer) []float64 {
129133
ioHIDServiceClientCopyEvent := common.GetFunc[common.IOHIDServiceClientCopyEventFunc](ta.ioKit, common.IOHIDServiceClientCopyEventSym)
130134
ioHIDEventGetFloatValue := common.GetFunc[common.IOHIDEventGetFloatValueFunc](ta.ioKit, common.IOHIDEventGetFloatValueSym)
131135

132-
system := ta.ioHIDEventSystemClientCreate(common.KCFAllocatorDefault)
133-
134136
ta.ioHIDEventSystemClientSetMatching(uintptr(system), uintptr(ta.sensors))
135137
matchingsrvs := ta.ioHIDEventSystemClientCopyServices(uintptr(system))
136138

137139
if matchingsrvs == nil {
138140
return nil
139141
}
142+
defer ta.cfRelease(uintptr(matchingsrvs))
140143

141144
count := ta.cfArrayGetCount(uintptr(matchingsrvs))
142145

@@ -155,7 +158,6 @@ func (ta *temperatureArm) getThermalValues() []float64 {
155158
values = append(values, temp)
156159
}
157160

158-
ta.cfRelease(uintptr(matchingsrvs))
159161
return values
160162
}
161163

0 commit comments

Comments
 (0)