cache.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463
  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__ = "128"
  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 self.mtime(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 (ValueError, KeyError):
  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. # Check file still exists
  171. if self.mtime(fn) == 0:
  172. bb.msg.debug(2, bb.msg.domain.Cache, "Cache: %s not longer exists" % fn)
  173. self.remove(fn)
  174. return False
  175. # Check the file's timestamp
  176. if bb.parse.cached_mtime(fn) > self.getVar("CACHETIMESTAMP", fn, True):
  177. bb.msg.debug(2, bb.msg.domain.Cache, "Cache: %s changed" % fn)
  178. self.remove(fn)
  179. return False
  180. # Check dependencies are still valid
  181. depends = self.getVar("__depends", fn, True)
  182. for f,old_mtime in depends:
  183. # Check if file still exists
  184. if self.mtime(f) == 0:
  185. self.remove(fn)
  186. return False
  187. new_mtime = bb.parse.cached_mtime(f)
  188. if (new_mtime > old_mtime):
  189. bb.msg.debug(2, bb.msg.domain.Cache, "Cache: %s's dependency %s changed" % (fn, f))
  190. self.remove(fn)
  191. return False
  192. #bb.msg.debug(2, bb.msg.domain.Cache, "Depends Cache: %s is clean" % fn)
  193. if not fn in self.clean:
  194. self.clean[fn] = ""
  195. return True
  196. def skip(self, fn):
  197. """
  198. Mark a fn as skipped
  199. Called from the parser
  200. """
  201. if not fn in self.depends_cache:
  202. self.depends_cache[fn] = {}
  203. self.depends_cache[fn]["SKIPPED"] = "1"
  204. def remove(self, fn):
  205. """
  206. Remove a fn from the cache
  207. Called from the parser in error cases
  208. """
  209. bb.msg.debug(1, bb.msg.domain.Cache, "Removing %s from cache" % fn)
  210. if fn in self.depends_cache:
  211. del self.depends_cache[fn]
  212. if fn in self.clean:
  213. del self.clean[fn]
  214. def sync(self):
  215. """
  216. Save the cache
  217. Called from the parser when complete (or exiting)
  218. """
  219. if not self.has_cache:
  220. return
  221. if self.cacheclean:
  222. bb.msg.note(1, bb.msg.domain.Cache, "Cache is clean, not saving.")
  223. return
  224. version_data = {}
  225. version_data['CACHE_VER'] = __cache_version__
  226. version_data['BITBAKE_VER'] = bb.__version__
  227. p = pickle.Pickler(file(self.cachefile, "wb" ), -1 )
  228. p.dump([self.depends_cache, version_data])
  229. def mtime(self, cachefile):
  230. return bb.parse.cached_mtime_noerror(cachefile)
  231. def handle_data(self, file_name, cacheData):
  232. """
  233. Save data we need into the cache
  234. """
  235. pn = self.getVar('PN', file_name, True)
  236. pe = self.getVar('PE', file_name, True) or "0"
  237. pv = self.getVar('PV', file_name, True)
  238. pr = self.getVar('PR', file_name, True)
  239. dp = int(self.getVar('DEFAULT_PREFERENCE', file_name, True) or "0")
  240. provides = Set([pn] + (self.getVar("PROVIDES", file_name, True) or "").split())
  241. depends = bb.utils.explode_deps(self.getVar("DEPENDS", file_name, True) or "")
  242. packages = (self.getVar('PACKAGES', file_name, True) or "").split()
  243. packages_dynamic = (self.getVar('PACKAGES_DYNAMIC', file_name, True) or "").split()
  244. rprovides = (self.getVar("RPROVIDES", file_name, True) or "").split()
  245. cacheData.task_deps[file_name] = self.getVar("_task_deps", file_name, True)
  246. # build PackageName to FileName lookup table
  247. if pn not in cacheData.pkg_pn:
  248. cacheData.pkg_pn[pn] = []
  249. cacheData.pkg_pn[pn].append(file_name)
  250. cacheData.stamp[file_name] = self.getVar('STAMP', file_name, True)
  251. # build FileName to PackageName lookup table
  252. cacheData.pkg_fn[file_name] = pn
  253. cacheData.pkg_pepvpr[file_name] = (pe,pv,pr)
  254. cacheData.pkg_dp[file_name] = dp
  255. # Build forward and reverse provider hashes
  256. # Forward: virtual -> [filenames]
  257. # Reverse: PN -> [virtuals]
  258. if pn not in cacheData.pn_provides:
  259. cacheData.pn_provides[pn] = Set()
  260. cacheData.pn_provides[pn] |= provides
  261. cacheData.fn_provides[file_name] = Set()
  262. for provide in provides:
  263. if provide not in cacheData.providers:
  264. cacheData.providers[provide] = []
  265. cacheData.providers[provide].append(file_name)
  266. cacheData.fn_provides[file_name].add(provide)
  267. cacheData.deps[file_name] = Set()
  268. for dep in depends:
  269. cacheData.all_depends.add(dep)
  270. cacheData.deps[file_name].add(dep)
  271. # Build reverse hash for PACKAGES, so runtime dependencies
  272. # can be be resolved (RDEPENDS, RRECOMMENDS etc.)
  273. for package in packages:
  274. if not package in cacheData.packages:
  275. cacheData.packages[package] = []
  276. cacheData.packages[package].append(file_name)
  277. rprovides += (self.getVar("RPROVIDES_%s" % package, file_name, 1) or "").split()
  278. for package in packages_dynamic:
  279. if not package in cacheData.packages_dynamic:
  280. cacheData.packages_dynamic[package] = []
  281. cacheData.packages_dynamic[package].append(file_name)
  282. for rprovide in rprovides:
  283. if not rprovide in cacheData.rproviders:
  284. cacheData.rproviders[rprovide] = []
  285. cacheData.rproviders[rprovide].append(file_name)
  286. # Build hash of runtime depends and rececommends
  287. def add_dep(deplist, deps):
  288. for dep in deps:
  289. if not dep in deplist:
  290. deplist[dep] = ""
  291. if not file_name in cacheData.rundeps:
  292. cacheData.rundeps[file_name] = {}
  293. if not file_name in cacheData.runrecs:
  294. cacheData.runrecs[file_name] = {}
  295. rdepends = bb.utils.explode_deps(self.getVar('RDEPENDS', file_name, True) or "")
  296. rrecommends = bb.utils.explode_deps(self.getVar('RRECOMMENDS', file_name, True) or "")
  297. for package in packages + [pn]:
  298. if not package in cacheData.rundeps[file_name]:
  299. cacheData.rundeps[file_name][package] = {}
  300. if not package in cacheData.runrecs[file_name]:
  301. cacheData.runrecs[file_name][package] = {}
  302. add_dep(cacheData.rundeps[file_name][package], rdepends)
  303. add_dep(cacheData.runrecs[file_name][package], rrecommends)
  304. add_dep(cacheData.rundeps[file_name][package], bb.utils.explode_deps(self.getVar("RDEPENDS_%s" % package, file_name, True) or ""))
  305. add_dep(cacheData.runrecs[file_name][package], bb.utils.explode_deps(self.getVar("RRECOMMENDS_%s" % package, file_name, True) or ""))
  306. # Collect files we may need for possible world-dep
  307. # calculations
  308. if not self.getVar('BROKEN', file_name, True) and not self.getVar('EXCLUDE_FROM_WORLD', file_name, True):
  309. cacheData.possible_world.append(file_name)
  310. def load_bbfile( self, bbfile , config):
  311. """
  312. Load and parse one .bb build file
  313. Return the data and whether parsing resulted in the file being skipped
  314. """
  315. import bb
  316. from bb import utils, data, parse, debug, event, fatal
  317. # expand tmpdir to include this topdir
  318. data.setVar('TMPDIR', data.getVar('TMPDIR', config, 1) or "", config)
  319. bbfile_loc = os.path.abspath(os.path.dirname(bbfile))
  320. oldpath = os.path.abspath(os.getcwd())
  321. if self.mtime(bbfile_loc):
  322. os.chdir(bbfile_loc)
  323. bb_data = data.init_db(config)
  324. try:
  325. bb_data = parse.handle(bbfile, bb_data) # read .bb data
  326. os.chdir(oldpath)
  327. return bb_data, False
  328. except bb.parse.SkipPackage:
  329. os.chdir(oldpath)
  330. return bb_data, True
  331. except:
  332. os.chdir(oldpath)
  333. raise
  334. def init(cooker):
  335. """
  336. The Objective: Cache the minimum amount of data possible yet get to the
  337. stage of building packages (i.e. tryBuild) without reparsing any .bb files.
  338. To do this, we intercept getVar calls and only cache the variables we see
  339. being accessed. We rely on the cache getVar calls being made for all
  340. variables bitbake might need to use to reach this stage. For each cached
  341. file we need to track:
  342. * Its mtime
  343. * The mtimes of all its dependencies
  344. * Whether it caused a parse.SkipPackage exception
  345. Files causing parsing errors are evicted from the cache.
  346. """
  347. return Cache(cooker)
  348. #============================================================================#
  349. # CacheData
  350. #============================================================================#
  351. class CacheData:
  352. """
  353. The data structures we compile from the cached data
  354. """
  355. def __init__(self):
  356. """
  357. Direct cache variables
  358. (from Cache.handle_data)
  359. """
  360. self.providers = {}
  361. self.rproviders = {}
  362. self.packages = {}
  363. self.packages_dynamic = {}
  364. self.possible_world = []
  365. self.pkg_pn = {}
  366. self.pkg_fn = {}
  367. self.pkg_pepvpr = {}
  368. self.pkg_dp = {}
  369. self.pn_provides = {}
  370. self.fn_provides = {}
  371. self.all_depends = Set()
  372. self.deps = {}
  373. self.rundeps = {}
  374. self.runrecs = {}
  375. self.task_queues = {}
  376. self.task_deps = {}
  377. self.stamp = {}
  378. self.preferred = {}
  379. """
  380. Indirect Cache variables
  381. (set elsewhere)
  382. """
  383. self.ignored_dependencies = []
  384. self.world_target = Set()
  385. self.bbfile_priority = {}
  386. self.bbfile_config_priorities = []