Files
scrutiny/vendor/github.com/jaypipes/ghw/memory_linux.go
T
2020-08-21 06:31:48 +00:00

184 lines
4.8 KiB
Go

// Use and distribution licensed under the Apache license version 2.
//
// See the COPYING file in the root project directory for full text.
//
package ghw
import (
"bufio"
"compress/gzip"
"fmt"
"io"
"io/ioutil"
"os"
"path/filepath"
"regexp"
"strconv"
"strings"
)
const (
_WARN_CANNOT_DETERMINE_PHYSICAL_MEMORY = `
Could not determine total physical bytes of memory. This may
be due to the host being a virtual machine or container with no
/var/log/syslog file, or the current user may not have necessary
privileges to read the syslog. We are falling back to setting the
total physical amount of memory to the total usable amount of memory
`
)
var (
// System log lines will look similar to the following:
// ... kernel: [0.000000] Memory: 24633272K/25155024K ...
_REGEX_SYSLOG_MEMLINE = regexp.MustCompile(`Memory:\s+\d+K\/(\d+)K`)
)
func (ctx *context) memFillInfo(info *MemoryInfo) error {
tub := ctx.memTotalUsableBytes()
if tub < 1 {
return fmt.Errorf("Could not determine total usable bytes of memory")
}
info.TotalUsableBytes = tub
tpb := ctx.memTotalPhysicalBytes()
info.TotalPhysicalBytes = tpb
if tpb < 1 {
warn(_WARN_CANNOT_DETERMINE_PHYSICAL_MEMORY)
info.TotalPhysicalBytes = tub
}
info.SupportedPageSizes = ctx.memSupportedPageSizes()
return nil
}
func (ctx *context) memTotalPhysicalBytes() int64 {
// In Linux, the total physical memory can be determined by looking at the
// output of dmidecode, however dmidecode requires root privileges to run,
// so instead we examine the system logs for startup information containing
// total physical memory and cache the results of this.
findPhysicalKb := func(line string) int64 {
matches := _REGEX_SYSLOG_MEMLINE.FindStringSubmatch(line)
if len(matches) == 2 {
i, err := strconv.Atoi(matches[1])
if err != nil {
return -1
}
return int64(i * 1024)
}
return -1
}
// /var/log will contain a file called syslog and 0 or more files called
// syslog.$NUMBER or syslog.$NUMBER.gz containing system log records. We
// search each, stopping when we match a system log record line that
// contains physical memory information.
logDir := ctx.pathVarLog()
logFiles, err := ioutil.ReadDir(logDir)
if err != nil {
return -1
}
for _, file := range logFiles {
if strings.HasPrefix(file.Name(), "syslog") {
fullPath := filepath.Join(logDir, file.Name())
unzip := strings.HasSuffix(file.Name(), ".gz")
var r io.ReadCloser
r, err = os.Open(fullPath)
if err != nil {
return -1
}
defer safeClose(r)
if unzip {
r, err = gzip.NewReader(r)
if err != nil {
return -1
}
}
scanner := bufio.NewScanner(r)
for scanner.Scan() {
line := scanner.Text()
size := findPhysicalKb(line)
if size > 0 {
return size
}
}
}
}
return -1
}
func (ctx *context) memTotalUsableBytes() int64 {
// In Linux, /proc/meminfo contains a set of memory-related amounts, with
// lines looking like the following:
//
// $ cat /proc/meminfo
// MemTotal: 24677596 kB
// MemFree: 21244356 kB
// MemAvailable: 22085432 kB
// ...
// HugePages_Total: 0
// HugePages_Free: 0
// HugePages_Rsvd: 0
// HugePages_Surp: 0
// ...
//
// It's worth noting that /proc/meminfo returns exact information, not
// "theoretical" information. For instance, on the above system, I have
// 24GB of RAM but MemTotal is indicating only around 23GB. This is because
// MemTotal contains the exact amount of *usable* memory after accounting
// for the kernel's resident memory size and a few reserved bits. For more
// information, see:
//
// https://www.kernel.org/doc/Documentation/filesystems/proc.txt
filePath := ctx.pathProcMeminfo()
r, err := os.Open(filePath)
if err != nil {
return -1
}
defer safeClose(r)
scanner := bufio.NewScanner(r)
for scanner.Scan() {
line := scanner.Text()
parts := strings.Fields(line)
key := strings.Trim(parts[0], ": \t")
if key != "MemTotal" {
continue
}
value, err := strconv.Atoi(strings.TrimSpace(parts[1]))
if err != nil {
return -1
}
inKb := (len(parts) == 3 && strings.TrimSpace(parts[2]) == "kB")
if inKb {
value = value * int(KB)
}
return int64(value)
}
return -1
}
func (ctx *context) memSupportedPageSizes() []uint64 {
// In Linux, /sys/kernel/mm/hugepages contains a directory per page size
// supported by the kernel. The directory name corresponds to the pattern
// 'hugepages-{pagesize}kb'
dir := ctx.pathSysKernelMMHugepages()
out := make([]uint64, 0)
files, err := ioutil.ReadDir(dir)
if err != nil {
return out
}
for _, file := range files {
parts := strings.Split(file.Name(), "-")
sizeStr := parts[1]
// Cut off the 'kb'
sizeStr = sizeStr[0 : len(sizeStr)-2]
size, err := strconv.Atoi(sizeStr)
if err != nil {
return out
}
out = append(out, uint64(size*int(KB)))
}
return out
}