| # These directories will be staged in the sysroot |
| SYSROOT_DIRS = " \ |
| ${includedir} \ |
| ${libdir} \ |
| ${base_libdir} \ |
| ${nonarch_base_libdir} \ |
| ${datadir} \ |
| " |
| |
| # These directories are also staged in the sysroot when they contain files that |
| # are usable on the build system |
| SYSROOT_DIRS_NATIVE = " \ |
| ${bindir} \ |
| ${sbindir} \ |
| ${base_bindir} \ |
| ${base_sbindir} \ |
| ${libexecdir} \ |
| ${sysconfdir} \ |
| ${localstatedir} \ |
| " |
| SYSROOT_DIRS_append_class-native = " ${SYSROOT_DIRS_NATIVE}" |
| SYSROOT_DIRS_append_class-cross = " ${SYSROOT_DIRS_NATIVE}" |
| SYSROOT_DIRS_append_class-crosssdk = " ${SYSROOT_DIRS_NATIVE}" |
| |
| # These directories will not be staged in the sysroot |
| SYSROOT_DIRS_BLACKLIST = " \ |
| ${mandir} \ |
| ${docdir} \ |
| ${infodir} \ |
| ${datadir}/locale \ |
| ${datadir}/applications \ |
| ${datadir}/fonts \ |
| ${datadir}/pixmaps \ |
| " |
| |
| sysroot_stage_dir() { |
| src="$1" |
| dest="$2" |
| # if the src doesn't exist don't do anything |
| if [ ! -d "$src" ]; then |
| return |
| fi |
| |
| mkdir -p "$dest" |
| ( |
| cd $src |
| find . -print0 | cpio --null -pdlu $dest |
| ) |
| } |
| |
| sysroot_stage_dirs() { |
| from="$1" |
| to="$2" |
| |
| for dir in ${SYSROOT_DIRS}; do |
| sysroot_stage_dir "$from$dir" "$to$dir" |
| done |
| |
| # Remove directories we do not care about |
| for dir in ${SYSROOT_DIRS_BLACKLIST}; do |
| rm -rf "$to$dir" |
| done |
| } |
| |
| sysroot_stage_all() { |
| sysroot_stage_dirs ${D} ${SYSROOT_DESTDIR} |
| } |
| |
| python sysroot_strip () { |
| import stat, errno |
| |
| dvar = d.getVar('SYSROOT_DESTDIR', True) |
| pn = d.getVar('PN', True) |
| |
| os.chdir(dvar) |
| |
| # Return type (bits): |
| # 0 - not elf |
| # 1 - ELF |
| # 2 - stripped |
| # 4 - executable |
| # 8 - shared library |
| # 16 - kernel module |
| def isELF(path): |
| type = 0 |
| ret, result = oe.utils.getstatusoutput("file \"%s\"" % path.replace("\"", "\\\"")) |
| |
| if ret: |
| bb.error("split_and_strip_files: 'file %s' failed" % path) |
| return type |
| |
| # Not stripped |
| if "ELF" in result: |
| type |= 1 |
| if "not stripped" not in result: |
| type |= 2 |
| if "executable" in result: |
| type |= 4 |
| if "shared" in result: |
| type |= 8 |
| return type |
| |
| |
| elffiles = {} |
| inodes = {} |
| libdir = os.path.abspath(dvar + os.sep + d.getVar("libdir", True)) |
| baselibdir = os.path.abspath(dvar + os.sep + d.getVar("base_libdir", True)) |
| if (d.getVar('INHIBIT_SYSROOT_STRIP', True) != '1'): |
| # |
| # First lets figure out all of the files we may have to process |
| # |
| for root, dirs, files in os.walk(dvar): |
| for f in files: |
| file = os.path.join(root, f) |
| |
| try: |
| ltarget = oe.path.realpath(file, dvar, False) |
| s = os.lstat(ltarget) |
| except OSError as e: |
| (err, strerror) = e.args |
| if err != errno.ENOENT: |
| raise |
| # Skip broken symlinks |
| continue |
| if not s: |
| continue |
| # Check its an excutable |
| if (s[stat.ST_MODE] & stat.S_IXUSR) or (s[stat.ST_MODE] & stat.S_IXGRP) or (s[stat.ST_MODE] & stat.S_IXOTH) \ |
| or ((file.startswith(libdir) or file.startswith(baselibdir)) and ".so" in f): |
| # If it's a symlink, and points to an ELF file, we capture the readlink target |
| if os.path.islink(file): |
| continue |
| |
| # It's a file (or hardlink), not a link |
| # ...but is it ELF, and is it already stripped? |
| elf_file = isELF(file) |
| if elf_file & 1: |
| if elf_file & 2: |
| if 'already-stripped' in (d.getVar('INSANE_SKIP_' + pn, True) or "").split(): |
| bb.note("Skipping file %s from %s for already-stripped QA test" % (file[len(dvar):], pn)) |
| else: |
| bb.warn("File '%s' from %s was already stripped, this will prevent future debugging!" % (file[len(dvar):], pn)) |
| continue |
| |
| if s.st_ino in inodes: |
| os.unlink(file) |
| os.link(inodes[s.st_ino], file) |
| else: |
| inodes[s.st_ino] = file |
| # break hardlink |
| bb.utils.copyfile(file, file) |
| elffiles[file] = elf_file |
| |
| # |
| # Now strip them (in parallel) |
| # |
| strip = d.getVar("STRIP", True) |
| sfiles = [] |
| for file in elffiles: |
| elf_file = int(elffiles[file]) |
| #bb.note("Strip %s" % file) |
| sfiles.append((file, elf_file, strip)) |
| |
| oe.utils.multiprocess_exec(sfiles, oe.package.runstrip) |
| } |
| |
| do_populate_sysroot[dirs] = "${SYSROOT_DESTDIR}" |
| do_populate_sysroot[umask] = "022" |
| |
| addtask populate_sysroot after do_install |
| |
| SYSROOT_PREPROCESS_FUNCS ?= "" |
| SYSROOT_DESTDIR = "${WORKDIR}/sysroot-destdir" |
| SYSROOT_LOCK = "${STAGING_DIR}/staging.lock" |
| |
| # We clean out any existing sstate from the sysroot if we rerun configure |
| python sysroot_cleansstate () { |
| ss = sstate_state_fromvars(d, "populate_sysroot") |
| sstate_clean(ss, d) |
| } |
| do_configure[prefuncs] += "sysroot_cleansstate" |
| |
| |
| BB_SETSCENE_VERIFY_FUNCTION2 = "sysroot_checkhashes2" |
| |
| def sysroot_checkhashes2(covered, tasknames, fns, d, invalidtasks): |
| problems = set() |
| configurefns = set() |
| for tid in invalidtasks: |
| if tasknames[tid] == "do_configure" and tid not in covered: |
| configurefns.add(fns[tid]) |
| for tid in covered: |
| if tasknames[tid] == "do_populate_sysroot" and fns[tid] in configurefns: |
| problems.add(tid) |
| return problems |
| |
| BB_SETSCENE_VERIFY_FUNCTION = "sysroot_checkhashes" |
| |
| def sysroot_checkhashes(covered, tasknames, fnids, fns, d, invalidtasks = None): |
| problems = set() |
| configurefnids = set() |
| if not invalidtasks: |
| invalidtasks = range(len(tasknames)) |
| for task in invalidtasks: |
| if tasknames[task] == "do_configure" and task not in covered: |
| configurefnids.add(fnids[task]) |
| for task in covered: |
| if tasknames[task] == "do_populate_sysroot" and fnids[task] in configurefnids: |
| problems.add(task) |
| return problems |
| |
| python do_populate_sysroot () { |
| bb.build.exec_func("sysroot_stage_all", d) |
| bb.build.exec_func("sysroot_strip", d) |
| for f in (d.getVar('SYSROOT_PREPROCESS_FUNCS', True) or '').split(): |
| bb.build.exec_func(f, d) |
| pn = d.getVar("PN", True) |
| multiprov = d.getVar("MULTI_PROVIDER_WHITELIST", True).split() |
| provdir = d.expand("${SYSROOT_DESTDIR}${base_prefix}/sysroot-providers/") |
| bb.utils.mkdirhier(provdir) |
| for p in d.getVar("PROVIDES", True).split(): |
| if p in multiprov: |
| continue |
| p = p.replace("/", "_") |
| with open(provdir + p, "w") as f: |
| f.write(pn) |
| } |
| |
| do_populate_sysroot[vardeps] += "${SYSROOT_PREPROCESS_FUNCS}" |
| do_populate_sysroot[vardepsexclude] += "MULTI_PROVIDER_WHITELIST" |
| |
| SSTATETASKS += "do_populate_sysroot" |
| do_populate_sysroot[cleandirs] = "${SYSROOT_DESTDIR}" |
| do_populate_sysroot[sstate-inputdirs] = "${SYSROOT_DESTDIR}" |
| do_populate_sysroot[sstate-outputdirs] = "${STAGING_DIR_HOST}/" |
| do_populate_sysroot[stamp-extra-info] = "${MACHINE}" |
| |
| python do_populate_sysroot_setscene () { |
| sstate_setscene(d) |
| } |
| addtask do_populate_sysroot_setscene |
| |
| |