cache.py 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474
  1. # ex:ts=4:sw=4:sts=4:et
  2. # -*- tab-width: 4; c-basic-offset: 4; indent-tabs-mode: nil -*-
  3. #
  4. # BitBake 'Event' implementation
  5. #
  6. # Caching of bitbake variables before task execution
  7. # Copyright (C) 2006 Richard Purdie
  8. # but small sections based on code from bin/bitbake:
  9. # Copyright (C) 2003, 2004 Chris Larson
  10. # Copyright (C) 2003, 2004 Phil Blundell
  11. # Copyright (C) 2003 - 2005 Michael 'Mickey' Lauer
  12. # Copyright (C) 2005 Holger Hans Peter Freyther
  13. # Copyright (C) 2005 ROAD GmbH
  14. #
  15. # This program is free software; you can redistribute it and/or modify
  16. # it under the terms of the GNU General Public License version 2 as
  17. # published by the Free Software Foundation.
  18. #
  19. # This program is distributed in the hope that it will be useful,
  20. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  21. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  22. # GNU General Public License for more details.
  23. #
  24. # You should have received a copy of the GNU General Public License along
  25. # with this program; if not, write to the Free Software Foundation, Inc.,
  26. # 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
  27. import os, re
  28. import bb.data
  29. import bb.utils
  30. from sets import Set
  31. try:
  32. import cPickle as pickle
  33. except ImportError:
  34. import pickle
  35. bb.msg.note(1, bb.msg.domain.Cache, "Importing cPickle failed. Falling back to a very slow implementation.")
  36. __cache_version__ = "129"
  37. class Cache:
  38. """
  39. BitBake Cache implementation
  40. """
  41. def __init__(self, cooker):
  42. self.cachedir = bb.data.getVar("CACHE", cooker.configuration.data, True)
  43. self.clean = {}
  44. self.checked = {}
  45. self.depends_cache = {}
  46. self.data = None
  47. self.data_fn = None
  48. self.cacheclean = True
  49. if self.cachedir in [None, '']:
  50. self.has_cache = False
  51. bb.msg.note(1, bb.msg.domain.Cache, "Not using a cache. Set CACHE = <directory> to enable.")
  52. else:
  53. self.has_cache = True
  54. self.cachefile = os.path.join(self.cachedir,"bb_cache.dat")
  55. bb.msg.debug(1, bb.msg.domain.Cache, "Using cache in '%s'" % self.cachedir)
  56. try:
  57. os.stat( self.cachedir )
  58. except OSError:
  59. bb.mkdirhier( self.cachedir )
  60. if not self.has_cache:
  61. return
  62. # If any of configuration.data's dependencies are newer than the
  63. # cache there isn't even any point in loading it...
  64. newest_mtime = 0
  65. deps = bb.data.getVar("__depends", cooker.configuration.data, True)
  66. for f,old_mtime in deps:
  67. if old_mtime > newest_mtime:
  68. newest_mtime = old_mtime
  69. if bb.parse.cached_mtime_noerror(self.cachefile) >= newest_mtime:
  70. try:
  71. p = pickle.Unpickler(file(self.cachefile, "rb"))
  72. self.depends_cache, version_data = p.load()
  73. if version_data['CACHE_VER'] != __cache_version__:
  74. raise ValueError, 'Cache Version Mismatch'
  75. if version_data['BITBAKE_VER'] != bb.__version__:
  76. raise ValueError, 'Bitbake Version Mismatch'
  77. except EOFError:
  78. bb.msg.note(1, bb.msg.domain.Cache, "Truncated cache found, rebuilding...")
  79. self.depends_cache = {}
  80. except:
  81. bb.msg.note(1, bb.msg.domain.Cache, "Invalid cache found, rebuilding...")
  82. self.depends_cache = {}
  83. else:
  84. bb.msg.note(1, bb.msg.domain.Cache, "Out of date cache found, rebuilding...")
  85. def getVar(self, var, fn, exp = 0):
  86. """
  87. Gets the value of a variable
  88. (similar to getVar in the data class)
  89. There are two scenarios:
  90. 1. We have cached data - serve from depends_cache[fn]
  91. 2. We're learning what data to cache - serve from data
  92. backend but add a copy of the data to the cache.
  93. """
  94. if fn in self.clean:
  95. return self.depends_cache[fn][var]
  96. if not fn in self.depends_cache:
  97. self.depends_cache[fn] = {}
  98. if fn != self.data_fn:
  99. # We're trying to access data in the cache which doesn't exist
  100. # yet setData hasn't been called to setup the right access. Very bad.
  101. bb.msg.error(bb.msg.domain.Cache, "Parsing error data_fn %s and fn %s don't match" % (self.data_fn, fn))
  102. self.cacheclean = False
  103. result = bb.data.getVar(var, self.data, exp)
  104. self.depends_cache[fn][var] = result
  105. return result
  106. def setData(self, fn, data):
  107. """
  108. Called to prime bb_cache ready to learn which variables to cache.
  109. Will be followed by calls to self.getVar which aren't cached
  110. but can be fulfilled from self.data.
  111. """
  112. self.data_fn = fn
  113. self.data = data
  114. # Make sure __depends makes the depends_cache
  115. self.getVar("__depends", fn, True)
  116. self.depends_cache[fn]["CACHETIMESTAMP"] = bb.parse.cached_mtime(fn)
  117. def loadDataFull(self, fn, cfgData):
  118. """
  119. Return a complete set of data for fn.
  120. To do this, we need to parse the file.
  121. """
  122. bb.msg.debug(1, bb.msg.domain.Cache, "Parsing %s (full)" % fn)
  123. bb_data, skipped = self.load_bbfile(fn, cfgData)
  124. return bb_data
  125. def loadData(self, fn, cfgData):
  126. """
  127. Load a subset of data for fn.
  128. If the cached data is valid we do nothing,
  129. To do this, we need to parse the file and set the system
  130. to record the variables accessed.
  131. Return the cache status and whether the file was skipped when parsed
  132. """
  133. if fn not in self.checked:
  134. self.cacheValidUpdate(fn)
  135. if self.cacheValid(fn):
  136. if "SKIPPED" in self.depends_cache[fn]:
  137. return True, True
  138. return True, False
  139. bb.msg.debug(1, bb.msg.domain.Cache, "Parsing %s" % fn)
  140. bb_data, skipped = self.load_bbfile(fn, cfgData)
  141. self.setData(fn, bb_data)
  142. return False, skipped
  143. def cacheValid(self, fn):
  144. """
  145. Is the cache valid for fn?
  146. Fast version, no timestamps checked.
  147. """
  148. # Is cache enabled?
  149. if not self.has_cache:
  150. return False
  151. if fn in self.clean:
  152. return True
  153. return False
  154. def cacheValidUpdate(self, fn):
  155. """
  156. Is the cache valid for fn?
  157. Make thorough (slower) checks including timestamps.
  158. """
  159. # Is cache enabled?
  160. if not self.has_cache:
  161. return False
  162. self.checked[fn] = ""
  163. # Pretend we're clean so getVar works
  164. self.clean[fn] = ""
  165. # File isn't in depends_cache
  166. if not fn in self.depends_cache:
  167. bb.msg.debug(2, bb.msg.domain.Cache, "Cache: %s is not cached" % fn)
  168. self.remove(fn)
  169. return False
  170. mtime = bb.parse.cached_mtime_noerror(fn)
  171. # Check file still exists
  172. if mtime == 0:
  173. bb.msg.debug(2, bb.msg.domain.Cache, "Cache: %s not longer exists" % fn)
  174. self.remove(fn)
  175. return False
  176. # Check the file's timestamp
  177. if mtime != self.getVar("CACHETIMESTAMP", fn, True):
  178. bb.msg.debug(2, bb.msg.domain.Cache, "Cache: %s changed" % fn)
  179. self.remove(fn)
  180. return False
  181. # Check dependencies are still valid
  182. depends = self.getVar("__depends", fn, True)
  183. if depends:
  184. for f,old_mtime in depends:
  185. fmtime = bb.parse.cached_mtime_noerror(f)
  186. # Check if file still exists
  187. if fmtime == 0:
  188. self.remove(fn)
  189. return False
  190. if (fmtime != old_mtime):
  191. bb.msg.debug(2, bb.msg.domain.Cache, "Cache: %s's dependency %s changed" % (fn, f))
  192. self.remove(fn)
  193. return False
  194. #bb.msg.debug(2, bb.msg.domain.Cache, "Depends Cache: %s is clean" % fn)
  195. if not fn in self.clean:
  196. self.clean[fn] = ""
  197. return True
  198. def skip(self, fn):
  199. """
  200. Mark a fn as skipped
  201. Called from the parser
  202. """
  203. if not fn in self.depends_cache:
  204. self.depends_cache[fn] = {}
  205. self.depends_cache[fn]["SKIPPED"] = "1"
  206. def remove(self, fn):
  207. """
  208. Remove a fn from the cache
  209. Called from the parser in error cases
  210. """
  211. bb.msg.debug(1, bb.msg.domain.Cache, "Removing %s from cache" % fn)
  212. if fn in self.depends_cache:
  213. del self.depends_cache[fn]
  214. if fn in self.clean:
  215. del self.clean[fn]
  216. def sync(self):
  217. """
  218. Save the cache
  219. Called from the parser when complete (or exiting)
  220. """
  221. import copy
  222. if not self.has_cache:
  223. return
  224. if self.cacheclean:
  225. bb.msg.note(1, bb.msg.domain.Cache, "Cache is clean, not saving.")
  226. return
  227. version_data = {}
  228. version_data['CACHE_VER'] = __cache_version__
  229. version_data['BITBAKE_VER'] = bb.__version__
  230. cache_data = copy.deepcopy(self.depends_cache)
  231. for fn in self.depends_cache:
  232. if '__BB_DONT_CACHE' in self.depends_cache[fn] and self.depends_cache[fn]['__BB_DONT_CACHE']:
  233. bb.msg.debug(2, bb.msg.domain.Cache, "Not caching %s, marked as not cacheable" % fn)
  234. del cache_data[fn]
  235. p = pickle.Pickler(file(self.cachefile, "wb" ), -1 )
  236. p.dump([cache_data, version_data])
  237. def mtime(self, cachefile):
  238. return bb.parse.cached_mtime_noerror(cachefile)
  239. def handle_data(self, file_name, cacheData):
  240. """
  241. Save data we need into the cache
  242. """
  243. pn = self.getVar('PN', file_name, True)
  244. pe = self.getVar('PE', file_name, True) or "0"
  245. pv = self.getVar('PV', file_name, True)
  246. pr = self.getVar('PR', file_name, True)
  247. dp = int(self.getVar('DEFAULT_PREFERENCE', file_name, True) or "0")
  248. depends = bb.utils.explode_deps(self.getVar("DEPENDS", file_name, True) or "")
  249. packages = (self.getVar('PACKAGES', file_name, True) or "").split()
  250. packages_dynamic = (self.getVar('PACKAGES_DYNAMIC', file_name, True) or "").split()
  251. rprovides = (self.getVar("RPROVIDES", file_name, True) or "").split()
  252. cacheData.task_deps[file_name] = self.getVar("_task_deps", file_name, True)
  253. # build PackageName to FileName lookup table
  254. if pn not in cacheData.pkg_pn:
  255. cacheData.pkg_pn[pn] = []
  256. cacheData.pkg_pn[pn].append(file_name)
  257. cacheData.stamp[file_name] = self.getVar('STAMP', file_name, True)
  258. # build FileName to PackageName lookup table
  259. cacheData.pkg_fn[file_name] = pn
  260. cacheData.pkg_pepvpr[file_name] = (pe,pv,pr)
  261. cacheData.pkg_dp[file_name] = dp
  262. provides = [pn]
  263. for provide in (self.getVar("PROVIDES", file_name, True) or "").split():
  264. if provide not in provides:
  265. provides.append(provide)
  266. # Build forward and reverse provider hashes
  267. # Forward: virtual -> [filenames]
  268. # Reverse: PN -> [virtuals]
  269. if pn not in cacheData.pn_provides:
  270. cacheData.pn_provides[pn] = []
  271. cacheData.fn_provides[file_name] = provides
  272. for provide in provides:
  273. if provide not in cacheData.providers:
  274. cacheData.providers[provide] = []
  275. cacheData.providers[provide].append(file_name)
  276. if not provide in cacheData.pn_provides[pn]:
  277. cacheData.pn_provides[pn].append(provide)
  278. cacheData.deps[file_name] = []
  279. for dep in depends:
  280. if not dep in cacheData.deps[file_name]:
  281. cacheData.deps[file_name].append(dep)
  282. if not dep in cacheData.all_depends:
  283. cacheData.all_depends.append(dep)
  284. # Build reverse hash for PACKAGES, so runtime dependencies
  285. # can be be resolved (RDEPENDS, RRECOMMENDS etc.)
  286. for package in packages:
  287. if not package in cacheData.packages:
  288. cacheData.packages[package] = []
  289. cacheData.packages[package].append(file_name)
  290. rprovides += (self.getVar("RPROVIDES_%s" % package, file_name, 1) or "").split()
  291. for package in packages_dynamic:
  292. if not package in cacheData.packages_dynamic:
  293. cacheData.packages_dynamic[package] = []
  294. cacheData.packages_dynamic[package].append(file_name)
  295. for rprovide in rprovides:
  296. if not rprovide in cacheData.rproviders:
  297. cacheData.rproviders[rprovide] = []
  298. cacheData.rproviders[rprovide].append(file_name)
  299. # Build hash of runtime depends and rececommends
  300. if not file_name in cacheData.rundeps:
  301. cacheData.rundeps[file_name] = {}
  302. if not file_name in cacheData.runrecs:
  303. cacheData.runrecs[file_name] = {}
  304. rdepends = self.getVar('RDEPENDS', file_name, True) or ""
  305. rrecommends = self.getVar('RRECOMMENDS', file_name, True) or ""
  306. for package in packages + [pn]:
  307. if not package in cacheData.rundeps[file_name]:
  308. cacheData.rundeps[file_name][package] = []
  309. if not package in cacheData.runrecs[file_name]:
  310. cacheData.runrecs[file_name][package] = []
  311. cacheData.rundeps[file_name][package] = rdepends + " " + (self.getVar("RDEPENDS_%s" % package, file_name, True) or "")
  312. cacheData.runrecs[file_name][package] = rrecommends + " " + (self.getVar("RRECOMMENDS_%s" % package, file_name, True) or "")
  313. # Collect files we may need for possible world-dep
  314. # calculations
  315. if not self.getVar('BROKEN', file_name, True) and not self.getVar('EXCLUDE_FROM_WORLD', file_name, True):
  316. cacheData.possible_world.append(file_name)
  317. # Touch this to make sure its in the cache
  318. self.getVar('__BB_DONT_CACHE', file_name, True)
  319. def load_bbfile( self, bbfile , config):
  320. """
  321. Load and parse one .bb build file
  322. Return the data and whether parsing resulted in the file being skipped
  323. """
  324. import bb
  325. from bb import utils, data, parse, debug, event, fatal
  326. # expand tmpdir to include this topdir
  327. data.setVar('TMPDIR', data.getVar('TMPDIR', config, 1) or "", config)
  328. bbfile_loc = os.path.abspath(os.path.dirname(bbfile))
  329. oldpath = os.path.abspath(os.getcwd())
  330. if bb.parse.cached_mtime_noerror(bbfile_loc):
  331. os.chdir(bbfile_loc)
  332. bb_data = data.init_db(config)
  333. try:
  334. bb_data = parse.handle(bbfile, bb_data) # read .bb data
  335. os.chdir(oldpath)
  336. return bb_data, False
  337. except bb.parse.SkipPackage:
  338. os.chdir(oldpath)
  339. return bb_data, True
  340. except:
  341. os.chdir(oldpath)
  342. raise
  343. def init(cooker):
  344. """
  345. The Objective: Cache the minimum amount of data possible yet get to the
  346. stage of building packages (i.e. tryBuild) without reparsing any .bb files.
  347. To do this, we intercept getVar calls and only cache the variables we see
  348. being accessed. We rely on the cache getVar calls being made for all
  349. variables bitbake might need to use to reach this stage. For each cached
  350. file we need to track:
  351. * Its mtime
  352. * The mtimes of all its dependencies
  353. * Whether it caused a parse.SkipPackage exception
  354. Files causing parsing errors are evicted from the cache.
  355. """
  356. return Cache(cooker)
  357. #============================================================================#
  358. # CacheData
  359. #============================================================================#
  360. class CacheData:
  361. """
  362. The data structures we compile from the cached data
  363. """
  364. def __init__(self):
  365. """
  366. Direct cache variables
  367. (from Cache.handle_data)
  368. """
  369. self.providers = {}
  370. self.rproviders = {}
  371. self.packages = {}
  372. self.packages_dynamic = {}
  373. self.possible_world = []
  374. self.pkg_pn = {}
  375. self.pkg_fn = {}
  376. self.pkg_pepvpr = {}
  377. self.pkg_dp = {}
  378. self.pn_provides = {}
  379. self.fn_provides = {}
  380. self.all_depends = []
  381. self.deps = {}
  382. self.rundeps = {}
  383. self.runrecs = {}
  384. self.task_queues = {}
  385. self.task_deps = {}
  386. self.stamp = {}
  387. self.preferred = {}
  388. """
  389. Indirect Cache variables
  390. (set elsewhere)
  391. """
  392. self.ignored_dependencies = []
  393. self.world_target = Set()
  394. self.bbfile_priority = {}
  395. self.bbfile_config_priorities = []