| 
									
										
										
										
											2024-05-03 11:45:46 -07:00
										 |  |  | import atexit | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | import errno | 
					
						
							|  |  |  | import os | 
					
						
							| 
									
										
										
										
											2013-09-05 20:46:49 +02:00
										 |  |  | import selectors | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | import signal | 
					
						
							|  |  |  | import socket | 
					
						
							|  |  |  | import struct | 
					
						
							|  |  |  | import sys | 
					
						
							|  |  |  | import threading | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  | import warnings | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | from . import connection | 
					
						
							|  |  |  | from . import process | 
					
						
							| 
									
										
										
										
											2016-09-09 18:03:10 -05:00
										 |  |  | from .context import reduction | 
					
						
							| 
									
										
										
										
											2019-05-10 22:59:08 +02:00
										 |  |  | from . import resource_tracker | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | from . import spawn | 
					
						
							|  |  |  | from . import util | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | __all__ = ['ensure_running', 'get_inherited_fds', 'connect_to_new_process', | 
					
						
							|  |  |  |            'set_forkserver_preload'] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MAXFDS_TO_SEND = 256 | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  | SIGNED_STRUCT = struct.Struct('q')     # large enough for pid_t | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  | # Forkserver class | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  | class ForkServer(object): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self): | 
					
						
							|  |  |  |         self._forkserver_address = None | 
					
						
							|  |  |  |         self._forkserver_alive_fd = None | 
					
						
							| 
									
										
										
										
											2017-11-03 13:34:22 +01:00
										 |  |  |         self._forkserver_pid = None | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  |         self._inherited_fds = None | 
					
						
							|  |  |  |         self._lock = threading.Lock() | 
					
						
							|  |  |  |         self._preload_modules = ['__main__'] | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-07-05 16:15:39 +02:00
										 |  |  |     def _stop(self): | 
					
						
							|  |  |  |         # Method used by unit tests to stop the server | 
					
						
							|  |  |  |         with self._lock: | 
					
						
							|  |  |  |             self._stop_unlocked() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _stop_unlocked(self): | 
					
						
							|  |  |  |         if self._forkserver_pid is None: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # close the "alive" file descriptor asks the server to stop | 
					
						
							|  |  |  |         os.close(self._forkserver_alive_fd) | 
					
						
							|  |  |  |         self._forkserver_alive_fd = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         os.waitpid(self._forkserver_pid, 0) | 
					
						
							|  |  |  |         self._forkserver_pid = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-09 13:48:01 +00:00
										 |  |  |         if not util.is_abstract_socket_namespace(self._forkserver_address): | 
					
						
							|  |  |  |             os.unlink(self._forkserver_address) | 
					
						
							| 
									
										
										
										
											2019-07-05 16:15:39 +02:00
										 |  |  |         self._forkserver_address = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  |     def set_forkserver_preload(self, modules_names): | 
					
						
							|  |  |  |         '''Set list of module names to try to load in forkserver process.''' | 
					
						
							| 
									
										
										
										
											2023-08-15 22:58:12 +09:00
										 |  |  |         if not all(type(mod) is str for mod in modules_names): | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  |             raise TypeError('module_names must be a list of strings') | 
					
						
							|  |  |  |         self._preload_modules = modules_names | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_inherited_fds(self): | 
					
						
							|  |  |  |         '''Return list of fds inherited from parent process.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         This returns None if the current process was not started by fork | 
					
						
							|  |  |  |         server. | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         return self._inherited_fds | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def connect_to_new_process(self, fds): | 
					
						
							|  |  |  |         '''Request forkserver to create a child process.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Returns a pair of fds (status_r, data_w).  The calling process can read | 
					
						
							|  |  |  |         the child process's pid and (eventually) its returncode from status_r. | 
					
						
							|  |  |  |         The calling process should write to data_w the pickled preparation and | 
					
						
							|  |  |  |         process data. | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         self.ensure_running() | 
					
						
							|  |  |  |         if len(fds) + 4 >= MAXFDS_TO_SEND: | 
					
						
							|  |  |  |             raise ValueError('too many fds') | 
					
						
							|  |  |  |         with socket.socket(socket.AF_UNIX) as client: | 
					
						
							|  |  |  |             client.connect(self._forkserver_address) | 
					
						
							|  |  |  |             parent_r, child_w = os.pipe() | 
					
						
							|  |  |  |             child_r, parent_w = os.pipe() | 
					
						
							|  |  |  |             allfds = [child_r, child_w, self._forkserver_alive_fd, | 
					
						
							| 
									
										
										
										
											2019-05-10 22:59:08 +02:00
										 |  |  |                       resource_tracker.getfd()] | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  |             allfds += fds | 
					
						
							| 
									
										
										
										
											2013-08-22 11:38:55 +01:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  |                 reduction.sendfds(client, allfds) | 
					
						
							|  |  |  |                 return parent_r, parent_w | 
					
						
							| 
									
										
										
										
											2013-08-22 11:38:55 +01:00
										 |  |  |             except: | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  |                 os.close(parent_r) | 
					
						
							|  |  |  |                 os.close(parent_w) | 
					
						
							| 
									
										
										
										
											2013-08-22 11:38:55 +01:00
										 |  |  |                 raise | 
					
						
							|  |  |  |             finally: | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  |                 os.close(child_r) | 
					
						
							|  |  |  |                 os.close(child_w) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def ensure_running(self): | 
					
						
							|  |  |  |         '''Make sure that a fork server is running.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         This can be called from any process.  Note that usually a child | 
					
						
							|  |  |  |         process will just reuse the forkserver started by its parent, so | 
					
						
							|  |  |  |         ensure_running() will do nothing. | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         with self._lock: | 
					
						
							| 
									
										
										
										
											2019-05-10 22:59:08 +02:00
										 |  |  |             resource_tracker.ensure_running() | 
					
						
							| 
									
										
										
										
											2017-11-03 13:34:22 +01:00
										 |  |  |             if self._forkserver_pid is not None: | 
					
						
							|  |  |  |                 # forkserver was launched before, is it still running? | 
					
						
							|  |  |  |                 pid, status = os.waitpid(self._forkserver_pid, os.WNOHANG) | 
					
						
							|  |  |  |                 if not pid: | 
					
						
							|  |  |  |                     # still alive | 
					
						
							|  |  |  |                     return | 
					
						
							|  |  |  |                 # dead, launch it again | 
					
						
							|  |  |  |                 os.close(self._forkserver_alive_fd) | 
					
						
							|  |  |  |                 self._forkserver_address = None | 
					
						
							|  |  |  |                 self._forkserver_alive_fd = None | 
					
						
							|  |  |  |                 self._forkserver_pid = None | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             cmd = ('from multiprocessing.forkserver import main; ' + | 
					
						
							|  |  |  |                    'main(%d, %d, %r, **%r)') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if self._preload_modules: | 
					
						
							|  |  |  |                 desired_keys = {'main_path', 'sys_path'} | 
					
						
							|  |  |  |                 data = spawn.get_preparation_data('ignore') | 
					
						
							| 
									
										
										
										
											2017-05-18 07:35:54 -07:00
										 |  |  |                 data = {x: y for x, y in data.items() if x in desired_keys} | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 data = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             with socket.socket(socket.AF_UNIX) as listener: | 
					
						
							|  |  |  |                 address = connection.arbitrary_address('AF_UNIX') | 
					
						
							|  |  |  |                 listener.bind(address) | 
					
						
							| 
									
										
										
										
											2020-03-09 13:48:01 +00:00
										 |  |  |                 if not util.is_abstract_socket_namespace(address): | 
					
						
							|  |  |  |                     os.chmod(address, 0o600) | 
					
						
							| 
									
										
										
										
											2014-07-23 19:28:13 +01:00
										 |  |  |                 listener.listen() | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 # all client processes own the write end of the "alive" pipe; | 
					
						
							|  |  |  |                 # when they all terminate the read end becomes ready. | 
					
						
							|  |  |  |                 alive_r, alive_w = os.pipe() | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     fds_to_pass = [listener.fileno(), alive_r] | 
					
						
							|  |  |  |                     cmd %= (listener.fileno(), alive_r, self._preload_modules, | 
					
						
							|  |  |  |                             data) | 
					
						
							|  |  |  |                     exe = spawn.get_executable() | 
					
						
							|  |  |  |                     args = [exe] + util._args_from_interpreter_flags() | 
					
						
							|  |  |  |                     args += ['-c', cmd] | 
					
						
							|  |  |  |                     pid = util.spawnv_passfds(exe, args, fds_to_pass) | 
					
						
							|  |  |  |                 except: | 
					
						
							|  |  |  |                     os.close(alive_w) | 
					
						
							|  |  |  |                     raise | 
					
						
							|  |  |  |                 finally: | 
					
						
							|  |  |  |                     os.close(alive_r) | 
					
						
							|  |  |  |                 self._forkserver_address = address | 
					
						
							|  |  |  |                 self._forkserver_alive_fd = alive_w | 
					
						
							| 
									
										
										
										
											2017-11-03 13:34:22 +01:00
										 |  |  |                 self._forkserver_pid = pid | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  | # | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | def main(listener_fd, alive_r, preload, main_path=None, sys_path=None): | 
					
						
							|  |  |  |     '''Run forkserver.''' | 
					
						
							|  |  |  |     if preload: | 
					
						
							| 
									
										
										
										
											2024-11-09 15:01:32 -08:00
										 |  |  |         if sys_path is not None: | 
					
						
							|  |  |  |             sys.path[:] = sys_path | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |         if '__main__' in preload and main_path is not None: | 
					
						
							|  |  |  |             process.current_process()._inheriting = True | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 spawn.import_main_path(main_path) | 
					
						
							|  |  |  |             finally: | 
					
						
							|  |  |  |                 del process.current_process()._inheriting | 
					
						
							|  |  |  |         for modname in preload: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 __import__(modname) | 
					
						
							|  |  |  |             except ImportError: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-25 09:29:50 +01:00
										 |  |  |     util._close_stdin() | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |     sig_r, sig_w = os.pipe() | 
					
						
							| 
									
										
										
										
											2017-06-13 09:46:06 +02:00
										 |  |  |     os.set_blocking(sig_r, False) | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |     os.set_blocking(sig_w, False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def sigchld_handler(*_unused): | 
					
						
							| 
									
										
										
										
											2017-06-13 09:46:06 +02:00
										 |  |  |         # Dummy signal handler, doesn't do anything | 
					
						
							|  |  |  |         pass | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-04 16:44:53 +02:00
										 |  |  |     handlers = { | 
					
						
							| 
									
										
										
										
											2017-11-03 13:34:22 +01:00
										 |  |  |         # unblocking SIGCHLD allows the wakeup fd to notify our event loop | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |         signal.SIGCHLD: sigchld_handler, | 
					
						
							| 
									
										
										
										
											2017-11-03 13:34:22 +01:00
										 |  |  |         # protect the process from ^C | 
					
						
							|  |  |  |         signal.SIGINT: signal.SIG_IGN, | 
					
						
							| 
									
										
										
										
											2017-05-04 16:44:53 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     old_handlers = {sig: signal.signal(sig, val) | 
					
						
							|  |  |  |                     for (sig, val) in handlers.items()} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-13 09:46:06 +02:00
										 |  |  |     # calling os.write() in the Python signal handler is racy | 
					
						
							|  |  |  |     signal.set_wakeup_fd(sig_w) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |     # map child pids to client fds | 
					
						
							|  |  |  |     pid_to_fd = {} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-05 20:46:49 +02:00
										 |  |  |     with socket.socket(socket.AF_UNIX, fileno=listener_fd) as listener, \ | 
					
						
							|  |  |  |          selectors.DefaultSelector() as selector: | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  |         _forkserver._forkserver_address = listener.getsockname() | 
					
						
							| 
									
										
										
										
											2013-09-05 20:46:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         selector.register(listener, selectors.EVENT_READ) | 
					
						
							|  |  |  |         selector.register(alive_r, selectors.EVENT_READ) | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |         selector.register(sig_r, selectors.EVENT_READ) | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         while True: | 
					
						
							|  |  |  |             try: | 
					
						
							| 
									
										
										
										
											2013-09-05 20:46:49 +02:00
										 |  |  |                 while True: | 
					
						
							|  |  |  |                     rfds = [key.fileobj for (key, events) in selector.select()] | 
					
						
							|  |  |  |                     if rfds: | 
					
						
							|  |  |  |                         break | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if alive_r in rfds: | 
					
						
							|  |  |  |                     # EOF because no more client processes left | 
					
						
							| 
									
										
										
										
											2017-08-29 17:52:18 -05:00
										 |  |  |                     assert os.read(alive_r, 1) == b'', "Not at EOF?" | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |                     raise SystemExit | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |                 if sig_r in rfds: | 
					
						
							|  |  |  |                     # Got SIGCHLD | 
					
						
							|  |  |  |                     os.read(sig_r, 65536)  # exhaust | 
					
						
							|  |  |  |                     while True: | 
					
						
							|  |  |  |                         # Scan for child processes | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |                         try: | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |                             pid, sts = os.waitpid(-1, os.WNOHANG) | 
					
						
							|  |  |  |                         except ChildProcessError: | 
					
						
							|  |  |  |                             break | 
					
						
							|  |  |  |                         if pid == 0: | 
					
						
							|  |  |  |                             break | 
					
						
							|  |  |  |                         child_w = pid_to_fd.pop(pid, None) | 
					
						
							|  |  |  |                         if child_w is not None: | 
					
						
							| 
									
										
										
										
											2020-04-01 18:49:29 +02:00
										 |  |  |                             returncode = os.waitstatus_to_exitcode(sts) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-24 19:22:23 +02:00
										 |  |  |                             # Send exit code to client process | 
					
						
							|  |  |  |                             try: | 
					
						
							|  |  |  |                                 write_signed(child_w, returncode) | 
					
						
							|  |  |  |                             except BrokenPipeError: | 
					
						
							|  |  |  |                                 # client vanished | 
					
						
							|  |  |  |                                 pass | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |                             os.close(child_w) | 
					
						
							|  |  |  |                         else: | 
					
						
							|  |  |  |                             # This shouldn't happen really | 
					
						
							|  |  |  |                             warnings.warn('forkserver: waitpid returned ' | 
					
						
							|  |  |  |                                           'unexpected pid %d' % pid) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if listener in rfds: | 
					
						
							|  |  |  |                     # Incoming fork request | 
					
						
							|  |  |  |                     with listener.accept()[0] as s: | 
					
						
							|  |  |  |                         # Receive fds from client | 
					
						
							|  |  |  |                         fds = reduction.recvfds(s, MAXFDS_TO_SEND + 1) | 
					
						
							| 
									
										
										
										
											2017-08-29 17:52:18 -05:00
										 |  |  |                         if len(fds) > MAXFDS_TO_SEND: | 
					
						
							|  |  |  |                             raise RuntimeError( | 
					
						
							|  |  |  |                                 "Too many ({0:n}) fds to send".format( | 
					
						
							|  |  |  |                                     len(fds))) | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |                         child_r, child_w, *fds = fds | 
					
						
							|  |  |  |                         s.close() | 
					
						
							|  |  |  |                         pid = os.fork() | 
					
						
							|  |  |  |                         if pid == 0: | 
					
						
							|  |  |  |                             # Child | 
					
						
							|  |  |  |                             code = 1 | 
					
						
							|  |  |  |                             try: | 
					
						
							|  |  |  |                                 listener.close() | 
					
						
							| 
									
										
										
										
											2017-07-22 13:22:54 +02:00
										 |  |  |                                 selector.close() | 
					
						
							|  |  |  |                                 unused_fds = [alive_r, child_w, sig_r, sig_w] | 
					
						
							|  |  |  |                                 unused_fds.extend(pid_to_fd.values()) | 
					
						
							| 
									
										
										
										
											2024-05-03 11:45:46 -07:00
										 |  |  |                                 atexit._clear() | 
					
						
							|  |  |  |                                 atexit.register(util._exit_function) | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |                                 code = _serve_one(child_r, fds, | 
					
						
							| 
									
										
										
										
											2017-07-22 13:22:54 +02:00
										 |  |  |                                                   unused_fds, | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |                                                   old_handlers) | 
					
						
							|  |  |  |                             except Exception: | 
					
						
							|  |  |  |                                 sys.excepthook(*sys.exc_info()) | 
					
						
							|  |  |  |                                 sys.stderr.flush() | 
					
						
							|  |  |  |                             finally: | 
					
						
							| 
									
										
										
										
											2024-05-03 11:45:46 -07:00
										 |  |  |                                 atexit._run_exitfuncs() | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |                                 os._exit(code) | 
					
						
							|  |  |  |                         else: | 
					
						
							| 
									
										
										
										
											2017-06-24 19:22:23 +02:00
										 |  |  |                             # Send pid to client process | 
					
						
							|  |  |  |                             try: | 
					
						
							|  |  |  |                                 write_signed(child_w, pid) | 
					
						
							|  |  |  |                             except BrokenPipeError: | 
					
						
							|  |  |  |                                 # client vanished | 
					
						
							|  |  |  |                                 pass | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |                             pid_to_fd[pid] = child_w | 
					
						
							|  |  |  |                             os.close(child_r) | 
					
						
							|  |  |  |                             for fd in fds: | 
					
						
							|  |  |  |                                 os.close(fd) | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             except OSError as e: | 
					
						
							|  |  |  |                 if e.errno != errno.ECONNABORTED: | 
					
						
							|  |  |  |                     raise | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | def _serve_one(child_r, fds, unused_fds, handlers): | 
					
						
							| 
									
										
										
										
											2017-05-04 16:44:53 +02:00
										 |  |  |     # close unnecessary stuff and reset signal handlers | 
					
						
							| 
									
										
										
										
											2017-06-13 09:46:06 +02:00
										 |  |  |     signal.set_wakeup_fd(-1) | 
					
						
							| 
									
										
										
										
											2017-05-04 16:44:53 +02:00
										 |  |  |     for sig, val in handlers.items(): | 
					
						
							|  |  |  |         signal.signal(sig, val) | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |     for fd in unused_fds: | 
					
						
							|  |  |  |         os.close(fd) | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |     (_forkserver._forkserver_alive_fd, | 
					
						
							| 
									
										
										
										
											2019-05-10 22:59:08 +02:00
										 |  |  |      resource_tracker._resource_tracker._fd, | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |      *_forkserver._inherited_fds) = fds | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |     # Run process object received over pipe | 
					
						
							| 
									
										
										
										
											2019-05-20 21:37:05 +02:00
										 |  |  |     parent_sentinel = os.dup(child_r) | 
					
						
							|  |  |  |     code = spawn._main(child_r, parent_sentinel) | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |     return code | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  | # Read and write signed numbers | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  | def read_signed(fd): | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |     data = b'' | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |     length = SIGNED_STRUCT.size | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |     while len(data) < length: | 
					
						
							| 
									
										
										
										
											2015-02-07 13:27:50 +00:00
										 |  |  |         s = os.read(fd, length - len(data)) | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |         if not s: | 
					
						
							|  |  |  |             raise EOFError('unexpected EOF') | 
					
						
							|  |  |  |         data += s | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  |     return SIGNED_STRUCT.unpack(data)[0] | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-12 15:28:19 +02:00
										 |  |  | def write_signed(fd, n): | 
					
						
							|  |  |  |     msg = SIGNED_STRUCT.pack(n) | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |     while msg: | 
					
						
							| 
									
										
										
										
											2015-02-07 13:27:50 +00:00
										 |  |  |         nbytes = os.write(fd, msg) | 
					
						
							| 
									
										
										
										
											2013-08-14 15:35:41 +01:00
										 |  |  |         if nbytes == 0: | 
					
						
							|  |  |  |             raise RuntimeError('should not get here') | 
					
						
							|  |  |  |         msg = msg[nbytes:] | 
					
						
							| 
									
										
										
										
											2013-10-16 16:41:56 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | _forkserver = ForkServer() | 
					
						
							|  |  |  | ensure_running = _forkserver.ensure_running | 
					
						
							|  |  |  | get_inherited_fds = _forkserver.get_inherited_fds | 
					
						
							|  |  |  | connect_to_new_process = _forkserver.connect_to_new_process | 
					
						
							|  |  |  | set_forkserver_preload = _forkserver.set_forkserver_preload |