3e90b9
#!/bin/bash
3e90b9
#
3e90b9
# grubby wrapper to manage BootLoaderSpec files
3e90b9
#
3e90b9
#
3e90b9
# Copyright 2018 Red Hat, Inc.  All rights reserved.
3e90b9
#
3e90b9
# This program is free software; you can redistribute it and/or modify
3e90b9
# it under the terms of the GNU General Public License as published by
3e90b9
# the Free Software Foundation; either version 2 of the License, or
3e90b9
# (at your option) any later version.
3e90b9
#
3e90b9
# This program is distributed in the hope that it will be useful,
3e90b9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
3e90b9
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
3e90b9
# GNU General Public License for more details.
3e90b9
#
3e90b9
# You should have received a copy of the GNU General Public License
3e90b9
# along with this program.  If not, see <http://www.gnu.org/licenses/>.
3e90b9
3e90b9
readonly SCRIPTNAME="${0##*/}"
3e90b9
3e90b9
CMDLINE_LINUX_DEBUG=" systemd.log_level=debug systemd.log_target=kmsg"
3e90b9
LINUX_DEBUG_VERSION_POSTFIX="_with_debugging"
3e90b9
LINUX_DEBUG_TITLE_POSTFIX=" with debugging"
3e90b9
3e90b9
declare -a bls_file
3e90b9
declare -a bls_title
3e90b9
declare -a bls_version
3e90b9
declare -a bls_linux
3e90b9
declare -a bls_initrd
3e90b9
declare -a bls_options
3e90b9
declare -a bls_id
3e90b9
3e90b9
[[ -f /etc/sysconfig/kernel ]] && . /etc/sysconfig/kernel
3e90b9
[[ -f /etc/os-release ]] && . /etc/os-release
3e90b9
read MACHINE_ID < /etc/machine-id
3e90b9
arch=$(uname -m)
3e90b9
3e90b9
if [[ $arch = 's390' || $arch = 's390x' ]]; then
3e90b9
    bootloader="zipl"
3e90b9
else
3e90b9
    bootloader="grub2"
3e90b9
fi
3e90b9
3e90b9
print_error() {
3e90b9
    echo "$1" >&2
3e90b9
    exit 1
3e90b9
}
3e90b9
3e90b9
print_info() {
3e90b9
    echo "$1" >&2
3e90b9
}
3e90b9
3e90b9
if [[ ${#} = 0 ]]; then
3e90b9
    print_error "no action specified"
3e90b9
fi
3e90b9
3e90b9
get_bls_value() {
3e90b9
    local bls="$1" && shift
3e90b9
    local key="$1" && shift
3e90b9
3e90b9
    echo "$(grep "^${key}[ \t]" "${bls}" | sed -e "s!^${key}[ \t]*!!")"
3e90b9
}
3e90b9
3e90b9
set_bls_value() {
3e90b9
    local bls="$1" && shift
3e90b9
    local key="$1" && shift
3e90b9
    local value="$1" && shift
3e90b9
3e90b9
    value=$(echo $value | sed -e 's/\//\\\//g')
3e90b9
    sed -i -e "s/^${key}.*/${key} ${value}/" "${bls}"
3e90b9
}
3e90b9
3e90b9
append_bls_value() {
3e90b9
    local bls="$1" && shift
3e90b9
    local key="$1" && shift
3e90b9
    local value="$1" && shift
3e90b9
3e90b9
    old_value="$(get_bls_value "${bls}" ${key})"
3e90b9
    set_bls_value "${bls}" "${key}" "${old_value}${value}"
3e90b9
}
3e90b9
3e90b9
get_bls_values() {
3e90b9
    count=0
3e90b9
    local -a files
3e90b9
    local IFS=$'\n'
3e90b9
    files=($(for bls in ${blsdir}/*.conf ; do
3e90b9
        if ! [[ -e "${bls}" ]] ; then
3e90b9
            continue
3e90b9
        fi
3e90b9
        bls="${bls%.conf}"
3e90b9
        bls="${bls##*/}"
3e90b9
        echo "${bls}"
3e90b9
    done | /usr/libexec/grubby/rpm-sort -c rpmnvrcmp 2>/dev/null | tac)) || :
3e90b9
3e90b9
    for bls in "${files[@]}" ; do
3e90b9
        blspath="${blsdir}/${bls}.conf"
3e90b9
        bls_file[$count]="${blspath}"
3e90b9
        bls_title[$count]="$(get_bls_value ${blspath} title)"
3e90b9
        bls_version[$count]="$(get_bls_value ${blspath} version)"
3e90b9
        bls_linux[$count]="$(get_bls_value ${blspath} linux)"
3e90b9
        bls_initrd[$count]="$(get_bls_value ${blspath} initrd)"
3e90b9
        bls_options[$count]="$(get_bls_value ${blspath} options)"
3e90b9
        bls_id[$count]="${bls}"
3e90b9
3e90b9
        count=$((count+1))
3e90b9
    done
3e90b9
}
3e90b9
3e90b9
get_default_index() {
3e90b9
    local default=""
3e90b9
    local index="-1"
3e90b9
    local title=""
3e90b9
    local version=""
3e90b9
    if [[ $bootloader = "grub2" ]]; then
3e90b9
	default="$(grep '^saved_entry=' ${env} | sed -e 's/^saved_entry=//')"
3e90b9
    else
3e90b9
	default="$(grep '^default=' ${zipl_config} | sed -e 's/^default=//')"
3e90b9
    fi
3e90b9
3e90b9
    if [[ -z $default ]]; then
3e90b9
	index=0
3e90b9
    elif [[ $default =~ ^[0-9]+$ ]]; then
3e90b9
	index="$default"
3e90b9
    fi
3e90b9
3e90b9
    for i in ${!bls_file[@]}; do
3e90b9
        if [[ $i -eq $index ]]; then
3e90b9
            echo $i
3e90b9
            return
3e90b9
        fi
3e90b9
3e90b9
        if [[ $default = ${bls_id[$i]} || $default = ${bls_title[$i]} ]]; then
3e90b9
            echo $i
3e90b9
            return
3e90b9
        fi
3e90b9
    done
3e90b9
}
3e90b9
3e90b9
display_default_value() {
3e90b9
    local prefix=$(get_prefix)
3e90b9
3e90b9
    case "$display_default" in
3e90b9
        kernel)
3e90b9
            echo "${prefix}${bls_linux[$default_index]}"
3e90b9
            exit 0
3e90b9
            ;;
3e90b9
        index)
3e90b9
            echo "$default_index"
3e90b9
            exit 0
3e90b9
            ;;
3e90b9
        title)
