bitbake-worker 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503
  1. #!/usr/bin/env python3
  2. import os
  3. import sys
  4. import warnings
  5. sys.path.insert(0, os.path.join(os.path.dirname(os.path.dirname(sys.argv[0])), 'lib'))
  6. from bb import fetch2
  7. import logging
  8. import bb
  9. import select
  10. import errno
  11. import signal
  12. import pickle
  13. import traceback
  14. import queue
  15. from multiprocessing import Lock
  16. from threading import Thread
  17. if sys.getfilesystemencoding() != "utf-8":
  18. sys.exit("Please use a locale setting which supports UTF-8 (such as LANG=en_US.UTF-8).\nPython can't change the filesystem locale after loading so we need a UTF-8 when Python starts or things won't work.")
  19. # Users shouldn't be running this code directly
  20. if len(sys.argv) != 2 or not sys.argv[1].startswith("decafbad"):
  21. print("bitbake-worker is meant for internal execution by bitbake itself, please don't use it standalone.")
  22. sys.exit(1)
  23. profiling = False
  24. if sys.argv[1].startswith("decafbadbad"):
  25. profiling = True
  26. try:
  27. import cProfile as profile
  28. except:
  29. import profile
  30. # Unbuffer stdout to avoid log truncation in the event
  31. # of an unorderly exit as well as to provide timely
  32. # updates to log files for use with tail
  33. try:
  34. if sys.stdout.name == '<stdout>':
  35. import fcntl
  36. fl = fcntl.fcntl(sys.stdout.fileno(), fcntl.F_GETFL)
  37. fl |= os.O_SYNC
  38. fcntl.fcntl(sys.stdout.fileno(), fcntl.F_SETFL, fl)
  39. #sys.stdout = os.fdopen(sys.stdout.fileno(), 'w', 0)
  40. except:
  41. pass
  42. logger = logging.getLogger("BitBake")
  43. worker_pipe = sys.stdout.fileno()
  44. bb.utils.nonblockingfd(worker_pipe)
  45. # Need to guard against multiprocessing being used in child processes
  46. # and multiple processes trying to write to the parent at the same time
  47. worker_pipe_lock = None
  48. handler = bb.event.LogHandler()
  49. logger.addHandler(handler)
  50. if 0:
  51. # Code to write out a log file of all events passing through the worker
  52. logfilename = "/tmp/workerlogfile"
  53. format_str = "%(levelname)s: %(message)s"
  54. conlogformat = bb.msg.BBLogFormatter(format_str)
  55. consolelog = logging.FileHandler(logfilename)
  56. bb.msg.addDefaultlogFilter(consolelog)
  57. consolelog.setFormatter(conlogformat)
  58. logger.addHandler(consolelog)
  59. worker_queue = queue.Queue()
  60. def worker_fire(event, d):
  61. data = b"<event>" + pickle.dumps(event) + b"</event>"
  62. worker_fire_prepickled(data)
  63. def worker_fire_prepickled(event):
  64. global worker_queue
  65. worker_queue.put(event)
  66. #
  67. # We can end up with write contention with the cooker, it can be trying to send commands
  68. # and we can be trying to send event data back. Therefore use a separate thread for writing
  69. # back data to cooker.
  70. #
  71. worker_thread_exit = False
  72. def worker_flush(worker_queue):
  73. worker_queue_int = b""
  74. global worker_pipe, worker_thread_exit
  75. while True:
  76. try:
  77. worker_queue_int = worker_queue_int + worker_queue.get(True, 1)
  78. except queue.Empty:
  79. pass
  80. while (worker_queue_int or not worker_queue.empty()):
  81. try:
  82. (_, ready, _) = select.select([], [worker_pipe], [], 1)
  83. if not worker_queue.empty():
  84. worker_queue_int = worker_queue_int + worker_queue.get()
  85. written = os.write(worker_pipe, worker_queue_int)
  86. worker_queue_int = worker_queue_int[written:]
  87. except (IOError, OSError) as e:
  88. if e.errno != errno.EAGAIN and e.errno != errno.EPIPE:
  89. raise
  90. if worker_thread_exit and worker_queue.empty() and not worker_queue_int:
  91. return
  92. worker_thread = Thread(target=worker_flush, args=(worker_queue,))
  93. worker_thread.start()
  94. def worker_child_fire(event, d):
  95. global worker_pipe
  96. global worker_pipe_lock
  97. data = b"<event>" + pickle.dumps(event) + b"</event>"
  98. try:
  99. worker_pipe_lock.acquire()
  100. worker_pipe.write(data)
  101. worker_pipe_lock.release()
  102. except IOError:
  103. sigterm_handler(None, None)
  104. raise
  105. bb.event.worker_fire = worker_fire
  106. lf = None
  107. #lf = open("/tmp/workercommandlog", "w+")
  108. def workerlog_write(msg):
  109. if lf:
  110. lf.write(msg)
  111. lf.flush()
  112. def sigterm_handler(signum, frame):
  113. signal.signal(signal.SIGTERM, signal.SIG_DFL)
  114. os.killpg(0, signal.SIGTERM)
  115. sys.exit()
  116. def fork_off_task(cfg, data, databuilder, workerdata, fn, task, taskname, appends, taskdepdata, extraconfigdata, quieterrors=False, dry_run_exec=False):
  117. # We need to setup the environment BEFORE the fork, since
  118. # a fork() or exec*() activates PSEUDO...
  119. envbackup = {}
  120. fakeenv = {}
  121. umask = None
  122. taskdep = workerdata["taskdeps"][fn]
  123. if 'umask' in taskdep and taskname in taskdep['umask']:
  124. # umask might come in as a number or text string..
  125. try:
  126. umask = int(taskdep['umask'][taskname],8)
  127. except TypeError:
  128. umask = taskdep['umask'][taskname]
  129. dry_run = cfg.dry_run or dry_run_exec
  130. # We can't use the fakeroot environment in a dry run as it possibly hasn't been built
  131. if 'fakeroot' in taskdep and taskname in taskdep['fakeroot'] and not dry_run:
  132. envvars = (workerdata["fakerootenv"][fn] or "").split()
  133. for key, value in (var.split('=') for var in envvars):
  134. envbackup[key] = os.environ.get(key)
  135. os.environ[key] = value
  136. fakeenv[key] = value
  137. fakedirs = (workerdata["fakerootdirs"][fn] or "").split()
  138. for p in fakedirs:
  139. bb.utils.mkdirhier(p)
  140. logger.debug(2, 'Running %s:%s under fakeroot, fakedirs: %s' %
  141. (fn, taskname, ', '.join(fakedirs)))
  142. else:
  143. envvars = (workerdata["fakerootnoenv"][fn] or "").split()
  144. for key, value in (var.split('=') for var in envvars):
  145. envbackup[key] = os.environ.get(key)
  146. os.environ[key] = value
  147. fakeenv[key] = value
  148. sys.stdout.flush()
  149. sys.stderr.flush()
  150. try:
  151. pipein, pipeout = os.pipe()
  152. pipein = os.fdopen(pipein, 'rb', 4096)
  153. pipeout = os.fdopen(pipeout, 'wb', 0)
  154. pid = os.fork()
  155. except OSError as e:
  156. logger.critical("fork failed: %d (%s)" % (e.errno, e.strerror))
  157. sys.exit(1)
  158. if pid == 0:
  159. def child():
  160. global worker_pipe
  161. global worker_pipe_lock
  162. pipein.close()
  163. bb.utils.signal_on_parent_exit("SIGTERM")
  164. # Save out the PID so that the event can include it the
  165. # events
  166. bb.event.worker_pid = os.getpid()
  167. bb.event.worker_fire = worker_child_fire
  168. worker_pipe = pipeout
  169. worker_pipe_lock = Lock()
  170. # Make the child the process group leader and ensure no
  171. # child process will be controlled by the current terminal
  172. # This ensures signals sent to the controlling terminal like Ctrl+C
  173. # don't stop the child processes.
  174. os.setsid()
  175. signal.signal(signal.SIGTERM, sigterm_handler)
  176. # Let SIGHUP exit as SIGTERM
  177. signal.signal(signal.SIGHUP, sigterm_handler)
  178. # No stdin
  179. newsi = os.open(os.devnull, os.O_RDWR)
  180. os.dup2(newsi, sys.stdin.fileno())
  181. if umask:
  182. os.umask(umask)
  183. try:
  184. bb_cache = bb.cache.NoCache(databuilder)
  185. (realfn, virtual, mc) = bb.cache.virtualfn2realfn(fn)
  186. the_data = databuilder.mcdata[mc]
  187. the_data.setVar("BB_WORKERCONTEXT", "1")
  188. the_data.setVar("BB_TASKDEPDATA", taskdepdata)
  189. if cfg.limited_deps:
  190. the_data.setVar("BB_LIMITEDDEPS", "1")
  191. the_data.setVar("BUILDNAME", workerdata["buildname"])
  192. the_data.setVar("DATE", workerdata["date"])
  193. the_data.setVar("TIME", workerdata["time"])
  194. for varname, value in extraconfigdata.items():
  195. the_data.setVar(varname, value)
  196. bb.parse.siggen.set_taskdata(workerdata["sigdata"])
  197. ret = 0
  198. the_data = bb_cache.loadDataFull(fn, appends)
  199. the_data.setVar('BB_TASKHASH', workerdata["runq_hash"][task])
  200. bb.utils.set_process_name("%s:%s" % (the_data.getVar("PN"), taskname.replace("do_", "")))
  201. # exported_vars() returns a generator which *cannot* be passed to os.environ.update()
  202. # successfully. We also need to unset anything from the environment which shouldn't be there
  203. exports = bb.data.exported_vars(the_data)
  204. bb.utils.empty_environment()
  205. for e, v in exports:
  206. os.environ[e] = v
  207. for e in fakeenv:
  208. os.environ[e] = fakeenv[e]
  209. the_data.setVar(e, fakeenv[e])
  210. the_data.setVarFlag(e, 'export', "1")
  211. task_exports = the_data.getVarFlag(taskname, 'exports')
  212. if task_exports:
  213. for e in task_exports.split():
  214. the_data.setVarFlag(e, 'export', '1')
  215. v = the_data.getVar(e)
  216. if v is not None:
  217. os.environ[e] = v
  218. if quieterrors:
  219. the_data.setVarFlag(taskname, "quieterrors", "1")
  220. except Exception:
  221. if not quieterrors:
  222. logger.critical(traceback.format_exc())
  223. os._exit(1)
  224. try:
  225. if dry_run:
  226. return 0
  227. return bb.build.exec_task(fn, taskname, the_data, cfg.profile)
  228. except:
  229. os._exit(1)
  230. if not profiling:
  231. os._exit(child())
  232. else:
  233. profname = "profile-%s.log" % (fn.replace("/", "-") + "-" + taskname)
  234. prof = profile.Profile()
  235. try:
  236. ret = profile.Profile.runcall(prof, child)
  237. finally:
  238. prof.dump_stats(profname)
  239. bb.utils.process_profilelog(profname)
  240. os._exit(ret)
  241. else:
  242. for key, value in iter(envbackup.items()):
  243. if value is None:
  244. del os.environ[key]
  245. else:
  246. os.environ[key] = value
  247. return pid, pipein, pipeout
  248. class runQueueWorkerPipe():
  249. """
  250. Abstraction for a pipe between a worker thread and the worker server
  251. """
  252. def __init__(self, pipein, pipeout):
  253. self.input = pipein
  254. if pipeout:
  255. pipeout.close()
  256. bb.utils.nonblockingfd(self.input)
  257. self.queue = b""
  258. def read(self):
  259. start = len(self.queue)
  260. try:
  261. self.queue = self.queue + (self.input.read(102400) or b"")
  262. except (OSError, IOError) as e:
  263. if e.errno != errno.EAGAIN:
  264. raise
  265. end = len(self.queue)
  266. index = self.queue.find(b"</event>")
  267. while index != -1:
  268. worker_fire_prepickled(self.queue[:index+8])
  269. self.queue = self.queue[index+8:]
  270. index = self.queue.find(b"</event>")
  271. return (end > start)
  272. def close(self):
  273. while self.read():
  274. continue
  275. if len(self.queue) > 0:
  276. print("Warning, worker child left partial message: %s" % self.queue)
  277. self.input.close()
  278. normalexit = False
  279. class BitbakeWorker(object):
  280. def __init__(self, din):
  281. self.input = din
  282. bb.utils.nonblockingfd(self.input)
  283. self.queue = b""
  284. self.cookercfg = None
  285. self.databuilder = None
  286. self.data = None
  287. self.extraconfigdata = None
  288. self.build_pids = {}
  289. self.build_pipes = {}
  290. signal.signal(signal.SIGTERM, self.sigterm_exception)
  291. # Let SIGHUP exit as SIGTERM
  292. signal.signal(signal.SIGHUP, self.sigterm_exception)
  293. if "beef" in sys.argv[1]:
  294. bb.utils.set_process_name("Worker (Fakeroot)")
  295. else:
  296. bb.utils.set_process_name("Worker")
  297. def sigterm_exception(self, signum, stackframe):
  298. if signum == signal.SIGTERM:
  299. bb.warn("Worker received SIGTERM, shutting down...")
  300. elif signum == signal.SIGHUP:
  301. bb.warn("Worker received SIGHUP, shutting down...")
  302. self.handle_finishnow(None)
  303. signal.signal(signal.SIGTERM, signal.SIG_DFL)
  304. os.kill(os.getpid(), signal.SIGTERM)
  305. def serve(self):
  306. while True:
  307. (ready, _, _) = select.select([self.input] + [i.input for i in self.build_pipes.values()], [] , [], 1)
  308. if self.input in ready:
  309. try:
  310. r = self.input.read()
  311. if len(r) == 0:
  312. # EOF on pipe, server must have terminated
  313. self.sigterm_exception(signal.SIGTERM, None)
  314. self.queue = self.queue + r
  315. except (OSError, IOError):
  316. pass
  317. if len(self.queue):
  318. self.handle_item(b"cookerconfig", self.handle_cookercfg)
  319. self.handle_item(b"extraconfigdata", self.handle_extraconfigdata)
  320. self.handle_item(b"workerdata", self.handle_workerdata)
  321. self.handle_item(b"runtask", self.handle_runtask)
  322. self.handle_item(b"finishnow", self.handle_finishnow)
  323. self.handle_item(b"ping", self.handle_ping)
  324. self.handle_item(b"quit", self.handle_quit)
  325. for pipe in self.build_pipes:
  326. if self.build_pipes[pipe].input in ready:
  327. self.build_pipes[pipe].read()
  328. if len(self.build_pids):
  329. while self.process_waitpid():
  330. continue
  331. def handle_item(self, item, func):
  332. if self.queue.startswith(b"<" + item + b">"):
  333. index = self.queue.find(b"</" + item + b">")
  334. while index != -1:
  335. func(self.queue[(len(item) + 2):index])
  336. self.queue = self.queue[(index + len(item) + 3):]
  337. index = self.queue.find(b"</" + item + b">")
  338. def handle_cookercfg(self, data):
  339. self.cookercfg = pickle.loads(data)
  340. self.databuilder = bb.cookerdata.CookerDataBuilder(self.cookercfg, worker=True)
  341. self.databuilder.parseBaseConfiguration()
  342. self.data = self.databuilder.data
  343. def handle_extraconfigdata(self, data):
  344. self.extraconfigdata = pickle.loads(data)
  345. def handle_workerdata(self, data):
  346. self.workerdata = pickle.loads(data)
  347. bb.msg.loggerDefaultDebugLevel = self.workerdata["logdefaultdebug"]
  348. bb.msg.loggerDefaultVerbose = self.workerdata["logdefaultverbose"]
  349. bb.msg.loggerVerboseLogs = self.workerdata["logdefaultverboselogs"]
  350. bb.msg.loggerDefaultDomains = self.workerdata["logdefaultdomain"]
  351. for mc in self.databuilder.mcdata:
  352. self.databuilder.mcdata[mc].setVar("PRSERV_HOST", self.workerdata["prhost"])
  353. def handle_ping(self, _):
  354. workerlog_write("Handling ping\n")
  355. logger.warning("Pong from bitbake-worker!")
  356. def handle_quit(self, data):
  357. workerlog_write("Handling quit\n")
  358. global normalexit
  359. normalexit = True
  360. sys.exit(0)
  361. def handle_runtask(self, data):
  362. fn, task, taskname, quieterrors, appends, taskdepdata, dry_run_exec = pickle.loads(data)
  363. workerlog_write("Handling runtask %s %s %s\n" % (task, fn, taskname))
  364. pid, pipein, pipeout = fork_off_task(self.cookercfg, self.data, self.databuilder, self.workerdata, fn, task, taskname, appends, taskdepdata, self.extraconfigdata, quieterrors, dry_run_exec)
  365. self.build_pids[pid] = task
  366. self.build_pipes[pid] = runQueueWorkerPipe(pipein, pipeout)
  367. def process_waitpid(self):
  368. """
  369. Return none is there are no processes awaiting result collection, otherwise
  370. collect the process exit codes and close the information pipe.
  371. """
  372. try:
  373. pid, status = os.waitpid(-1, os.WNOHANG)
  374. if pid == 0 or os.WIFSTOPPED(status):
  375. return False
  376. except OSError:
  377. return False
  378. workerlog_write("Exit code of %s for pid %s\n" % (status, pid))
  379. if os.WIFEXITED(status):
  380. status = os.WEXITSTATUS(status)
  381. elif os.WIFSIGNALED(status):
  382. # Per shell conventions for $?, when a process exits due to
  383. # a signal, we return an exit code of 128 + SIGNUM
  384. status = 128 + os.WTERMSIG(status)
  385. task = self.build_pids[pid]
  386. del self.build_pids[pid]
  387. self.build_pipes[pid].close()
  388. del self.build_pipes[pid]
  389. worker_fire_prepickled(b"<exitcode>" + pickle.dumps((task, status)) + b"</exitcode>")
  390. return True
  391. def handle_finishnow(self, _):
  392. if self.build_pids:
  393. logger.info("Sending SIGTERM to remaining %s tasks", len(self.build_pids))
  394. for k, v in iter(self.build_pids.items()):
  395. try:
  396. os.kill(-k, signal.SIGTERM)
  397. os.waitpid(-1, 0)
  398. except:
  399. pass
  400. for pipe in self.build_pipes:
  401. self.build_pipes[pipe].read()
  402. try:
  403. worker = BitbakeWorker(os.fdopen(sys.stdin.fileno(), 'rb'))
  404. if not profiling:
  405. worker.serve()
  406. else:
  407. profname = "profile-worker.log"
  408. prof = profile.Profile()
  409. try:
  410. profile.Profile.runcall(prof, worker.serve)
  411. finally:
  412. prof.dump_stats(profname)
  413. bb.utils.process_profilelog(profname)
  414. except BaseException as e:
  415. if not normalexit:
  416. import traceback
  417. sys.stderr.write(traceback.format_exc())
  418. sys.stderr.write(str(e))
  419. worker_thread_exit = True
  420. worker_thread.join()
  421. workerlog_write("exitting")
  422. sys.exit(0)