cgroups: Update systemd to match fs backend
This updates systemd.Apply to match the fs backend by: * Always join blockio controller (for stats) * Support CpusetCpus * Support MemorySwap Also, it removes the generic UnitProperties in favour of a single option to set the slice. Docker-DCO-1.1-Signed-off-by: Alexander Larsson <alexl@redhat.com> (github: alexlarsson)
This commit is contained in:
parent
bbc3c913a9
commit
b5b239af1d
3 changed files with 142 additions and 13 deletions
|
@ -22,7 +22,7 @@ type Cgroup struct {
|
||||||
CpusetCpus string `json:"cpuset_cpus,omitempty"` // CPU to use
|
CpusetCpus string `json:"cpuset_cpus,omitempty"` // CPU to use
|
||||||
Freezer string `json:"freezer,omitempty"` // set the freeze value for the process
|
Freezer string `json:"freezer,omitempty"` // set the freeze value for the process
|
||||||
|
|
||||||
UnitProperties [][2]string `json:"unit_properties,omitempty"` // systemd unit properties
|
Slice string `json:"slice,omitempty"` // Parent slice to use for systemd
|
||||||
}
|
}
|
||||||
|
|
||||||
type ActiveCgroup interface {
|
type ActiveCgroup interface {
|
||||||
|
|
|
@ -11,6 +11,6 @@ func UseSystemd() bool {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
func systemdApply(c *Cgroup, pid int) (cgroups.ActiveCgroup, error) {
|
func Apply(c *Cgroup, pid int) (cgroups.ActiveCgroup, error) {
|
||||||
return nil, fmt.Errorf("Systemd not supported")
|
return nil, fmt.Errorf("Systemd not supported")
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,9 +3,10 @@
|
||||||
package systemd
|
package systemd
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
|
@ -16,6 +17,7 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
type systemdCgroup struct {
|
type systemdCgroup struct {
|
||||||
|
cleanupDirs []string
|
||||||
}
|
}
|
||||||
|
|
||||||
type DeviceAllow struct {
|
type DeviceAllow struct {
|
||||||
|
@ -69,20 +71,42 @@ func getIfaceForUnit(unitName string) string {
|
||||||
return "Unit"
|
return "Unit"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type cgroupArg struct {
|
||||||
|
File string
|
||||||
|
Value string
|
||||||
|
}
|
||||||
|
|
||||||
func Apply(c *cgroups.Cgroup, pid int) (cgroups.ActiveCgroup, error) {
|
func Apply(c *cgroups.Cgroup, pid int) (cgroups.ActiveCgroup, error) {
|
||||||
var (
|
var (
|
||||||
unitName = c.Parent + "-" + c.Name + ".scope"
|
unitName = c.Parent + "-" + c.Name + ".scope"
|
||||||
slice = "system.slice"
|
slice = "system.slice"
|
||||||
properties []systemd1.Property
|
properties []systemd1.Property
|
||||||
|
cpuArgs []cgroupArg
|
||||||
|
cpusetArgs []cgroupArg
|
||||||
|
memoryArgs []cgroupArg
|
||||||
|
res systemdCgroup
|
||||||
)
|
)
|
||||||
|
|
||||||
for _, v := range c.UnitProperties {
|
// First set up things not supported by systemd
|
||||||
switch v[0] {
|
|
||||||
case "Slice":
|
// -1 disables memorySwap
|
||||||
slice = v[1]
|
if c.MemorySwap >= 0 && (c.Memory != 0 || c.MemorySwap > 0) {
|
||||||
default:
|
memorySwap := c.MemorySwap
|
||||||
return nil, fmt.Errorf("Unknown unit propery %s", v[0])
|
|
||||||
|
if memorySwap == 0 {
|
||||||
|
// By default, MemorySwap is set to twice the size of RAM.
|
||||||
|
memorySwap = c.Memory * 2
|
||||||
}
|
}
|
||||||
|
|
||||||
|
memoryArgs = append(memoryArgs, cgroupArg{"memory.memsw.limit_in_bytes", strconv.FormatInt(memorySwap, 10)})
|
||||||
|
}
|
||||||
|
|
||||||
|
if c.CpusetCpus != "" {
|
||||||
|
cpusetArgs = append(cpusetArgs, cgroupArg{"cpuset.cpus", c.CpusetCpus})
|
||||||
|
}
|
||||||
|
|
||||||
|
if c.Slice != "" {
|
||||||
|
slice = c.Slice
|
||||||
}
|
}
|
||||||
|
|
||||||
properties = append(properties,
|
properties = append(properties,
|
||||||
|
@ -111,11 +135,12 @@ func Apply(c *cgroups.Cgroup, pid int) (cgroups.ActiveCgroup, error) {
|
||||||
})})
|
})})
|
||||||
}
|
}
|
||||||
|
|
||||||
// Always enable accounting, this gets us the same behaviour as the raw implementation,
|
// Always enable accounting, this gets us the same behaviour as the fs implementation,
|
||||||
// plus the kernel has some problems with joining the memory cgroup at a later time.
|
// plus the kernel has some problems with joining the memory cgroup at a later time.
|
||||||
properties = append(properties,
|
properties = append(properties,
|
||||||
systemd1.Property{"MemoryAccounting", dbus.MakeVariant(true)},
|
systemd1.Property{"MemoryAccounting", dbus.MakeVariant(true)},
|
||||||
systemd1.Property{"CPUAccounting", dbus.MakeVariant(true)})
|
systemd1.Property{"CPUAccounting", dbus.MakeVariant(true)},
|
||||||
|
systemd1.Property{"BlockIOAccounting", dbus.MakeVariant(true)})
|
||||||
|
|
||||||
if c.Memory != 0 {
|
if c.Memory != 0 {
|
||||||
properties = append(properties,
|
properties = append(properties,
|
||||||
|
@ -162,10 +187,114 @@ func Apply(c *cgroups.Cgroup, pid int) (cgroups.ActiveCgroup, error) {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return &systemdCgroup{}, nil
|
|
||||||
|
if len(cpuArgs) != 0 {
|
||||||
|
mountpoint, err := cgroups.FindCgroupMountpoint("cpu")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
path := filepath.Join(mountpoint, cgroup)
|
||||||
|
|
||||||
|
for _, arg := range cpuArgs {
|
||||||
|
if err := ioutil.WriteFile(filepath.Join(path, arg.File), []byte(arg.Value), 0700); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(memoryArgs) != 0 {
|
||||||
|
mountpoint, err := cgroups.FindCgroupMountpoint("memory")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
path := filepath.Join(mountpoint, cgroup)
|
||||||
|
|
||||||
|
for _, arg := range memoryArgs {
|
||||||
|
if err := ioutil.WriteFile(filepath.Join(path, arg.File), []byte(arg.Value), 0700); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(cpusetArgs) != 0 {
|
||||||
|
// systemd does not atm set up the cpuset controller, so we must manually
|
||||||
|
// join it. Additionally that is a very finicky controller where each
|
||||||
|
// level must have a full setup as the default for a new directory is "no cpus",
|
||||||
|
// so we avoid using any hierarchies here, creating a toplevel directory.
|
||||||
|
mountpoint, err := cgroups.FindCgroupMountpoint("cpuset")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
initPath, err := cgroups.GetInitCgroupDir("cpuset")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
rootPath := filepath.Join(mountpoint, initPath)
|
||||||
|
|
||||||
|
path := filepath.Join(mountpoint, initPath, c.Parent+"-"+c.Name)
|
||||||
|
|
||||||
|
res.cleanupDirs = append(res.cleanupDirs, path)
|
||||||
|
|
||||||
|
if err := os.MkdirAll(path, 0755); err != nil && !os.IsExist(err) {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
foundCpus := false
|
||||||
|
foundMems := false
|
||||||
|
|
||||||
|
for _, arg := range cpusetArgs {
|
||||||
|
if arg.File == "cpuset.cpus" {
|
||||||
|
foundCpus = true
|
||||||
|
}
|
||||||
|
if arg.File == "cpuset.mems" {
|
||||||
|
foundMems = true
|
||||||
|
}
|
||||||
|
if err := ioutil.WriteFile(filepath.Join(path, arg.File), []byte(arg.Value), 0700); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// These are required, if not specified inherit from parent
|
||||||
|
if !foundCpus {
|
||||||
|
s, err := ioutil.ReadFile(filepath.Join(rootPath, "cpuset.cpus"))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := ioutil.WriteFile(filepath.Join(path, "cpuset.cpus"), s, 0700); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// These are required, if not specified inherit from parent
|
||||||
|
if !foundMems {
|
||||||
|
s, err := ioutil.ReadFile(filepath.Join(rootPath, "cpuset.mems"))
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := ioutil.WriteFile(filepath.Join(path, "cpuset.mems"), s, 0700); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := ioutil.WriteFile(filepath.Join(path, "cgroup.procs"), []byte(strconv.Itoa(pid)), 0700); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return &res, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *systemdCgroup) Cleanup() error {
|
func (c *systemdCgroup) Cleanup() error {
|
||||||
// systemd cleans up, we don't need to do anything
|
// systemd cleans up, we don't need to do much
|
||||||
|
|
||||||
|
for _, path := range c.cleanupDirs {
|
||||||
|
os.RemoveAll(path)
|
||||||
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue