##// END OF EJS Templates
worker: support parallelization of functions with return values...
Valentin Gatien-Baron -
r42648:55b724df default draft
parent child Browse files
Show More
@@ -83,7 +83,8 def worthwhile(ui, costperop, nops, thre
83 83 benefit = linear - (_STARTUP_COST * workers + linear / workers)
84 84 return benefit >= 0.15
85 85
86 def worker(ui, costperarg, func, staticargs, args, threadsafe=True):
86 def worker(ui, costperarg, func, staticargs, args, hasretval=False,
87 threadsafe=True):
87 88 '''run a function, possibly in parallel in multiple worker
88 89 processes.
89 90
@@ -91,23 +92,27 def worker(ui, costperarg, func, statica
91 92
92 93 costperarg - cost of a single task
93 94
94 func - function to run
95 func - function to run. It is expected to return a progress iterator.
95 96
96 97 staticargs - arguments to pass to every invocation of the function
97 98
98 99 args - arguments to split into chunks, to pass to individual
99 100 workers
100 101
102 hasretval - when True, func and the current function return an progress
103 iterator then a list (encoded as an iterator that yield many (False, ..)
104 then a (True, list)). The resulting list is in the natural order.
105
101 106 threadsafe - whether work items are thread safe and can be executed using
102 107 a thread-based worker. Should be disabled for CPU heavy tasks that don't
103 108 release the GIL.
104 109 '''
105 110 enabled = ui.configbool('worker', 'enabled')
106 111 if enabled and worthwhile(ui, costperarg, len(args), threadsafe=threadsafe):
107 return _platformworker(ui, func, staticargs, args)
112 return _platformworker(ui, func, staticargs, args, hasretval)
108 113 return func(*staticargs + (args,))
109 114
110 def _posixworker(ui, func, staticargs, args):
115 def _posixworker(ui, func, staticargs, args, hasretval):
111 116 workers = _numworkers(ui)
112 117 oldhandler = signal.getsignal(signal.SIGINT)
113 118 signal.signal(signal.SIGINT, signal.SIG_IGN)
@@ -157,7 +162,8 def _posixworker(ui, func, staticargs, a
157 162 ui.flush()
158 163 parentpid = os.getpid()
159 164 pipes = []
160 for pargs in partition(args, workers):
165 retvals = []
166 for i, pargs in enumerate(partition(args, workers)):
161 167 # Every worker gets its own pipe to send results on, so we don't have to
162 168 # implement atomic writes larger than PIPE_BUF. Each forked process has
163 169 # its own pipe's descriptors in the local variables, and the parent
@@ -165,6 +171,7 def _posixworker(ui, func, staticargs, a
165 171 # care what order they're in).
166 172 rfd, wfd = os.pipe()
167 173 pipes.append((rfd, wfd))
174 retvals.append(None)
168 175 # make sure we use os._exit in all worker code paths. otherwise the
169 176 # worker may do some clean-ups which could cause surprises like
170 177 # deadlock. see sshpeer.cleanup for example.
@@ -185,7 +192,7 def _posixworker(ui, func, staticargs, a
185 192 os.close(w)
186 193 os.close(rfd)
187 194 for result in func(*(staticargs + (pargs,))):
188 os.write(wfd, util.pickle.dumps(result))
195 os.write(wfd, util.pickle.dumps((i, result)))
189 196 return 0
190 197
191 198 ret = scmutil.callcatch(ui, workerfunc)
@@ -219,7 +226,11 def _posixworker(ui, func, staticargs, a
219 226 while openpipes > 0:
220 227 for key, events in selector.select():
221 228 try:
222 yield util.pickle.load(key.fileobj)
229 i, res = util.pickle.load(key.fileobj)
230 if hasretval and res[0]:
231 retvals[i] = res[1]
232 else:
233 yield res
223 234 except EOFError:
224 235 selector.unregister(key.fileobj)
225 236 key.fileobj.close()
@@ -237,6 +248,8 def _posixworker(ui, func, staticargs, a
237 248 if status < 0:
238 249 os.kill(os.getpid(), -status)
239 250 sys.exit(status)
251 if hasretval:
252 yield True, sum(retvals, [])
240 253
241 254 def _posixexitstatus(code):
242 255 '''convert a posix exit status into the same form returned by
@@ -248,7 +261,7 def _posixexitstatus(code):
248 261 elif os.WIFSIGNALED(code):
249 262 return -os.WTERMSIG(code)
250 263
251 def _windowsworker(ui, func, staticargs, args):
264 def _windowsworker(ui, func, staticargs, args, hasretval):
252 265 class Worker(threading.Thread):
253 266 def __init__(self, taskqueue, resultqueue, func, staticargs, *args,
254 267 **kwargs):
@@ -268,9 +281,9 def _windowsworker(ui, func, staticargs,
268 281 try:
269 282 while not self._taskqueue.empty():
270 283 try:
271 args = self._taskqueue.get_nowait()
284 i, args = self._taskqueue.get_nowait()
272 285 for res in self._func(*self._staticargs + (args,)):
273 self._resultqueue.put(res)
286 self._resultqueue.put((i, res))
274 287 # threading doesn't provide a native way to
275 288 # interrupt execution. handle it manually at every
276 289 # iteration.
@@ -305,9 +318,11 def _windowsworker(ui, func, staticargs,
305 318 workers = _numworkers(ui)
306 319 resultqueue = pycompat.queue.Queue()
307 320 taskqueue = pycompat.queue.Queue()
321 retvals = []
308 322 # partition work to more pieces than workers to minimize the chance
309 323 # of uneven distribution of large tasks between the workers
310 for pargs in partition(args, workers * 20):
324 for pargs in enumerate(partition(args, workers * 20)):
325 retvals.append(None)
311 326 taskqueue.put(pargs)
312 327 for _i in range(workers):
313 328 t = Worker(taskqueue, resultqueue, func, staticargs)
@@ -316,7 +331,11 def _windowsworker(ui, func, staticargs,
316 331 try:
317 332 while len(threads) > 0:
318 333 while not resultqueue.empty():
319 yield resultqueue.get()
334 (i, res) = resultqueue.get()
335 if hasretval and res[0]:
336 retvals[i] = res[1]
337 else:
338 yield res
320 339 threads[0].join(0.05)
321 340 finishedthreads = [_t for _t in threads if not _t.is_alive()]
322 341 for t in finishedthreads:
@@ -327,7 +346,13 def _windowsworker(ui, func, staticargs,
327 346 trykillworkers()
328 347 raise
329 348 while not resultqueue.empty():
330 yield resultqueue.get()
349 (i, res) = resultqueue.get()
350 if hasretval and res[0]:
351 retvals[i] = res[1]
352 else:
353 yield res
354 if hasretval:
355 yield True, sum(retvals, [])
331 356
332 357 if pycompat.iswindows:
333 358 _platformworker = _windowsworker
General Comments 0
You need to be logged in to leave comments. Login now