3e90b9
            echo "${bls_title[$default_index]}"
3e90b9
            exit 0
3e90b9
            ;;
3e90b9
    esac
3e90b9
}
3e90b9
3e90b9
param_to_indexes() {
3e90b9
    local param="$1"
3e90b9
    local indexes=""
3e90b9
3e90b9
    if [[ $param = "ALL" ]]; then
3e90b9
        for i in ${!bls_file[@]}; do
3e90b9
            indexes="$indexes $i"
3e90b9
        done
3e90b9
        echo -n $indexes
3e90b9
        return
3e90b9
    fi
3e90b9
3e90b9
    if [[ $param = "DEFAULT" ]]; then
3e90b9
        echo -n $default_index
3e90b9
        return
3e90b9
    fi
3e90b9
3e90b9
    for i in ${!bls_file[@]}; do
3e90b9
        if [[ $param = "${bls_linux[$i]}" || "/${param##*/}" = "${bls_linux[$i]}" ]]; then
3e90b9
            indexes="$indexes $i"
3e90b9
        fi
3e90b9
3e90b9
        if [[ $param = "TITLE=${bls_title[$i]}" ]]; then
3e90b9
            indexes="$indexes $i"
3e90b9
        fi
3e90b9
3e90b9
	if [[ $param = $i ]]; then
3e90b9
	    indexes="$indexes $i"
3e90b9
	fi
3e90b9
    done
3e90b9
3e90b9
    if [[ -n $indexes ]]; then
3e90b9
        echo -n $indexes
3e90b9
        return
3e90b9
    fi
3e90b9
3e90b9
    echo -n "-1"
3e90b9
}
3e90b9
3e90b9
get_prefix() {
3e90b9
    if [[ $bootloader = grub2 ]] && mountpoint -q /boot; then
3e90b9
	echo "/boot"
3e90b9
    else
3e90b9
	echo ""
3e90b9
    fi
3e90b9
}
3e90b9
3e90b9
expand_var() {
3e90b9
    local var=$1
3e90b9
3e90b9
    if [[ $bootloader == "grub2" ]]; then
3e90b9
        local value="$(grub2-editenv "${env}" list | grep ${var##$} | sed -e "s/${var##$}=//")"
3e90b9
        value="$(echo ${value} | sed -e 's/\//\\\//g')"
3e90b9
        if [[ -n $value ]]; then
3e90b9
            var="$value"
3e90b9
        fi
3e90b9
    fi
3e90b9
3e90b9
    echo $var
3e90b9
}
3e90b9
3e90b9
has_kernelopts()
3e90b9
{
3e90b9
    local args=${bls_options[$1]}
3e90b9
    local opts=(${args})
3e90b9
3e90b9
    for opt in ${opts[*]}; do
3e90b9
        [[ $opt = "\$kernelopts" ]] && echo "true"
3e90b9
    done
3e90b9
3e90b9
    echo "false"
3e90b9
}
3e90b9
3e90b9
get_bls_args() {
3e90b9
    local args=${bls_options[$1]}
3e90b9
    local opts=(${args})
3e90b9
3e90b9
    for opt in ${opts[*]}; do
3e90b9
        if [[ $opt = "\$kernelopts" ]]; then
3e90b9
            value="$(expand_var $opt)"
3e90b9
            args="$(echo ${args} | sed -e "s/${opt}/${value}/")"
3e90b9
        fi
3e90b9
    done
3e90b9
3e90b9
    echo ${args}
3e90b9
}
3e90b9
3e90b9
display_info_values() {
3e90b9
    local indexes=($(param_to_indexes "$1"))
3e90b9
    local prefix=$(get_prefix)
3e90b9
3e90b9
    if [[ $indexes = "-1" ]]; then
3e90b9
        print_error "The param $1 is incorrect"
3e90b9
    fi
3e90b9
3e90b9
    for i in ${indexes[*]}; do
3e90b9
        local root=""
3e90b9
        local value=""
3e90b9
        local args="$(get_bls_args "$i")"
3e90b9
3e90b9
        local opts=(${args})
3e90b9
3e90b9
        for opt in ${opts[*]}; do
3e90b9
            if echo $opt | grep -q "^root="; then
3e90b9
                root="$(echo $opt | sed -e 's/root=//')"
3e90b9
                value="$(echo ${opt} | sed -e 's/\//\\\//g')"
3e90b9
                args="$(echo ${args} | sed -e "s/${value}[ \t]*//")"
3e90b9
                break
3e90b9
            fi
3e90b9
        done
3e90b9
3e90b9
        echo "index=$i"
3e90b9
        echo "kernel=\"${prefix}${bls_linux[$i]}\""
3e90b9
        echo "args=\"${args}\""
3e90b9
3e90b9
        if [[ -n $root ]]; then
3e90b9
            echo "root=\"${root}\""
3e90b9
        fi
3e90b9
3e90b9
        echo "initrd=\"${prefix}${bls_initrd[$i]}\""
3e90b9
        echo "title=\"${bls_title[$i]}\""
3e90b9
        echo "id=\"${bls_id[$i]}\""
3e90b9
    done
3e90b9
    exit 0
3e90b9
}
3e90b9
3e90b9
mkbls() {
3e90b9
    local kernel=$1 && shift
3e90b9
    local kernelver=$1 && shift
3e90b9
    local datetime=$1 && shift
3e90b9
3e90b9
    local debugname=""
3e90b9
    local flavor=""
3e90b9
    local prefix=""
3e90b9
3e90b9
    if [[ $(get_prefix) = "" ]]; then
3e90b9
	prefix="/boot"
3e90b9
    fi
3e90b9
3e90b9
    if [[ $kernelver == *\+* ]] ; then
3e90b9
        local flavor=-"${kernelver##*+}"
3e90b9
        if [[ $flavor == "-debug" ]]; then
3e90b9
            local debugname="with debugging"
3e90b9
	    local debugid="-debug"
3e90b9
        fi
3e90b9
    fi
3e90b9
3e90b9
    cat <
3e90b9
title ${NAME} (${kernelver}) ${VERSION}${debugname}
3e90b9
version ${kernelver}${debugid}
3e90b9
linux ${kernel}
3e90b9
initrd ${prefix}/initramfs-${kernelver}.img
3e90b9
options \$kernelopts
3e90b9
id ${ID}-${datetime}-${kernelver}${debugid}
3e90b9
grub_users \$grub_users
3e90b9
grub_arg --unrestricted
3e90b9
grub_class kernel${flavor}
3e90b9
EOF
3e90b9
}
3e90b9
3e90b9
unset_default_bls()
3e90b9
{
3e90b9
    if [[ $bootloader = grub2 ]]; then
3e90b9
        grub2-editenv "${env}" unset saved_entry
3e90b9
    else
3e90b9
        sed -i -e "/^default=.*/d" "${zipl_config}"
3e90b9
    fi
3e90b9
}
3e90b9
3e90b9
remove_bls_fragment() {
3e90b9
    local indexes=($(param_to_indexes "$1"))
3e90b9
3e90b9
    if [[ $indexes = "-1" ]]; then
3e90b9
	print_error "The param $(get_prefix)$1 is incorrect"
3e90b9
    fi
3e90b9
3e90b9
    for i in "${indexes[@]}"; do
3e90b9
        if [[ $default_index = $i ]]; then
3e90b9
            unset_default_bls
3e90b9
        fi
3e90b9
        rm -f "${bls_file[$i]}"
3e90b9
    done
3e90b9
3e90b9
    get_bls_values
3e90b9
3e90b9
    update_grubcfg
3e90b9
}
3e90b9
3e90b9
get_custom_bls_filename() {
3e90b9
    local kernelver=$1
3e90b9
    local bls_target="${blsdir}/${MACHINE_ID}-${kernelver}.conf"
3e90b9
    count=0
3e90b9
    local -a files
3e90b9
    local IFS=$'\n'
3e90b9
3e90b9
    prefix="${bls_target%%.conf}"
3e90b9
    prefix="${bls_target%%${arch}}"
3e90b9
    prefix="${prefix%.*}"
3e90b9
3e90b9
    last=($(for bls in ${prefix}.*~custom*.conf ; do
3e90b9
        if ! [[ -e "${bls}" ]] ; then
3e90b9
            continue
3e90b9
        fi
3e90b9
        bls="${bls##${prefix}.}"
3e90b9
        bls="${bls%%~custom*}"
3e90b9
        echo "${bls}"
3e90b9
    done | tail -n1)) || :
3e90b9
3e90b9
    if [[ -z $last ]]; then
3e90b9
        last="0"
3e90b9
    else
3e90b9
        last=$((last+1))
3e90b9
    fi
3e90b9
3e90b9
    echo "${bls_target}" | sed -e "s!${prefix}!${prefix}.${last}~custom!"
3e90b9
}
3e90b9
3e90b9
add_bls_fragment() {
3e90b9
    local kernel="$1" && shift
3e90b9
    local title="$1" && shift
3e90b9
    local options="$1" && shift
3e90b9
    local initrd="$1" && shift
3e90b9
    local extra_initrd="$1" && shift
3e90b9
3e90b9
    if [[ $kernel = *"vmlinuz-"* ]]; then
3e90b9
	kernelver="${kernel##*/vmlinuz-}"
3e90b9
	prefix="vmlinuz-"
3e90b9
    else
3e90b9
	kernelver="${kernel##*/}"
3e90b9
    fi
3e90b9
3e90b9
    if [[ ! -f "/boot/${prefix}${kernelver}" ]] &&
3e90b9
       [[ $bad_image != "true" ]]; then
3e90b9
        print_error "The ${kernelver} kernel isn't installed in the machine"
3e90b9
    fi
3e90b9
3e90b9
    if [[ -z $title ]]; then
3e90b9
	print_error "The kernel title must be specified"
3e90b9
    fi
3e90b9
3e90b9
    if [[ ! -d $blsdir ]]; then
3e90b9
        install -m 700 -d "${blsdir}"
3e90b9
    fi
3e90b9
3e90b9
    bls_target="${blsdir}/${MACHINE_ID}-${kernelver}.conf"
3e90b9
3e90b9
    if [[ -e ${bls_target} ]]; then
3e90b9
        bls_target="$(get_custom_bls_filename "${kernelver}")"
3e90b9
        print_info "An entry for kernel ${kernelver} already exists, adding ${bls_target}"
3e90b9
    fi
3e90b9
3e90b9
    kernel_dir="/lib/modules/${kernelver}"
3e90b9
    if [[ -d $kernel_dir ]]; then
3e90b9
        datetime="$(date -u +%Y%m%d%H%M%S -d "$(stat -c '%y' "${kernel_dir}")")"
3e90b9
    else
3e90b9
        datetime=0
3e90b9
    fi
3e90b9
    mkbls "${kernel}" "${kernelver}" "${datetime}" > "${bls_target}"
3e90b9
3e90b9
    if [[ -n $title ]]; then
3e90b9
        set_bls_value "${bls_target}" "title" "${title}"
3e90b9
    fi
3e90b9
3e90b9
    if [[ -n $options ]]; then
3e90b9
        set_bls_value "${bls_target}" "options" "${options}"
3e90b9
    fi
3e90b9
3e90b9
    if [[ -n $initrd ]]; then
3e90b9
        set_bls_value "${bls_target}" "initrd" "${initrd}"
3e90b9
    fi
3e90b9
3e90b9
    if [[ -n $extra_initrd ]]; then
3e90b9
        append_bls_value "${bls_target}" "initrd" " ${extra_initrd}"
3e90b9
    fi
3e90b9
3e90b9
    if [[ $MAKEDEBUG = "yes" ]]; then
3e90b9
        bls_debug="$(echo ${bls_target} | sed -e "s/${kernelver}/${kernelver}~debug/")"
3e90b9
        cp -aT  "${bls_target}" "${bls_debug}"
3e90b9
        append_bls_value "${bls_debug}" "title" "${LINUX_DEBUG_TITLE_POSTFIX}"
3e90b9
        append_bls_value "${bls_debug}" "version" "${LINUX_DEBUG_VERSION_POSTFIX}"
3e90b9
        append_bls_value "${bls_debug}" "options" "${CMDLINE_LINUX_DEBUG}"
3e90b9
        blsid="$(get_bls_value ${bls_debug} "id" | sed -e "s/${kernelver}/${kernelver}~debug/")"
3e90b9
        set_bls_value "${bls_debug}" "id" "${blsid}"
3e90b9
    fi
3e90b9
3e90b9
    get_bls_values
3e90b9
3e90b9
    if [[ $make_default = "true" ]]; then
3e90b9
        set_default_bls "TITLE=${title}"
3e90b9
    fi
3e90b9
3e90b9
    update_grubcfg
3e90b9
3e90b9
    exit 0
3e90b9
}
3e90b9
3e90b9
update_args() {
3e90b9
    local args=$1 && shift
3e90b9
    local remove_args=($1) && shift
3e90b9
    local add_args=($1) && shift
3e90b9
3e90b9
    for arg in ${remove_args[*]}; do
3e90b9
        arg="$(echo $arg | sed -e 's/\//\\\//g')"
3e90b9
        if [[ $arg = *"="* ]]; then
3e90b9
            args="$(echo $args | sed -E "s/(^|[[:space:]])$arg([[:space:]]|$)/ /")"
3e90b9
        else
3e90b9
            args="$(echo $args | sed -E "s/(^|[[:space:]])$arg(([[:space:]]|$)|([=][^ ]*([$]*)))/ /g")"
3e90b9
        fi
3e90b9
    done
3e90b9
3e90b9
    for arg in ${add_args[*]}; do
3e90b9
        arg="${arg%%=*}"
3e90b9
        arg="$(echo $arg | sed -e 's/\//\\\//g')"
3e90b9
        args="$(echo $args | sed -E "s/(^|[[:space:]])$arg(([[:space:]]|$)|([=][^ ]*([$]*)))/ /g")"
3e90b9
    done
3e90b9
3e90b9
    for arg in ${add_args[*]}; do
3e90b9
        args="$args $arg"
3e90b9
    done
3e90b9
3e90b9
    echo ${args}
3e90b9
}
3e90b9
3e90b9
update_bls_fragment() {
3e90b9
    local param="$1"
3e90b9
    local indexes=($(param_to_indexes "$1")) && shift
3e90b9
    local remove_args=$1 && shift
3e90b9
    local add_args=$1 && shift
3e90b9
    local initrd=$1 && shift
3e90b9
    local opts
3e90b9
3e90b9
    if [[ $indexes = "-1" ]]; then
3e90b9
        print_error "The param $(get_prefix)${param} is incorrect"
3e90b9
    fi
3e90b9
3e90b9
    if [[ $param = "ALL" && $bootloader = grub2 ]] && [[ -n $remove_args || -n $add_args ]]; then
3e90b9
        local old_args=""
3e90b9
3e90b9
        if [[ -z $no_etc_update ]] && [[ -e ${grub_etc_default} ]]; then
3e90b9
            old_args="$(source ${grub_etc_default}; echo ${GRUB_CMDLINE_LINUX})"
3e90b9
            if [[ -n $old_args ]]; then
3e90b9
                opts="$(update_args "${old_args}" "${remove_args}" "${add_args}")"
3e90b9
                opts="$(echo "$opts" | sed -e 's/\//\\\//g')"
3e90b9
                sed -i -e "s/^GRUB_CMDLINE_LINUX.*/GRUB_CMDLINE_LINUX=\\\"${opts}\\\"/" "${grub_etc_default}"
3e90b9
            fi
3e90b9
        fi
3e90b9
3e90b9
        old_args="$(grub2-editenv "${env}" list | grep kernelopts | sed -e "s/kernelopts=//")"
3e90b9
        if [[ -n $old_args ]]; then
3e90b9
            opts="$(update_args "${old_args}" "${remove_args}" "${add_args}")"
3e90b9
            grub2-editenv "${env}" set kernelopts="${opts}"
3e90b9
        fi
3e90b9
    elif [[ $bootloader = grub2 ]]; then
3e90b9
        opts="$(grub2-editenv "${env}" list | grep kernelopts | sed -e "s/kernelopts=//")"
3e90b9
    fi
3e90b9
3e90b9
    for i in ${indexes[*]}; do
3e90b9
	if [[ -n $remove_args || -n $add_args ]]; then
3e90b9
            local old_args="$(get_bls_args "$i")"
3e90b9
            local new_args="$(update_args "${old_args}" "${remove_args}" "${add_args}")"
3e90b9
3e90b9
            if [[ $param != "ALL" || "$(has_kernelopts "$i")" = "false" ]]; then
3e90b9
                set_bls_value "${bls_file[$i]}" "options" "${new_args}"
3e90b9
            fi
3e90b9
	fi
3e90b9
3e90b9
	if [[ -n $initrd ]]; then
3e90b9
	    set_bls_value "${bls_file[$i]}" "initrd" "${initrd}"
3e90b9
	fi
3e90b9
    done
3e90b9
e35461
    if [[ $param = "ALL" ]] && [[ -n $remove_args || -n $add_args ]]; then
e35461
	if [[ ! -f /etc/kernel/cmdline ]]; then
e35461
	    # anaconda could pre-populate this file, but until then, most of
e35461
	    # the time we'll just want the most recent one.  This is pretty
e35461
	    # close to the current almost-correct behavior of falling back to
e35461
	    # /proc/cmdline anyhow.
e35461
	    echo "$(get_bls_args -1)" > /etc/kernel/cmdline
e35461
	fi
e35461
e35461
	read old_args < /etc/kernel/cmdline
e35461
	local new_args="$(update_args "${old_args}" "${remove_args}" "${add_args}")"
e35461
	echo "$new_args" > /etc/kernel/cmdline
e35461
    fi
e35461
3e90b9
    update_grubcfg
3e90b9
}
3e90b9
3e90b9
set_default_bls() {
3e90b9
    local index=($(param_to_indexes "$1"))
3e90b9
3e90b9
    if [[ $index = "-1" ]]; then
3e90b9
        print_error "The param $1 is incorrect"
3e90b9
    fi
3e90b9
3e90b9
    if [[ $bootloader = grub2 ]]; then
3e90b9
        grub2-editenv "${env}" set saved_entry="${bls_id[$index]}"
3e90b9
    else
3e90b9
        local default="${bls_title[$index]}"
3e90b9
        local current="$(grep '^default=' ${zipl_config} | sed -e 's/^default=//')"
3e90b9
        if [[ -n $current ]]; then
3e90b9
            sed -i -e "s,^default=.*,default=${default}," "${zipl_config}"
3e90b9
        else
3e90b9
            echo "default=${default}" >> "${zipl_config}"
3e90b9
        fi
3e90b9
    fi
3e90b9
3e90b9
    print_info "The default is ${bls_file[$index]} with index $index and kernel $(get_prefix)${bls_linux[$index]}"
3e90b9
}
3e90b9
3e90b9
remove_var_prefix() {
3e90b9
    local prefix="$1"
3e90b9
3e90b9
    [ -z "${prefix}" ] && return
3e90b9
3e90b9
    if [[ -n $remove_kernel && $remove_kernel =~ ^/ ]]; then
3e90b9
       remove_kernel="/${remove_kernel##${prefix}/}"
3e90b9
    fi
3e90b9
3e90b9
    if [[ -n $initrd ]]; then
3e90b9
	initrd="/${initrd##${prefix}/}"
3e90b9
    fi
3e90b9
3e90b9
    if [[ -n $extra_initrd ]]; then
3e90b9
	extra_initrd="/${extra_initrd##${prefix}/}"
3e90b9
    fi
3e90b9
3e90b9
    if [[ -n $kernel ]]; then
3e90b9
	kernel="/${kernel##${prefix}/}"
3e90b9
    fi
3e90b9
3e90b9
    if [[ -n $update_kernel && $update_kernel =~ ^/ ]]; then
3e90b9
	update_kernel="/${update_kernel##${prefix}/}"
3e90b9
    fi
3e90b9
}
3e90b9
3e90b9
update_grubcfg()
3e90b9
{
3e90b9
    # Older ppc64le OPAL firmware (petitboot version < 1.8.0) don't have BLS support
3e90b9
    # so grub2-mkconfig has to be run to generate a config with menuentry commands.
3e90b9
    if [ "${arch}" = "ppc64le" ] && [ -d /sys/firmware/opal ]; then
3e90b9
        RUN_MKCONFIG="true"
3e90b9
        petitboot_path="/sys/firmware/devicetree/base/ibm,firmware-versions/petitboot"
3e90b9
3e90b9
        if test -e ${petitboot_path}; then
3e90b9
            read -r -d '' petitboot_version < ${petitboot_path}
3e90b9
            petitboot_version="$(echo ${petitboot_version//v})"
3e90b9
3e90b9
            if test -n ${petitboot_version}; then
3e90b9
                major_version="$(echo ${petitboot_version} | cut -d . -f1)"
3e90b9
                minor_version="$(echo ${petitboot_version} | cut -d . -f2)"
3e90b9
3e90b9
                re='^[0-9]+$'
3e90b9
                if [[ $major_version =~ $re ]] && [[ $minor_version =~ $re ]] &&
3e90b9
                       ([[ ${major_version} -gt 1 ]] ||
3e90b9
                            [[ ${major_version} -eq 1 &&
3e90b9
                                 ${minor_version} -ge 8  ]]); then
3e90b9
                    RUN_MKCONFIG="false"
3e90b9
                fi
3e90b9
            fi
3e90b9
        fi
3e90b9
    fi
3e90b9
3e90b9
    if [[ $RUN_MKCONFIG = "true" ]]; then
3e90b9
        grub2-mkconfig --no-grubenv-update -o "${grub_config}" >& /dev/null
3e90b9
    fi
3e90b9
}
3e90b9
3e90b9
print_usage()
3e90b9
{
3e90b9
    cat <
3e90b9
Usage: grubby [OPTION...]
3e90b9
      --add-kernel=kernel-path            add an entry for the specified kernel
3e90b9
      --args=args                         default arguments for the new kernel or new arguments for kernel being updated)
3e90b9
      --bad-image-okay                    don't sanity check images in boot entries (for testing only)
3e90b9
  -c, --config-file=path                  path to grub config file to update ("-" for stdin)
3e90b9
      --copy-default                      use the default boot entry as a template for the new entry being added; if the default is not a linux image, or if the kernel referenced by the default image does not exist, the
3e90b9
                                          first linux entry whose kernel does exist is used as the template
3e90b9
      --default-kernel                    display the path of the default kernel
3e90b9
      --default-index                     display the index of the default kernel
3e90b9
      --default-title                     display the title of the default kernel
3e90b9
      --env=path                          path for environment data
3e90b9
      --grub2                             configure grub2 bootloader
3e90b9
      --info=kernel-path                  display boot information for specified kernel
3e90b9
      --initrd=initrd-path                initrd image for the new kernel
3e90b9
  -i, --extra-initrd=initrd-path          auxiliary initrd image for things other than the new kernel
3e90b9
      --make-default                      make the newly added entry the default boot entry
3e90b9
      --remove-args=STRING                remove kernel arguments
3e90b9
      --remove-kernel=kernel-path         remove all entries for the specified kernel
3e90b9
      --set-default=kernel-path           make the first entry referencing the specified kernel the default
3e90b9
      --set-default-index=entry-index     make the given entry index the default entry
3e90b9
      --title=entry-title                 title to use for the new kernel entry
3e90b9
      --update-kernel=kernel-path         updated information for the specified kernel
3e90b9
      --zipl                              configure zipl bootloader
3e90b9
  -b, --bls-directory                     path to directory containing the BootLoaderSpec fragment files
3e90b9
      --no-etc-grub-update                don't update the GRUB_CMDLINE_LINUX variable in /etc/default/grub
3e90b9
3e90b9
Help options:
3e90b9
  -h, --help                              Show this help message
3e90b9
3e90b9
EOF
3e90b9
}
3e90b9
3e90b9
OPTS="$(getopt -o hc:i:b:? --long help,add-kernel:,args:,bad-image-okay,\
3e90b9
config-file:,copy-default,default-kernel,default-index,default-title,env:,\
3e90b9
grub2,info:,initrd:,extra-initrd:,make-default,remove-args:,\
3e90b9
remove-kernel:,set-default:,set-default-index:,title:,update-kernel:,zipl,\
3e90b9
bls-directory:,no-etc-grub-update,add-multiboot:,mbargs:,mounts:,boot-filesystem:,\
3e90b9
bootloader-probe,debug,devtree,devtreedir:,elilo,efi,extlinux,grub,lilo,\
3e90b9
output-file:,remove-mbargs:,remove-multiboot:,silo,yaboot -n ${SCRIPTNAME} -- "$@")"
3e90b9
3e90b9
[[ $? = 0 ]] || exit 1
3e90b9
3e90b9
eval set -- "$OPTS"
3e90b9
3e90b9
while [ ${#} -gt 0 ]; do
3e90b9
    case "$1" in
3e90b9
        --help|-h)
3e90b9
            print_usage
3e90b9
            exit 0
3e90b9
            ;;
3e90b9
        --add-kernel)
