blob: 1ee947d584fc48e0cd7f962efe21db034612a45b [file] [log] [blame]
Brad Bishopc342db32019-05-15 21:57:59 -04001#
2# SPDX-License-Identifier: GPL-2.0-only
3#
4
Brad Bishop6e60e8b2018-02-01 10:27:11 -05005import subprocess
Brad Bishop1a4b7ee2018-12-16 17:11:34 -08006import multiprocessing
7import traceback
Patrick Williamsc124f4f2015-09-15 14:41:29 -05008
9def read_file(filename):
10 try:
11 f = open( filename, "r" )
12 except IOError as reason:
13 return "" # WARNING: can't raise an error now because of the new RDEPENDS handling. This is a bit ugly. :M:
14 else:
15 data = f.read().strip()
16 f.close()
17 return data
18 return None
19
20def ifelse(condition, iftrue = True, iffalse = False):
21 if condition:
22 return iftrue
23 else:
24 return iffalse
25
26def conditional(variable, checkvalue, truevalue, falsevalue, d):
Brad Bishop6e60e8b2018-02-01 10:27:11 -050027 if d.getVar(variable) == checkvalue:
Patrick Williamsc124f4f2015-09-15 14:41:29 -050028 return truevalue
29 else:
30 return falsevalue
31
Brad Bishop1a4b7ee2018-12-16 17:11:34 -080032def vartrue(var, iftrue, iffalse, d):
33 import oe.types
34 if oe.types.boolean(d.getVar(var)):
35 return iftrue
36 else:
37 return iffalse
38
Patrick Williamsc124f4f2015-09-15 14:41:29 -050039def less_or_equal(variable, checkvalue, truevalue, falsevalue, d):
Brad Bishop6e60e8b2018-02-01 10:27:11 -050040 if float(d.getVar(variable)) <= float(checkvalue):
Patrick Williamsc124f4f2015-09-15 14:41:29 -050041 return truevalue
42 else:
43 return falsevalue
44
45def version_less_or_equal(variable, checkvalue, truevalue, falsevalue, d):
Brad Bishop6e60e8b2018-02-01 10:27:11 -050046 result = bb.utils.vercmp_string(d.getVar(variable), checkvalue)
Patrick Williamsc124f4f2015-09-15 14:41:29 -050047 if result <= 0:
48 return truevalue
49 else:
50 return falsevalue
51
52def both_contain(variable1, variable2, checkvalue, d):
Brad Bishop6e60e8b2018-02-01 10:27:11 -050053 val1 = d.getVar(variable1)
54 val2 = d.getVar(variable2)
Patrick Williamsc124f4f2015-09-15 14:41:29 -050055 val1 = set(val1.split())
56 val2 = set(val2.split())
Patrick Williamsc0f7c042017-02-23 20:41:17 -060057 if isinstance(checkvalue, str):
Patrick Williamsc124f4f2015-09-15 14:41:29 -050058 checkvalue = set(checkvalue.split())
59 else:
60 checkvalue = set(checkvalue)
61 if checkvalue.issubset(val1) and checkvalue.issubset(val2):
62 return " ".join(checkvalue)
63 else:
64 return ""
65
66def set_intersect(variable1, variable2, d):
67 """
68 Expand both variables, interpret them as lists of strings, and return the
69 intersection as a flattened string.
70
71 For example:
72 s1 = "a b c"
73 s2 = "b c d"
74 s3 = set_intersect(s1, s2)
75 => s3 = "b c"
76 """
Brad Bishop6e60e8b2018-02-01 10:27:11 -050077 val1 = set(d.getVar(variable1).split())
78 val2 = set(d.getVar(variable2).split())
Patrick Williamsc124f4f2015-09-15 14:41:29 -050079 return " ".join(val1 & val2)
80
81def prune_suffix(var, suffixes, d):
82 # See if var ends with any of the suffixes listed and
83 # remove it if found
84 for suffix in suffixes:
Brad Bishopd89cb5f2019-04-10 09:02:41 -040085 if suffix and var.endswith(suffix):
86 var = var[:-len(suffix)]
Patrick Williamsc124f4f2015-09-15 14:41:29 -050087
Brad Bishop6e60e8b2018-02-01 10:27:11 -050088 prefix = d.getVar("MLPREFIX")
Patrick Williamsc124f4f2015-09-15 14:41:29 -050089 if prefix and var.startswith(prefix):
Brad Bishopd89cb5f2019-04-10 09:02:41 -040090 var = var[len(prefix):]
Patrick Williamsc124f4f2015-09-15 14:41:29 -050091
92 return var
93
94def str_filter(f, str, d):
95 from re import match
Patrick Williamsc0f7c042017-02-23 20:41:17 -060096 return " ".join([x for x in str.split() if match(f, x, 0)])
Patrick Williamsc124f4f2015-09-15 14:41:29 -050097
98def str_filter_out(f, str, d):
99 from re import match
Patrick Williamsc0f7c042017-02-23 20:41:17 -0600100 return " ".join([x for x in str.split() if not match(f, x, 0)])
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500101
Brad Bishop6e60e8b2018-02-01 10:27:11 -0500102def build_depends_string(depends, task):
103 """Append a taskname to a string of dependencies as used by the [depends] flag"""
104 return " ".join(dep + ":" + task for dep in depends.split())
105
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500106def inherits(d, *classes):
107 """Return True if the metadata inherits any of the specified classes"""
108 return any(bb.data.inherits_class(cls, d) for cls in classes)
109
110def features_backfill(var,d):
111 # This construct allows the addition of new features to variable specified
112 # as var
113 # Example for var = "DISTRO_FEATURES"
114 # This construct allows the addition of new features to DISTRO_FEATURES
115 # that if not present would disable existing functionality, without
116 # disturbing distributions that have already set DISTRO_FEATURES.
117 # Distributions wanting to elide a value in DISTRO_FEATURES_BACKFILL should
118 # add the feature to DISTRO_FEATURES_BACKFILL_CONSIDERED
Brad Bishop6e60e8b2018-02-01 10:27:11 -0500119 features = (d.getVar(var) or "").split()
120 backfill = (d.getVar(var+"_BACKFILL") or "").split()
121 considered = (d.getVar(var+"_BACKFILL_CONSIDERED") or "").split()
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500122
123 addfeatures = []
124 for feature in backfill:
125 if feature not in features and feature not in considered:
126 addfeatures.append(feature)
127
128 if addfeatures:
129 d.appendVar(var, " " + " ".join(addfeatures))
130
Brad Bishopd7bf8c12018-02-25 22:55:05 -0500131def all_distro_features(d, features, truevalue="1", falsevalue=""):
132 """
133 Returns truevalue if *all* given features are set in DISTRO_FEATURES,
134 else falsevalue. The features can be given as single string or anything
135 that can be turned into a set.
136
137 This is a shorter, more flexible version of
138 bb.utils.contains("DISTRO_FEATURES", features, truevalue, falsevalue, d).
139
140 Without explicit true/false values it can be used directly where
141 Python expects a boolean:
142 if oe.utils.all_distro_features(d, "foo bar"):
143 bb.fatal("foo and bar are mutually exclusive DISTRO_FEATURES")
144
145 With just a truevalue, it can be used to include files that are meant to be
146 used only when requested via DISTRO_FEATURES:
147 require ${@ oe.utils.all_distro_features(d, "foo bar", "foo-and-bar.inc")
148 """
149 return bb.utils.contains("DISTRO_FEATURES", features, truevalue, falsevalue, d)
150
151def any_distro_features(d, features, truevalue="1", falsevalue=""):
152 """
153 Returns truevalue if at least *one* of the given features is set in DISTRO_FEATURES,
154 else falsevalue. The features can be given as single string or anything
155 that can be turned into a set.
156
157 This is a shorter, more flexible version of
158 bb.utils.contains_any("DISTRO_FEATURES", features, truevalue, falsevalue, d).
159
160 Without explicit true/false values it can be used directly where
161 Python expects a boolean:
162 if not oe.utils.any_distro_features(d, "foo bar"):
163 bb.fatal("foo, bar or both must be set in DISTRO_FEATURES")
164
165 With just a truevalue, it can be used to include files that are meant to be
166 used only when requested via DISTRO_FEATURES:
167 require ${@ oe.utils.any_distro_features(d, "foo bar", "foo-or-bar.inc")
168
169 """
170 return bb.utils.contains_any("DISTRO_FEATURES", features, truevalue, falsevalue, d)
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500171
Andrew Geissler82c905d2020-04-13 13:39:40 -0500172def parallel_make(d, makeinst=False):
Brad Bishop316dfdd2018-06-25 12:45:53 -0400173 """
174 Return the integer value for the number of parallel threads to use when
175 building, scraped out of PARALLEL_MAKE. If no parallelization option is
176 found, returns None
177
178 e.g. if PARALLEL_MAKE = "-j 10", this will return 10 as an integer.
179 """
Andrew Geissler82c905d2020-04-13 13:39:40 -0500180 if makeinst:
181 pm = (d.getVar('PARALLEL_MAKEINST') or '').split()
182 else:
183 pm = (d.getVar('PARALLEL_MAKE') or '').split()
Brad Bishop316dfdd2018-06-25 12:45:53 -0400184 # look for '-j' and throw other options (e.g. '-l') away
185 while pm:
186 opt = pm.pop(0)
187 if opt == '-j':
188 v = pm.pop(0)
189 elif opt.startswith('-j'):
190 v = opt[2:].strip()
191 else:
192 continue
193
194 return int(v)
195
Andrew Geisslerd1e89492021-02-12 15:35:20 -0600196 return ''
Brad Bishop316dfdd2018-06-25 12:45:53 -0400197
Andrew Geissler82c905d2020-04-13 13:39:40 -0500198def parallel_make_argument(d, fmt, limit=None, makeinst=False):
Brad Bishop316dfdd2018-06-25 12:45:53 -0400199 """
200 Helper utility to construct a parallel make argument from the number of
201 parallel threads specified in PARALLEL_MAKE.
202
203 Returns the input format string `fmt` where a single '%d' will be expanded
204 with the number of parallel threads to use. If `limit` is specified, the
205 number of parallel threads will be no larger than it. If no parallelization
206 option is found in PARALLEL_MAKE, returns an empty string
207
208 e.g. if PARALLEL_MAKE = "-j 10", parallel_make_argument(d, "-n %d") will return
209 "-n 10"
210 """
Andrew Geissler82c905d2020-04-13 13:39:40 -0500211 v = parallel_make(d, makeinst)
Brad Bishop316dfdd2018-06-25 12:45:53 -0400212 if v:
213 if limit:
214 v = min(limit, v)
215 return fmt % v
216 return ''
217
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500218def packages_filter_out_system(d):
219 """
220 Return a list of packages from PACKAGES with the "system" packages such as
221 PN-dbg PN-doc PN-locale-eb-gb removed.
222 """
Brad Bishop6e60e8b2018-02-01 10:27:11 -0500223 pn = d.getVar('PN')
Andrew Geissler9aee5002022-03-30 16:27:02 +0000224 pkgfilter = [pn + suffix for suffix in ('', '-dbg', '-dev', '-doc', '-locale', '-staticdev', '-src')]
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500225 localepkg = pn + "-locale-"
226 pkgs = []
227
Brad Bishop6e60e8b2018-02-01 10:27:11 -0500228 for pkg in d.getVar('PACKAGES').split():
Andrew Geissler9aee5002022-03-30 16:27:02 +0000229 if pkg not in pkgfilter and localepkg not in pkg:
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500230 pkgs.append(pkg)
231 return pkgs
232
233def getstatusoutput(cmd):
Brad Bishop6e60e8b2018-02-01 10:27:11 -0500234 return subprocess.getstatusoutput(cmd)
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500235
236
237def trim_version(version, num_parts=2):
238 """
239 Return just the first <num_parts> of <version>, split by periods. For
240 example, trim_version("1.2.3", 2) will return "1.2".
241 """
242 if type(version) is not str:
243 raise TypeError("Version should be a string")
244 if num_parts < 1:
245 raise ValueError("Cannot split to parts < 1")
246
247 parts = version.split(".")
248 trimmed = ".".join(parts[:num_parts])
249 return trimmed
250
Andrew Geissler595f6302022-01-24 19:11:47 +0000251def cpu_count(at_least=1, at_most=64):
Andrew Geisslerc3d88e42020-10-02 09:45:00 -0500252 cpus = len(os.sched_getaffinity(0))
Andrew Geissler595f6302022-01-24 19:11:47 +0000253 return max(min(cpus, at_most), at_least)
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500254
255def execute_pre_post_process(d, cmds):
256 if cmds is None:
257 return
258
259 for cmd in cmds.strip().split(';'):
260 cmd = cmd.strip()
261 if cmd != '':
262 bb.note("Executing %s ..." % cmd)
263 bb.build.exec_func(cmd, d)
264
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800265# For each item in items, call the function 'target' with item as the first
266# argument, extraargs as the other arguments and handle any exceptions in the
267# parent thread
268def multiprocess_launch(target, items, d, extraargs=None):
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500269
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800270 class ProcessLaunch(multiprocessing.Process):
271 def __init__(self, *args, **kwargs):
272 multiprocessing.Process.__init__(self, *args, **kwargs)
273 self._pconn, self._cconn = multiprocessing.Pipe()
274 self._exception = None
275 self._result = None
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500276
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800277 def run(self):
278 try:
279 ret = self._target(*self._args, **self._kwargs)
280 self._cconn.send((None, ret))
281 except Exception as e:
282 tb = traceback.format_exc()
283 self._cconn.send((e, tb))
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500284
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800285 def update(self):
286 if self._pconn.poll():
287 (e, tb) = self._pconn.recv()
288 if e is not None:
289 self._exception = (e, tb)
290 else:
291 self._result = tb
Brad Bishopd7bf8c12018-02-25 22:55:05 -0500292
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800293 @property
294 def exception(self):
295 self.update()
296 return self._exception
Brad Bishopd7bf8c12018-02-25 22:55:05 -0500297
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800298 @property
299 def result(self):
300 self.update()
301 return self._result
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500302
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800303 max_process = int(d.getVar("BB_NUMBER_THREADS") or os.cpu_count() or 1)
304 launched = []
305 errors = []
306 results = []
307 items = list(items)
308 while (items and not errors) or launched:
309 if not errors and items and len(launched) < max_process:
310 args = (items.pop(),)
311 if extraargs is not None:
312 args = args + extraargs
313 p = ProcessLaunch(target=target, args=args)
314 p.start()
315 launched.append(p)
316 for q in launched:
Brad Bishop19323692019-04-05 15:28:33 -0400317 # Have to manually call update() to avoid deadlocks. The pipe can be full and
318 # transfer stalled until we try and read the results object but the subprocess won't exit
319 # as it still has data to write (https://bugs.python.org/issue8426)
320 q.update()
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800321 # The finished processes are joined when calling is_alive()
322 if not q.is_alive():
323 if q.exception:
324 errors.append(q.exception)
325 if q.result:
326 results.append(q.result)
327 launched.remove(q)
328 # Paranoia doesn't hurt
329 for p in launched:
330 p.join()
331 if errors:
332 msg = ""
333 for (e, tb) in errors:
Brad Bishopc342db32019-05-15 21:57:59 -0400334 if isinstance(e, subprocess.CalledProcessError) and e.output:
335 msg = msg + str(e) + "\n"
336 msg = msg + "Subprocess output:"
337 msg = msg + e.output.decode("utf-8", errors="ignore")
338 else:
339 msg = msg + str(e) + ": " + str(tb) + "\n"
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800340 bb.fatal("Fatal errors occurred in subprocesses:\n%s" % msg)
Brad Bishopd7bf8c12018-02-25 22:55:05 -0500341 return results
342
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500343def squashspaces(string):
344 import re
Brad Bishop19323692019-04-05 15:28:33 -0400345 return re.sub(r"\s+", " ", string).strip()
Patrick Williamsc124f4f2015-09-15 14:41:29 -0500346
Andrew Geisslerd159c7f2021-09-02 21:05:58 -0500347def rprovides_map(pkgdata_dir, pkg_dict):
348 # Map file -> pkg provider
349 rprov_map = {}
350
351 for pkg in pkg_dict:
352 path_to_pkgfile = os.path.join(pkgdata_dir, 'runtime-reverse', pkg)
353 if not os.path.isfile(path_to_pkgfile):
354 continue
355 with open(path_to_pkgfile) as f:
356 for line in f:
357 if line.startswith('RPROVIDES') or line.startswith('FILERPROVIDES'):
358 # List all components provided by pkg.
359 # Exclude version strings, i.e. those starting with (
360 provides = [x for x in line.split()[1:] if not x.startswith('(')]
361 for prov in provides:
362 if prov in rprov_map:
363 rprov_map[prov].append(pkg)
364 else:
365 rprov_map[prov] = [pkg]
366
367 return rprov_map
368
369def format_pkg_list(pkg_dict, ret_format=None, pkgdata_dir=None):
Patrick Williamsd8c66bc2016-06-20 12:57:21 -0500370 output = []
371
372 if ret_format == "arch":
373 for pkg in sorted(pkg_dict):
374 output.append("%s %s" % (pkg, pkg_dict[pkg]["arch"]))
375 elif ret_format == "file":
376 for pkg in sorted(pkg_dict):
377 output.append("%s %s %s" % (pkg, pkg_dict[pkg]["filename"], pkg_dict[pkg]["arch"]))
378 elif ret_format == "ver":
379 for pkg in sorted(pkg_dict):
380 output.append("%s %s %s" % (pkg, pkg_dict[pkg]["arch"], pkg_dict[pkg]["ver"]))
381 elif ret_format == "deps":
Andrew Geisslerd159c7f2021-09-02 21:05:58 -0500382 rprov_map = rprovides_map(pkgdata_dir, pkg_dict)
Patrick Williamsd8c66bc2016-06-20 12:57:21 -0500383 for pkg in sorted(pkg_dict):
384 for dep in pkg_dict[pkg]["deps"]:
Andrew Geisslerd159c7f2021-09-02 21:05:58 -0500385 if dep in rprov_map:
386 # There could be multiple providers within the image
387 for pkg_provider in rprov_map[dep]:
388 output.append("%s|%s * %s [RPROVIDES]" % (pkg, pkg_provider, dep))
389 else:
390 output.append("%s|%s" % (pkg, dep))
Patrick Williamsd8c66bc2016-06-20 12:57:21 -0500391 else:
392 for pkg in sorted(pkg_dict):
393 output.append(pkg)
394
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800395 output_str = '\n'.join(output)
Patrick Williamsd8c66bc2016-06-20 12:57:21 -0500396
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800397 if output_str:
398 # make sure last line is newline terminated
399 output_str += '\n'
400
401 return output_str
402
Andrew Geissler82c905d2020-04-13 13:39:40 -0500403
404# Helper function to get the host compiler version
405# Do not assume the compiler is gcc
406def get_host_compiler_version(d, taskcontextonly=False):
407 import re, subprocess
408
409 if taskcontextonly and d.getVar('BB_WORKERCONTEXT') != '1':
410 return
411
412 compiler = d.getVar("BUILD_CC")
413 # Get rid of ccache since it is not present when parsing.
414 if compiler.startswith('ccache '):
415 compiler = compiler[7:]
416 try:
417 env = os.environ.copy()
418 # datastore PATH does not contain session PATH as set by environment-setup-...
419 # this breaks the install-buildtools use-case
420 # env["PATH"] = d.getVar("PATH")
421 output = subprocess.check_output("%s --version" % compiler, \
422 shell=True, env=env, stderr=subprocess.STDOUT).decode("utf-8")
423 except subprocess.CalledProcessError as e:
424 bb.fatal("Error running %s --version: %s" % (compiler, e.output.decode("utf-8")))
425
426 match = re.match(r".* (\d+\.\d+)\.\d+.*", output.split('\n')[0])
427 if not match:
428 bb.fatal("Can't get compiler version from %s --version output" % compiler)
429
430 version = match.group(1)
431 return compiler, version
432
433
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800434def host_gcc_version(d, taskcontextonly=False):
Brad Bishop37a0e4d2017-12-04 01:01:44 -0500435 import re, subprocess
436
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800437 if taskcontextonly and d.getVar('BB_WORKERCONTEXT') != '1':
438 return
439
Brad Bishop6e60e8b2018-02-01 10:27:11 -0500440 compiler = d.getVar("BUILD_CC")
Brad Bishop19323692019-04-05 15:28:33 -0400441 # Get rid of ccache since it is not present when parsing.
442 if compiler.startswith('ccache '):
443 compiler = compiler[7:]
Brad Bishop37a0e4d2017-12-04 01:01:44 -0500444 try:
445 env = os.environ.copy()
Brad Bishop6e60e8b2018-02-01 10:27:11 -0500446 env["PATH"] = d.getVar("PATH")
Brad Bishop19323692019-04-05 15:28:33 -0400447 output = subprocess.check_output("%s --version" % compiler, \
448 shell=True, env=env, stderr=subprocess.STDOUT).decode("utf-8")
Brad Bishop37a0e4d2017-12-04 01:01:44 -0500449 except subprocess.CalledProcessError as e:
450 bb.fatal("Error running %s --version: %s" % (compiler, e.output.decode("utf-8")))
451
Andrew Geissler82c905d2020-04-13 13:39:40 -0500452 match = re.match(r".* (\d+\.\d+)\.\d+.*", output.split('\n')[0])
Brad Bishop37a0e4d2017-12-04 01:01:44 -0500453 if not match:
454 bb.fatal("Can't get compiler version from %s --version output" % compiler)
455
456 version = match.group(1)
457 return "-%s" % version if version in ("4.8", "4.9") else ""
458
Brad Bishop316dfdd2018-06-25 12:45:53 -0400459
460def get_multilib_datastore(variant, d):
461 localdata = bb.data.createCopy(d)
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800462 if variant:
463 overrides = localdata.getVar("OVERRIDES", False) + ":virtclass-multilib-" + variant
464 localdata.setVar("OVERRIDES", overrides)
465 localdata.setVar("MLPREFIX", variant + "-")
466 else:
467 origdefault = localdata.getVar("DEFAULTTUNE_MULTILIB_ORIGINAL")
468 if origdefault:
469 localdata.setVar("DEFAULTTUNE", origdefault)
470 overrides = localdata.getVar("OVERRIDES", False).split(":")
471 overrides = ":".join([x for x in overrides if not x.startswith("virtclass-multilib-")])
472 localdata.setVar("OVERRIDES", overrides)
473 localdata.setVar("MLPREFIX", "")
Brad Bishop316dfdd2018-06-25 12:45:53 -0400474 return localdata
475
Brad Bishop08902b02019-08-20 09:16:51 -0400476class ImageQAFailed(Exception):
Patrick Williamsc0f7c042017-02-23 20:41:17 -0600477 def __init__(self, description, name=None, logfile=None):
478 self.description = description
479 self.name = name
480 self.logfile=logfile
481
482 def __str__(self):
483 msg = 'Function failed: %s' % self.name
484 if self.description:
485 msg = msg + ' (%s)' % self.description
486
487 return msg
Brad Bishop1a4b7ee2018-12-16 17:11:34 -0800488
Brad Bishop19323692019-04-05 15:28:33 -0400489def sh_quote(string):
490 import shlex
491 return shlex.quote(string)
Andrew Geissler95ac1b82021-03-31 14:34:31 -0500492
493def directory_size(root, blocksize=4096):
494 """
495 Calculate the size of the directory, taking into account hard links,
496 rounding up every size to multiples of the blocksize.
497 """
498 def roundup(size):
499 """
500 Round the size up to the nearest multiple of the block size.
501 """
502 import math
503 return math.ceil(size / blocksize) * blocksize
504
505 def getsize(filename):
506 """
507 Get the size of the filename, not following symlinks, taking into
508 account hard links.
509 """
510 stat = os.lstat(filename)
511 if stat.st_ino not in inodes:
512 inodes.add(stat.st_ino)
513 return stat.st_size
514 else:
515 return 0
516
517 inodes = set()
518 total = 0
519 for root, dirs, files in os.walk(root):
520 total += sum(roundup(getsize(os.path.join(root, name))) for name in files)
521 total += roundup(getsize(root))
522 return total