forked from libretro/Lakka-LibreELEC
402 lines
14 KiB
Python
Executable File
402 lines
14 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
# -*- coding: utf-8 -*-
|
|
|
|
# SPDX-License-Identifier: GPL-2.0
|
|
# Copyright (C) 2019-present Team LibreELEC (https://libreelec.tv)
|
|
|
|
import sys, os, codecs, json, argparse, re
|
|
|
|
ROOT_PKG = "__root__"
|
|
|
|
class LibreELEC_Package:
|
|
def __init__(self, name, section):
|
|
self.name = name
|
|
self.section = section
|
|
self.deps = {"bootstrap": [],
|
|
"init": [],
|
|
"host": [],
|
|
"target": []}
|
|
self.wants = []
|
|
self.wantedby = []
|
|
|
|
self.unpacks = []
|
|
|
|
def __repr__(self):
|
|
s = f"{name:<9}: {self.name}"
|
|
s = f"{s}\n{section:<9}: {self.section}"
|
|
|
|
for t in self.deps:
|
|
s = f"{s}\n{t:<9}: {self.deps[t]}"
|
|
|
|
s = f"{s}\n{'UNPACKS':<9}: {self.unpacks}"
|
|
|
|
s = f"{s}\n{'NEEDS':<9}: {self.wants}"
|
|
s = f"{s}\n{'WANTED BY':<9}: {self.wantedby}"
|
|
|
|
return s
|
|
|
|
def addDependencies(self, target, packages):
|
|
for d in " ".join(packages.split()).split():
|
|
self.deps[target].append(d)
|
|
name = d.split(":")[0]
|
|
if name not in self.wants and name != self.name:
|
|
self.wants.append(name)
|
|
|
|
def delDependency(self, target, package):
|
|
if package in self.deps[target]:
|
|
self.deps[target].remove(package)
|
|
name = package.split(":")[0]
|
|
if name in self.wants:
|
|
self.wants.remove(name)
|
|
|
|
def addReference(self, package):
|
|
name = package.split(":")[0]
|
|
if name not in self.wantedby:
|
|
self.wantedby.append(name)
|
|
|
|
def delReference(self, package):
|
|
name = package.split(":")[0]
|
|
if name in self.wantedby:
|
|
self.wantedby.remove(name)
|
|
|
|
def addUnpack(self, packages):
|
|
if packages.strip():
|
|
self.unpacks = packages.strip().split()
|
|
|
|
def isReferenced(self):
|
|
return False if self.wants == [] else True
|
|
|
|
def isWanted(self):
|
|
return False if self.wantedby == [] else True
|
|
|
|
def references(self, package):
|
|
return package in self.wants
|
|
|
|
# Reference material:
|
|
# https://www.electricmonk.nl/docs/dependency_resolving_algorithm/dependency_resolving_algorithm.html
|
|
class Node:
|
|
def __init__(self, name, target, section):
|
|
self.name = name
|
|
self.target = target
|
|
self.section = section
|
|
self.fqname = f"{name}:{target}"
|
|
self.edges = []
|
|
|
|
def appendEdges(self, node):
|
|
# Add the node itself...
|
|
if node not in self.edges:
|
|
self.edges.append(node)
|
|
# as well as its edges
|
|
for e in node.edges:
|
|
if e not in self.edges:
|
|
self.edges.append(e)
|
|
|
|
# Return True if the dependencies of the specified node are met by this node
|
|
def satisfies(self, node):
|
|
for e in node.edges:
|
|
if e not in self.edges:
|
|
return False
|
|
return True
|
|
|
|
def __repr__(self):
|
|
s = f"{'name':<9}: {self.name}"
|
|
s = f"{s}\n{'target':<9}: {self.target}"
|
|
s = f"{s}\n{'fqname':<9}: {self.fqname}"
|
|
s = f"{s}\n{'common':<9}: {self.commonName()}"
|
|
s = f"{s}\n{'section':<9}: {self.section}"
|
|
|
|
for e in self.edges:
|
|
s = f"{s}\nEDGE: {e.fqname}"
|
|
|
|
return s
|
|
|
|
def commonName(self):
|
|
return self.name if self.target == "target" else f"{self.name}:{self.target}"
|
|
|
|
def addEdge(self, node):
|
|
if node not in self.edges:
|
|
self.edges.append(node)
|
|
|
|
def eprint(*args, **kwargs):
|
|
print(*args, file=sys.stderr, **kwargs)
|
|
|
|
# Read a JSON list of all possible packages from stdin, removing newlines
|
|
def loadPackages():
|
|
jdata = json.loads(f"[{sys.stdin.read().replace(chr(10),'')[:-1]}]")
|
|
|
|
map = {}
|
|
|
|
# Load "global" packages first
|
|
for pkg in jdata:
|
|
if pkg["hierarchy"] == "global":
|
|
map[pkg["name"]] = initPackage(pkg)
|
|
|
|
# Then the "local" packages, as these will replace any matching "global" packages
|
|
for pkg in jdata:
|
|
if pkg["hierarchy"] == "local":
|
|
map[pkg["name"]] = initPackage(pkg)
|
|
|
|
return map
|
|
|
|
# Create a fully formed LibreELEC_Package object
|
|
def initPackage(package):
|
|
pkg = LibreELEC_Package(package["name"], package["section"])
|
|
|
|
for target in ["bootstrap", "init", "host", "target"]:
|
|
pkg.addDependencies(target, package[target])
|
|
|
|
pkg.addUnpack(package["unpack"])
|
|
|
|
return pkg
|
|
|
|
# Split name:target into components
|
|
def split_package(name):
|
|
parts = name.split(":")
|
|
pn = parts[0]
|
|
pt = parts[1] if len(parts) != 1 else "target"
|
|
return (pn, pt)
|
|
|
|
# Return a list of packages of the specified type
|
|
def get_packages_by_target(target, list):
|
|
newlist = []
|
|
|
|
for p in list:
|
|
(pn, pt) = split_package(p)
|
|
if target in ["target", "init"] and pt in ["target", "init"]:
|
|
newlist.append(p)
|
|
elif target in ["bootstrap", "host"] and pt in ["bootstrap", "host"]:
|
|
newlist.append(p)
|
|
|
|
return newlist
|
|
|
|
# For the specified node iterate over the list of scheduled nodes and return the first
|
|
# position where we could possibly build this node (ie. all dependencies satisfied).
|
|
def findbuildpos(node, list):
|
|
|
|
# Keep a running total of all dependencies as we progress through the list
|
|
alldeps = Node("", "", "")
|
|
|
|
candidate = None
|
|
for n in list:
|
|
alldeps.appendEdges(n)
|
|
if alldeps.satisfies(node):
|
|
if len(n.edges) > len(node.edges):
|
|
if candidate == None:
|
|
candidate = n
|
|
break
|
|
candidate = n
|
|
|
|
return list.index(candidate) + 1 if candidate else -1
|
|
|
|
# Resolve dependencies for a node
|
|
def dep_resolve(node, resolved, unresolved):
|
|
unresolved.append(node)
|
|
|
|
for edge in node.edges:
|
|
if edge not in resolved:
|
|
if edge in unresolved:
|
|
raise Exception((
|
|
f"Circular reference detected: {node.fqname} -> {edge.commonName()}\n"
|
|
f"Remove {edge.commonName()} from {node.name} package.mk::PKG_DEPENDS_{node.target.upper()}"
|
|
))
|
|
dep_resolve(edge, resolved, unresolved)
|
|
|
|
if node not in resolved:
|
|
resolved.append(node)
|
|
|
|
unresolved.remove(node)
|
|
|
|
# Return a list of build steps for the trigger packages
|
|
def get_build_steps(args, nodes):
|
|
resolved = []
|
|
unresolved = []
|
|
|
|
# When building the image the :target packages must be installed.
|
|
#
|
|
# However, if we are not building the image then only build the packages
|
|
# and don't install them as it's likely we will be building discrete add-ons
|
|
# which are installed outside of the image.
|
|
#
|
|
install = True if "image" in args.build else False
|
|
|
|
for pkgname in [x for x in args.build if x]:
|
|
if pkgname.find(":") == -1:
|
|
pkgname = f"{pkgname}:target"
|
|
|
|
if pkgname in nodes:
|
|
dep_resolve(nodes[pkgname], resolved, unresolved)
|
|
|
|
# Abort if any references remain unresolved
|
|
if unresolved != []:
|
|
eprint("The following dependencies have not been resolved:")
|
|
for dep in unresolved:
|
|
eprint(f" {dep}")
|
|
raise("Unresolved references")
|
|
|
|
# Output list of resolved dependencies
|
|
for pkg in resolved:
|
|
task = "build" if pkg.fqname.endswith(":host") or pkg.fqname.endswith(":init") or not install else "install"
|
|
yield(task, pkg.fqname)
|
|
|
|
# Reduce the complete list of packages to a map of those packages that will
|
|
# be needed for the build.
|
|
def processPackages(args, packages):
|
|
# Add dummy package to ensure build/install dependencies are not culled
|
|
pkg = {
|
|
"name": ROOT_PKG,
|
|
"section": "virtual",
|
|
"hierarchy": "global",
|
|
"bootstrap": "",
|
|
"init": "",
|
|
"host": " ".join(get_packages_by_target("host", args.build)),
|
|
"target": " ".join(get_packages_by_target("target", args.build)),
|
|
"unpack": ""
|
|
}
|
|
|
|
packages[pkg["name"]] = initPackage(pkg)
|
|
|
|
# Resolve reverse references that we can use to ignore unreferenced packages
|
|
for pkgname in packages:
|
|
for opkgname in packages:
|
|
opkg = packages[opkgname]
|
|
if opkg.references(pkgname):
|
|
if pkgname in packages:
|
|
packages[pkgname].addReference(opkgname)
|
|
|
|
# Identify unused packages
|
|
while True:
|
|
changed = False
|
|
for pkgname in packages:
|
|
pkg = packages[pkgname]
|
|
if pkg.isWanted():
|
|
for opkgname in pkg.wantedby:
|
|
if opkgname != ROOT_PKG:
|
|
if not packages[opkgname].isWanted():
|
|
pkg.delReference(opkgname)
|
|
changed = True
|
|
if not changed:
|
|
break
|
|
|
|
# Create a new map of "needed" packages
|
|
needed_map = {}
|
|
for pkgname in packages:
|
|
pkg = packages[pkgname]
|
|
if pkg.isWanted() or pkgname == ROOT_PKG:
|
|
needed_map[pkgname] = pkg
|
|
|
|
# Validate package dependency references
|
|
if not args.ignore_invalid:
|
|
for pkgname in needed_map:
|
|
pkg = needed_map[pkgname]
|
|
for t in pkg.deps:
|
|
for d in pkg.deps[t]:
|
|
if split_package(d)[0] not in needed_map:
|
|
msg = f'Invalid package reference: dependency {d} in package {pkgname}::PKG_DEPENDS_{t.upper()} is not valid'
|
|
if args.warn_invalid:
|
|
eprint(f"WARNING: {msg}")
|
|
else:
|
|
raise Exception(msg)
|
|
|
|
node_map = {}
|
|
|
|
# Convert all packages to target-specific nodes
|
|
for pkgname in needed_map:
|
|
pkg = needed_map[pkgname]
|
|
for target in pkg.deps:
|
|
if pkg.deps[target]:
|
|
node = Node(pkgname, target, pkg.section)
|
|
node_map[node.fqname] = node
|
|
|
|
# Ensure all referenced dependencies exist as a basic node
|
|
for pkgname in needed_map:
|
|
pkg = needed_map[pkgname]
|
|
for target in pkg.deps:
|
|
for dep in pkg.deps[target]:
|
|
dfq = dep if dep.find(":") != -1 else f"{dep}:target"
|
|
if dfq not in node_map:
|
|
(dfq_p, dfq_t) = split_package(dfq)
|
|
if dfq_p in packages:
|
|
dpkg = packages[dfq_p]
|
|
node_map[dfq] = Node(dfq_p, dfq_t, dpkg.section)
|
|
elif not args.ignore_invalid:
|
|
raise Exception(f"Invalid package! Package {dfq_p} cannot be found for this PROJECT/DEVICE/ARCH")
|
|
|
|
# To each target-specific node, add the corresponding
|
|
# target-specific dependency nodes ("edges")
|
|
for name in node_map:
|
|
node = node_map[name]
|
|
if node.name not in needed_map:
|
|
if args.warn_invalid:
|
|
continue
|
|
else:
|
|
raise Exception(f"Invalid package! Package {node.name} cannot be found for this PROJECT/DEVICE/ARCH")
|
|
for dep in needed_map[node.name].deps[node.target]:
|
|
dfq = dep if dep.find(":") != -1 else f"{dep}:target"
|
|
if dfq in node_map:
|
|
node.addEdge(node_map[dfq])
|
|
|
|
return node_map
|
|
|
|
#---------------------------------------------
|
|
parser = argparse.ArgumentParser(description="Generate package dependency list for the requested build/install packages. \
|
|
Package data will be read from stdin in JSON format.", \
|
|
formatter_class=lambda prog: argparse.HelpFormatter(prog,max_help_position=25,width=90))
|
|
|
|
parser.add_argument("-b", "--build", nargs="+", metavar="PACKAGE", required=True, \
|
|
help="Space-separated list of build trigger packages, either for host or target. Required property - specify at least one package.")
|
|
|
|
parser.add_argument("--warn-invalid", action="store_true", default=False, \
|
|
help="Warn about invalid/missing dependency packages, perhaps excluded by a PKG_ARCH incompatability. Default is to abort.")
|
|
|
|
parser.add_argument("--ignore-invalid", action="store_true", default=False, \
|
|
help="Ignore invalid packages.")
|
|
|
|
group = parser.add_mutually_exclusive_group()
|
|
group.add_argument("--show-wants", action="store_true", \
|
|
help="Output \"wants\" dependencies for each step.")
|
|
group.add_argument("--hide-wants", action="store_false", dest="show_wants", default=True, \
|
|
help="Disable --show-wants. This is the default.")
|
|
|
|
parser.add_argument("--with-json", metavar="FILE", \
|
|
help="File into which JSON formatted plan will be written.")
|
|
|
|
args = parser.parse_args()
|
|
|
|
ALL_PACKAGES = loadPackages()
|
|
|
|
loaded = len(ALL_PACKAGES)
|
|
|
|
REQUIRED_PKGS = processPackages(args, ALL_PACKAGES)
|
|
|
|
# Identify list of packages to build/install
|
|
steps = [step for step in get_build_steps(args, REQUIRED_PKGS)]
|
|
|
|
eprint(f"Packages loaded : {loaded}")
|
|
eprint(f"Build trigger(s): {len(args.build)} [{' '.join(args.build)}]")
|
|
eprint(f"Package steps : {len(steps)}")
|
|
eprint("")
|
|
|
|
# Write the JSON build plan (with dependencies)
|
|
if args.with_json:
|
|
plan = []
|
|
for step in steps:
|
|
(pkg_name, target) = split_package(step[1])
|
|
plan.append({"task": step[0],
|
|
"name": step[1],
|
|
"section": ALL_PACKAGES[pkg_name].section,
|
|
"wants": [d.fqname for d in REQUIRED_PKGS[step[1]].edges],
|
|
"unpacks": ALL_PACKAGES[pkg_name].unpacks if pkg_name in ALL_PACKAGES else []})
|
|
|
|
with open(args.with_json, "w") as out:
|
|
print(json.dumps(plan, indent=2, sort_keys=False), file=out)
|
|
|
|
# Output build/install steps
|
|
if args.show_wants:
|
|
for step in steps:
|
|
node = (REQUIRED_PKGS[step[1]])
|
|
wants = [edge.fqname for edge in node.edges]
|
|
print(f"{step[0]:<7} {step[1].replace(':target',''):<25} (wants: {', '.join(wants).replace(':target','')})")
|
|
else:
|
|
for step in steps:
|
|
print(f"{step[0]:<7} {step[1].replace(':target','')}")
|