112 lines
2.7 KiB
Go
112 lines
2.7 KiB
Go
|
// +build darwin
|
||
|
// +build cgo
|
||
|
|
||
|
package cpu
|
||
|
|
||
|
/*
|
||
|
#include <stdlib.h>
|
||
|
#include <sys/sysctl.h>
|
||
|
#include <sys/mount.h>
|
||
|
#include <mach/mach_init.h>
|
||
|
#include <mach/mach_host.h>
|
||
|
#include <mach/host_info.h>
|
||
|
#if TARGET_OS_MAC
|
||
|
#include <libproc.h>
|
||
|
#endif
|
||
|
#include <mach/processor_info.h>
|
||
|
#include <mach/vm_map.h>
|
||
|
*/
|
||
|
import "C"
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"encoding/binary"
|
||
|
"fmt"
|
||
|
"unsafe"
|
||
|
)
|
||
|
|
||
|
// these CPU times for darwin is borrowed from influxdb/telegraf.
|
||
|
|
||
|
func perCPUTimes() ([]TimesStat, error) {
|
||
|
var (
|
||
|
count C.mach_msg_type_number_t
|
||
|
cpuload *C.processor_cpu_load_info_data_t
|
||
|
ncpu C.natural_t
|
||
|
)
|
||
|
|
||
|
status := C.host_processor_info(C.host_t(C.mach_host_self()),
|
||
|
C.PROCESSOR_CPU_LOAD_INFO,
|
||
|
&ncpu,
|
||
|
(*C.processor_info_array_t)(unsafe.Pointer(&cpuload)),
|
||
|
&count)
|
||
|
|
||
|
if status != C.KERN_SUCCESS {
|
||
|
return nil, fmt.Errorf("host_processor_info error=%d", status)
|
||
|
}
|
||
|
|
||
|
// jump through some cgo casting hoops and ensure we properly free
|
||
|
// the memory that cpuload points to
|
||
|
target := C.vm_map_t(C.mach_task_self_)
|
||
|
address := C.vm_address_t(uintptr(unsafe.Pointer(cpuload)))
|
||
|
defer C.vm_deallocate(target, address, C.vm_size_t(ncpu))
|
||
|
|
||
|
// the body of struct processor_cpu_load_info
|
||
|
// aka processor_cpu_load_info_data_t
|
||
|
var cpu_ticks [C.CPU_STATE_MAX]uint32
|
||
|
|
||
|
// copy the cpuload array to a []byte buffer
|
||
|
// where we can binary.Read the data
|
||
|
size := int(ncpu) * binary.Size(cpu_ticks)
|
||
|
buf := (*[1 << 30]byte)(unsafe.Pointer(cpuload))[:size:size]
|
||
|
|
||
|
bbuf := bytes.NewBuffer(buf)
|
||
|
|
||
|
var ret []TimesStat
|
||
|
|
||
|
for i := 0; i < int(ncpu); i++ {
|
||
|
err := binary.Read(bbuf, binary.LittleEndian, &cpu_ticks)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
c := TimesStat{
|
||
|
CPU: fmt.Sprintf("cpu%d", i),
|
||
|
User: float64(cpu_ticks[C.CPU_STATE_USER]) / ClocksPerSec,
|
||
|
System: float64(cpu_ticks[C.CPU_STATE_SYSTEM]) / ClocksPerSec,
|
||
|
Nice: float64(cpu_ticks[C.CPU_STATE_NICE]) / ClocksPerSec,
|
||
|
Idle: float64(cpu_ticks[C.CPU_STATE_IDLE]) / ClocksPerSec,
|
||
|
}
|
||
|
|
||
|
ret = append(ret, c)
|
||
|
}
|
||
|
|
||
|
return ret, nil
|
||
|
}
|
||
|
|
||
|
func allCPUTimes() ([]TimesStat, error) {
|
||
|
var count C.mach_msg_type_number_t
|
||
|
var cpuload C.host_cpu_load_info_data_t
|
||
|
|
||
|
count = C.HOST_CPU_LOAD_INFO_COUNT
|
||
|
|
||
|
status := C.host_statistics(C.host_t(C.mach_host_self()),
|
||
|
C.HOST_CPU_LOAD_INFO,
|
||
|
C.host_info_t(unsafe.Pointer(&cpuload)),
|
||
|
&count)
|
||
|
|
||
|
if status != C.KERN_SUCCESS {
|
||
|
return nil, fmt.Errorf("host_statistics error=%d", status)
|
||
|
}
|
||
|
|
||
|
c := TimesStat{
|
||
|
CPU: "cpu-total",
|
||
|
User: float64(cpuload.cpu_ticks[C.CPU_STATE_USER]) / ClocksPerSec,
|
||
|
System: float64(cpuload.cpu_ticks[C.CPU_STATE_SYSTEM]) / ClocksPerSec,
|
||
|
Nice: float64(cpuload.cpu_ticks[C.CPU_STATE_NICE]) / ClocksPerSec,
|
||
|
Idle: float64(cpuload.cpu_ticks[C.CPU_STATE_IDLE]) / ClocksPerSec,
|
||
|
}
|
||
|
|
||
|
return []TimesStat{c}, nil
|
||
|
|
||
|
}
|