3e90b9
            kernel="${2}"
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --args)
3e90b9
            args="${2}"
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --bad-image-okay)
3e90b9
            bad_image=true
3e90b9
            ;;
3e90b9
        --config-file|-c)
3e90b9
            grub_config="${2}"
3e90b9
            zipl_config="${2}"
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --copy-default)
3e90b9
            copy_default=true
3e90b9
            ;;
3e90b9
        --default-kernel)
3e90b9
            display_default="kernel"
3e90b9
            ;;
3e90b9
        --default-index)
3e90b9
            display_default="index"
3e90b9
            ;;
3e90b9
        --default-title)
3e90b9
            display_default="title"
3e90b9
            ;;
3e90b9
        --env)
3e90b9
            env="${2}"
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --grub2)
3e90b9
            bootloader="grub2"
3e90b9
            ;;
3e90b9
        --info)
3e90b9
            display_info="${2}"
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --initrd)
3e90b9
            initrd="${2}"
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --extra-initrd|-i)
3e90b9
            extra_initrd="${2}"
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --make-default)
3e90b9
            make_default=true
3e90b9
            ;;
3e90b9
        --remove-args)
3e90b9
            remove_args="${2}"
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --remove-kernel)
3e90b9
            remove_kernel="${2}"
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --set-default)
3e90b9
            set_default="${2}"
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --set-default-index)
3e90b9
            set_default="${2}"
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --title)
3e90b9
            title="${2}"
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --update-kernel)
3e90b9
            update_kernel="${2}"
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --zipl)
3e90b9
            bootloader="zipl"
