cpuacct.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123
  1. // +build linux
  2. package fs
  3. import (
  4. "fmt"
  5. "io/ioutil"
  6. "path/filepath"
  7. "strconv"
  8. "strings"
  9. "github.com/opencontainers/runc/libcontainer/cgroups"
  10. "github.com/opencontainers/runc/libcontainer/cgroups/fscommon"
  11. "github.com/opencontainers/runc/libcontainer/configs"
  12. "github.com/opencontainers/runc/libcontainer/system"
  13. )
  14. const (
  15. cgroupCpuacctStat = "cpuacct.stat"
  16. nanosecondsInSecond = 1000000000
  17. )
  18. var clockTicks = uint64(system.GetClockTicks())
  19. type CpuacctGroup struct {
  20. }
  21. func (s *CpuacctGroup) Name() string {
  22. return "cpuacct"
  23. }
  24. func (s *CpuacctGroup) Apply(d *cgroupData) error {
  25. // we just want to join this group even though we don't set anything
  26. if _, err := d.join("cpuacct"); err != nil && !cgroups.IsNotFound(err) {
  27. return err
  28. }
  29. return nil
  30. }
  31. func (s *CpuacctGroup) Set(path string, cgroup *configs.Cgroup) error {
  32. return nil
  33. }
  34. func (s *CpuacctGroup) Remove(d *cgroupData) error {
  35. return removePath(d.path("cpuacct"))
  36. }
  37. func (s *CpuacctGroup) GetStats(path string, stats *cgroups.Stats) error {
  38. userModeUsage, kernelModeUsage, err := getCpuUsageBreakdown(path)
  39. if err != nil {
  40. return err
  41. }
  42. totalUsage, err := fscommon.GetCgroupParamUint(path, "cpuacct.usage")
  43. if err != nil {
  44. return err
  45. }
  46. percpuUsage, err := getPercpuUsage(path)
  47. if err != nil {
  48. return err
  49. }
  50. stats.CpuStats.CpuUsage.TotalUsage = totalUsage
  51. stats.CpuStats.CpuUsage.PercpuUsage = percpuUsage
  52. stats.CpuStats.CpuUsage.UsageInUsermode = userModeUsage
  53. stats.CpuStats.CpuUsage.UsageInKernelmode = kernelModeUsage
  54. return nil
  55. }
  56. // Returns user and kernel usage breakdown in nanoseconds.
  57. func getCpuUsageBreakdown(path string) (uint64, uint64, error) {
  58. userModeUsage := uint64(0)
  59. kernelModeUsage := uint64(0)
  60. const (
  61. userField = "user"
  62. systemField = "system"
  63. )
  64. // Expected format:
  65. // user <usage in ticks>
  66. // system <usage in ticks>
  67. data, err := ioutil.ReadFile(filepath.Join(path, cgroupCpuacctStat))
  68. if err != nil {
  69. return 0, 0, err
  70. }
  71. fields := strings.Fields(string(data))
  72. if len(fields) < 4 {
  73. return 0, 0, fmt.Errorf("failure - %s is expected to have at least 4 fields", filepath.Join(path, cgroupCpuacctStat))
  74. }
  75. if fields[0] != userField {
  76. return 0, 0, fmt.Errorf("unexpected field %q in %q, expected %q", fields[0], cgroupCpuacctStat, userField)
  77. }
  78. if fields[2] != systemField {
  79. return 0, 0, fmt.Errorf("unexpected field %q in %q, expected %q", fields[2], cgroupCpuacctStat, systemField)
  80. }
  81. if userModeUsage, err = strconv.ParseUint(fields[1], 10, 64); err != nil {
  82. return 0, 0, err
  83. }
  84. if kernelModeUsage, err = strconv.ParseUint(fields[3], 10, 64); err != nil {
  85. return 0, 0, err
  86. }
  87. return (userModeUsage * nanosecondsInSecond) / clockTicks, (kernelModeUsage * nanosecondsInSecond) / clockTicks, nil
  88. }
  89. func getPercpuUsage(path string) ([]uint64, error) {
  90. percpuUsage := []uint64{}
  91. data, err := ioutil.ReadFile(filepath.Join(path, "cpuacct.usage_percpu"))
  92. if err != nil {
  93. return percpuUsage, err
  94. }
  95. for _, value := range strings.Fields(string(data)) {
  96. value, err := strconv.ParseUint(value, 10, 64)
  97. if err != nil {
  98. return percpuUsage, fmt.Errorf("Unable to convert param value to uint64: %s", err)
  99. }
  100. percpuUsage = append(percpuUsage, value)
  101. }
  102. return percpuUsage, nil
  103. }