cpu.go 2.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182
  1. package cgroup
  2. import (
  3. "fmt"
  4. "os"
  5. "path"
  6. "strconv"
  7. "strings"
  8. )
  9. type CPUStat struct {
  10. UsageSeconds float64
  11. ThrottledTimeSeconds float64
  12. LimitCores float64
  13. }
  14. func (cg Cgroup) CpuStat() (*CPUStat, error) {
  15. if cg.Version == V1 {
  16. return cg.cpuStatV1()
  17. }
  18. return cg.cpuStatV2()
  19. }
  20. func (cg Cgroup) cpuStatV1() (*CPUStat, error) {
  21. throttling, err := readVariablesFromFile(path.Join(cgRoot, "cpu", cg.subsystems["cpu"], "cpu.stat"))
  22. if err != nil {
  23. return nil, err
  24. }
  25. usageNs, err := readIntFromFile(path.Join(cgRoot, "cpuacct", cg.subsystems["cpuacct"], "cpuacct.usage"))
  26. if err != nil {
  27. return nil, err
  28. }
  29. periodUs, err := readIntFromFile(path.Join(cgRoot, "cpu", cg.subsystems["cpu"], "cpu.cfs_period_us"))
  30. if err != nil {
  31. return nil, err
  32. }
  33. quotaUs, err := readIntFromFile(path.Join(cgRoot, "cpu", cg.subsystems["cpu"], "cpu.cfs_quota_us"))
  34. if err != nil {
  35. return nil, err
  36. }
  37. res := &CPUStat{
  38. UsageSeconds: float64(usageNs) / 1e9,
  39. ThrottledTimeSeconds: float64(throttling["throttled_time"]) / 1e9,
  40. }
  41. if quotaUs > 0 {
  42. res.LimitCores = float64(quotaUs) / float64(periodUs)
  43. }
  44. return res, nil
  45. }
  46. func (cg Cgroup) cpuStatV2() (*CPUStat, error) {
  47. vars, err := readVariablesFromFile(path.Join(cgRoot, cg.subsystems[""], "cpu.stat"))
  48. if err != nil {
  49. return nil, err
  50. }
  51. res := &CPUStat{
  52. UsageSeconds: float64(vars["usage_usec"]) / 1e6,
  53. ThrottledTimeSeconds: float64(vars["throttled_usec"]) / 1e6,
  54. }
  55. if payload, err := os.ReadFile(path.Join(cgRoot, cg.subsystems[""], "cpu.max")); err == nil {
  56. data := strings.TrimSpace(string(payload))
  57. parts := strings.Fields(data)
  58. if len(parts) != 2 {
  59. return nil, fmt.Errorf("invalid cpu.max payload: %s", data)
  60. }
  61. if parts[0] == "max" { //no limit
  62. return res, nil
  63. }
  64. quotaUs, err := strconv.ParseUint(parts[0], 10, 64)
  65. if err != nil {
  66. return nil, fmt.Errorf("invalid quota value in cpu.max: %s", parts[0])
  67. }
  68. periodUs, err := strconv.ParseUint(parts[1], 10, 64)
  69. if err != nil {
  70. return nil, fmt.Errorf("invalid period value in cpu.max: %s", parts[1])
  71. }
  72. if periodUs > 0 {
  73. res.LimitCores = float64(quotaUs) / float64(periodUs)
  74. }
  75. }
  76. return res, nil
  77. }