blob: a44456c92bb0bb05a3d576871e591af7003f9759 [file] [log] [blame]
# Copyright 2017 The ChromiumOS Authors
# Distributed under the terms of the GNU General Public License v2
# Check for EAPI 5+
case "${EAPI:-0}" in
5|6|7) ;;
*) die "unsupported EAPI (${EAPI}) in eclass (${ECLASS})" ;;
esac
# In order to rebuild packages when config changes happen, we need a
# subslot dependency on on the bsp packages that provide the actual
# configs. We do this here instead of in the ebuilds so that we don't
# pollute the tree with this complex dependency syntax.
case "${CATEGORY}/${PN}" in
# Can't depend on ourselves, special case this away.
chromeos-base/chromeos-config-bsp* ) ;;
* )
IUSE="
has_chromeos_config_bsp
has_chromeos_config_bsp_private
has_chromeos_config_bsp_baseboard
has_chromeos_config_bsp_baseboard_private
"
DEPEND="
has_chromeos_config_bsp? ( chromeos-base/chromeos-config-bsp:= )
has_chromeos_config_bsp_private? (
chromeos-base/chromeos-config-bsp-private:=
)
has_chromeos_config_bsp_baseboard? (
chromeos-base/chromeos-config-bsp-baseboard:=
)
has_chromeos_config_bsp_baseboard_private? (
chromeos-base/chromeos-config-bsp-baseboard-private:=
)
"
# Again, can't depend on ourselves.
if [[ "${CATEGORY}/${PN}" != "chromeos-base/chromeos-config" ]]; then
IUSE+=" unibuild "
DEPEND+="
unibuild? ( chromeos-base/chromeos-config:= )
"
fi
RDEPEND="${DEPEND}"
# Let's also provide an environment variable that
# cros_config_host can use, that way it can validate we got
# the proper dependencies to use it.
export CROS_UNIBUILD_ECLASS=1
esac
# Additionally, if we support EAPI=7, let's be proper and add a
# BDEPEND.
case "${EAPI}" in
7 )
BDEPEND="chromeos-base/chromeos-config-host:="
;;
esac
# @ECLASS-VARIABLE: UNIBOARD_CROS_CONFIG_DIR
# @DESCRIPTION:
# This is the installation directory of cros-config data.
UNIBOARD_CROS_CONFIG_DIR="/usr/share/chromeos-config"
# @ECLASS-VARIABLE: UNIBOARD_JSON_INSTALL_PATH
# @DESCRIPTION:
# This is the filename of the master configuration for use with doins.
UNIBOARD_JSON_INSTALL_PATH="${UNIBOARD_CROS_CONFIG_DIR}/config.json"
# @ECLASS-VARIABLE: UNIBOARD_YAML_DIR
# @DESCRIPTION:
# This is the installation directory of the yaml source files.
UNIBOARD_YAML_DIR="${UNIBOARD_CROS_CONFIG_DIR}/yaml"
# @ECLASS-VARIABLE: UNIBOARD_YAML_CONFIG
# @DESCRIPTION:
# This is the installation path to the YAML source file.
UNIBOARD_YAML_CONFIG="${UNIBOARD_YAML_DIR}/config.yaml"
# @FUNCTION: _find_configs
# @USAGE: <directory> <extension>
# @INTERNAL
# @DESCRIPTION:
# Find .json/.yaml files in a given directory tree.
# Args:
# $1: Directory to search.
# $2: Extension to search for (.json or .yaml).
# Returns:
# Exports a 'files' variable containing the array of files found.
# TODO(gmeinke): rename this to something more generic.
_find_configs() {
local file
while read -d $'\0' -r file; do
files+=( "${file}" )
done < <(find -H "$1" -name "*$2" -print0)
}
# @FUNCTION: _insert_config_files
# @USAGE: <files array> <output directory>
# @INTERNAL
# @DESCRIPTION:
# Do a newins to the output directory for each file in the input array.
# Args:
# $1: Files array to insert.
# $2: Output directory to newins into.
# $3: prefix to prepend to filename.
# Returns:
# None.
_insert_config_files() {
[[ $# -eq 3 ]] || die "${FUNCNAME}: takes three arguments"
local ins_files=($1)
local output_dir="$2"
local prefix="$3"
if [[ "${#ins_files[@]}" -gt 0 ]]; then
einfo "Installing ${#ins_files[@]} files to ${output_dir}"
# Avoid polluting callers with our newins.
(
insinto "${output_dir}"
for f in "${ins_files[@]}"; do
local dest="${prefix}${f##*/}"
einfo "Copying ${f} -> ${dest}"
newins "${f}" "${dest}"
done
)
fi
}
# @FUNCTION: _install_model_files
# @USAGE: [prefix]
# @INTERNAL
# @DESCRIPTION:
# Find .yaml files in a given directory tree.
# Install model files with a given prefix:
# Args:
# $1: Prefix to use (either "" or "private-")
_install_model_files() {
[[ $# -eq 1 ]] || die "${FUNCNAME}: takes one arguments"
local prefix="$1"
local files
files=()
_find_configs "${FILESDIR}" ".yaml"
_insert_config_files "${files[*]}" "${UNIBOARD_YAML_DIR}" "${prefix}"
}
# @FUNCTION: install_private_model_files
# @USAGE:
# @DESCRIPTION:
# Installs all .yaml files for the current board. This is intended to be called
# from the chromeos-config-<board> private ebuild. The files are named
# "private-<fname>.yaml".
install_private_model_files() {
[[ $# -eq 0 ]] || die "${FUNCNAME}: takes no arguments"
_install_model_files "private-"
}
# @FUNCTION: install_model_files
# @USAGE:
# @DESCRIPTION:
# Installs all .yaml files for the current board. This is intended to be called
# from the chromeos-config-<board> public ebuild. The files are named
# "<fname>.yaml".
install_model_files() {
[[ $# -eq 0 ]] || die "${FUNCNAME}: takes no arguments"
_install_model_files ""
}
# @FUNCTION: install_generated_config_files
# @USAGE:
# @DESCRIPTION:
# Installs generated JSON payloads and C files for the current board. This is
# intended to be called from the chromeos-config-<board> public and private
# ebuilds.
install_generated_config_files() {
[[ $# -eq 0 ]] || die "${FUNCNAME[0]}: takes no arguments"
# Install build config as "config.yaml".
# Consumers of the payloads expect these names and locations.
insinto "${UNIBOARD_YAML_DIR}"
newins "${FILESDIR}/generated/build_config.json" "config.yaml"
doins "${FILESDIR}/generated/config.c"
}
# @FUNCTION: verify_file_match
# @USAGE: [expected_file] [actual_file]
# @DESCRIPTION:
# Verifies the expected file matches the actual file.
# $1: Filename of expected file contents
# $2: Filename of actual file contents
verify_file_match() {
local expected_file="$1"
local actual_file="$2"
einfo "Verifying ${expected_file} matches ${actual_file}"
local expected_cksum="$(cksum "${expected_file}" | cut -d ' ' -f 1)"
local actual_cksum="$(cksum "${actual_file}" | cut -d ' ' -f 1)"
if [[ "${expected_cksum}" -ne "${actual_cksum}" ]]; then
eerror "Generated file doesn't match expected file. \n" \
"Generated file is available at: ${actual_file}\n" \
"If this is an expected change, copy this change to the expected" \
"$(basename "${expected_file}") file and commit with your CL.\n"
die
fi
einfo "Successfully verified ${expected_file} matches ${actual_file}"
}
# @FUNCTION: _unibuild_common_install
# @USAGE: command [config_file]
# @INTERNAL
# @DESCRIPTION:
# Common installation function.
# Install files from a relative path in FILESDIR to an absolute path in the
# root.
# Args:
# $1: Command to pass to cros_config_host to get the files
# $2: (optional) Config file used by cros_config_host
_unibuild_common_install() {
[[ $# -gt 0 ]] || die "${FUNCNAME}: cros_config_host command required"
[[ $# -lt 3 ]] || die "${FUNCNAME}: Only optional config file arg allowed"
local cmd="$1"
local config_files_path="${FILESDIR}"
local config="${SYSROOT}${UNIBOARD_YAML_DIR}/config.yaml"
if [[ $# -gt 1 ]]; then
config_files_path="."
config="$2"
fi
einfo "unibuild: Installing ${cmd} based on ${config}"
local source dest origfile
(cros_config_host -c "${config}" "${cmd}" || die) |
while read -r source; do
read -r dest
einfo " - ${config_files_path}/${source}"
insinto "$(dirname "${dest}")"
# From EAPI4+ symbolic links are not dereferenced when
# installing, but we want to dereference our links when
# installing config files. Common config files may be linked
# in the source directory, but the installed file should not be
# a link especially since the installed folder structure is
# different.
origfile="$(readlink -f "${config_files_path}/${source}")" ||
die "readlink -f ${config_files_path}/${source} failed"
newins "${origfile}" "$(basename "${dest}")"
done
}
# @FUNCTION: _unibuild_install_fw
# @USAGE: [source] [dest] [symlink path]
# @INTERNAL
# @DESCRIPTION:
# Install the firmware and create a symlink for 'request firmware' hotplug.
# $1: Source filename
# $2: Destination filename (in /opt/google)
# $3: Full path to symlink in /lib/firmware
_unibuild_install_fw() {
local source="$1"
local dest="$2"
local symlink="$3"
elog " - ${source} with symlink from ${symlink}"
insinto "$(dirname "${dest}")"
newins "${source}" "$(basename "${dest}")"
dosym "${dest}" "${symlink}"
}
# @FUNCTION: _unibuild_install_fw_common
# @USAGE: [cmd] [config_file]
# @INTERNAL
# @DESCRIPTION:
# Install the firmware and create a symlink for the files which query from
# the cros_config_host.
# Args:
# $1: Command to pass to cros_config_host to get the files
# $2: (optional) Config file used by cros_config_host
_unibuild_install_fw_common() {
[[ $# -gt 0 ]] || die "${FUNCNAME}: cros_config_host command required"
[[ $# -lt 3 ]] || die "${FUNCNAME}: Only optional config file arg allowed"
local cmd="$1"
local files_path="${FILESDIR}"
local config="${SYSROOT}${UNIBOARD_YAML_DIR}/config.yaml"
if [[ $# -gt 1 ]]; then
files_path="."
config="$2"
fi
# Determine if using files or workdir for the path.
if [[ "${cmd}" == *"-workdir" ]]; then
cmd="${cmd%-workdir}"
files_path="${WORKDIR}"
fi
einfo "unibuild: Installing ${cmd} based on ${config}"
set -o pipefail
cros_config_host -c "${config}" "${cmd}" |
( while read -r source; do
read -r dest
read -r symlink
_unibuild_install_fw "${files_path}/${source}" "${dest}" "${symlink}"
done ) || die "Failed to read config"
}
# @FUNCTION: unibuild_install_touch_files
# @USAGE: [config_file]
# @DESCRIPTION:
# Install files related to touch firmware. This includes firmware for the
# touchscreen, touchpad and stylus. These files are expected in files dir.
# Args:
# $1: (optional) Config file used by cros_config_host
unibuild_install_touch_files() {
[[ $# -lt 2 ]] || die "${FUNCNAME}: Only optional config file arg allowed"
_unibuild_install_fw_common "get-touch-firmware-files" "$@"
}
# @FUNCTION: unibuild_install_touch_workdir
# @USAGE: [config_file]
# @DESCRIPTION:
# Install files related to touch firmware. This includes firmware for the
# touchscreen, touchpad and stylus. These files are expected in workdir.
# Args:
# $1: (optional) Config file used by cros_config_host
unibuild_install_touch_workdir() {
[[ $# -lt 2 ]] || die "${FUNCNAME}: Only optional config file arg allowed"
_unibuild_install_fw_common "get-touch-firmware-files-workdir" "$@"
}
# @FUNCTION: unibuild_install_detachable_base_files
# @USAGE: [config_file]
# @DESCRIPTION:
# Install files related to detachable base firmware. This includes firmware
# for the detachable base ec and touchpad binary
# Args:
# $1: (optional) Config file used by cros_config_host
unibuild_install_detachable_base_files() {
[[ $# -lt 2 ]] || die "${FUNCNAME}: Only optional config file arg allowed"
_unibuild_install_fw_common "get-detachable-base-firmware-files" "$@"
}
# @FUNCTION: unibuild_install_detachable_base_files_workdir
# @USAGE: [config_file]
# @DESCRIPTION:
# Install files related to detachable base firmware. This includes firmware
# for the detachable base ec and touchpad binary.
# These files are expected in workdir.
# Args:
# $1: (optional) Config file used by cros_config_host
unibuild_install_detachable_base_files_workdir() {
[[ $# -lt 2 ]] || die "${FUNCNAME}: Only optional config file arg allowed"
_unibuild_install_fw_common "get-detachable-base-firmware-files-workdir" "$@"
}
# @FUNCTION: unibuild_build_configfs_file
# @USAGE:
# @DESCRIPTION:
# Build configfs img file.
unibuild_build_configfs_file() {
[[ $# -eq 0 ]] || die "${FUNCNAME}: takes no arguments"
einfo "unibuild: Build configfs img file"
local yaml="${FILESDIR}/generated/build_config.json"
local configfs_image="${WORKDIR}/configfs.img"
cros_config_schema -c "${yaml}" \
--configfs-output "${configfs_image}" -g "${WORKDIR}" -f "True" \
|| die "cros_config_schema failed for configfs."
}
# @FUNCTION: unibuild_install_configfs_file
# @USAGE:
# @DESCRIPTION:
# Install configfs img file.
unibuild_install_configfs_file() {
[[ $# -eq 0 ]] || die "${FUNCNAME}: takes no arguments"
einfo "unibuild: Install configfs img file"
insinto "${UNIBOARD_CROS_CONFIG_DIR}"
doins "${WORKDIR}/configfs.img"
}
# @FUNCTION: platform_json_compile
# @USAGE:
# @DESCRIPTION:
# Compile platform json file project-config.json. This is done by finding all
# project-config.json within $S and using cros_config_schema to combine them
# into a single project-config.json.
platform_json_compile() {
[[ $# -eq 0 ]] || die "${FUNCNAME}: takes no arguments"
einfo "Compiling platform json file project-config.json."
local files
files=()
_find_configs "${S}" "project-config.json"
cros_config_schema \
-o "${WORKDIR}/project-config.json" \
-m "${files[@]}" \
|| die "cros_config_schema failed for build config."
}
# @FUNCTION: platform_json_install
# @USAGE:
# @DESCRIPTION:
# Install platform json file project-config.json.
platform_json_install() {
[[ $# -eq 0 ]] || die "${FUNCNAME}: takes no arguments"
einfo "Installing platform json file project-config.json."
insinto "${UNIBOARD_YAML_DIR}"
doins "${WORKDIR}/project-config.json"
}
# @FUNCTION: platform_merged_install
# @USAGE:
# @DESCRIPTION:
# Install project-specific joined.jsonproto files into tmp dir.
# These files will be available to the Build API as artifacts.
# Args:
# $1: Name of the project being installed
platform_merged_install() {
[[ $# -eq 1 ]] || die "${FUNCNAME}: takes a single argument"
einfo "Installing project-specific joined.jsonproto file."
insinto "/build/share/cros-unibuild/$1/generated"
doins "${S}/$1/generated/joined.jsonproto"
}
# @FUNCTION: unibuild_install_files
# @USAGE: [config_file]
# @DESCRIPTION:
# Install files as specified in project config.
# Args:
# $1: Which files to install, translates to a command to pass to
# cros_config_host to get the files
# $2: (optional) Config file used by cros_config_host
unibuild_install_files() {
[[ $# -gt 0 ]] || die "${FUNCNAME}: files to install arg required"
[[ $# -lt 3 ]] || die "${FUNCNAME}: Only optional config file arg allowed"
local file_set="$1"
shift
_unibuild_common_install "get-${file_set}" "$@"
}