From 049288a09cc3627b9957cc6af60002304323e13e Mon Sep 17 00:00:00 2001 From: Ryan Friedman <25047695+Ryanf55@users.noreply.github.com> Date: Sun, 15 Dec 2024 21:56:44 -0700 Subject: [PATCH] mavproxy.py: Clean up log_writer exit * Use threading.Event instead of daemon thread to avoid uncaught exception on ctrl+C exit * Daemon threads are not recommended anymore Signed-off-by: Ryan Friedman <25047695+Ryanf55@users.noreply.github.com> --- MAVProxy/mavproxy.py | 34 +++++++++++++++++++--------------- 1 file changed, 19 insertions(+), 15 deletions(-) diff --git a/MAVProxy/mavproxy.py b/MAVProxy/mavproxy.py index fab768c50d..acdafa9c25 100755 --- a/MAVProxy/mavproxy.py +++ b/MAVProxy/mavproxy.py @@ -87,7 +87,8 @@ def __init__(self): self.mav_error = 0 self.altitude = 0 self.last_distance_announce = 0.0 - self.exit = False + # A newer thread event to stop MAVProxy + self.stop_event = threading.Event() self.flightmode = 'MAV' self.last_mode_announce = 0 self.last_mode_announced = 'MAV' @@ -807,7 +808,7 @@ def process_stdin(line): print("%-15s : %s" % (cmd, help)) return if cmd == 'exit' and mpstate.settings.requireexit: - mpstate.status.exit = True + mpstate.status.stop_event.set() return if cmd not in command_map: @@ -936,8 +937,12 @@ def mkdir_p(dir): def log_writer(): '''log writing thread''' - while not mpstate.status.exit: - mpstate.logfile_raw.write(bytearray(mpstate.logqueue_raw.get())) + while not mpstate.status.stop_event.is_set(): + if not mpstate.logqueue_raw.empty(): + bytes = mpstate.logqueue_raw.get(block=False) + mpstate.logfile_raw.write(bytearray(bytes)) + + # TODO consider wait() the stop event instead timeout = time.time() + 10 while not mpstate.logqueue_raw.empty() and time.time() < timeout: mpstate.logfile_raw.write(mpstate.logqueue_raw.get()) @@ -948,6 +953,7 @@ def log_writer(): mpstate.logfile_raw.flush() + # If state_basedir is NOT set then paths for logs and aircraft # directories are relative to mavproxy's cwd def log_paths(): @@ -1009,18 +1015,17 @@ def open_telemetry_logs(logpath_telem, logpath_telem_raw): stat = os.statvfs(logpath_telem) if stat.f_bfree*stat.f_bsize < 209715200: print("ERROR: Not enough free disk space for logfile") - mpstate.status.exit = True + mpstate.status.stop_event.set() return # use a separate thread for writing to the logfile to prevent # delays during disk writes (important as delays can be long if camera # app is running) t = threading.Thread(target=log_writer, name='log_writer') - t.daemon = True t.start() except Exception as e: print("ERROR: opening log file for writing: %s" % e) - mpstate.status.exit = True + mpstate.status.stop_event.set() return @@ -1121,7 +1126,7 @@ def main_loop(): set_stream_rates() while True: - if mpstate is None or mpstate.status.exit: + if mpstate is None or mpstate.status.stop_event.is_set(): return # enable or disable screensaver: @@ -1218,12 +1223,12 @@ def main_loop(): def input_loop(): '''wait for user input''' - while mpstate.status.exit is not True: + while not mpstate.status.stop_event.is_set(): try: line = mpstate.rl.input() mpstate.input_queue.put(line) except (EOFError, IOError): - mpstate.status.exit = True + not mpstate.status.stop_event.set() def run_script(scriptfile): @@ -1407,7 +1412,6 @@ def run_startup_scripts(): # global mavproxy state mpstate = MPState() - mpstate.status.exit = False mpstate.command_map = command_map mpstate.continue_mode = opts.continue_mode # queues for logging @@ -1447,11 +1451,11 @@ def run_startup_scripts(): def quit_handler(signum=None, frame=None): # print('Signal handler called with signal', signum) - if mpstate.status.exit: + if mpstate.status.stop_event.is_set(): print('Clean shutdown impossible, forcing an exit') sys.exit(0) else: - mpstate.status.exit = True + mpstate.status.stop_event.set() # Listen for kill signals to cleanly shutdown modules fatalsignals = [signal.SIGTERM] @@ -1584,7 +1588,7 @@ def quit_handler(signum=None, frame=None): # use main program for input. This ensures the terminal cleans # up on exit - while (mpstate.status.exit is not True): + while not mpstate.status.stop_event.is_set(): try: if opts.daemon or opts.non_interactive: time.sleep(0.1) @@ -1606,7 +1610,7 @@ def quit_handler(signum=None, frame=None): m.init(mpstate) else: - mpstate.status.exit = True + mpstate.status.stop_event.set() sys.exit(1) if opts.profile: