reduction.py 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. #
  2. # Module to allow connection and socket objects to be transferred
  3. # between processes
  4. #
  5. # multiprocessing/reduction.py
  6. #
  7. # Copyright (c) 2006-2008, R Oudkerk
  8. # All rights reserved.
  9. #
  10. # Redistribution and use in source and binary forms, with or without
  11. # modification, are permitted provided that the following conditions
  12. # are met:
  13. #
  14. # 1. Redistributions of source code must retain the above copyright
  15. # notice, this list of conditions and the following disclaimer.
  16. # 2. Redistributions in binary form must reproduce the above copyright
  17. # notice, this list of conditions and the following disclaimer in the
  18. # documentation and/or other materials provided with the distribution.
  19. # 3. Neither the name of author nor the names of any contributors may be
  20. # used to endorse or promote products derived from this software
  21. # without specific prior written permission.
  22. #
  23. # THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS "AS IS" AND
  24. # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  25. # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  26. # ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
  27. # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  28. # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
  29. # OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
  30. # HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
  31. # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
  32. # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
  33. # SUCH DAMAGE.
  34. #
  35. __all__ = []
  36. import os
  37. import sys
  38. import socket
  39. import threading
  40. import _multiprocessing
  41. from multiprocessing import current_process
  42. from multiprocessing.forking import Popen, duplicate, close, ForkingPickler
  43. from multiprocessing.util import register_after_fork, debug, sub_debug
  44. from multiprocessing.connection import Client, Listener
  45. #
  46. #
  47. #
  48. if not(sys.platform == 'win32' or hasattr(_multiprocessing, 'recvfd')):
  49. raise ImportError('pickling of connections not supported')
  50. #
  51. # Platform specific definitions
  52. #
  53. if sys.platform == 'win32':
  54. import _subprocess
  55. from _multiprocessing import win32
  56. def send_handle(conn, handle, destination_pid):
  57. process_handle = win32.OpenProcess(
  58. win32.PROCESS_ALL_ACCESS, False, destination_pid
  59. )
  60. try:
  61. new_handle = duplicate(handle, process_handle)
  62. conn.send(new_handle)
  63. finally:
  64. close(process_handle)
  65. def recv_handle(conn):
  66. return conn.recv()
  67. else:
  68. def send_handle(conn, handle, destination_pid):
  69. _multiprocessing.sendfd(conn.fileno(), handle)
  70. def recv_handle(conn):
  71. return _multiprocessing.recvfd(conn.fileno())
  72. #
  73. # Support for a per-process server thread which caches pickled handles
  74. #
  75. _cache = set()
  76. def _reset(obj):
  77. global _lock, _listener, _cache
  78. for h in _cache:
  79. close(h)
  80. _cache.clear()
  81. _lock = threading.Lock()
  82. _listener = None
  83. _reset(None)
  84. register_after_fork(_reset, _reset)
  85. def _get_listener():
  86. global _listener
  87. if _listener is None:
  88. _lock.acquire()
  89. try:
  90. if _listener is None:
  91. debug('starting listener and thread for sending handles')
  92. _listener = Listener(authkey=current_process().authkey)
  93. t = threading.Thread(target=_serve)
  94. t.daemon = True
  95. t.start()
  96. finally:
  97. _lock.release()
  98. return _listener
  99. def _serve():
  100. from .util import is_exiting, sub_warning
  101. while 1:
  102. try:
  103. conn = _listener.accept()
  104. handle_wanted, destination_pid = conn.recv()
  105. _cache.remove(handle_wanted)
  106. send_handle(conn, handle_wanted, destination_pid)
  107. close(handle_wanted)
  108. conn.close()
  109. except:
  110. if not is_exiting():
  111. import traceback
  112. sub_warning(
  113. 'thread for sharing handles raised exception :\n' +
  114. '-'*79 + '\n' + traceback.format_exc() + '-'*79
  115. )
  116. #
  117. # Functions to be used for pickling/unpickling objects with handles
  118. #
  119. def reduce_handle(handle):
  120. if Popen.thread_is_spawning():
  121. return (None, Popen.duplicate_for_child(handle), True)
  122. dup_handle = duplicate(handle)
  123. _cache.add(dup_handle)
  124. sub_debug('reducing handle %d', handle)
  125. return (_get_listener().address, dup_handle, False)
  126. def rebuild_handle(pickled_data):
  127. address, handle, inherited = pickled_data
  128. if inherited:
  129. return handle
  130. sub_debug('rebuilding handle %d', handle)
  131. conn = Client(address, authkey=current_process().authkey)
  132. conn.send((handle, os.getpid()))
  133. new_handle = recv_handle(conn)
  134. conn.close()
  135. return new_handle
  136. #
  137. # Register `_multiprocessing.Connection` with `ForkingPickler`
  138. #
  139. def reduce_connection(conn):
  140. rh = reduce_handle(conn.fileno())
  141. return rebuild_connection, (rh, conn.readable, conn.writable)
  142. def rebuild_connection(reduced_handle, readable, writable):
  143. handle = rebuild_handle(reduced_handle)
  144. return _multiprocessing.Connection(
  145. handle, readable=readable, writable=writable
  146. )
  147. ForkingPickler.register(_multiprocessing.Connection, reduce_connection)
  148. #
  149. # Register `socket.socket` with `ForkingPickler`
  150. #
  151. def fromfd(fd, family, type_, proto=0):
  152. s = socket.fromfd(fd, family, type_, proto)
  153. if s.__class__ is not socket.socket:
  154. s = socket.socket(_sock=s)
  155. return s
  156. def reduce_socket(s):
  157. reduced_handle = reduce_handle(s.fileno())
  158. return rebuild_socket, (reduced_handle, s.family, s.type, s.proto)
  159. def rebuild_socket(reduced_handle, family, type_, proto):
  160. fd = rebuild_handle(reduced_handle)
  161. _sock = fromfd(fd, family, type_, proto)
  162. close(fd)
  163. return _sock
  164. ForkingPickler.register(socket.socket, reduce_socket)
  165. #
  166. # Register `_multiprocessing.PipeConnection` with `ForkingPickler`
  167. #
  168. if sys.platform == 'win32':
  169. def reduce_pipe_connection(conn):
  170. rh = reduce_handle(conn.fileno())
  171. return rebuild_pipe_connection, (rh, conn.readable, conn.writable)
  172. def rebuild_pipe_connection(reduced_handle, readable, writable):
  173. handle = rebuild_handle(reduced_handle)
  174. return _multiprocessing.PipeConnection(
  175. handle, readable=readable, writable=writable
  176. )
  177. ForkingPickler.register(_multiprocessing.PipeConnection, reduce_pipe_connection)