mem_solaris.go 2.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. package mem
  2. import (
  3. "context"
  4. "errors"
  5. "fmt"
  6. "os/exec"
  7. "regexp"
  8. "strconv"
  9. "strings"
  10. "github.com/shirou/gopsutil/internal/common"
  11. )
  12. // VirtualMemory for Solaris is a minimal implementation which only returns
  13. // what Nomad needs. It does take into account global vs zone, however.
  14. func VirtualMemory() (*VirtualMemoryStat, error) {
  15. return VirtualMemoryWithContext(context.Background())
  16. }
  17. func VirtualMemoryWithContext(ctx context.Context) (*VirtualMemoryStat, error) {
  18. result := &VirtualMemoryStat{}
  19. zoneName, err := zoneName()
  20. if err != nil {
  21. return nil, err
  22. }
  23. if zoneName == "global" {
  24. cap, err := globalZoneMemoryCapacity()
  25. if err != nil {
  26. return nil, err
  27. }
  28. result.Total = cap
  29. } else {
  30. cap, err := nonGlobalZoneMemoryCapacity()
  31. if err != nil {
  32. return nil, err
  33. }
  34. result.Total = cap
  35. }
  36. return result, nil
  37. }
  38. func SwapMemory() (*SwapMemoryStat, error) {
  39. return SwapMemoryWithContext(context.Background())
  40. }
  41. func SwapMemoryWithContext(ctx context.Context) (*SwapMemoryStat, error) {
  42. return nil, common.ErrNotImplementedError
  43. }
  44. func zoneName() (string, error) {
  45. zonename, err := exec.LookPath("/usr/bin/zonename")
  46. if err != nil {
  47. return "", err
  48. }
  49. ctx := context.Background()
  50. out, err := invoke.CommandWithContext(ctx, zonename)
  51. if err != nil {
  52. return "", err
  53. }
  54. return strings.TrimSpace(string(out)), nil
  55. }
  56. var globalZoneMemoryCapacityMatch = regexp.MustCompile(`memory size: ([\d]+) Megabytes`)
  57. func globalZoneMemoryCapacity() (uint64, error) {
  58. prtconf, err := exec.LookPath("/usr/sbin/prtconf")
  59. if err != nil {
  60. return 0, err
  61. }
  62. ctx := context.Background()
  63. out, err := invoke.CommandWithContext(ctx, prtconf)
  64. if err != nil {
  65. return 0, err
  66. }
  67. match := globalZoneMemoryCapacityMatch.FindAllStringSubmatch(string(out), -1)
  68. if len(match) != 1 {
  69. return 0, errors.New("memory size not contained in output of /usr/sbin/prtconf")
  70. }
  71. totalMB, err := strconv.ParseUint(match[0][1], 10, 64)
  72. if err != nil {
  73. return 0, err
  74. }
  75. return totalMB * 1024 * 1024, nil
  76. }
  77. var kstatMatch = regexp.MustCompile(`([^\s]+)[\s]+([^\s]*)`)
  78. func nonGlobalZoneMemoryCapacity() (uint64, error) {
  79. kstat, err := exec.LookPath("/usr/bin/kstat")
  80. if err != nil {
  81. return 0, err
  82. }
  83. ctx := context.Background()
  84. out, err := invoke.CommandWithContext(ctx, kstat, "-p", "-c", "zone_memory_cap", "memory_cap:*:*:physcap")
  85. if err != nil {
  86. return 0, err
  87. }
  88. kstats := kstatMatch.FindAllStringSubmatch(string(out), -1)
  89. if len(kstats) != 1 {
  90. return 0, fmt.Errorf("expected 1 kstat, found %d", len(kstats))
  91. }
  92. memSizeBytes, err := strconv.ParseUint(kstats[0][2], 10, 64)
  93. if err != nil {
  94. return 0, err
  95. }
  96. return memSizeBytes, nil
  97. }