
stdeout and stderr content returned by subprocess API has different types in Python 3(bytes) and Python 2(string). Decoding it to 'utf-8' makes it unicode on both pythons. (From OE-Core rev: ae4d36375683b6cfd48af25bfca70b2e77f7ade5) Signed-off-by: Ed Bartosh <ed.bartosh@linux.intel.com> Signed-off-by: Richard Purdie <richard.purdie@linuxfoundation.org>
6.0 KiB
Executable File
#!/usr/bin/env python
Copyright (c) 2012 Wind River Systems, Inc.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License version 2 as
published by the Free Software Foundation.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
import os import sys import optparse import re import subprocess import shutil
pkg_cur_dirs = {} obsolete_dirs = [] parser = None
def err_quit(msg): print(msg) parser.print_usage() sys.exit(1)
def parse_version(verstr): elems = verstr.split(':') epoch = elems[0] if len(epoch) == 0: return elems[1] else: return epoch + '_' + elems[1]
def run_command(cmd): pipe = subprocess.Popen(cmd, stdout=subprocess.PIPE, shell=True) output = pipe.communicate()[0] if pipe.returncode != 0: print("Execute command '%s' failed." % cmd) sys.exit(1) return output.decode('utf-8')
def get_cur_arch_dirs(workdir, arch_dirs): pattern = workdir + '/(.*?)/'
cmd = "bitbake -e | grep ^SDK_ARCH="
output = run_command(cmd)
sdk_arch = output.split('"')[1]
# select thest 5 packages to get the dirs of current arch
pkgs = ['hicolor-icon-theme', 'base-files', 'acl-native', 'binutils-crosssdk-' + sdk_arch, 'nativesdk-autoconf']
for pkg in pkgs:
cmd = "bitbake -e " + pkg + " | grep ^IMAGE_ROOTFS="
output = run_command(cmd)
output = output.split('"')[1]
m = re.match(pattern, output)
arch_dirs.append(m.group(1))
def main(): global parser parser = optparse.OptionParser( usage = """%prog
%prog removes the obsolete packages' build directories in WORKDIR. This script must be ran under BUILDDIR after source file "oe-init-build-env".
Any file or directory under WORKDIR which is not created by Yocto will be deleted. Be CAUTIOUS.""")
options, args = parser.parse_args(sys.argv)
builddir = run_command('echo $BUILDDIR').strip()
if len(builddir) == 0:
err_quit("Please source file \"oe-init-build-env\" first.\n")
if os.getcwd() != builddir:
err_quit("Please run %s under: %s\n" % (os.path.basename(args[0]), builddir))
print('Updating bitbake caches...')
cmd = "bitbake -s"
output = run_command(cmd)
output = output.split('\n')
index = 0
while len(output[index]) > 0:
index += 1
alllines = output[index+1:]
for line in alllines:
# empty again means end of the versions output
if len(line) == 0:
break
line = line.strip()
line = re.sub('\s+', ' ', line)
elems = line.split(' ')
if len(elems) == 2:
version = parse_version(elems[1])
else:
version = parse_version(elems[2])
pkg_cur_dirs[elems[0]] = version
cmd = "bitbake -e"
output = run_command(cmd)
tmpdir = None
image_rootfs = None
output = output.split('\n')
for line in output:
if tmpdir and image_rootfs:
break
if not tmpdir:
m = re.match('TMPDIR="(.*)"', line)
if m:
tmpdir = m.group(1)
if not image_rootfs:
m = re.match('IMAGE_ROOTFS="(.*)"', line)
if m:
image_rootfs = m.group(1)
# won't fail just in case
if not tmpdir or not image_rootfs:
print("Can't get TMPDIR or IMAGE_ROOTFS.")
return 1
pattern = tmpdir + '/(.*?)/(.*?)/'
m = re.match(pattern, image_rootfs)
if not m:
print("Can't get WORKDIR.")
return 1
workdir = os.path.join(tmpdir, m.group(1))
# we only deal the dirs of current arch, total numbers of dirs are 6
cur_arch_dirs = [m.group(2)]
get_cur_arch_dirs(workdir, cur_arch_dirs)
for workroot, dirs, files in os.walk(workdir):
# For the files, they should NOT exist in WORKDIR. Remove them.
for f in files:
obsolete_dirs.append(os.path.join(workroot, f))
for d in dirs:
if d not in cur_arch_dirs:
continue
for pkgroot, pkgdirs, filenames in os.walk(os.path.join(workroot, d)):
for f in filenames:
obsolete_dirs.append(os.path.join(pkgroot, f))
for pkgdir in sorted(pkgdirs):
if pkgdir not in pkg_cur_dirs:
obsolete_dirs.append(os.path.join(pkgroot, pkgdir))
else:
for verroot, verdirs, verfiles in os.walk(os.path.join(pkgroot, pkgdir)):
for f in verfiles:
obsolete_dirs.append(os.path.join(pkgroot, f))
for v in sorted(verdirs):
if v not in pkg_cur_dirs[pkgdir]:
obsolete_dirs.append(os.path.join(pkgroot, pkgdir, v))
break
# just process the top dir of every package under tmp/work/*/,
# then jump out of the above os.walk()
break
# it is convenient to use os.walk() to get dirs and files at same time
# both of them have been dealed in the loop, so jump out
break
for d in obsolete_dirs:
print("Deleting %s" % d)
shutil.rmtree(d, True)
if len(obsolete_dirs):
print('\nTotal %d items.' % len(obsolete_dirs))
else:
print('\nNo obsolete directory found under %s.' % workdir)
return 0
if name == 'main': try: ret = main() except Exception: ret = 2 import traceback traceback.print_exc() sys.exit(ret)