Merge pull request #935 from vishh/cgo

Use update time to detect if kmem limits have been set
This commit is contained in:
Qiang Huang 2016-07-11 10:17:28 +08:00 committed by GitHub
commit 4eb8c2fb1d
3 changed files with 42 additions and 45 deletions

View File

@ -5,15 +5,14 @@ package fs
import ( import (
"bufio" "bufio"
"fmt" "fmt"
"math"
"os" "os"
"path/filepath" "path/filepath"
"strconv" "strconv"
"strings" "strings"
"time"
"github.com/opencontainers/runc/libcontainer/cgroups" "github.com/opencontainers/runc/libcontainer/cgroups"
"github.com/opencontainers/runc/libcontainer/configs" "github.com/opencontainers/runc/libcontainer/configs"
"github.com/opencontainers/runc/libcontainer/system"
) )
type MemoryGroup struct { type MemoryGroup struct {
@ -28,53 +27,65 @@ func (s *MemoryGroup) Apply(d *cgroupData) (err error) {
if err != nil && !cgroups.IsNotFound(err) { if err != nil && !cgroups.IsNotFound(err) {
return err return err
} }
if memoryAssigned(d.config) { // reset error.
if path != "" { err = nil
if err := os.MkdirAll(path, 0755); err != nil { if path == "" {
return err // Invalid input.
return fmt.Errorf("invalid path for memory cgroups: %+v", d)
} }
}
// We have to set kernel memory here, as we can't change it once
// processes have been attached to the cgroup.
if err := s.SetKernelMemory(path, d.config); err != nil {
return err
}
}
defer func() { defer func() {
if err != nil { if err != nil {
os.RemoveAll(path) os.RemoveAll(path)
} }
}() }()
if !cgroups.PathExists(path) {
if err = os.MkdirAll(path, 0755); err != nil {
return err
}
}
if memoryAssigned(d.config) {
// We have to set kernel memory here, as we can't change it once
// processes have been attached to the cgroup.
if err = s.SetKernelMemory(path, d.config); err != nil {
return err
}
}
// We need to join memory cgroup after set memory limits, because // We need to join memory cgroup after set memory limits, because
// kmem.limit_in_bytes can only be set when the cgroup is empty. // kmem.limit_in_bytes can only be set when the cgroup is empty.
_, err = d.join("memory") if _, jerr := d.join("memory"); jerr != nil && !cgroups.IsNotFound(jerr) {
if err != nil && !cgroups.IsNotFound(err) { err = jerr
return err return err
} }
return nil return nil
} }
func getModifyTime(path string) (time.Time, error) {
stat, err := os.Stat(path)
if err != nil {
return time.Time{}, fmt.Errorf("failed to get memory cgroups creation time: %v", err)
}
return stat.ModTime(), nil
}
func (s *MemoryGroup) SetKernelMemory(path string, cgroup *configs.Cgroup) error { func (s *MemoryGroup) SetKernelMemory(path string, cgroup *configs.Cgroup) error {
// This has to be done separately because it has special // This has to be done separately because it has special
// constraints (it can only be initialized before setting up a // constraints (it can only be initialized before setting up a
// hierarchy or adding a task to the cgroups. However, if // hierarchy or adding a task to the cgroups. However, if
// sucessfully initialized, it can be updated anytime afterwards) // sucessfully initialized, it can be updated anytime afterwards)
if cgroup.Resources.KernelMemory != 0 { if cgroup.Resources.KernelMemory != 0 {
kmemInitialized := false
// Is kmem.limit_in_bytes already set? // Is kmem.limit_in_bytes already set?
kmemValue, err := getCgroupParamUint(path, "memory.kmem.limit_in_bytes") // memory.kmem.max_usage_in_bytes is a read-only file. Use it to get cgroups creation time.
kmemCreationTime, err := getModifyTime(filepath.Join(path, "memory.kmem.max_usage_in_bytes"))
if err != nil { if err != nil {
return err return err
} }
switch system.GetLongBit() { kmemLimitsUpdateTime, err := getModifyTime(filepath.Join(path, "memory.kmem.limit_in_bytes"))
case 32: if err != nil {
kmemInitialized = uint32(kmemValue) != uint32(math.MaxUint32) return err
case 64:
kmemInitialized = kmemValue != uint64(math.MaxUint64)
} }
// kmem.limit_in_bytes has already been set if its update time is after that of creation time.
// We use `!=` op instead of `>` because updates are losing precision compared to creation.
kmemInitialized := !kmemLimitsUpdateTime.Equal(kmemCreationTime)
if !kmemInitialized { if !kmemInitialized {
// If there's already tasks in the cgroup, we can't change the limit either // If there's already tasks in the cgroup, we can't change the limit either
tasks, err := getCgroupParamString(path, "tasks") tasks, err := getCgroupParamString(path, "tasks")
@ -85,7 +96,6 @@ func (s *MemoryGroup) SetKernelMemory(path string, cgroup *configs.Cgroup) error
return fmt.Errorf("cannot set kmem.limit_in_bytes after task have joined this cgroup") return fmt.Errorf("cannot set kmem.limit_in_bytes after task have joined this cgroup")
} }
} }
if err := writeFile(path, "memory.kmem.limit_in_bytes", strconv.FormatInt(cgroup.Resources.KernelMemory, 10)); err != nil { if err := writeFile(path, "memory.kmem.limit_in_bytes", strconv.FormatInt(cgroup.Resources.KernelMemory, 10)); err != nil {
return err return err
} }

View File

@ -227,6 +227,12 @@ func TestMemorySetKernelMemory(t *testing.T) {
helper.writeFileContents(map[string]string{ helper.writeFileContents(map[string]string{
"memory.kmem.limit_in_bytes": strconv.Itoa(kernelMemoryBefore), "memory.kmem.limit_in_bytes": strconv.Itoa(kernelMemoryBefore),
}) })
helper.writeFileContents(map[string]string{
"memory.kmem.max_usage_in_bytes": strconv.Itoa(kernelMemoryBefore),
})
helper.writeFileContents(map[string]string{
"tasks": "",
})
helper.CgroupData.config.Resources.KernelMemory = kernelMemoryAfter helper.CgroupData.config.Resources.KernelMemory = kernelMemoryAfter
memory := &MemoryGroup{} memory := &MemoryGroup{}

View File

@ -4,28 +4,9 @@ package system
/* /*
#include <unistd.h> #include <unistd.h>
#include <limits.h>
int GetLongBit() {
#ifdef _SC_LONG_BIT
int longbits;
longbits = sysconf(_SC_LONG_BIT);
if (longbits < 0) {
longbits = (CHAR_BIT * sizeof(long));
}
return longbits;
#else
return (CHAR_BIT * sizeof(long));
#endif
}
*/ */
import "C" import "C"
func GetClockTicks() int { func GetClockTicks() int {
return int(C.sysconf(C._SC_CLK_TCK)) return int(C.sysconf(C._SC_CLK_TCK))
} }
func GetLongBit() int {
return int(C.GetLongBit())
}