2018-10-19 18:33:23 +00:00
|
|
|
package mem
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"os/exec"
|
|
|
|
"regexp"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
2020-11-07 01:48:38 +00:00
|
|
|
"github.com/shirou/gopsutil/v3/internal/common"
|
2018-10-19 18:33:23 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// VirtualMemory for Solaris is a minimal implementation which only returns
|
|
|
|
// what Nomad needs. It does take into account global vs zone, however.
|
|
|
|
func VirtualMemory() (*VirtualMemoryStat, error) {
|
|
|
|
return VirtualMemoryWithContext(context.Background())
|
|
|
|
}
|
|
|
|
|
|
|
|
func VirtualMemoryWithContext(ctx context.Context) (*VirtualMemoryStat, error) {
|
|
|
|
result := &VirtualMemoryStat{}
|
|
|
|
|
|
|
|
zoneName, err := zoneName()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if zoneName == "global" {
|
|
|
|
cap, err := globalZoneMemoryCapacity()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
result.Total = cap
|
|
|
|
} else {
|
|
|
|
cap, err := nonGlobalZoneMemoryCapacity()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
result.Total = cap
|
|
|
|
}
|
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func SwapMemory() (*SwapMemoryStat, error) {
|
|
|
|
return SwapMemoryWithContext(context.Background())
|
|
|
|
}
|
|
|
|
|
|
|
|
func SwapMemoryWithContext(ctx context.Context) (*SwapMemoryStat, error) {
|
|
|
|
return nil, common.ErrNotImplementedError
|
|
|
|
}
|
|
|
|
|
|
|
|
func zoneName() (string, error) {
|
2020-07-01 12:47:56 +00:00
|
|
|
zonename, err := exec.LookPath("zonename")
|
2018-10-19 18:33:23 +00:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
out, err := invoke.CommandWithContext(ctx, zonename)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
return strings.TrimSpace(string(out)), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var globalZoneMemoryCapacityMatch = regexp.MustCompile(`memory size: ([\d]+) Megabytes`)
|
|
|
|
|
|
|
|
func globalZoneMemoryCapacity() (uint64, error) {
|
2020-07-01 12:47:56 +00:00
|
|
|
prtconf, err := exec.LookPath("prtconf")
|
2018-10-19 18:33:23 +00:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
out, err := invoke.CommandWithContext(ctx, prtconf)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
match := globalZoneMemoryCapacityMatch.FindAllStringSubmatch(string(out), -1)
|
|
|
|
if len(match) != 1 {
|
|
|
|
return 0, errors.New("memory size not contained in output of /usr/sbin/prtconf")
|
|
|
|
}
|
|
|
|
|
|
|
|
totalMB, err := strconv.ParseUint(match[0][1], 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return totalMB * 1024 * 1024, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var kstatMatch = regexp.MustCompile(`([^\s]+)[\s]+([^\s]*)`)
|
|
|
|
|
|
|
|
func nonGlobalZoneMemoryCapacity() (uint64, error) {
|
2020-07-01 12:47:56 +00:00
|
|
|
kstat, err := exec.LookPath("kstat")
|
2018-10-19 18:33:23 +00:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ctx := context.Background()
|
|
|
|
out, err := invoke.CommandWithContext(ctx, kstat, "-p", "-c", "zone_memory_cap", "memory_cap:*:*:physcap")
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
kstats := kstatMatch.FindAllStringSubmatch(string(out), -1)
|
|
|
|
if len(kstats) != 1 {
|
|
|
|
return 0, fmt.Errorf("expected 1 kstat, found %d", len(kstats))
|
|
|
|
}
|
|
|
|
|
|
|
|
memSizeBytes, err := strconv.ParseUint(kstats[0][2], 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return memSizeBytes, nil
|
|
|
|
}
|