Add prctl set child subreaper
Signed-off-by: Michael Crosby <crosbymichael@gmail.com>
This commit is contained in:
		
							parent
							
								
									ae9b2bafd5
								
							
						
					
					
						commit
						a10aa91051
					
				
					 2 changed files with 82 additions and 47 deletions
				
			
		|  | @ -4,17 +4,14 @@ import ( | |||
| 	"log" | ||||
| 	"net/http" | ||||
| 	"os" | ||||
| 	"os/signal" | ||||
| 	"runtime" | ||||
| 	"sync" | ||||
| 	"syscall" | ||||
| 	"time" | ||||
| 
 | ||||
| 	"github.com/Sirupsen/logrus" | ||||
| 	"github.com/codegangsta/cli" | ||||
| 	"github.com/docker/containerd" | ||||
| 	"github.com/docker/containerd/api/v1" | ||||
| 	"github.com/opencontainers/runc/libcontainer/utils" | ||||
| 	"github.com/rcrowley/go-metrics" | ||||
| ) | ||||
| 
 | ||||
|  | @ -89,6 +86,9 @@ func daemon(stateDir string, concurrency, bufferSize int) error { | |||
| 		w := containerd.NewWorker(supervisor, wg) | ||||
| 		go w.Start() | ||||
| 	} | ||||
| 	if err := setSubReaper(); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	// start the signal handler in the background. | ||||
| 	go startSignalHandler(supervisor, bufferSize) | ||||
| 	if err := supervisor.Start(); err != nil { | ||||
|  | @ -97,47 +97,3 @@ func daemon(stateDir string, concurrency, bufferSize int) error { | |||
| 	server := v1.NewServer(supervisor) | ||||
| 	return http.ListenAndServe("localhost:8888", server) | ||||
| } | ||||
| 
 | ||||
| func startSignalHandler(supervisor *containerd.Supervisor, bufferSize int) { | ||||
| 	logrus.Debug("containerd: starting signal handler") | ||||
| 	signals := make(chan os.Signal, bufferSize) | ||||
| 	signal.Notify(signals) | ||||
| 	for s := range signals { | ||||
| 		switch s { | ||||
| 		case syscall.SIGTERM, syscall.SIGINT, syscall.SIGSTOP: | ||||
| 			supervisor.Close() | ||||
| 			os.Exit(0) | ||||
| 		case syscall.SIGCHLD: | ||||
| 			exits, err := reap() | ||||
| 			if err != nil { | ||||
| 				logrus.WithField("error", err).Error("containerd: reaping child processes") | ||||
| 			} | ||||
| 			for _, e := range exits { | ||||
| 				supervisor.SendEvent(e) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func reap() (exits []*containerd.Event, err error) { | ||||
| 	var ( | ||||
| 		ws  syscall.WaitStatus | ||||
| 		rus syscall.Rusage | ||||
| 	) | ||||
| 	for { | ||||
| 		pid, err := syscall.Wait4(-1, &ws, syscall.WNOHANG, &rus) | ||||
| 		if err != nil { | ||||
| 			if err == syscall.ECHILD { | ||||
| 				return exits, nil | ||||
| 			} | ||||
| 			return exits, err | ||||
| 		} | ||||
| 		if pid <= 0 { | ||||
| 			return exits, nil | ||||
| 		} | ||||
| 		e := containerd.NewEvent(containerd.ExitEventType) | ||||
| 		e.Pid = pid | ||||
| 		e.Status = utils.ExitStatus(ws) | ||||
| 		exits = append(exits, e) | ||||
| 	} | ||||
| } | ||||
|  |  | |||
							
								
								
									
										79
									
								
								containerd/reap_linux.go
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								containerd/reap_linux.go
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,79 @@ | |||
| // +build linux | ||||
| 
 | ||||
| package main | ||||
| 
 | ||||
| import ( | ||||
| 	"os" | ||||
| 	"os/signal" | ||||
| 	"syscall" | ||||
| 
 | ||||
| 	"github.com/Sirupsen/logrus" | ||||
| 	"github.com/docker/containerd" | ||||
| 	"github.com/opencontainers/runc/libcontainer/utils" | ||||
| ) | ||||
| 
 | ||||
| // http://man7.org/linux/man-pages/man2/prctl.2.html | ||||
| // | ||||
| // If arg2 is nonzero, set the "child subreaper" attribute of the | ||||
| // calling process; if arg2 is zero, unset the attribute.  When a | ||||
| // process is marked as a child subreaper, all of the children | ||||
| // that it creates, and their descendants, will be marked as | ||||
| // having a subreaper.  In effect, a subreaper fulfills the role | ||||
| // of init(1) for its descendant processes.  Upon termination of | ||||
| // a process that is orphaned (i.e., its immediate parent has | ||||
| // already terminated) and marked as having a subreaper, the | ||||
| // nearest still living ancestor subreaper will receive a SIGCHLD | ||||
| // signal and be able to wait(2) on the process to discover its | ||||
| // termination status. | ||||
| const PR_SET_CHILD_SUBREAPER = 36 | ||||
| 
 | ||||
| func startSignalHandler(supervisor *containerd.Supervisor, bufferSize int) { | ||||
| 	logrus.Debug("containerd: starting signal handler") | ||||
| 	signals := make(chan os.Signal, bufferSize) | ||||
| 	signal.Notify(signals) | ||||
| 	for s := range signals { | ||||
| 		switch s { | ||||
| 		case syscall.SIGTERM, syscall.SIGINT: | ||||
| 			supervisor.Close() | ||||
| 			os.Exit(0) | ||||
| 		case syscall.SIGCHLD: | ||||
| 			exits, err := reap() | ||||
| 			if err != nil { | ||||
| 				logrus.WithField("error", err).Error("containerd: reaping child processes") | ||||
| 			} | ||||
| 			for _, e := range exits { | ||||
| 				supervisor.SendEvent(e) | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func reap() (exits []*containerd.Event, err error) { | ||||
| 	var ( | ||||
| 		ws  syscall.WaitStatus | ||||
| 		rus syscall.Rusage | ||||
| 	) | ||||
| 	for { | ||||
| 		pid, err := syscall.Wait4(-1, &ws, syscall.WNOHANG, &rus) | ||||
| 		if err != nil { | ||||
| 			if err == syscall.ECHILD { | ||||
| 				return exits, nil | ||||
| 			} | ||||
| 			return exits, err | ||||
| 		} | ||||
| 		if pid <= 0 { | ||||
| 			return exits, nil | ||||
| 		} | ||||
| 		e := containerd.NewEvent(containerd.ExitEventType) | ||||
| 		e.Pid = pid | ||||
| 		e.Status = utils.ExitStatus(ws) | ||||
| 		exits = append(exits, e) | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| func setSubReaper() error { | ||||
| 	if _, _, err := syscall.RawSyscall(syscall.SYS_PRCTL, PR_SET_CHILD_SUBREAPER, 1, 0); err != 0 { | ||||
| 		return err | ||||
| 	} | ||||
| 	return nil | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue