daemon: execdriver: lxc: fix cgroup paths

When running LXC dind (outer docker is started with native driver)
cgroup paths point to `/docker/CID` inside `/proc/self/mountinfo` but
these paths aren't mounted (root is wrong). This fix just discard the
cgroup dir from mountinfo and set it to root `/`.
This patch fixes/skip OOM LXC tests that were failing.
Fix #16520

Signed-off-by: Antonio Murdaca <runcom@linux.com>
Signed-off-by: Antonio Murdaca <amurdaca@redhat.com>
This commit is contained in:
Antonio Murdaca 2015-09-24 12:13:19 +02:00 коммит произвёл Jessica Frazelle
Родитель 06f0d03ced
Коммит cfcddefacd
3 изменённых файлов: 27 добавлений и 12 удалений

Просмотреть файл

@ -324,24 +324,20 @@ func (d *Driver) Run(c *execdriver.Command, pipes *execdriver.Pipes, hooks execd
c.ContainerPid = pid c.ContainerPid = pid
oomKill := false
oomKillNotification, err := notifyOnOOM(cgroupPaths)
if hooks.Start != nil { if hooks.Start != nil {
logrus.Debugf("Invoking startCallback") logrus.Debugf("Invoking startCallback")
hooks.Start(&c.ProcessConfig, pid, oomKillNotification) chOOM := make(chan struct{})
close(chOOM)
hooks.Start(&c.ProcessConfig, pid, chOOM)
} }
oomKillNotification := notifyChannelOOM(cgroupPaths)
<-waitLock <-waitLock
exitCode := getExitCode(c) exitCode := getExitCode(c)
if err == nil { _, oomKill := <-oomKillNotification
_, oomKill = <-oomKillNotification logrus.Debugf("oomKill error: %v, waitErr: %v", oomKill, waitErr)
logrus.Debugf("oomKill error: %v, waitErr: %v", oomKill, waitErr)
} else {
logrus.Warnf("Your kernel does not support OOM notifications: %s", err)
}
// check oom error // check oom error
if oomKill { if oomKill {
@ -351,6 +347,17 @@ func (d *Driver) Run(c *execdriver.Command, pipes *execdriver.Pipes, hooks execd
return execdriver.ExitStatus{ExitCode: exitCode, OOMKilled: oomKill}, waitErr return execdriver.ExitStatus{ExitCode: exitCode, OOMKilled: oomKill}, waitErr
} }
func notifyChannelOOM(paths map[string]string) <-chan struct{} {
oom, err := notifyOnOOM(paths)
if err != nil {
logrus.Warnf("Your kernel does not support OOM notifications: %s", err)
c := make(chan struct{})
close(c)
return c
}
return oom
}
// copy from libcontainer // copy from libcontainer
func notifyOnOOM(paths map[string]string) (<-chan struct{}, error) { func notifyOnOOM(paths map[string]string) (<-chan struct{}, error) {
dir := paths["memory"] dir := paths["memory"]
@ -386,11 +393,13 @@ func notifyOnOOM(paths map[string]string) (<-chan struct{}, error) {
buf := make([]byte, 8) buf := make([]byte, 8)
for { for {
if _, err := eventfd.Read(buf); err != nil { if _, err := eventfd.Read(buf); err != nil {
logrus.Warn(err)
return return
} }
// When a cgroup is destroyed, an event is sent to eventfd. // When a cgroup is destroyed, an event is sent to eventfd.
// So if the control path is gone, return instead of notifying. // So if the control path is gone, return instead of notifying.
if _, err := os.Lstat(eventControlPath); os.IsNotExist(err) { if _, err := os.Lstat(eventControlPath); os.IsNotExist(err) {
logrus.Warn(err)
return return
} }
ch <- struct{}{} ch <- struct{}{}
@ -424,6 +433,11 @@ func cgroupPaths(containerID string) (map[string]string, error) {
//unsupported subystem //unsupported subystem
continue continue
} }
// if we are running dind
dockerPathIdx := strings.LastIndex(cgroupDir, "docker")
if dockerPathIdx != -1 {
cgroupDir = cgroupDir[:dockerPathIdx-1]
}
path := filepath.Join(cgroupRoot, cgroupDir, "lxc", containerID) path := filepath.Join(cgroupRoot, cgroupDir, "lxc", containerID)
paths[subsystem] = path paths[subsystem] = path
} }

Просмотреть файл

@ -167,7 +167,6 @@ func (d *Driver) Run(c *execdriver.Command, pipes *execdriver.Pipes, hooks execd
oom := notifyOnOOM(cont) oom := notifyOnOOM(cont)
if hooks.Start != nil { if hooks.Start != nil {
pid, err := p.Pid() pid, err := p.Pid()
if err != nil { if err != nil {
p.Signal(os.Kill) p.Signal(os.Kill)

Просмотреть файл

@ -56,6 +56,7 @@ func (s *DockerSuite) TestEventsRedirectStdout(c *check.C) {
func (s *DockerSuite) TestEventsOOMDisableFalse(c *check.C) { func (s *DockerSuite) TestEventsOOMDisableFalse(c *check.C) {
testRequires(c, DaemonIsLinux) testRequires(c, DaemonIsLinux)
testRequires(c, NativeExecDriver)
testRequires(c, oomControl) testRequires(c, oomControl)
errChan := make(chan error) errChan := make(chan error)
@ -103,6 +104,7 @@ func (s *DockerSuite) TestEventsOOMDisableFalse(c *check.C) {
func (s *DockerSuite) TestEventsOOMDisableTrue(c *check.C) { func (s *DockerSuite) TestEventsOOMDisableTrue(c *check.C) {
testRequires(c, DaemonIsLinux) testRequires(c, DaemonIsLinux)
testRequires(c, NativeExecDriver)
testRequires(c, oomControl) testRequires(c, oomControl)
errChan := make(chan error) errChan := make(chan error)