UI: exit gracefully (#25388)

* UI: exit gracefully

* cleanup

Co-authored-by: Comma Device <device@comma.ai>
pull/25390/head
Adeeb Shihadeh 3 years ago committed by GitHub
parent 49044b5399
commit 2cff05be43
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
  1. 15
      selfdrive/manager/test/test_manager.py
  2. 9
      selfdrive/ui/qt/util.cc

@ -7,7 +7,7 @@ import unittest
import selfdrive.manager.manager as manager import selfdrive.manager.manager as manager
from selfdrive.manager.process import DaemonProcess from selfdrive.manager.process import DaemonProcess
from selfdrive.manager.process_config import managed_processes from selfdrive.manager.process_config import managed_processes
from system.hardware import AGNOS, HARDWARE from system.hardware import HARDWARE
os.environ['FAKEUPLOAD'] = "1" os.environ['FAKEUPLOAD'] = "1"
@ -35,8 +35,10 @@ class TestManager(unittest.TestCase):
t = time.monotonic() - start t = time.monotonic() - start
assert t < MAX_STARTUP_TIME, f"startup took {t}s, expected <{MAX_STARTUP_TIME}s" assert t < MAX_STARTUP_TIME, f"startup took {t}s, expected <{MAX_STARTUP_TIME}s"
# ensure all processes exit cleanly
def test_clean_exit(self): def test_clean_exit(self):
"""
Ensure all processes exit cleanly when stopped.
"""
HARDWARE.set_power_save(False) HARDWARE.set_power_save(False)
manager.manager_prepare() manager.manager_prepare()
for p in ALL_PROCESSES: for p in ALL_PROCESSES:
@ -45,19 +47,18 @@ class TestManager(unittest.TestCase):
time.sleep(10) time.sleep(10)
for p in reversed(ALL_PROCESSES): for p in reversed(ALL_PROCESSES):
with self.subTest(proc=p):
state = managed_processes[p].get_process_state_msg() state = managed_processes[p].get_process_state_msg()
self.assertTrue(state.running, f"{p} not running") self.assertTrue(state.running, f"{p} not running")
exit_code = managed_processes[p].stop(retry=False) exit_code = managed_processes[p].stop(retry=False)
if (AGNOS and p in ['ui',]):
# TODO: make Qt UI exit gracefully self.assertTrue(exit_code is not None, f"{p} failed to exit")
continue
# TODO: interrupted blocking read exits with 1 in cereal. use a more unique return code # TODO: interrupted blocking read exits with 1 in cereal. use a more unique return code
exit_codes = [0, 1] exit_codes = [0, 1]
if managed_processes[p].sigkill: if managed_processes[p].sigkill:
exit_codes = [-signal.SIGKILL] exit_codes = [-signal.SIGKILL]
assert exit_code in exit_codes, f"{p} died with {exit_code}" self.assertIn(exit_code, exit_codes, f"{p} died with {exit_code}")
if __name__ == "__main__": if __name__ == "__main__":

@ -106,10 +106,19 @@ void setQtSurfaceFormat() {
QSurfaceFormat::setDefaultFormat(fmt); QSurfaceFormat::setDefaultFormat(fmt);
} }
void sigTermHandler(int s) {
std::signal(s, SIG_DFL);
qApp->quit();
}
void initApp(int argc, char *argv[]) { void initApp(int argc, char *argv[]) {
Hardware::set_display_power(true); Hardware::set_display_power(true);
Hardware::set_brightness(65); Hardware::set_brightness(65);
// setup signal handlers to exit gracefully
std::signal(SIGINT, sigTermHandler);
std::signal(SIGTERM, sigTermHandler);
#ifdef __APPLE__ #ifdef __APPLE__
{ {
// Get the devicePixelRatio, and scale accordingly to maintain 1:1 rendering // Get the devicePixelRatio, and scale accordingly to maintain 1:1 rendering

Loading…
Cancel
Save