3e90b9
            ;;
3e90b9
        --bls-directory|-b)
3e90b9
            blsdir="${2}"
3e90b9
	    shift
3e90b9
	    ;;
3e90b9
        --no-etc-grub-update)
3e90b9
            no_etc_update=true
3e90b9
            shift
3e90b9
            ;;
3e90b9
        --add-multiboot|--mbargs|--mounts|--boot-filesystem|\
3e90b9
        --bootloader-probe|--debug|--devtree|--devtreedir|--elilo|--efi|\
3e90b9
        --extlinux|--grub|--lilo|--output-file|--remove-mbargs|--silo|\
3e90b9
        --remove-multiboot|--slilo|--yaboot)
3e90b9
            echo
3e90b9
            echo "${SCRIPTNAME}: the option \"${1}\" was deprecated" >&2
3e90b9
            echo "Try '${SCRIPTNAME} --help' to list supported options" >&2
3e90b9
            echo
3e90b9
	    exit 1
3e90b9
	    ;;
3e90b9
        --)
3e90b9
            shift
3e90b9
            break
3e90b9
            ;;
3e90b9
        *)
3e90b9
            echo
3e90b9
            echo "${SCRIPTNAME}: invalid option \"${1}\"" >&2
3e90b9
            echo "Try '${SCRIPTNAME} --help' for more information" >&2
