nixpkgs/pkgs/build-support/setup-hooks/multiple-outputs.sh

157 lines
5.3 KiB
Bash
Raw Normal View History

# The base package for automatic multiple-output splitting. Used in stdenv as well.
preConfigureHooks+=(_multioutConfig)
2014-08-25 17:30:46 +04:00
preFixupHooks+=(_multioutDocs)
2014-08-27 03:14:09 +04:00
preFixupHooks+=(_multioutDevs)
2014-08-25 17:30:46 +04:00
postFixupHooks+=(_multioutPropagateDev)
2014-08-27 03:14:09 +04:00
# Assign the first string containing nonempty variable to the variable named $1
_assignFirst() {
local varName="$1"
shift
2014-08-27 03:14:09 +04:00
while [ $# -ge 1 ]; do
if [ -n "${!1}" ]; then eval "${varName}"="$1"; return; fi
shift
done
return 1 # none found
}
2014-08-30 10:43:46 +04:00
# Same as _assignFirst, but only if "$1" = ""
_overrideFirst() {
if [ -z "${!1}" ]; then
_assignFirst "$@"
fi
}
# Setup chains of sane default values with easy overridability.
# The variables are global to be usable anywhere during the build.
2014-08-27 03:14:09 +04:00
2014-08-30 10:43:46 +04:00
_overrideFirst outputDev "dev" "out"
_overrideFirst outputBin "bin" "out"
2014-08-30 10:43:46 +04:00
_overrideFirst outputInclude "$outputDev"
# so-libs are often among the main things to keep, and so go to $out
2014-08-30 10:43:46 +04:00
_overrideFirst outputLib "lib" "out"
2014-08-30 10:43:46 +04:00
_overrideFirst outputDoc "doc" "out"
# man and info pages are small and often useful to distribute with binaries
2014-08-30 10:43:46 +04:00
_overrideFirst outputMan "man" "doc" "$outputBin"
_overrideFirst outputInfo "info" "doc" "$outputMan"
2014-08-27 03:14:09 +04:00
2014-08-25 17:30:46 +04:00
# Add standard flags to put files into the desired outputs.
_multioutConfig() {
if [ "$outputs" = "out" ] || [ -z "${setOutputFlags-1}" ]; then return; fi;
2014-08-27 03:14:09 +04:00
configureFlags="\
--bindir=${!outputBin}/bin --sbindir=${!outputBin}/sbin \
--includedir=${!outputInclude}/include --oldincludedir=${!outputInclude}/include \
--mandir=${!outputMan}/share/man --infodir=${!outputInfo}/share/info \
--docdir=${!outputDoc}/share/doc \
--libdir=${!outputLib}/lib --libexecdir=${!outputLib}/libexec \
$configureFlags"
installFlags="\
pkgconfigdir=${!outputDev}/lib/pkgconfig \
m4datadir=${!outputDev}/share/aclocal aclocaldir=${!outputDev}/share/aclocal \
$installFlags"
}
# Add rpath prefixes to library paths, and avoid stdenv doing it for $out.
2014-08-27 03:14:09 +04:00
_addRpathPrefix "${!outputLib}"
NIX_NO_SELF_RPATH=1
2014-08-27 03:14:09 +04:00
# Move subpaths that match pattern $1 from under any output/ to the $2 output/
# Beware: only * ? [..] patterns are accepted.
2014-08-27 03:14:09 +04:00
_moveToOutput() {
local patt="$1"
local dstOut="$2"
local output
for output in $outputs; do
if [ "${!output}" = "$dstOut" ]; then continue; fi
2014-08-27 03:14:09 +04:00
local srcPath
for srcPath in ${!output}/$patt; do
if [ ! -e "$srcPath" ]; then continue; fi
local dstPath="$dstOut${srcPath#${!output}}"
echo "moving $srcPath to $dstPath"
if [ -d "$dstPath" ] && [ -d "$srcPath" ]
then # attempt directory merge
mv -t "$dstPath" "$srcPath"/*
rmdir "$srcPath"
else # usual move
mkdir -p $(readlink -m "$dstPath/..") # create the parent for $dstPath
mv "$srcPath" "$dstPath"
fi
done
done
}
# Move documentation to the desired outputs.
2014-08-25 17:30:46 +04:00
_multioutDocs() {
if [ "$outputs" = "out" ]; then return; fi;
2014-08-27 03:14:09 +04:00
echo "Looking for documentation to move between outputs"
_moveToOutput share/man "${!outputMan}"
_moveToOutput share/info "${!outputInfo}"
_moveToOutput share/doc "${!outputDoc}"
# outputs TODO: perhaps have outputDevDoc for developer docs
# and maybe allow _moveToOutput move to "/dev/trash" or similar
_moveToOutput share/gtk-doc "${!outputDoc}"
2014-08-25 17:30:46 +04:00
# Remove empty share directory.
if [ -d "$out/share" ]; then
rmdir "$out/share" --ignore-fail-on-non-empty
2014-08-25 17:30:46 +04:00
fi
}
2014-08-27 03:14:09 +04:00
# Move development-only stuff to the desired outputs.
_multioutDevs() {
if [ "$outputs" = "out" ] || [ -z "${moveToDev-1}" ]; then return; fi;
echo "Looking for development-only stuff to move to $outputDev"
2014-08-27 03:14:09 +04:00
_moveToOutput include "${!outputInclude}"
_moveToOutput lib/pkgconfig "${!outputDev}"
_moveToOutput share/pkgconfig "${!outputDev}"
# don't move libtool files yet
#_moveToOutput "lib/*.la" "${!outputDev}"
for f in "${!outputDev}"/{lib,share}/pkgconfig/*.pc; do
echo "Patching '$f' includedir to output ${!outputInclude}"
sed -i "/^includedir=/s,=\${prefix},=${!outputInclude}," "$f"
done
2014-08-25 17:30:46 +04:00
}
# Make the first output (typically "dev") propagate other outputs needed for development.
# Take the first, because that's what one gets when putting the package into buildInputs.
2014-08-27 03:14:09 +04:00
# Note: during the build, probably only the "native" development packages are useful.
# With current cross-building setup, all packages are "native" if not cross-building.
2014-08-25 17:30:46 +04:00
_multioutPropagateDev() {
if [ "$outputs" = "out" ]; then return; fi;
local outputFirst
for outputFirst in $outputs; do
break
done
# Default value: propagate binaries, includes and libraries
if [ -z "${propagatedOutputs+1}" ]; then
local po_dirty="$outputBin $outputInclude $outputLib"
set +o pipefail
propagatedOutputs=`echo "$po_dirty" \
| tr -s ' ' '\n' | grep -v -F "$outputFirst" \
| sort -u | tr '\n' ' ' `
set -o pipefail
elif [ -z "$propagatedOutputs" ]; then
return # variable was explicitly set to empty
2014-08-25 17:30:46 +04:00
fi
mkdir -p "${!outputFirst}"/nix-support
for output in $propagatedOutputs; do
echo -n " ${!output}" >> "${!outputFirst}"/nix-support/propagated-native-build-inputs
done
2014-08-25 17:30:46 +04:00
}