blob: ed38cae4de4e17c2de888eac3ec06765a4d7b655 [file] [log] [blame]
# Copyright 2018 The ChromiumOS Authors
# Distributed under the terms of the GNU General Public License v2
# NOTE: If you make changes to this file that require Rust code to
# be rebuilt, you can change the revision on virtual/rust-binaries
# to make that rebuild happen on the next build_packages run.
# @ECLASS: cros-rust.eclass
# @MAINTAINER:
# The ChromiumOS Authors <chromium-os-dev@chromium.org>
# @BUGREPORTS:
# Please report bugs via https://crbug.com/new (with component "Tools>ChromeOS-Toolchain")
# @VCSURL: https://chromium.googlesource.com/chromiumos/overlays/chromiumos-overlay/+/HEAD/eclass/@ECLASS@
# @BLURB: Eclass for fetching, building, and installing Rust packages.
if [[ -z ${_ECLASS_CROS_RUST} ]]; then
_ECLASS_CROS_RUST="1"
# Check for EAPI 7+.
case "${EAPI:-0}" in
[0123456]) die "unsupported EAPI (${EAPI}) in eclass (${ECLASS})" ;;
esac
# @ECLASS-VARIABLE: CROS_RUST_CRATE_NAME
# @DESCRIPTION:
# The name of the crate used by Cargo. This defaults to the package name.
: "${CROS_RUST_CRATE_NAME:=${PN}}"
# @ECLASS-VARIABLE: CROS_RUST_CRATE_VERSION
# @DESCRIPTION:
# The version of the crate used by Cargo. This defaults to PV. Note that
# cros-rust_get_crate_version can be used to get this information from the
# Cargo.toml but that is only available in src_* functions. Also, for -9999
# ebuilds this is handled in a special way; A symbolic link is used to point to
# the installed crate so it can be removed correctly.
: "${CROS_RUST_CRATE_VERSION:=${PV}}"
# @ECLASS-VARIABLE: CROS_RUST_OVERFLOW_CHECKS
# @PRE_INHERIT
# @DESCRIPTION:
# Enable integer overflow checks for this package. Packages that wish to
# disable integer overflow checks should set this value to 0. Integer overflow
# checks are always enabled when the cros-debug flag is set.
: "${CROS_RUST_OVERFLOW_CHECKS:=1}"
# @ECLASS-VARIABLE: CROS_RUST_REMOVE_DEV_DEPS
# @PRE_INHERIT
# @DESCRIPTION:
# Removes all the dev-dependencies from the Cargo.toml. This can break circular
# dependencies and help minimize how many dependent packages need to be added.
: "${CROS_RUST_REMOVE_DEV_DEPS:=}"
# @ECLASS-VARIABLE: CROS_RUST_REMOVE_TARGET_CFG
# @PRE_INHERIT
# @DESCRIPTION:
# Removes all the target. sections from the Cargo.toml except cfg(unix),
# cfg(linux), cfg(not(windows), and *-linux-gnu. Note that this does not handle
# more complicated cfg strings, so those cases should be handled manually
# instead of using this option.
: "${CROS_RUST_REMOVE_TARGET_CFG:=}"
# @ECLASS-VARIABLE: CROS_RUST_SUBDIR
# @DESCRIPTION:
# Subdir where the package is located. Only used by cros-workon ebuilds.
: "${CROS_RUST_SUBDIR:=${CROS_RUST_CRATE_NAME}}"
# @ECLASS-VARIABLE: CROS_RUST_TESTS
# @DESCRIPTION:
# An array of test executables to be run, which defaults to empty value and is
# set by invoking cros-rust_get_test_executables.
: "${CROS_RUST_TESTS:=}"
# @ECLASS-VARIABLE: CROS_RUST_HOST_TESTS
# @DESCRIPTION:
# An array of test executables that are built for cros-host, which defaults to
# empty value and is set by invoking cros-rust_get_host_test_executables.
# If it is empty when cros-rust_get_test_executables is called, it will be set
# to include tests not compiled for ${CHOST}.
: "${CROS_RUST_HOST_TESTS:=}"
# @ECLASS-VARIABLE: CROS_RUST_PLATFORM_TEST_ARGS
# @DESCRIPTION:
# An array of arguments to pass to platform2_test.py such as --no-ns-net,
# --no-ns-pid, or --run_as_root.
# @ECLASS-VARIABLE: CROS_RUST_TEST_DIRECT_EXEC_ONLY
# @DESCRIPTION:
# If set to yes, run the test only for amd64 and x86 (i.e. no emulation).
: "${CROS_RUST_TEST_DIRECT_EXEC_ONLY:="no"}"
# @ECLASS-VARIABLE: CROS_RUST_TEST_MULTIPROCESS
# @PRE_INHERIT
# @DESCRIPTION:
# If set to yes, run test binaries in parallel but without affecting
# `--test-threads`` on individual test binaries.
# TODO(b/293327433): Temporarily disabled due to flaky tests
: "${CROS_RUST_TEST_MULTIPROCESS:="no"}"
# @ECLASS-VARIABLE: CROS_RUST_PACKAGE_IS_HOT
# @DESCRIPTION:
# If set to a nonempty value, we will consider the binaries we compile to be
# hot, and optimize them more aggressively for speed. Please use the
# `cros_optimize_package_for_speed` function to set this, as that also applies
# the same settings for C and C++ code.
: "${CROS_RUST_PACKAGE_IS_HOT:=}"
# @ECLASS-VARIABLE: CROS_RUST_PREINSTALLED_REGISTRY_CRATE
# @DESCRIPTION:
# If set to a nonempty value, `cros-rust_src_unpack` will also copy sources from
# `${CROS_RUST_REGISTRY_DIR}` into `${S}`, and suppress any automatic publishing
# of Rust sources.
#
# TODO(gbiv): This should ideally `ln` from the registry, rather than `cp`.
# There's quite a bit that wants to write to the crate root though, and the
# registry should be immutable, so a cleanup is needed.
: "${CROS_RUST_PREINSTALLED_REGISTRY_CRATE:=}"
# @ECLASS-VARIABLE: CROS_RUST_NO_COVERAGE
# @DESCRIPTION:
# If set to a nonempty value, code coverage flags will be disabled regardless of
# the value of the `rust-coverage` setting. This is generally only useful if
# you're using a nonstandard Rust toolchain.
: "${CROS_RUST_COVERAGE_DISABLED:=}"
# @ECLASS-VARIABLE: CROS_RUST_INSTALL_SOURCES_ONLY
# @PRE_INHERIT
# @DESCRIPTION:
# If set to a nonempty value, cros-rust assumes that the package in question
# only exists to install source files. Setting this makes it an error to use
# `ecargo` in most ebuild functions. As a convenience, usage of `ecargo` is
# allowed in `src_test` of `FEATURES=test` builds. This allows packages to
# easily self-check using Cargo.
: "${CROS_RUST_INSTALL_SOURCES_ONLY:=}"
# @ECLASS-VARIABLE: CROS_RUST_FORCE_STATIC_LINK
# @DESCRIPTION:
# If set to a nonempty value, libstd will be forcibly statically linked. This
# harms code size, so is not recommended if your ebuild builds a binary that
# lands in ChromeOS' userland.
: "${CROS_RUST_FORCE_STATIC_LINK:=}"
# @ECLASS-VARIABLE: CROS_RUST_DISABLE_EDITION_CHECKS
# @PRE_INHERIT
# @DESCRIPTION:
# If set to a nonempty value, edition checks won't be enabled on this package.
# Our edition checks help keep CrOS on the newest Rust editions, which often
# enable cleaner code.
: "${CROS_RUST_DISABLE_EDITION_CHECKS:=}"
# @ECLASS-VARIABLE: CROS_RUST_EXTRA_ALLOWED_FEATURES
# @DESCRIPTION:
# Array of extra allowed Rust unstable features used during build. Please get
# approval from the toolchain team before setting it, as unstable features
# might be easily broken by toolchain updates.
if [[ ! -v CROS_RUST_EXTRA_ALLOWED_FEATURES ]]; then
CROS_RUST_EXTRA_ALLOWED_FEATURES=()
fi
inherit multiprocessing toolchain-funcs cros-constants cros-debug cros-sanitizers
IUSE="asan rust-coverage cros_host fuzzer lsan +lto msan +panic-abort test tsan ubsan"
REQUIRED_USE="?? ( asan lsan msan tsan )"
EXPORT_FUNCTIONS pkg_setup src_unpack src_prepare src_configure src_compile src_test src_install pkg_preinst pkg_postinst pkg_prerm
# virtual/rust-binaries is listed in both DEPEND and RDEPEND. Changing the
# version of virtual/rust-binaries forces a rebuild of everything that
# depends on it (that is, all compiled Rust code in ChromeOS).
DEPEND="
virtual/rust:1=
virtual/rust-binaries:=
"
RDEPEND="
virtual/rust-binaries:=
"
# Pull in dynamically linked libraries if we're going to use `-Cprefer-dynamic`.
# `-Cprefer-dynamic` is disabled for fuzzing and sanitizers since its usability
# is unclear with those, and it's disabled for the host since we care less about
# binary size there than on devices.
if [[ -z "${CROS_RUST_PACKAGE_IS_HOT}" && -z "${CROS_RUST_FORCE_STATIC_LINK}" ]]; then
RDEPEND+="
!cros_host? (
!test? ( !asan? ( !lsan? ( !msan? ( !tsan? ( !fuzzer? ( dev-lang/rust-dylibs:= ) ) ) ) ) )
)
"
fi
BDEPEND="
!cros_host? (
arm? ( cross-armv7a-cros-linux-gnueabihf/rust:= )
arm64? ( cross-aarch64-cros-linux-gnu/rust:= )
amd64? ( cross-x86_64-cros-linux-gnu/rust:= )
)
dev-lang/rust-host:=
"
_cros-rust_enable_edition_checks() {
[[ "${PV}" == 9999 && -z "${CROS_RUST_DISABLE_EDITION_CHECKS}" ]]
}
_cros-rust_enable_edition_checks && BDEPEND+="
dev-util/rust-edition-checker
"
# If we're just installing sources, no need to force dependencies on Rust
# (unless we're testing, as outlined in docs for
# CROS_RUST_INSTALL_SOURCES_ONLY).
if [[ -n "${CROS_RUST_INSTALL_SOURCES_ONLY}" ]]; then
DEPEND="test? ( ${DEPEND} )"
RDEPEND="test? ( ${RDEPEND} )"
BDEPEND="test? ( ${BDEPEND} )"
fi
CROS_RUST_REGISTRY_BASE="/usr/lib/cros_rust_registry"
ECARGO_HOME="${WORKDIR}/cargo_home"
CROS_RUST_REGISTRY_DIR="${CROS_RUST_REGISTRY_BASE}/store"
CROS_RUST_REGISTRY_INST_DIR="${CROS_RUST_REGISTRY_BASE}/registry"
# Crate owners directory. This has one file per crate in
# CROS_RUST_REGISTRY_INST_DIR that describes the package which installed the
# crate's link in CROS_RUST_REGISTRY_INST_DIR. This is needed to support our
# current preinst/postinst/prerm functions without introducing race conditions:
# - prerm will delete a symlink if the symlink is owned by the current package
# - preinst will delete a symlink regardless of ownership
# - postinst installs a new symlink and declares ownership of it
CROS_RUST_REGISTRY_OWNER_DIR="${CROS_RUST_REGISTRY_BASE}/owners"
# Ignore odr violations in unit tests in asan builds
# (https://github.com/rust-lang/rust/issues/41807).
export ASAN_OPTIONS="detect_odr_violation=0"
_cros-rust_acquire_fd_lock() {
local fd="$1"
local extra_args=( "${@:2}" )
flock --timeout=15 --conflict-exit-code=200 "${extra_args[@]}" "${fd}"
local returncode="$?"
case "${returncode}" in
0 )
return
;;
200 )
einfo "Acquiring the registry lock is taking a while."
einfo "If this command hangs indefinitely, you might have old processes hanging onto the lock."
flock "${extra_args[@]}" "${fd}" || die
;;
* )
die "Unexpected flock return code: ${returncode}"
;;
esac
}
# @FUNCTION: cros-rust_get_reg_lock
# @DESCRIPTION:
# Return the path to our rust registry lock file used to prevent races.
cros-rust_get_reg_lock() {
echo "${ROOT}${CROS_RUST_REGISTRY_BASE}/lock"
}
# @FUNCTION: cros-rust_pkg_setup
# @DESCRIPTION:
# Sets up the package. Particularly, makes sure the rust registry lock exits.
cros-rust_pkg_setup() {
debug-print-function "${FUNCNAME[0]}" "$@"
# This triggers a linter error SC2154 which says:
# "EBUILD_PHASE_FUNC is used but not defined inside this file"
# Since EBUILD_PHASE_FUNC comes from outside the file, that's ok
# shellcheck disable=SC2154
if [[ "${EBUILD_PHASE_FUNC}" != "pkg_setup" ]]; then
die "${FUNCNAME[0]}() should only be used in pkg_setup() phase"
fi
addwrite "$(cros-rust_get_reg_lock)"
_cros-rust_prepare_lock
# This is needed for CROS_WORKON_INCREMENTAL_BUILD to be honored.
if [[ -n "${CROS_WORKON_PROJECT}" ]]; then
cros-workon_pkg_setup
fi
}
# @FUNCTION: cros-rust_src_unpack
# @DESCRIPTION:
# Unpacks the package
cros-rust_src_unpack() {
debug-print-function "${FUNCNAME[0]}" "$@"
# If this is a cros-workon ebuild and hasn't been unpacked, then unpack it.
if [[ -n "${CROS_WORKON_PROJECT}" && ! -e "${S}" ]]; then
cros-workon_src_unpack
S+="/${CROS_RUST_SUBDIR}"
fi
if [[ -n "${CROS_RUST_PREINSTALLED_REGISTRY_CRATE}" ]]; then
local registry_dir="${ROOT}${CROS_RUST_REGISTRY_DIR}/${CROS_RUST_CRATE_NAME}-${CROS_RUST_CRATE_VERSION}"
[[ -d "${registry_dir}" ]] || die "Registry directory ${registry_dir} doesn't exist."
cp -r "${registry_dir}" "${S}" || die
fi
local archive
for archive in ${A}; do
case "${archive}" in
*.crate)
ebegin "Unpacking ${archive}"
ln -s "${DISTDIR}/${archive}" "${archive}.tar"
unpack "./${archive}.tar"
rm "${archive}.tar"
eend $?
;;
*)
unpack "${archive}"
;;
esac
done
if [[ -z "${LICENSE}" ]]; then
die "Missing LICENSE= setting in ebuild"
fi
if [[ "${LICENSE}" == "metapackage" ]]; then
die "LICENSE=metapackage is not allowed in crate ebuilds"
fi
# Set up the cargo config.
mkdir -p "${ECARGO_HOME}"
cat <<- EOF > "${ECARGO_HOME}/config"
[source.chromeos]
directory = "${SYSROOT}${CROS_RUST_REGISTRY_INST_DIR}"
[source.crates-io]
replace-with = "chromeos"
local-registry = "/nonexistent"
[build]
jobs = $(makeopts_jobs)
EOF
# When the target environment is different from the host environment,
# add a setting for the target environment.
if tc-is-cross-compiler; then
cat <<- EOF >> "${ECARGO_HOME}/config"
[target.${CBUILD}]
linker = "$(tc-getBUILD_CC)"
EOF
fi
# Tell cargo not to use terminal colors if NOCOLOR is set.
# Shellcheck thinks NOCOLOR is never defined.
# shellcheck disable=SC2154
if [[ "${NOCOLOR}" == true || "${NOCOLOR}" == yes ]]; then
cat <<- EOF >> "${ECARGO_HOME}/config"
[term]
color = "never"
EOF
fi
}
# @FUNCTION: cros-rust-patch-cargo-toml
# @USAGE: <path to Cargo.toml file>
# @DESCRIPTION:
# Patches the Cargo.toml at "${1}". This function supports
# "# provided by ebuild" macro and "# ignored by ebuild" macro for replacing
# and removing path dependencies.
#
# NOTE: the Cargo.toml will be modified in place. This is not compatible with
# CROS_WORKON_OUTOFTREE_BUILD.
cros-rust-patch-cargo-toml() {
local cargo_toml_path="${1}"
[[ -e "${cargo_toml_path}" ]] || die "Provided path doesn't exist"
# shellcheck disable=SC2154
if [[ "${CROS_WORKON_OUTOFTREE_BUILD}" == 1 ]]; then
die "CROS_WORKON_OUTOFTREE_BUILD=1 must not be set when using" \
"\`provided by ebuild\`"
fi
# '# provided by ebuild'
# Replace path dependencies with ones provided by their ebuild.
#
# For local developer builds, we want Cargo.toml to contain path
# dependencies on sibling crates within the same repository or elsewhere
# in the Chrome OS source tree. This enables developers to run `cargo
# build` and have dependencies resolve correctly to their locally
# checked out code.
#
# At the same time, some crates contained within the crosvm repository
# have their own ebuild independent of the crosvm ebuild so that they
# are usable from outside of crosvm. Ebuilds of downstream crates won't
# be able to depend on these crates by path dependency because that
# violates the build sandbox. We perform a sed replacement to eliminate
# the path dependency during ebuild of the downstream crates.
#
# The sed command says: in any line containing `# provided by ebuild`,
# please replace `path = "..."` with `version = "*"`. The intended usage
# is like this:
#
# [dependencies]
# data_model = { path = "../data_model" } # provided by ebuild
#
# This also works with `git` attributes:
# [dependencies]
# bar = { git = "https://www.foo.com", branch = "a" } # provided by ebuild
# foo = { git = "https://www.foo.com", rev = "1234567" } # provided by ebuild
# foo = { git = "https://www.foo.com" } # provided by ebuild
#
# '# ignored by ebuild'
# Emerge ignores "out-of-sandbox" [patch.crates-io] lines in Cargo.toml.
sed -i \
-e '/# ignored by ebuild/d' \
-e '/# provided by ebuild$/ {
s/\(path\|git\) = "[^"]*"/version = "*"/
s/,\? *\(branch\|rev\) = "[^"]*"//
}' \
"${cargo_toml_path}" || die
}
# @FUNCTION: cros-rust_src_prepare
# @DESCRIPTION:
# Prepares the src. This function supports "# provided by ebuild" macro and
# "# ignored by ebuild" macro for replacing and removing path dependencies
# with ones provided by their ebuild in Cargo.toml
# and Cargo.toml will be modified in place. If the macro is used in
# ${S}/Cargo.toml, CROS_WORKON_OUTOFTREE_BUILD can't be set to 1 in its ebuild.
cros-rust_src_prepare() {
debug-print-function "${FUNCNAME[0]}" "$@"
if grep -q "# provided by ebuild\|# ignored by ebuild" "${S}/Cargo.toml"; then
cros-rust-patch-cargo-toml "${S}/Cargo.toml"
fi
# Remove dev-dependencies and target.cfg sections within the Cargo.toml file
#
# The awk program reads the file line by line. If any line matches one of the
# matched section headers, it will skip every line a new section header is
# found that does not match one of the matched section headers.
#
# Awk cannot do in-place editing, so we write the result to a temporary
# file before replacing the input with that temp file.
if [[ "${CROS_RUST_REMOVE_DEV_DEPS}" == 1 ]] || [[ "${CROS_RUST_REMOVE_TARGET_CFG}" == 1 ]]; then
awk -v rm_dev_dep="${CROS_RUST_REMOVE_DEV_DEPS}" \
-v rm_target_cfg="${CROS_RUST_REMOVE_TARGET_CFG}" \
'{
# Stop skipping for a new section header, but check for another match.
if ($0 ~ /^\[/) {
skip = 0
}
# If rm_dev_dep is set, match section headers of the following forms:
# [token.dev-dependencies]
# [dev-dependencies.token]
# [dev-dependencies]
if (rm_dev_dep && ($0 ~ /^\[([^][]+\.)?dev-dependencies(\.[^][]+)?\]$/)) {
skip = 1
next
}
# If rm_target_cfg is set, match section headers prefixed by `[target.`,
# but exclude matches that contain any of `cfg(unix`, `cfg(linux`,
# `cfg(not(windows)`, or `-linux-gnu`.
if (rm_target_cfg && ($0 ~ /^\[target[.]/) && ($0 !~ /cfg[(](unix|linux|not[(]windows[)])|-linux-gnu/)) {
skip = 1
next
}
if (skip == 0) {
print
}
}' "${S}/Cargo.toml" > "${S}/Cargo.toml.stripped" || die
mv "${S}/Cargo.toml.stripped" "${S}/Cargo.toml"|| die
fi
default
}
_cros-rust_find_default_libstd_rlib() {
local libstds
read -r -a libstds < <(shopt -s nullglob; echo "/usr/lib/rustlib/${CHOST}/lib"/libstd-*.rlib) || die "no libstd.so found"
[[ "${#libstds[@]}" -ne 1 ]] && die "Expected exactly one libstd; found ${libstds[*]}"
echo "${libstds[0]}"
}
# @FUNCTION: cros-rust_configure_cargo
# @DESCRIPTION:
# Sets up cargo configuration and exports any environment variables needed
# during the build.
cros-rust_configure_cargo() {
debug-print-function "${FUNCNAME[0]}"
sanitizers-setup-env
cros-debug-add-NDEBUG
if [[ -n "${CROS_WORKON_PROJECT}" ]]; then
# Use a sub directory to avoid unintended interactions with platform.eclass.
export CARGO_TARGET_DIR="$(cros-workon_get_build_dir)/cros-rust"
mkdir -p "${CARGO_TARGET_DIR}"
else
export CARGO_TARGET_DIR="${WORKDIR}"
fi
export CARGO_HOME="${ECARGO_HOME}"
export HOST="${CBUILD}"
export HOST_CC="$(tc-getBUILD_CC)"
# PKG_CONFIG_ALLOW_CROSS is required by pkg-config.
# https://github.com/rust-lang/pkg-config-rs/issues/41.
# Since cargo will overwrites $HOST with "" when building pkg-config, we
# need to set it regardless of the value of tc-is-cross-compiler here.
export PKG_CONFIG_ALLOW_CROSS=1
export PKG_CONFIG="$(tc-getPKG_CONFIG)"
export TARGET="${CHOST}"
export TARGET_CC="$(tc-getCC)"
# Intended use case:
# - Crate A generates sources when it is emerged from input files
# that are only accessible when it emerges.
# - Crate B depends on crate A, and this is reflected in the
# ebuild for crate B.
# (Examples: cros-dbus-bindings or bindgen for *-sys)
#
# The following scenarios are supported and need to work:
# - local `cargo build` for crate A
# - local `cargo build` for crate B
# - emerge A
# - emerge B
#
# Add CROS_RUST environment variable to support the `emerge B`
# case, since crate B can't access pre-generated source
# in emerge, the build.rs script for crate A will skip the
# source generation if both of the following are true:
# - The generated source exists
# - `CROS_RUST=1`
export CROS_RUST="1"
# There is a memory leak in libbacktrace:
# https://github.com/rust-lang/rust/issues/59125
cros-rust_use_sanitizers || export RUST_BACKTRACE=1
# NOTE(b/309651697): panic_info_message is specifically allowed for
# libchromeos-rs. Once that library no longer has use for this feature,
# it will be removed from this list.
local unstable_features="sanitizer,panic_info_message"
[[ "${#CROS_RUST_EXTRA_ALLOWED_FEATURES[@]}" -ne 0 ]] && \
unstable_features+=$(printf ",%s" "${CROS_RUST_EXTRA_ALLOWED_FEATURES[@]}")
# We want to split the flags since it's a command line as a scalar.
# shellcheck disable=SC2206
local rustflags=(
${CROS_BASE_RUSTFLAGS}
# We want debug info even in release builds.
"-Cdebuginfo=2"
"-Cstrip=none"
# Please get approval from the toolchain team before setting
# CROS_RUST_EXTRA_ALLOWED_FEATURES in your package, as unstable features
# might be easily broken by toolchain updates.
"-Zallow-features=${unstable_features}"
)
# RUSTFLAGS that should only apply to binaries compiled for '${CHOST}'.
# The 'rustflags' variable applies to all binaries, incl. '${CBUILD}'
# and others (e.g., baremetal)
local rustflags_chost=()
local skip_dyn_link=false opt_level=s
# Treat all host binaries as hot enough to opt for speed. The difference
# in size is minuscule compared to the rest of the SDK, and we don't
# have hotness data to guide which to optimize.
if [[ -n "${CROS_RUST_PACKAGE_IS_HOT}" ]] || use cros_host; then
skip_dyn_link=true
opt_level=3
elif use test || use asan || use msan || use lsan || use tsan || use fuzzer; then
skip_dyn_link=true
elif ! use panic-abort; then
ewarn "USE=panic-abort is recommended to save binary size"
skip_dyn_link=true
elif [[ -n "${CROS_RUST_FORCE_STATIC_LINK}" ]]; then
skip_dyn_link=true
fi
rustflags+=( "-Copt-level=${opt_level}" )
if "${skip_dyn_link}"; then
use lto && rustflags+=(
"-Clto=thin"
"-Cllvm-args=--import-instr-limit=30"
# Cargo sets -Cembed-bitcode to no because it does not
# know that we want to use LTO. Because -Clto requires
# -Cembed-bitcode=yes, set it explicitly.
"-Cembed-bitcode=yes"
)
# On targets, libstd.so from the panic=abort sysroot will be
# visible even when panic=unwind. rustc errors out over this
# ambiguity; explicitly specify the panic=unwind one.
if ! use cros_host; then
rustflags_chost+=(
"--extern"
"std=$(_cros-rust_find_default_libstd_rlib)"
)
fi
else
rustflags+=(
"-Cprefer-dynamic"
"-Clto=no"
"--sysroot=/usr/lib/rust-sysroots/panic-abort"
)
fi
# Set the panic=abort flag if it is turned on for the package.
if use panic-abort; then
# But never abort during tests.
use test || rustflags+=( -Cpanic=abort )
fi
if use cros-debug || [[ "${CROS_RUST_OVERFLOW_CHECKS}" == "1" ]]; then
rustflags+=( -Coverflow-checks=on )
fi
use cros-debug && rustflags+=( -Cdebug-assertions=on )
if use rust-coverage && [[ -z "${CROS_RUST_COVERAGE_DISABLED}" ]]; then
# TODO(b/215596245) Use rust-coverage use flag for rust packages.
rustflags+=( -Cinstrument-coverage )
else
# Remap source directories because of the following:
# * crashes from panics are grouped across different boards
# * the remapped strings are shorter resulting in smaller binaries
# NOTE: this is disabled with code coverage enabled since it is
# incompatible.
rustflags+=(
# This shouldn't be needed because cargo includes local sources
# with relative paths, but just-in-case remap the source directory.
"--remap-path-prefix=${S}=[${PN}]"
# Remap the cros_rust_registry/registry directory.
"--remap-path-prefix=${SYSROOT}${CROS_RUST_REGISTRY_INST_DIR}=[REGISTRY]"
# Remap the target directory for generated sources.
"--remap-path-prefix=${CARGO_TARGET_DIR}=[TARGET]"
)
fi
# Rust compiler is not exporting the __asan_* symbols needed in
# asan builds. Force export-dynamic linker flag to export __asan_* symbols
# https://crbug.com/1085546
use asan && rustflags+=( -Zsanitizer=address -Clink-arg="-Wl,-export-dynamic" )
use lsan && rustflags+=( -Zsanitizer=leak )
use msan && rustflags+=( -Zsanitizer=memory -Clink-arg="-Wl,--allow-shlib-undefined")
use tsan && rustflags+=( -Zsanitizer=thread )
use ubsan && rustflags+=( -Clink-arg=-fsanitize=undefined )
if use fuzzer; then
rustflags+=(
# We can get segfaults unless we turn this off; see
# https://github.com/rust-lang/rust/issues/99886
# Presumably we can remove this once that bug is
# resolved.
-Cllvm-args=-experimental-debug-variable-locations=0
--cfg fuzzing
-Cpasses=sancov-module
-Cllvm-args=-sanitizer-coverage-level=4
-Cllvm-args=-sanitizer-coverage-inline-8bit-counters
-Cllvm-args=-sanitizer-coverage-trace-compares
-Cllvm-args=-sanitizer-coverage-pc-table
-Cllvm-args=-sanitizer-coverage-trace-divs
-Cllvm-args=-sanitizer-coverage-trace-geps
-Cllvm-args=-sanitizer-coverage-prune-blocks=0
-Clink-arg="-Wl,--no-gc-sections"
)
fi
# Add EXTRA_RUSTFLAGS to the current rustflags. This lets us emerge rust
# packages with locally exported flags for testing purposes as:
# `EXTRA_RUSTFLAGS="<flags>" emerge-$BOARD <package>`
# We want to split the flags since it's a command line as a scalar.
# shellcheck disable=SC2206
rustflags+=( ${EXTRA_RUSTFLAGS:=} )
# Ensure RUSTFLAGS is *not* set in the environment.
# If it is, it will override the flags we configure below. See:
# https://doc.rust-lang.org/cargo/reference/config.html#buildrustflags
# Ebuilds should set their custom rustflags in cargo configuration.
# Developers can pass EXTRA_RUSTFLAGS for one-off builds as above.
unset RUSTFLAGS
# Add rustflags to the cargo configuration.
# This 'cfg(all())' [target] section will apply to *all* targets, CHOST
# and CBUILD.
# TODO(b/330537088): some flags above are not applicable to all targets,
# they should be configured into suitable [target] sections.
local rustflags_list=$(printf " %s,\n" "${rustflags[@]@Q}")
local rustflags_chost_list=""
if [[ "${#rustflags_chost[@]}" -ne 0 ]]; then
rustflags_chost_list=$(printf " %s,\n" "${rustflags_chost[@]@Q}")
fi
cat <<- EOF >> "${ECARGO_HOME}/config"
[target.'cfg(all())']
rustflags = [
${rustflags_list}
]
[target.${CHOST}]
linker = "$(tc-getCC)"
rustflags = [
${rustflags_chost_list}
]
EOF
}
# @FUNCTION: cros-rust_update_cargo_lock
# @DESCRIPTION:
# Regenerates/removes the Cargo.lock file to ensure cargo uses the dependency
# versions from our local registry, and checks the rustc version to make sure
# intermediates aren't mixed across rustc versions.
cros-rust_update_cargo_lock() {
debug-print-function "${FUNCNAME[0]}"
if [[ -n "${CROS_WORKON_PROJECT}" ]]; then
# Force an update the Cargo.lock file.
ecargo generate-lockfile
# Shellcheck thinks CROS_WORKON_INCREMENTAL_BUILD is never
# defined.
# shellcheck disable=SC2154
if [[ "${CROS_WORKON_INCREMENTAL_BUILD}" == "1" ]]; then
local previous_lockfile="${CARGO_TARGET_DIR}/Cargo.lock.prev"
local previous_rustc="${CARGO_TARGET_DIR}/rustc.ver"
local rustc_ver="$(rust-toolchain-version)"
# If any of the dependencies have changed, clear the incremental results.
if [[ ! -f "${previous_lockfile}" ]] ||
[[ ! -f "${previous_rustc}" ]] ||
[[ "$(< "${previous_rustc}")" != "${rustc_ver}" ]] ||
! cmp Cargo.lock "${previous_lockfile}" ; then
# This will print errors for the .crate files, but that is OK.
rm -rf "${CARGO_TARGET_DIR}"
mkdir -p "${CARGO_TARGET_DIR}"
cp Cargo.lock "${previous_lockfile}" || die
echo "${rustc_ver}" > "${previous_rustc}" || die
fi
fi
else
# Remove 3rd party lockfiles.
rm -f Cargo.lock
fi
}
# @FUNCTION: cros-rust_src_configure
# @DESCRIPTION:
# Configures the source and exports any environment variables needed during the
# build.
cros-rust_src_configure() {
debug-print-function "${FUNCNAME[0]}"
cros-rust_configure_cargo
cros-rust_update_cargo_lock
if _cros-rust_enable_edition_checks; then
local message
message="$(cd "${S}" && rust-edition-checker)" || die
[[ -n "${message}" ]] && ewarn "${message}"
fi
default
}
# @FUNCTION: cros-rust_use_sanitizers
# @DESCRIPTION:
# Checks whether sanitizers are being used.
cros-rust_use_sanitizers() {
use_sanitizers || use lsan
}
# @FUNCTION: ecargo
# @USAGE: <args to cargo>
# @DESCRIPTION:
# Call cargo with the specified command line options. This is an error to call
# if CROS_RUST_INSTALL_SOURCES_ONLY is set, unless it's called during the
# src_test phase func.
ecargo() {
if [[ -n "${CROS_RUST_INSTALL_SOURCES_ONLY}" && "${EBUILD_PHASE_FUNC}" != "src_test" ]]; then
die "CROS_RUST_INSTALL_SOURCES_ONLY packages may only call cargo from src_test."
fi
local args=( "$@" )
debug-print-function "${FUNCNAME[0]}" "${args[@]}"
addwrite Cargo.lock
echo cargo -v "${args[@]}" >&2
(
# Acquire a shared (read only) lock since this does not modify
# the registry.
_cros-rust_acquire_fd_lock 100 --shared
cargo -v "${args[@]}" || die
) 100<"$(cros-rust_get_reg_lock)"
}
# @FUNCTION: write_clippy
# @INTERNAL
# @DESCRIPTION:
# Executes cargo clippy and writes lints to file
_ecargo_write_clippy() {
# TODO(crbug.com/1194200): we should stop using /tmp for this sort of thing
local clippy_output_base="/tmp/cargo_clippy/${CATEGORY}"
mkdir -p "${clippy_output_base}"
# FIXME(crbug.com/1195313): rustc sysroot may not contain dependencies
local sysroot_old="${SYSROOT}"
SYSROOT=$(rustc --print sysroot)
echo "{\"package_path\":\"${S}\"}" > "${clippy_output_base}/${PF}.json"
ecargo clippy --message-format json --target="${CHOST}" --release \
--manifest-path="${S}/Cargo.toml" >> "${clippy_output_base}/${PF}.json"
export SYSROOT="${sysroot_old}"
}
# @FUNCTION: ecargo_build
# @USAGE: <args to cargo build>
# @DESCRIPTION:
# Call `cargo build` with the specified command line options.
ecargo_build() {
ecargo build --target="${CHOST}" --release "$@"
# FIXME(b/191687433): refactor ENABLE_RUST_CLIPPY to be easier to enable/disable then remove the platform2 check
if [[ -n "${ENABLE_RUST_CLIPPY}" && "${CROS_WORKON_PROJECT}" == "chromiumos/platform2" ]]; then
_ecargo_write_clippy
fi
}
# @FUNCTION: ecargo_build_fuzzer
# @DESCRIPTION:
# Call `cargo build` with fuzzing options enabled.
ecargo_build_fuzzer() {
local fuzzer_libdir="$(dirname "$($(tc-getCC) -print-libgcc-file-name)")"
local fuzzer_arch="${ARCH}"
if [[ "${ARCH}" == "amd64" ]]; then
fuzzer_arch="x86_64"
fi
local link_args=(
-Clink-arg="-L${fuzzer_libdir}"
-Clink-arg="-lclang_rt.fuzzer-${fuzzer_arch}"
-Clink-arg="-lc++"
-Clink-arg="-Wl,-export-dynamic"
)
# The `rustc` subcommand for cargo allows us to set some extra flags for
# the current package without setting them for all `rustc` invocations.
# On the other hand the flags in the RUSTFLAGS environment variable are set
# for all `rustc` invocations.
ecargo rustc --target="${CHOST}" --release "$@" -- "${link_args[@]}"
}
# @FUNCTION: cros_rust_platform_test_command
# @USAGE: <action> <bin> [-- [<test-args> ...]]
# @DESCRIPTION:
# Prints the platform2_test.py command line to execute the specified test binary
cros_rust_platform_test_command() {
local platform2_test_py="${CHROOT_SOURCE_ROOT}/src/platform2/common-mk/platform2_test.py"
local action="$1"
local bin="$2"
if [[ "$#" -gt 2 && "$3" != "--" ]]; then
die "Need to use -- to separate program args"
fi
local cmd=(
"${platform2_test_py}"
--action="${action}"
--strategy=unprivileged
--user=root
)
if use cros_host || has "${bin}" "${CROS_RUST_HOST_TESTS[@]}"; then
cmd+=(
"--host"
--sysroot="${BROOT}"
)
else
cmd+=( --sysroot="${SYSROOT}" )
fi
cmd+=( "${CROS_RUST_PLATFORM_TEST_ARGS[@]}" )
if [[ -n "${bin}" ]]; then
# $3 is "--" and anything that follows is passed to the test.
cmd+=(
"--"
"${bin}"
"${@:4}"
)
fi
printf "%q " "${cmd[@]}"
}
# @FUNCTION: ecargo_test
# @USAGE: <args to cargo test>
# @DESCRIPTION:
# Call `cargo test` with the specified command line options.
ecargo_test() {
local test_dir="${CARGO_TARGET_DIR}/ecargo-test"
local profile_flag=""
if ! has "--profile" "$@"; then
profile_flag="--release"
fi
if has "--no-run" "$@"; then
debug-print-function ecargo test --target="${CHOST}" --target-dir \
"${test_dir}" "${profile_flag}" "$@"
ecargo test --target="${CHOST}" --target-dir \
"${test_dir}" "${profile_flag}" "$@"
else
cros-rust_get_test_executables "$@"
local x=0
for (( x = 0; x <= $#; x++ )); do
if [[ ${!x} == "--" ]]; then
break
fi
done
local test_args=( "${@:x}" )
# Make sure there is a separator before --test-threads.
if [[ "${#test_args[@]}" == 0 ]]; then
test_args=( -- )
fi
# Limit the number of test threads if they are not limited already.
if [[ " ${test_args[*]}" != *" --test-threads"* ]]; then
test_args+=( "--test-threads=$(makeopts_jobs)" )
fi
local jobs=1
if [[ "${CROS_RUST_TEST_MULTIPROCESS}" == "yes" ]]; then
jobs="$(makeopts_jobs)"
fi
local testfile
for testfile in "${CROS_RUST_TESTS[@]}"; do
cros_rust_platform_test_command run "${testfile}" "${test_args[@]}"
# Print a NUL delimiter to separate each command.
printf "\0"
done | xargs -0 -P "${jobs}" --verbose -I '{}' bash -x -c "{}" || die
fi
}
# @FUNCTION: cros-rust_get_test_executables
# @USAGE: <args to cargo test>
# @DESCRIPTION:
# Call `ecargo_test` with '--no-run' and '--message-format=json' arguments.
# Then, use jq to parse and store all the test executables in a global array.
cros-rust_get_test_executables() {
# Make sure all the targets are built before generating the json. This ensures
# any error messages will not be hidden.
ecargo_test --no-run "$@" || die
mapfile -t CROS_RUST_TESTS < \
<(ecargo_test --no-run --message-format=json "$@" | \
jq -r 'select(.profile.test == true) | .filenames[]')
# Cargo puts tests not compiled for the SYSROOT in ecargo-test/release.
if [[ -z "${CROS_RUST_HOST_TESTS}" ]]; then
local testfile
for testfile in "${CROS_RUST_TESTS[@]}"; do
if [[ "${testfile}" == "${CARGO_TARGET_DIR}/ecargo-test/release"* ]]; then
CROS_RUST_HOST_TESTS+=( "${testfile}" )
fi
done
fi
}
# @FUNCTION: cros-rust_get_host_test_executables
# @USAGE: <args to cargo test>
# @DESCRIPTION:
# Call `ecargo_test` with '--no-run' and '--message-format=json' arguments.
# Then, use jq to parse and store the test executables in a global array.
cros-rust_get_host_test_executables() {
mapfile -t CROS_RUST_HOST_TESTS < \
<(ecargo_test --no-run --message-format=json "$@" | \
jq -r 'select(.profile.test == true) | .filenames[]')
}
# @FUNCTION: cros-rust_publish
# @USAGE: [crate name] [crate version]
# @DESCRIPTION:
# Install a library crate to the local registry store. Should only be called
# from within a src_install() function.
# This triggers a linter error SC2120 which says:
# "rust_publish references arguments, but none are ever passed"
# In this case, we will use without arguments to get a default value, but other
# usages exist in other files that do use arguments, so there is no problem.
# shellcheck disable=SC2120
cros-rust_publish() {
debug-print-function "${FUNCNAME[0]}" "$@"
[[ -n "${CROS_RUST_PREINSTALLED_REGISTRY_CRATE}" ]] && \
die "cros-rust_publish should not be called for preinstalled registry crates"
if [[ "${EBUILD_PHASE_FUNC}" != "src_install" ]]; then
die "${FUNCNAME[0]}() should only be used in src_install() phase"
fi
local default_version="${CROS_RUST_CRATE_VERSION}"
if [[ "${default_version}" == "9999" ]]; then
# This triggers a linter error SC2119 which says:
# "Use foo "$@" if function's $1 should mean script's $1"
# In this case, cros-rust_get_crate_version without arguments retrieves the
# default value which is desired, so this warning can be ignored.
# shellcheck disable=SC2119
default_version="$(cros-rust_get_crate_version)"
fi
local name="${1:-${CROS_RUST_CRATE_NAME}}"
local version="${2:-${default_version}}"
# Cargo.toml.orig is now reserved by `cargo package`.
if [[ -e Cargo.toml.orig ]]; then
# Don't try to delete it if it isn't present, because that can
# be a permission error in the Portage sandbox.
rm -f Cargo.toml.orig || die
fi
if [[ -n "${CROS_WORKON_PROJECT}" ]]; then
[[ -e "${FILESDIR}/chromeos-version.sh" ]] || die \
"Missing chromeos-version.sh. Please add one for installation to work properly."
fi
# Create the .crate file.
ecargo package --allow-dirty --no-metadata --no-verify --offline || die
# Unpack the crate we just created into the directory registry.
local crate="${CARGO_TARGET_DIR}/package/${name}-${version}.crate"
mkdir -p "${D}/${CROS_RUST_REGISTRY_DIR}"
pushd "${D}/${CROS_RUST_REGISTRY_DIR}" > /dev/null || die
tar xf "${crate}" || die
# Calculate the sha256sum since cargo will want this later.
local shasum="$(sha256sum "${crate}" | cut -d ' ' -f 1)"
local dir="${name}-${version}"
local checksum="${T}/${name}-${version}-checksum.json"
# Calculate the sha256 hashes of all the files in the crate. Sort the
# files for determinism.
# This triggers a linter error SC2207 which says:
# "Prefer mapfile or read -a to split command
# output (or quote to avoid splitting)."
# In this case, cros-rust_get_crate_version no argument retrieves the
# default value which is desired, so this warning can be ignored.
# shellcheck disable=SC2207
local files=( $(find "${dir}" -type f | sort) )
[[ "${#files[@]}" == "0" ]] && die "Could not find crate files for ${name}"
# Now start filling out the checksum file.
printf '{\n\t"package": "%s",\n\t"files": {\n' "${shasum}" > "${checksum}"
local idx=0
local f
for f in "${files[@]}"; do
shasum="$(sha256sum "${f}" | cut -d ' ' -f 1)"
printf '\t\t"%s": "%s"' "${f#"${dir}"/}" "${shasum}" >> "${checksum}"
# The json parser is unnecessarily strict about not allowing
# commas on the last line so we have to track this ourselves.
idx="$((idx+1))"
if [[ "${idx}" == "${#files[@]}" ]]; then
printf '\n' >> "${checksum}"
else
printf ',\n' >> "${checksum}"
fi
done
printf "\t}\n}\n" >> "${checksum}"
popd > /dev/null || die
insinto "${CROS_RUST_REGISTRY_DIR}/${name}-${version}"
newins "${checksum}" .cargo-checksum.json
# We want the Cargo.toml.orig file to be world readable.
fperms 0644 "${CROS_RUST_REGISTRY_DIR}/${name}-${version}/Cargo.toml.orig"
# Symlink the 9999 version to the version installed by the crate.
if [[ "${CROS_RUST_CRATE_VERSION}" == "9999" && "${version}" != "9999" ]]; then
dosym "${name}-${version}" "${CROS_RUST_REGISTRY_DIR}/${name}-9999"
fi
}
# @FUNCTION: cros-rust_get_build_dir
# @DESCRIPTION:
# Return the path to the directory where build artifacts are available.
cros-rust_get_build_dir() {
echo "${CARGO_TARGET_DIR}/${CHOST}/release"
}
# @FUNCTION: cros_rust_is_direct_exec
# @DESCRIPTION:
# Return true if the compiled executables are expected to run on this platform.
cros_rust_is_direct_exec() {
use amd64 || use x86
}
cros-rust_src_compile() {
debug-print-function "${FUNCNAME[0]}" "$@"
# Skip non cros-workon packages.
[[ -z "${CROS_WORKON_PROJECT}" ]] && return 0
ecargo_build "$@"
}
cros-rust_src_test() {
debug-print-function "${FUNCNAME[0]}" "$@"
if [[ "${CROS_RUST_TEST_DIRECT_EXEC_ONLY}" == "yes" ]] && ! cros_rust_is_direct_exec; then
ewarn "Skipping unittests for non-x86: ${PN}"
return 0
fi
eval "$(cros_rust_platform_test_command "pre_test")"
ecargo_test "$@"
}
cros-rust_src_install() {
debug-print-function "${FUNCNAME[0]}" "$@"
# This triggers a linter error SC2119 which says:
# "Use cros-rust_publish "$@" if function's $1 should mean script's $1"
# Here we will use without arguments to get a default value so there is no problem
# shellcheck disable=SC2119
cros-rust_publish
}
# @FUNCTION: _cros-rust_prepare_lock
# @INTERNAL
# @DESCRIPTION:
# Create registry lock files. This should only be called inside pkg_* functions
# to ensure the lock file is owned by root. The permissions are set to 644 so
# that $PORTAGE_USERNAME:portage will be able to obtain a shared lock inside
# src_* functions.
_cros-rust_prepare_lock() {
if [[ "$(id -u)" -ne 0 ]]; then
die "_cros-rust_prepare_lock should only be called inside pkg_* functions."
fi
local lock="$(cros-rust_get_reg_lock)"
local dir="$(dirname "${lock}")"
mkdir -p "${dir}" || die
chmod 755 "${dir}" || die
touch "${lock}" || die
chmod 644 "${lock}" || die
}
# @FUNCTION: _cleanup_registry_link
# @INTERNAL
# @USAGE: force [crate name] [crate version]
# @DESCRIPTION:
# Unlink a library crate from the local registry. This is repeated in the prerm
# and preinst stages. If force is nonempty, the link will be cleaned up
# regardless of declared ownership. Otherwise, ownership will be respected.
_cleanup_registry_link() {
local force="$1"
local name="${2:-${CROS_RUST_CRATE_NAME}}"
local version="${3:-${CROS_RUST_CRATE_VERSION}}"
local crate="${name}-${version}"
local crate_dir="${ROOT}${CROS_RUST_REGISTRY_DIR}/${crate}"
if [[ "${version}" == "9999" && -L "${crate_dir}" ]]; then
crate="$(basename "$(readlink -f "${crate_dir}")")"
fi
local registry_dir="${ROOT}${CROS_RUST_REGISTRY_INST_DIR}"
local link="${registry_dir}/${crate}"
# Add a check to avoid spamming when it doesn't exist (e.g. binary crates).
if [[ -L "${link}" ]]; then
# Acquire a exclusive lock since this modifies the registry.
_cros-rust_prepare_lock
(
local owner="${ROOT}${CROS_RUST_REGISTRY_OWNER_DIR}/${crate}"
local removed
_cros-rust_acquire_fd_lock 100 --exclusive
if [[ -n ${force} ]] || [[ $(< "${owner}") == "${PF}" ]]; then
rm -f "${link}" "${owner}" || die
removed=1
fi
flock -u 100 || die
if [[ -n "${removed}" ]]; then
einfo "Removed ${crate} from Cargo registry"
else
einfo "${crate} removal from Cargo registry" \
"skipped due to new symlink owner"
fi
) 100<"$(cros-rust_get_reg_lock)"
fi
}
# @FUNCTION: _create_registry_link
# @INTERNAL
# @USAGE: [crate name] [crate version]
# @DESCRIPTION:
# Link a library crate from the local registry. This is performed in the
# postinst stage.
_create_registry_link() {
local name="${1:-${CROS_RUST_CRATE_NAME}}"
local version="${2:-${CROS_RUST_CRATE_VERSION}}"
local crate="${name}-${version}"
local crate_dir="${ROOT}${CROS_RUST_REGISTRY_DIR}/${crate}"
local registry_dir="${ROOT}${CROS_RUST_REGISTRY_INST_DIR}"
if [[ "${version}" == "9999" && -L "${crate_dir}" ]]; then
crate_dir="$(readlink -f "${crate_dir}")"
crate="$(basename "${crate_dir}")"
fi
# Only install the link if there is a library crates to register. This
# avoids dangling symlinks in the case that this only installs
# executables.
if [[ -e "${crate_dir}" ]]; then
local owners_dir="${ROOT}${CROS_RUST_REGISTRY_OWNER_DIR}"
einfo "Linking ${crate} into Cargo registry at ${registry_dir}"
mkdir -p "${registry_dir}" "${owners_dir}"
# A redundant link presence check is used inside the lock
# because we do not want to lock if we don't have to, but there
# is a time-of-check to time-of-use issue that shows up if the
# link presence check is not in the lock (two ebuilds may try to
# create the same lock with one succeeding and the other failing
# because the link already exists).
(
local dest="${registry_dir}/${crate}"
local owners="${owners_dir}/${crate}"
_cros-rust_acquire_fd_lock 100 --exclusive
if [[ ! -L "${dest}" ]]; then
ln -srT "${crate_dir}" "${dest}" || die
fi
echo -n "${PF}" > "${owners}" || die
) 100<"$(cros-rust_get_reg_lock)"
fi
}
# @FUNCTION: cros-rust_cleanup_vendor_registry_links
# @DESCRIPTION: force [crate name ...]
# This cleans up the given vendor directories. If force is nonempty, their links
# will be cleaned up regardless of declared ownership. Otherwise, ownership will
# be respected.
cros-rust_cleanup_vendor_registry_links() {
local force="$1"
shift
local dirs=( "$@" )
local owner_dir="${ROOT}${CROS_RUST_REGISTRY_OWNER_DIR}"
# The registry might not exist. In that case, great; skip everything.
# Check the owner dir rather than the registry dir, since the registry
# dir is created before the owner dir, and both are needed for the logic
# below.
[[ -e "${owner_dir}" ]] || return 0
local dir remove_paths=()
for dir in "${dirs[@]}"; do
remove_paths+=( "${dir##*/}" )
done
(
local owned_files=()
cd "${owner_dir}" || die
_cros-rust_acquire_fd_lock 100 --exclusive
if [[ -n "${force}" ]]; then
owned_files=( "${remove_paths[@]}" )
else
for path in "${remove_paths[@]}"; do
if [[ "$(< "${path}" 2>/dev/null)" == "${PF}" ]]; then
owned_files+=( "${path}" )
fi
done
fi
rm -f "${owned_files[@]}" || die
cd "${ROOT}${CROS_RUST_REGISTRY_INST_DIR}" || die
rm -f "${owned_files[@]}" || die
) 100<"$(cros-rust_get_reg_lock)"
}
# @FUNCTION: cros-rust_create_vendor_registry_links
# @DESCRIPTION: [crate name ...]
# creates a registry link for every crate in [vendor tree base]. [vendor tree
# base] should be a path to the root of a Cargo vendor/ directory. Intended
# specifically for use by third-party-crates-src.
#
# This assumes that all of the crates in [vendor tree base] have been installed
# in the registry directory.
cros-rust_create_vendor_registry_links() {
local dirs=( "$@" )
# If the registry itself doesn't exist, portage has masked
# installations to it (e.g., we're in `build_image`, and installing
# registry symlinks is useless). Skip it.
[[ -e "${ROOT}${CROS_RUST_REGISTRY_DIR}" ]] || return 0
local registry_dir="${ROOT}${CROS_RUST_REGISTRY_INST_DIR}"
local owner_dir="${ROOT}${CROS_RUST_REGISTRY_OWNER_DIR}"
mkdir -p "${registry_dir}" "${owner_dir}" || die
# Use a subshell so we can conveniently lock the registry lock only once.
(
local crate_srcs="${ROOT}${CROS_RUST_REGISTRY_DIR}"
local crate crate_src
local crates_dne=()
_cros-rust_acquire_fd_lock 100 --exclusive
for crate in "${dirs[@]}"; do
crate_src="${crate_srcs}/${crate}"
# Ensure crates exist prior to creating links. These
# should always exist.
if [[ -e "${crate_src}" ]]; then
ln -srTf "${crate_src}" "${registry_dir}/${crate}" || die
echo "${PF}" > "${owner_dir}/${crate}" || die
else
crates_dne+=( "${crate_src}" )
fi
done
if [[ "${#crates_dne[@]}" -ne 0 ]]; then
die "Created links with crates that DNE: ${crates_dne[*]}"
fi
) 100<"$(cros-rust_get_reg_lock)"
}
# @FUNCTION: cros-rust_pkg_preinst
# @USAGE: [crate name] [crate version]
# @DESCRIPTION:
# Make sure a library crate isn't linked in the local registry prior to the
# install step to avoid races.
cros-rust_pkg_preinst() {
[[ -n "${CROS_RUST_PREINSTALLED_REGISTRY_CRATE}" ]] && return
debug-print-function "${FUNCNAME[0]}" "$@"
if [[ "${EBUILD_PHASE_FUNC}" != "pkg_preinst" ]]; then
die "${FUNCNAME[0]}() should only be used in pkg_preinst() phase"
fi
# Forcibly remove any existing link.
_cleanup_registry_link 1 "$@"
}
# @FUNCTION: cros-rust_pkg_postinst
# @USAGE: [crate name] [crate version]
# @DESCRIPTION:
# Install a library crate in the local registry store into the registry,
# making it visible to Cargo.
cros-rust_pkg_postinst() {
debug-print-function "${FUNCNAME[0]}" "$@"
[[ -n "${CROS_RUST_PREINSTALLED_REGISTRY_CRATE}" ]] && return
if [[ "${EBUILD_PHASE_FUNC}" != "pkg_postinst" ]]; then
die "${FUNCNAME[0]}() should only be used in pkg_postinst() phase"
fi
_create_registry_link "$@"
}
# @FUNCTION: cros-rust_pkg_prerm
# @USAGE: [crate name] [crate version]
# @DESCRIPTION:
# Unlink a library crate from the local registry unless another package now owns
# the link.
cros-rust_pkg_prerm() {
debug-print-function "${FUNCNAME[0]}" "$@"
[[ -n "${CROS_RUST_PREINSTALLED_REGISTRY_CRATE}" ]] && return
if [[ "${EBUILD_PHASE_FUNC}" != "pkg_prerm" ]]; then
die "${FUNCNAME[0]}() should only be used in pkg_prerm() phase"
fi
# Clean the link only if it's still owned by us
_cleanup_registry_link "" "$@"
}
# @FUNCTION: cros-rust_get_crate_version
# @USAGE: <path to crate>
# @DESCRIPTION:
# Returns the version for a crate by finding the first 'version =' line in the
# Cargo.toml in the crate.
# This triggers a linter error SC2120 which says:
# "rust_get_crate_version references arguments, but none are ever passed"
# In this case, we will use without arguments to get a default value, but other
# usages exist in other files that do use arguments, so there is no problem.
# shellcheck disable=SC2120
cros-rust_get_crate_version() {
local crate="${1:-${S}}"
[[ $# -gt 1 ]] && die "${FUNCNAME[0]}: incorrect number of arguments"
awk '/^version = / { print $3 }' "${crate}/Cargo.toml" | head -n1 | tr -d '"'
}
fi