3e90b9
            echo
3e90b9
            exit 1
3e90b9
            ;;
3e90b9
    esac
3e90b9
    shift
3e90b9
done
3e90b9
3e90b9
if [[ -z $update_kernel && -z $kernel ]] && [[ -n $args || -n $remove_args ]]; then
3e90b9
    print_error "no action specified"
3e90b9
fi
3e90b9
3e90b9
if [[ -z $blsdir ]]; then
3e90b9
    blsdir="/boot/loader/entries"
3e90b9
fi
3e90b9
3e90b9
if [[ -z $env ]]; then
3e90b9
    env="/boot/grub2/grubenv"
3e90b9
fi
3e90b9
3e90b9
if [[ -z $zipl_config ]]; then
3e90b9
    zipl_config="/etc/zipl.conf"
3e90b9
fi
3e90b9
3e90b9
if [[ -z $grub_config ]]; then
3e90b9
    grub_config="/boot/grub2/grub.cfg"
3e90b9
fi
3e90b9
3e90b9
if [[ -z $grub_etc_default ]]; then
3e90b9
    grub_etc_default="/etc/default/grub"
3e90b9
fi
3e90b9
3e90b9
get_bls_values
3e90b9
3e90b9
default_index="$(get_default_index)"
3e90b9
3e90b9
if [[ -n $display_default ]]; then
3e90b9
    display_default_value
