ast.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508
  1. # ex:ts=4:sw=4:sts=4:et
  2. # -*- tab-width: 4; c-basic-offset: 4; indent-tabs-mode: nil -*-
  3. """
  4. AbstractSyntaxTree classes for the Bitbake language
  5. """
  6. # Copyright (C) 2003, 2004 Chris Larson
  7. # Copyright (C) 2003, 2004 Phil Blundell
  8. # Copyright (C) 2009 Holger Hans Peter Freyther
  9. #
  10. # This program is free software; you can redistribute it and/or modify
  11. # it under the terms of the GNU General Public License version 2 as
  12. # published by the Free Software Foundation.
  13. #
  14. # This program is distributed in the hope that it will be useful,
  15. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  16. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  17. # GNU General Public License for more details.
  18. #
  19. # You should have received a copy of the GNU General Public License along
  20. # with this program; if not, write to the Free Software Foundation, Inc.,
  21. # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  22. import re
  23. import string
  24. import logging
  25. import bb
  26. import itertools
  27. from bb import methodpool
  28. from bb.parse import logger
  29. _bbversions_re = re.compile(r"\[(?P<from>[0-9]+)-(?P<to>[0-9]+)\]")
  30. class StatementGroup(list):
  31. def eval(self, data):
  32. for statement in self:
  33. statement.eval(data)
  34. class AstNode(object):
  35. def __init__(self, filename, lineno):
  36. self.filename = filename
  37. self.lineno = lineno
  38. class IncludeNode(AstNode):
  39. def __init__(self, filename, lineno, what_file, force):
  40. AstNode.__init__(self, filename, lineno)
  41. self.what_file = what_file
  42. self.force = force
  43. def eval(self, data):
  44. """
  45. Include the file and evaluate the statements
  46. """
  47. s = data.expand(self.what_file)
  48. logger.debug(2, "CONF %s:%s: including %s", self.filename, self.lineno, s)
  49. # TODO: Cache those includes... maybe not here though
  50. if self.force:
  51. bb.parse.ConfHandler.include(self.filename, s, self.lineno, data, "include required")
  52. else:
  53. bb.parse.ConfHandler.include(self.filename, s, self.lineno, data, False)
  54. class ExportNode(AstNode):
  55. def __init__(self, filename, lineno, var):
  56. AstNode.__init__(self, filename, lineno)
  57. self.var = var
  58. def eval(self, data):
  59. data.setVarFlag(self.var, "export", 1, op = 'exported')
  60. class UnsetNode(AstNode):
  61. def __init__(self, filename, lineno, var):
  62. AstNode.__init__(self, filename, lineno)
  63. self.var = var
  64. def eval(self, data):
  65. loginfo = {
  66. 'variable': self.var,
  67. 'file': self.filename,
  68. 'line': self.lineno,
  69. }
  70. data.delVar(self.var,**loginfo)
  71. class UnsetFlagNode(AstNode):
  72. def __init__(self, filename, lineno, var, flag):
  73. AstNode.__init__(self, filename, lineno)
  74. self.var = var
  75. self.flag = flag
  76. def eval(self, data):
  77. loginfo = {
  78. 'variable': self.var,
  79. 'file': self.filename,
  80. 'line': self.lineno,
  81. }
  82. data.delVarFlag(self.var, self.flag, **loginfo)
  83. class DataNode(AstNode):
  84. """
  85. Various data related updates. For the sake of sanity
  86. we have one class doing all this. This means that all
  87. this need to be re-evaluated... we might be able to do
  88. that faster with multiple classes.
  89. """
  90. def __init__(self, filename, lineno, groupd):
  91. AstNode.__init__(self, filename, lineno)
  92. self.groupd = groupd
  93. def getFunc(self, key, data):
  94. if 'flag' in self.groupd and self.groupd['flag'] != None:
  95. return data.getVarFlag(key, self.groupd['flag'], expand=False, noweakdefault=True)
  96. else:
  97. return data.getVar(key, False, noweakdefault=True, parsing=True)
  98. def eval(self, data):
  99. groupd = self.groupd
  100. key = groupd["var"]
  101. loginfo = {
  102. 'variable': key,
  103. 'file': self.filename,
  104. 'line': self.lineno,
  105. }
  106. if "exp" in groupd and groupd["exp"] != None:
  107. data.setVarFlag(key, "export", 1, op = 'exported', **loginfo)
  108. op = "set"
  109. if "ques" in groupd and groupd["ques"] != None:
  110. val = self.getFunc(key, data)
  111. op = "set?"
  112. if val == None:
  113. val = groupd["value"]
  114. elif "colon" in groupd and groupd["colon"] != None:
  115. e = data.createCopy()
  116. bb.data.update_data(e)
  117. op = "immediate"
  118. val = e.expand(groupd["value"], key + "[:=]")
  119. elif "append" in groupd and groupd["append"] != None:
  120. op = "append"
  121. val = "%s %s" % ((self.getFunc(key, data) or ""), groupd["value"])
  122. elif "prepend" in groupd and groupd["prepend"] != None:
  123. op = "prepend"
  124. val = "%s %s" % (groupd["value"], (self.getFunc(key, data) or ""))
  125. elif "postdot" in groupd and groupd["postdot"] != None:
  126. op = "postdot"
  127. val = "%s%s" % ((self.getFunc(key, data) or ""), groupd["value"])
  128. elif "predot" in groupd and groupd["predot"] != None:
  129. op = "predot"
  130. val = "%s%s" % (groupd["value"], (self.getFunc(key, data) or ""))
  131. else:
  132. val = groupd["value"]
  133. flag = None
  134. if 'flag' in groupd and groupd['flag'] != None:
  135. flag = groupd['flag']
  136. elif groupd["lazyques"]:
  137. flag = "_defaultval"
  138. loginfo['op'] = op
  139. loginfo['detail'] = groupd["value"]
  140. if flag:
  141. data.setVarFlag(key, flag, val, **loginfo)
  142. else:
  143. data.setVar(key, val, parsing=True, **loginfo)
  144. class MethodNode(AstNode):
  145. tr_tbl = str.maketrans('/.+-@%&', '_______')
  146. def __init__(self, filename, lineno, func_name, body, python, fakeroot):
  147. AstNode.__init__(self, filename, lineno)
  148. self.func_name = func_name
  149. self.body = body
  150. self.python = python
  151. self.fakeroot = fakeroot
  152. def eval(self, data):
  153. text = '\n'.join(self.body)
  154. funcname = self.func_name
  155. if self.func_name == "__anonymous":
  156. funcname = ("__anon_%s_%s" % (self.lineno, self.filename.translate(MethodNode.tr_tbl)))
  157. self.python = True
  158. text = "def %s(d):\n" % (funcname) + text
  159. bb.methodpool.insert_method(funcname, text, self.filename, self.lineno - len(self.body))
  160. anonfuncs = data.getVar('__BBANONFUNCS', False) or []
  161. anonfuncs.append(funcname)
  162. data.setVar('__BBANONFUNCS', anonfuncs)
  163. if data.getVar(funcname, False):
  164. # clean up old version of this piece of metadata, as its
  165. # flags could cause problems
  166. data.delVarFlag(funcname, 'python')
  167. data.delVarFlag(funcname, 'fakeroot')
  168. if self.python:
  169. data.setVarFlag(funcname, "python", "1")
  170. if self.fakeroot:
  171. data.setVarFlag(funcname, "fakeroot", "1")
  172. data.setVarFlag(funcname, "func", 1)
  173. data.setVar(funcname, text, parsing=True)
  174. data.setVarFlag(funcname, 'filename', self.filename)
  175. data.setVarFlag(funcname, 'lineno', str(self.lineno - len(self.body)))
  176. class PythonMethodNode(AstNode):
  177. def __init__(self, filename, lineno, function, modulename, body):
  178. AstNode.__init__(self, filename, lineno)
  179. self.function = function
  180. self.modulename = modulename
  181. self.body = body
  182. def eval(self, data):
  183. # Note we will add root to parsedmethods after having parse
  184. # 'this' file. This means we will not parse methods from
  185. # bb classes twice
  186. text = '\n'.join(self.body)
  187. bb.methodpool.insert_method(self.modulename, text, self.filename, self.lineno - len(self.body) - 1)
  188. data.setVarFlag(self.function, "func", 1)
  189. data.setVarFlag(self.function, "python", 1)
  190. data.setVar(self.function, text, parsing=True)
  191. data.setVarFlag(self.function, 'filename', self.filename)
  192. data.setVarFlag(self.function, 'lineno', str(self.lineno - len(self.body) - 1))
  193. class ExportFuncsNode(AstNode):
  194. def __init__(self, filename, lineno, fns, classname):
  195. AstNode.__init__(self, filename, lineno)
  196. self.n = fns.split()
  197. self.classname = classname
  198. def eval(self, data):
  199. for func in self.n:
  200. calledfunc = self.classname + "_" + func
  201. if data.getVar(func, False) and not data.getVarFlag(func, 'export_func', False):
  202. continue
  203. if data.getVar(func, False):
  204. data.setVarFlag(func, 'python', None)
  205. data.setVarFlag(func, 'func', None)
  206. for flag in [ "func", "python" ]:
  207. if data.getVarFlag(calledfunc, flag, False):
  208. data.setVarFlag(func, flag, data.getVarFlag(calledfunc, flag, False))
  209. for flag in [ "dirs" ]:
  210. if data.getVarFlag(func, flag, False):
  211. data.setVarFlag(calledfunc, flag, data.getVarFlag(func, flag, False))
  212. data.setVarFlag(func, "filename", "autogenerated")
  213. data.setVarFlag(func, "lineno", 1)
  214. if data.getVarFlag(calledfunc, "python", False):
  215. data.setVar(func, " bb.build.exec_func('" + calledfunc + "', d)\n", parsing=True)
  216. else:
  217. if "-" in self.classname:
  218. bb.fatal("The classname %s contains a dash character and is calling an sh function %s using EXPORT_FUNCTIONS. Since a dash is illegal in sh function names, this cannot work, please rename the class or don't use EXPORT_FUNCTIONS." % (self.classname, calledfunc))
  219. data.setVar(func, " " + calledfunc + "\n", parsing=True)
  220. data.setVarFlag(func, 'export_func', '1')
  221. class AddTaskNode(AstNode):
  222. def __init__(self, filename, lineno, func, before, after):
  223. AstNode.__init__(self, filename, lineno)
  224. self.func = func
  225. self.before = before
  226. self.after = after
  227. def eval(self, data):
  228. bb.build.addtask(self.func, self.before, self.after, data)
  229. class DelTaskNode(AstNode):
  230. def __init__(self, filename, lineno, func):
  231. AstNode.__init__(self, filename, lineno)
  232. self.func = func
  233. def eval(self, data):
  234. bb.build.deltask(self.func, data)
  235. class BBHandlerNode(AstNode):
  236. def __init__(self, filename, lineno, fns):
  237. AstNode.__init__(self, filename, lineno)
  238. self.hs = fns.split()
  239. def eval(self, data):
  240. bbhands = data.getVar('__BBHANDLERS', False) or []
  241. for h in self.hs:
  242. bbhands.append(h)
  243. data.setVarFlag(h, "handler", 1)
  244. data.setVar('__BBHANDLERS', bbhands)
  245. class InheritNode(AstNode):
  246. def __init__(self, filename, lineno, classes):
  247. AstNode.__init__(self, filename, lineno)
  248. self.classes = classes
  249. def eval(self, data):
  250. bb.parse.BBHandler.inherit(self.classes, self.filename, self.lineno, data)
  251. def handleInclude(statements, filename, lineno, m, force):
  252. statements.append(IncludeNode(filename, lineno, m.group(1), force))
  253. def handleExport(statements, filename, lineno, m):
  254. statements.append(ExportNode(filename, lineno, m.group(1)))
  255. def handleUnset(statements, filename, lineno, m):
  256. statements.append(UnsetNode(filename, lineno, m.group(1)))
  257. def handleUnsetFlag(statements, filename, lineno, m):
  258. statements.append(UnsetFlagNode(filename, lineno, m.group(1), m.group(2)))
  259. def handleData(statements, filename, lineno, groupd):
  260. statements.append(DataNode(filename, lineno, groupd))
  261. def handleMethod(statements, filename, lineno, func_name, body, python, fakeroot):
  262. statements.append(MethodNode(filename, lineno, func_name, body, python, fakeroot))
  263. def handlePythonMethod(statements, filename, lineno, funcname, modulename, body):
  264. statements.append(PythonMethodNode(filename, lineno, funcname, modulename, body))
  265. def handleExportFuncs(statements, filename, lineno, m, classname):
  266. statements.append(ExportFuncsNode(filename, lineno, m.group(1), classname))
  267. def handleAddTask(statements, filename, lineno, m):
  268. func = m.group("func")
  269. before = m.group("before")
  270. after = m.group("after")
  271. if func is None:
  272. return
  273. statements.append(AddTaskNode(filename, lineno, func, before, after))
  274. def handleDelTask(statements, filename, lineno, m):
  275. func = m.group("func")
  276. if func is None:
  277. return
  278. statements.append(DelTaskNode(filename, lineno, func))
  279. def handleBBHandlers(statements, filename, lineno, m):
  280. statements.append(BBHandlerNode(filename, lineno, m.group(1)))
  281. def handleInherit(statements, filename, lineno, m):
  282. classes = m.group(1)
  283. statements.append(InheritNode(filename, lineno, classes))
  284. def finalize(fn, d, variant = None):
  285. saved_handlers = bb.event.get_handlers().copy()
  286. for var in d.getVar('__BBHANDLERS', False) or []:
  287. # try to add the handler
  288. handlerfn = d.getVarFlag(var, "filename", False)
  289. if not handlerfn:
  290. bb.fatal("Undefined event handler function '%s'" % var)
  291. handlerln = int(d.getVarFlag(var, "lineno", False))
  292. bb.event.register(var, d.getVar(var, False), (d.getVarFlag(var, "eventmask", True) or "").split(), handlerfn, handlerln)
  293. bb.event.fire(bb.event.RecipePreFinalise(fn), d)
  294. bb.data.expandKeys(d)
  295. bb.data.update_data(d)
  296. code = []
  297. for funcname in d.getVar("__BBANONFUNCS", False) or []:
  298. code.append("%s(d)" % funcname)
  299. bb.utils.better_exec("\n".join(code), {"d": d})
  300. bb.data.update_data(d)
  301. tasklist = d.getVar('__BBTASKS', False) or []
  302. bb.build.add_tasks(tasklist, d)
  303. bb.parse.siggen.finalise(fn, d, variant)
  304. d.setVar('BBINCLUDED', bb.parse.get_file_depends(d))
  305. bb.event.fire(bb.event.RecipeParsed(fn), d)
  306. bb.event.set_handlers(saved_handlers)
  307. def _create_variants(datastores, names, function, onlyfinalise):
  308. def create_variant(name, orig_d, arg = None):
  309. if onlyfinalise and name not in onlyfinalise:
  310. return
  311. new_d = bb.data.createCopy(orig_d)
  312. function(arg or name, new_d)
  313. datastores[name] = new_d
  314. for variant in list(datastores.keys()):
  315. for name in names:
  316. if not variant:
  317. # Based on main recipe
  318. create_variant(name, datastores[""])
  319. else:
  320. create_variant("%s-%s" % (variant, name), datastores[variant], name)
  321. def _expand_versions(versions):
  322. def expand_one(version, start, end):
  323. for i in range(start, end + 1):
  324. ver = _bbversions_re.sub(str(i), version, 1)
  325. yield ver
  326. versions = iter(versions)
  327. while True:
  328. try:
  329. version = next(versions)
  330. except StopIteration:
  331. break
  332. range_ver = _bbversions_re.search(version)
  333. if not range_ver:
  334. yield version
  335. else:
  336. newversions = expand_one(version, int(range_ver.group("from")),
  337. int(range_ver.group("to")))
  338. versions = itertools.chain(newversions, versions)
  339. def multi_finalize(fn, d):
  340. appends = (d.getVar("__BBAPPEND") or "").split()
  341. for append in appends:
  342. logger.debug(1, "Appending .bbappend file %s to %s", append, fn)
  343. bb.parse.BBHandler.handle(append, d, True)
  344. onlyfinalise = d.getVar("__ONLYFINALISE", False)
  345. safe_d = d
  346. d = bb.data.createCopy(safe_d)
  347. try:
  348. finalize(fn, d)
  349. except bb.parse.SkipRecipe as e:
  350. d.setVar("__SKIPPED", e.args[0])
  351. datastores = {"": safe_d}
  352. versions = (d.getVar("BBVERSIONS") or "").split()
  353. if versions:
  354. pv = orig_pv = d.getVar("PV")
  355. baseversions = {}
  356. def verfunc(ver, d, pv_d = None):
  357. if pv_d is None:
  358. pv_d = d
  359. overrides = d.getVar("OVERRIDES").split(":")
  360. pv_d.setVar("PV", ver)
  361. overrides.append(ver)
  362. bpv = baseversions.get(ver) or orig_pv
  363. pv_d.setVar("BPV", bpv)
  364. overrides.append(bpv)
  365. d.setVar("OVERRIDES", ":".join(overrides))
  366. versions = list(_expand_versions(versions))
  367. for pos, version in enumerate(list(versions)):
  368. try:
  369. pv, bpv = version.split(":", 2)
  370. except ValueError:
  371. pass
  372. else:
  373. versions[pos] = pv
  374. baseversions[pv] = bpv
  375. if pv in versions and not baseversions.get(pv):
  376. versions.remove(pv)
  377. else:
  378. pv = versions.pop()
  379. # This is necessary because our existing main datastore
  380. # has already been finalized with the old PV, we need one
  381. # that's been finalized with the new PV.
  382. d = bb.data.createCopy(safe_d)
  383. verfunc(pv, d, safe_d)
  384. try:
  385. finalize(fn, d)
  386. except bb.parse.SkipRecipe as e:
  387. d.setVar("__SKIPPED", e.args[0])
  388. _create_variants(datastores, versions, verfunc, onlyfinalise)
  389. extended = d.getVar("BBCLASSEXTEND") or ""
  390. if extended:
  391. # the following is to support bbextends with arguments, for e.g. multilib
  392. # an example is as follows:
  393. # BBCLASSEXTEND = "multilib:lib32"
  394. # it will create foo-lib32, inheriting multilib.bbclass and set
  395. # BBEXTENDCURR to "multilib" and BBEXTENDVARIANT to "lib32"
  396. extendedmap = {}
  397. variantmap = {}
  398. for ext in extended.split():
  399. eext = ext.split(':', 2)
  400. if len(eext) > 1:
  401. extendedmap[ext] = eext[0]
  402. variantmap[ext] = eext[1]
  403. else:
  404. extendedmap[ext] = ext
  405. pn = d.getVar("PN")
  406. def extendfunc(name, d):
  407. if name != extendedmap[name]:
  408. d.setVar("BBEXTENDCURR", extendedmap[name])
  409. d.setVar("BBEXTENDVARIANT", variantmap[name])
  410. else:
  411. d.setVar("PN", "%s-%s" % (pn, name))
  412. bb.parse.BBHandler.inherit(extendedmap[name], fn, 0, d)
  413. safe_d.setVar("BBCLASSEXTEND", extended)
  414. _create_variants(datastores, extendedmap.keys(), extendfunc, onlyfinalise)
  415. for variant in datastores.keys():
  416. if variant:
  417. try:
  418. if not onlyfinalise or variant in onlyfinalise:
  419. finalize(fn, datastores[variant], variant)
  420. except bb.parse.SkipRecipe as e:
  421. datastores[variant].setVar("__SKIPPED", e.args[0])
  422. datastores[""] = d
  423. return datastores