916ca8f2b0
This make the process of applying overlays more reliable by: 1. Ignoring dtb files that are not really device trees. [^1] 2. Adding a `filter` option (per-overlay, there already is a global one) to limit the files to which the overlay applies. This is useful in cases where the `compatible` string is ambiguous and multiple unrelated files match. Previously the script would fail in both cases. [^1]: For example, there is dtbs/overlays/overlay_map.dtb in the Raspberry Pi 1 kernel.
45 lines
1.4 KiB
Nix
45 lines
1.4 KiB
Nix
{ lib, stdenvNoCC, dtc }:
|
|
|
|
with lib; {
|
|
applyOverlays = (base: overlays': stdenvNoCC.mkDerivation {
|
|
name = "device-tree-overlays";
|
|
nativeBuildInputs = [ dtc ];
|
|
buildCommand = let
|
|
overlays = toList overlays';
|
|
in ''
|
|
mkdir -p $out
|
|
cd "${base}"
|
|
find . -type f -name '*.dtb' -print0 \
|
|
| xargs -0 cp -v --no-preserve=mode --target-directory "$out" --parents
|
|
|
|
for dtb in $(find "$out" -type f -name '*.dtb'); do
|
|
dtbCompat=$(fdtget -t s "$dtb" / compatible 2>/dev/null || true)
|
|
# skip files without `compatible` string
|
|
test -z "$dtbCompat" && continue
|
|
|
|
${flip (concatMapStringsSep "\n") overlays (o: ''
|
|
overlayCompat="$(fdtget -t s "${o.dtboFile}" / compatible)"
|
|
|
|
# skip incompatible and non-matching overlays
|
|
if [[ ! "$dtbCompat" =~ "$overlayCompat" ]]; then
|
|
echo -n "Skipping overlay ${o.name}: incompatible with $(basename "$dtb")"
|
|
continue
|
|
fi
|
|
${optionalString (o.filter != null) ''
|
|
if [[ "''${dtb//${o.filter}/}" == "$dtb" ]]; then
|
|
echo -n "Skipping overlay ${o.name}: filter does not match $(basename "$dtb")"
|
|
continue
|
|
fi
|
|
''}
|
|
|
|
echo -n "Applying overlay ${o.name} to $(basename "$dtb")... "
|
|
mv "$dtb"{,.in}
|
|
fdtoverlay -o "$dtb" -i "$dtb.in" "${o.dtboFile}"
|
|
echo "ok"
|
|
rm "$dtb.in"
|
|
'')}
|
|
|
|
done
|
|
'';
|
|
});
|
|
}
|