3e90b9
fi
3e90b9
3e90b9
if [[ -n $display_info ]]; then
3e90b9
    display_info_values "${display_info}"
3e90b9
fi
3e90b9
3e90b9
remove_var_prefix "$(get_prefix)"
3e90b9
3e90b9
if [[ -n $kernel ]]; then
3e90b9
    if [[ $copy_default = "true" ]]; then
3e90b9
	opts="${bls_options[$default_index]}"
3e90b9
	if [[ -n $args ]]; then
3e90b9
	    opts="${opts} ${args}"
3e90b9
	fi
3e90b9
    else
3e90b9
	opts="${args}"
3e90b9
    fi
3e90b9
3e90b9
    add_bls_fragment "${kernel}" "${title}" "${opts}" "${initrd}" \
3e90b9
                     "${extra_initrd}"
3e90b9
fi
3e90b9
3e90b9
if [[ -n $remove_kernel ]]; then
3e90b9
    remove_bls_fragment "${remove_kernel}"
3e90b9
fi
3e90b9
3e90b9
if [[ -n $update_kernel ]]; then
3e90b9
    update_bls_fragment "${update_kernel}" "${remove_args}" "${args}" "${initrd}"
3e90b9
fi
3e90b9
3e90b9
if [[ -n $set_default ]]; then
3e90b9
    set_default_bls "${set_default}"
3e90b9
fi
3e90b9
3e90b9
exit 0