//go:build darwin || dragonfly || freebsd || linux || netbsd || openbsd package machine import ( "errors" "fmt" "syscall" "time" "github.com/containers/podman/v5/pkg/machine/define" psutil "github.com/shirou/gopsutil/v3/process" "github.com/sirupsen/logrus" "golang.org/x/sys/unix" ) const ( loops = 8 sleepTime = time.Millisecond * 1 ) // backoffForProcess checks if the process still exists, for something like // sigterm. If the process still exists after loops and sleep time are exhausted, // an error is returned func backoffForProcess(p *psutil.Process) error { sleepInterval := sleepTime for i := 0; i < loops; i++ { running, err := p.IsRunning() if err != nil { // It is possible that while in our loop, the PID vaporize triggering // an input/output error (#21845) if errors.Is(err, unix.EIO) { return nil } return fmt.Errorf("checking if process running: %w", err) } if !running { return nil } time.Sleep(sleepInterval) // double the time sleepInterval += sleepInterval } return fmt.Errorf("process %d has not ended", p.Pid) } // / waitOnProcess takes a pid and sends a sigterm to it. it then waits for the // process to not exist. if the sigterm does not end the process after an interval, // then sigkill is sent. it also waits for the process to exit after the sigkill too. func waitOnProcess(processID int) error { logrus.Infof("Going to stop gvproxy (PID %d)", processID) p, err := psutil.NewProcess(int32(processID)) if err != nil { return fmt.Errorf("looking up PID %d: %w", processID, err) } running, err := p.IsRunning() if err != nil { return fmt.Errorf("checking if gvproxy is running: %w", err) } if !running { return nil } if err := p.Kill(); err != nil { if errors.Is(err, syscall.ESRCH) { logrus.Debugf("Gvproxy already dead, exiting cleanly") return nil } return err } return backoffForProcess(p) } // removeGVProxyPIDFile is just a wrapper to vmfile delete so we handle differently // on windows func removeGVProxyPIDFile(f define.VMFile) error { return f.Delete() }