Brad Bishop | c342db3 | 2019-05-15 21:57:59 -0400 | [diff] [blame] | 1 | # |
Patrick Williams | 92b42cb | 2022-09-03 06:53:57 -0500 | [diff] [blame^] | 2 | # Copyright BitBake Contributors |
| 3 | # |
Brad Bishop | c342db3 | 2019-05-15 21:57:59 -0400 | [diff] [blame] | 4 | # SPDX-License-Identifier: GPL-2.0-only |
| 5 | # |
| 6 | |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 7 | """ |
| 8 | Python Daemonizing helper |
| 9 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 10 | Originally based on code Copyright (C) 2005 Chad J. Schroeder but now heavily modified |
| 11 | to allow a function to be daemonized and return for bitbake use by Richard Purdie |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 12 | """ |
| 13 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 14 | import os |
| 15 | import sys |
| 16 | import io |
| 17 | import traceback |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 18 | |
Andrew Geissler | c9f7865 | 2020-09-18 14:11:35 -0500 | [diff] [blame] | 19 | import bb |
| 20 | |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 21 | def createDaemon(function, logfile): |
| 22 | """ |
| 23 | Detach a process from the controlling terminal and run it in the |
| 24 | background as a daemon, returning control to the caller. |
| 25 | """ |
| 26 | |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 27 | # Ensure stdout/stderror are flushed before forking to avoid duplicate output |
| 28 | sys.stdout.flush() |
| 29 | sys.stderr.flush() |
| 30 | |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 31 | try: |
| 32 | # Fork a child process so the parent can exit. This returns control to |
| 33 | # the command-line or shell. It also guarantees that the child will not |
| 34 | # be a process group leader, since the child receives a new process ID |
| 35 | # and inherits the parent's process group ID. This step is required |
| 36 | # to insure that the next call to os.setsid is successful. |
| 37 | pid = os.fork() |
| 38 | except OSError as e: |
| 39 | raise Exception("%s [%d]" % (e.strerror, e.errno)) |
| 40 | |
| 41 | if (pid == 0): # The first child. |
| 42 | # To become the session leader of this new session and the process group |
| 43 | # leader of the new process group, we call os.setsid(). The process is |
| 44 | # also guaranteed not to have a controlling terminal. |
| 45 | os.setsid() |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 46 | try: |
| 47 | # Fork a second child and exit immediately to prevent zombies. This |
| 48 | # causes the second child process to be orphaned, making the init |
| 49 | # process responsible for its cleanup. And, since the first child is |
| 50 | # a session leader without a controlling terminal, it's possible for |
| 51 | # it to acquire one by opening a terminal in the future (System V- |
| 52 | # based systems). This second fork guarantees that the child is no |
| 53 | # longer a session leader, preventing the daemon from ever acquiring |
| 54 | # a controlling terminal. |
| 55 | pid = os.fork() # Fork a second child. |
| 56 | except OSError as e: |
| 57 | raise Exception("%s [%d]" % (e.strerror, e.errno)) |
| 58 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 59 | if (pid != 0): |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 60 | # Parent (the first child) of the second child. |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 61 | # exit() or _exit()? |
| 62 | # _exit is like exit(), but it doesn't call any functions registered |
| 63 | # with atexit (and on_exit) or any registered signal handlers. It also |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 64 | # closes any open file descriptors, but doesn't flush any buffered output. |
| 65 | # Using exit() may cause all any temporary files to be unexpectedly |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 66 | # removed. It's therefore recommended that child branches of a fork() |
| 67 | # and the parent branch(es) of a daemon use _exit(). |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 68 | os._exit(0) |
| 69 | else: |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 70 | os.waitpid(pid, 0) |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 71 | return |
| 72 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 73 | # The second child. |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 74 | |
Brad Bishop | d7bf8c1 | 2018-02-25 22:55:05 -0500 | [diff] [blame] | 75 | # Replace standard fds with our own |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 76 | with open('/dev/null', 'r') as si: |
| 77 | os.dup2(si.fileno(), sys.stdin.fileno()) |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 78 | |
Andrew Geissler | 595f630 | 2022-01-24 19:11:47 +0000 | [diff] [blame] | 79 | with open(logfile, 'a+') as so: |
| 80 | try: |
| 81 | os.dup2(so.fileno(), sys.stdout.fileno()) |
| 82 | os.dup2(so.fileno(), sys.stderr.fileno()) |
| 83 | except io.UnsupportedOperation: |
| 84 | sys.stdout = so |
Brad Bishop | 1a4b7ee | 2018-12-16 17:11:34 -0800 | [diff] [blame] | 85 | |
Andrew Geissler | 595f630 | 2022-01-24 19:11:47 +0000 | [diff] [blame] | 86 | # Have stdout and stderr be the same so log output matches chronologically |
Andrew Geissler | 7e0e3c0 | 2022-02-25 20:34:39 +0000 | [diff] [blame] | 87 | # and there aren't two separate buffers |
Andrew Geissler | 595f630 | 2022-01-24 19:11:47 +0000 | [diff] [blame] | 88 | sys.stderr = sys.stdout |
Patrick Williams | c124f4f | 2015-09-15 14:41:29 -0500 | [diff] [blame] | 89 | |
Andrew Geissler | 595f630 | 2022-01-24 19:11:47 +0000 | [diff] [blame] | 90 | try: |
| 91 | function() |
| 92 | except Exception as e: |
| 93 | traceback.print_exc() |
| 94 | finally: |
| 95 | bb.event.print_ui_queue() |
| 96 | # os._exit() doesn't flush open files like os.exit() does. Manually flush |
| 97 | # stdout and stderr so that any logging output will be seen, particularly |
| 98 | # exception tracebacks. |
| 99 | sys.stdout.flush() |
| 100 | sys.stderr.flush() |
| 101 | os._exit(0) |