ast.py 17 KB

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