|  | #!/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 | 
|  |  | 
|  | 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(3) | 
|  | sys.exit(ret) |