cpu_darwin_cgo.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. // +build darwin
  2. // +build cgo
  3. package cpu
  4. /*
  5. #include <stdlib.h>
  6. #include <sys/sysctl.h>
  7. #include <sys/mount.h>
  8. #include <mach/mach_init.h>
  9. #include <mach/mach_host.h>
  10. #include <mach/host_info.h>
  11. #if TARGET_OS_MAC
  12. #include <libproc.h>
  13. #endif
  14. #include <mach/processor_info.h>
  15. #include <mach/vm_map.h>
  16. */
  17. import "C"
  18. import (
  19. "bytes"
  20. "encoding/binary"
  21. "fmt"
  22. "unsafe"
  23. )
  24. // these CPU times for darwin is borrowed from influxdb/telegraf.
  25. func perCPUTimes() ([]TimesStat, error) {
  26. var (
  27. count C.mach_msg_type_number_t
  28. cpuload *C.processor_cpu_load_info_data_t
  29. ncpu C.natural_t
  30. )
  31. status := C.host_processor_info(C.host_t(C.mach_host_self()),
  32. C.PROCESSOR_CPU_LOAD_INFO,
  33. &ncpu,
  34. (*C.processor_info_array_t)(unsafe.Pointer(&cpuload)),
  35. &count)
  36. if status != C.KERN_SUCCESS {
  37. return nil, fmt.Errorf("host_processor_info error=%d", status)
  38. }
  39. // jump through some cgo casting hoops and ensure we properly free
  40. // the memory that cpuload points to
  41. target := C.vm_map_t(C.mach_task_self_)
  42. address := C.vm_address_t(uintptr(unsafe.Pointer(cpuload)))
  43. defer C.vm_deallocate(target, address, C.vm_size_t(ncpu))
  44. // the body of struct processor_cpu_load_info
  45. // aka processor_cpu_load_info_data_t
  46. var cpu_ticks [C.CPU_STATE_MAX]uint32
  47. // copy the cpuload array to a []byte buffer
  48. // where we can binary.Read the data
  49. size := int(ncpu) * binary.Size(cpu_ticks)
  50. buf := (*[1 << 30]byte)(unsafe.Pointer(cpuload))[:size:size]
  51. bbuf := bytes.NewBuffer(buf)
  52. var ret []TimesStat
  53. for i := 0; i < int(ncpu); i++ {
  54. err := binary.Read(bbuf, binary.LittleEndian, &cpu_ticks)
  55. if err != nil {
  56. return nil, err
  57. }
  58. c := TimesStat{
  59. CPU: fmt.Sprintf("cpu%d", i),
  60. User: float64(cpu_ticks[C.CPU_STATE_USER]) / ClocksPerSec,
  61. System: float64(cpu_ticks[C.CPU_STATE_SYSTEM]) / ClocksPerSec,
  62. Nice: float64(cpu_ticks[C.CPU_STATE_NICE]) / ClocksPerSec,
  63. Idle: float64(cpu_ticks[C.CPU_STATE_IDLE]) / ClocksPerSec,
  64. }
  65. ret = append(ret, c)
  66. }
  67. return ret, nil
  68. }
  69. func allCPUTimes() ([]TimesStat, error) {
  70. var count C.mach_msg_type_number_t
  71. var cpuload C.host_cpu_load_info_data_t
  72. count = C.HOST_CPU_LOAD_INFO_COUNT
  73. status := C.host_statistics(C.host_t(C.mach_host_self()),
  74. C.HOST_CPU_LOAD_INFO,
  75. C.host_info_t(unsafe.Pointer(&cpuload)),
  76. &count)
  77. if status != C.KERN_SUCCESS {
  78. return nil, fmt.Errorf("host_statistics error=%d", status)
  79. }
  80. c := TimesStat{
  81. CPU: "cpu-total",
  82. User: float64(cpuload.cpu_ticks[C.CPU_STATE_USER]) / ClocksPerSec,
  83. System: float64(cpuload.cpu_ticks[C.CPU_STATE_SYSTEM]) / ClocksPerSec,
  84. Nice: float64(cpuload.cpu_ticks[C.CPU_STATE_NICE]) / ClocksPerSec,
  85. Idle: float64(cpuload.cpu_ticks[C.CPU_STATE_IDLE]) / ClocksPerSec,
  86. }
  87. return []TimesStat{c}, nil
  88. }