ab224c
#!/bin/sh
ab224c
ab224c
exec &> /dev/console
ab224c
. /lib/dracut-lib.sh
ab224c
. /lib/kdump-lib.sh
ab224c
ab224c
if [ -f "$initdir/lib/dracut/no-emergency-shell" ]; then
ab224c
    rm -f -- $initdir/lib/dracut/no-emergency-shell
ab224c
fi
ab224c
ab224c
set -o pipefail
ab224c
KDUMP_PATH="/var/crash"
ab224c
CORE_COLLECTOR=""
ab224c
DEFAULT_CORE_COLLECTOR="makedumpfile -c --message-level 1 -d 31"
ab224c
DMESG_COLLECTOR="/sbin/vmcore-dmesg"
ab224c
DEFAULT_ACTION="reboot -f"
ab224c
DATEDIR=`date +%Y.%m.%d-%T`
ab224c
HOST_IP='127.0.0.1'
ab224c
DUMP_INSTRUCTION=""
ab224c
SSH_KEY_LOCATION="/root/.ssh/kdump_id_rsa"
ab224c
KDUMP_SCRIPT_DIR="/kdumpscripts"
ab224c
DD_BLKSIZE=512
ab224c
FINAL_ACTION="reboot -f"
ab224c
DUMP_RETVAL=0
ab224c
conf_file="/etc/kdump.conf"
ab224c
KDUMP_PRE=""
ab224c
KDUMP_POST=""
ab224c
MOUNTS=""
ab224c
ab224c
export PATH=$PATH:$KDUMP_SCRIPT_DIR
ab224c
ab224c
do_dump()
ab224c
{
ab224c
    local _ret
ab224c
ab224c
    eval $DUMP_INSTRUCTION
ab224c
    _ret=$?
ab224c
ab224c
    if [ $_ret -ne 0 ]; then
ab224c
        echo "kdump: saving vmcore failed"
ab224c
    fi
ab224c
ab224c
    return $_ret
ab224c
}
ab224c
ab224c
do_umount()
ab224c
{
ab224c
    if [ -n "$MOUNTS" ]; then
ab224c
        for mount in $MOUNTS; do
ab224c
            ismounted $mount && umount -R $mount
ab224c
        done
ab224c
    fi
ab224c
}
ab224c
ab224c
do_final_action()
ab224c
{
ab224c
    do_umount
ab224c
    eval $FINAL_ACTION
ab224c
}
ab224c
ab224c
do_default_action()
ab224c
{
ab224c
    wait_for_loginit
ab224c
    eval $DEFAULT_ACTION
ab224c
}
ab224c
ab224c
do_kdump_pre()
ab224c
{
ab224c
    if [ -n "$KDUMP_PRE" ]; then
ab224c
        "$KDUMP_PRE"
ab224c
    fi
ab224c
}
ab224c
ab224c
do_kdump_post()
ab224c
{
ab224c
    if [ -n "$KDUMP_POST" ]; then
ab224c
        "$KDUMP_POST" "$1"
ab224c
    fi
ab224c
}
ab224c
ab224c
add_dump_code()
ab224c
{
ab224c
    DUMP_INSTRUCTION=$1
ab224c
}
ab224c
ab224c
# dump_fs <mount point| device>
ab224c
dump_fs()
ab224c
{
ab224c
    local _dev=$(findmnt -k -f -n -r -o SOURCE $1)
ab224c
    local _mp=$(findmnt -k -f -n -r -o TARGET $1)
ab224c
ab224c
    echo "kdump: dump target is $_dev"
ab224c
ab224c
    if [ -z "$_mp" ]; then
ab224c
        echo "kdump: error: Dump target $_dev is not mounted."
ab224c
        return 1
ab224c
    fi
ab224c
    MOUNTS="$MOUNTS $_mp"
ab224c
ab224c
    # Remove -F in makedumpfile case. We don't want a flat format dump here.
ab224c
    [[ $CORE_COLLECTOR = *makedumpfile* ]] && CORE_COLLECTOR=`echo $CORE_COLLECTOR | sed -e "s/-F//g"`
ab224c
ab224c
    echo "kdump: saving to $_mp/$KDUMP_PATH/$HOST_IP-$DATEDIR/"
ab224c
ab224c
    mount -o remount,rw $_mp || return 1
ab224c
    mkdir -p $_mp/$KDUMP_PATH/$HOST_IP-$DATEDIR || return 1
ab224c
ab224c
    save_vmcore_dmesg_fs ${DMESG_COLLECTOR} "$_mp/$KDUMP_PATH/$HOST_IP-$DATEDIR/"
ab224c
ab224c
    echo "kdump: saving vmcore"
ab224c
    $CORE_COLLECTOR /proc/vmcore $_mp/$KDUMP_PATH/$HOST_IP-$DATEDIR/vmcore-incomplete || return 1
ab224c
    mv $_mp/$KDUMP_PATH/$HOST_IP-$DATEDIR/vmcore-incomplete $_mp/$KDUMP_PATH/$HOST_IP-$DATEDIR/vmcore
ab224c
    sync
ab224c
ab224c
    echo "kdump: saving vmcore complete"
ab224c
    return 0
ab224c
}
ab224c
ab224c
dump_raw()
ab224c
{
ab224c
    local _raw=$1
ab224c
ab224c
    [ -b "$_raw" ] || return 1
ab224c
ab224c
    echo "kdump: saving to raw disk $_raw"
ab224c
ab224c
    if ! $(echo -n $CORE_COLLECTOR|grep -q makedumpfile); then
ab224c
        _src_size=`ls -l /proc/vmcore | cut -d' ' -f5`
ab224c
        _src_size_mb=$(($_src_size / 1048576))
ab224c
        monitor_dd_progress $_src_size_mb &
ab224c
    fi
ab224c
ab224c
    echo "kdump: saving vmcore"
ab224c
    $CORE_COLLECTOR /proc/vmcore | dd of=$_raw bs=$DD_BLKSIZE >> /tmp/dd_progress_file 2>&1 || return 1
ab224c
    sync
ab224c
ab224c
    echo "kdump: saving vmcore complete"
ab224c
    return 0
ab224c
}
ab224c
ab224c
dump_ssh()
ab224c
{
ab224c
    local _opt="-i $1 -o BatchMode=yes -o StrictHostKeyChecking=yes"
ab224c
    local _dir="$KDUMP_PATH/$HOST_IP-$DATEDIR"
ab224c
    local _host=$2
ab224c
ab224c
    echo "kdump: saving to $_host:$_dir"
ab224c
ab224c
    cat /var/lib/random-seed > /dev/urandom
ab224c
    ssh -q $_opt $_host mkdir -p $_dir || return 1
ab224c
ab224c
    save_vmcore_dmesg_ssh ${DMESG_COLLECTOR} ${_dir} "${_opt}" $_host
ab224c
ab224c
    echo "kdump: saving vmcore"
ab224c
ab224c
    if [ "${CORE_COLLECTOR%%[[:blank:]]*}" = "scp" ]; then
ab224c
        scp -q $_opt /proc/vmcore "$_host:$_dir/vmcore-incomplete" || return 1
ab224c
        ssh $_opt $_host "mv $_dir/vmcore-incomplete $_dir/vmcore" || return 1
ab224c
    else
ab224c
        $CORE_COLLECTOR /proc/vmcore | ssh $_opt $_host "dd bs=512 of=$_dir/vmcore-incomplete" || return 1
ab224c
        ssh $_opt $_host "mv $_dir/vmcore-incomplete $_dir/vmcore.flat" || return 1
ab224c
    fi
ab224c
ab224c
    echo "kdump: saving vmcore complete"
ab224c
    return 0
ab224c
}
ab224c
ab224c
save_vmcore_dmesg_fs() {
ab224c
    local _dmesg_collector=$1
ab224c
    local _path=$2
ab224c
ab224c
    echo "kdump: saving vmcore-dmesg.txt"
ab224c
    $_dmesg_collector /proc/vmcore > ${_path}/vmcore-dmesg-incomplete.txt
ab224c
    _exitcode=$?
ab224c
    if [ $_exitcode -eq 0 ]; then
ab224c
        mv ${_path}/vmcore-dmesg-incomplete.txt ${_path}/vmcore-dmesg.txt
ab224c
        echo "kdump: saving vmcore-dmesg.txt complete"
ab224c
    else
ab224c
        echo "kdump: saving vmcore-dmesg.txt failed"
ab224c
    fi
ab224c
}
ab224c
ab224c
save_vmcore_dmesg_ssh() {
ab224c
    local _dmesg_collector=$1
ab224c
    local _path=$2
ab224c
    local _opts="$3"
ab224c
    local _location=$4
ab224c
ab224c
    echo "kdump: saving vmcore-dmesg.txt"
ab224c
    $_dmesg_collector /proc/vmcore | ssh $_opts $_location "dd of=$_path/vmcore-dmesg-incomplete.txt"
ab224c
    _exitcode=$?
ab224c
ab224c
    if [ $_exitcode -eq 0 ]; then
ab224c
        ssh -q $_opts $_location mv $_path/vmcore-dmesg-incomplete.txt $_path/vmcore-dmesg.txt
ab224c
        echo "kdump: saving vmcore-dmesg.txt complete"
ab224c
    else
ab224c
        echo "kdump: saving vmcore-dmesg.txt failed"
ab224c
    fi
ab224c
}
ab224c
ab224c
ab224c
get_host_ip()
ab224c
{
ab224c
    local _host
ab224c
    if is_nfs_dump_target || is_ssh_dump_target
ab224c
    then
ab224c
        kdumpnic=$(getarg kdumpnic=)
ab224c
        [ -z "$kdumpnic" ] && echo "kdump: failed to get kdumpnic!" && return 1
ab224c
        _host=`ip addr show dev $kdumpnic|grep 'inet '`
ab224c
        [ $? -ne 0 ] && echo "kdump: wrong kdumpnic: $kdumpnic" && return 1
ab224c
        _host="${_host##*inet }"
ab224c
        _host="${_host%%/*}"
ab224c
        [ -z "$_host" ] && echo "kdump: wrong kdumpnic: $kdumpnic" && return 1
ab224c
        HOST_IP=$_host
ab224c
    fi
ab224c
    return 0
ab224c
}
ab224c
ab224c
read_kdump_conf()
ab224c
{
ab224c
    if [ ! -f "$conf_file" ]; then
ab224c
        echo "kdump: $conf_file not found"
ab224c
        return
ab224c
    fi
ab224c
ab224c
    # first get the necessary variables
ab224c
    while read config_opt config_val;
ab224c
    do
ab224c
        # remove inline comments after the end of a directive.
ab224c
        config_val=$(strip_comments $config_val)
ab224c
        case "$config_opt" in
ab224c
        path)
ab224c
        KDUMP_PATH="$config_val"
ab224c
            ;;
ab224c
        core_collector)
ab224c
            [ -n "$config_val" ] && CORE_COLLECTOR="$config_val"
ab224c
            ;;
ab224c
        sshkey)
ab224c
            if [ -f "$config_val" ]; then
ab224c
                SSH_KEY_LOCATION=$config_val
ab224c
            fi
ab224c
            ;;
ab224c
        kdump_pre)
ab224c
            KDUMP_PRE="$config_val"
ab224c
            ;;
ab224c
        kdump_post)
ab224c
            KDUMP_POST="$config_val"
ab224c
            ;;
ab224c
        default)
ab224c
            case $config_val in
ab224c
                shell)
ab224c
                    DEFAULT_ACTION="_emergency_shell kdump"
ab224c
                    ;;
ab224c
                reboot)
ab224c
                    DEFAULT_ACTION="do_umount; reboot -f"
ab224c
                    ;;
ab224c
                halt)
ab224c
                    DEFAULT_ACTION="do_umount; halt -f"
ab224c
                    ;;
ab224c
                poweroff)
ab224c
                    DEFAULT_ACTION="do_umount; poweroff -f"
ab224c
                    ;;
ab224c
                dump_to_rootfs)
ab224c
                    DEFAULT_ACTION="dump_fs $NEWROOT"
ab224c
                    ;;
ab224c
            esac
ab224c
            ;;
ab224c
        esac
ab224c
    done < $conf_file
ab224c
ab224c
    # rescan for add code for dump target
ab224c
    while read config_opt config_val;
ab224c
    do
ab224c
        # remove inline comments after the end of a directive.
ab224c
        config_val=$(strip_comments $config_val)
ab224c
        case "$config_opt" in
ab224c
        ext[234]|xfs|btrfs|minix|nfs)
ab224c
            add_dump_code "dump_fs $config_val"
ab224c
            ;;
ab224c
        raw)
ab224c
            add_dump_code "dump_raw $config_val"
ab224c
            ;;
ab224c
        ssh)
ab224c
            add_dump_code "dump_ssh $SSH_KEY_LOCATION $config_val"
ab224c
            ;;
ab224c
        esac
ab224c
    done < $conf_file
ab224c
}
ab224c
ab224c
read_kdump_conf
ab224c
ab224c
if [ -z "$CORE_COLLECTOR" ];then
ab224c
    CORE_COLLECTOR=$DEFAULT_CORE_COLLECTOR
ab224c
    if is_ssh_dump_target || is_raw_dump_target; then
ab224c
        CORE_COLLECTOR="$CORE_COLLECTOR -F"
ab224c
    fi
ab224c
fi
ab224c
ab224c
get_host_ip
ab224c
if [ $? -ne 0 ]; then
ab224c
    echo "kdump: get_host_ip exited with non-zero status!"
ab224c
    do_default_action
ab224c
    do_final_action
ab224c
fi
ab224c
ab224c
if [ -z "$DUMP_INSTRUCTION" ]; then
ab224c
    add_dump_code "dump_fs $NEWROOT"
ab224c
fi
ab224c
ab224c
do_kdump_pre
ab224c
if [ $? -ne 0 ]; then
ab224c
    echo "kdump: kdump_pre script exited with non-zero status!"
ab224c
    do_final_action
ab224c
fi
ab224c
ab224c
do_dump
ab224c
DUMP_RETVAL=$?
ab224c
ab224c
do_kdump_post $DUMP_RETVAL
ab224c
if [ $? -ne 0 ]; then
ab224c
    echo "kdump: kdump_post script exited with non-zero status!"
ab224c
fi
ab224c
ab224c
if [ $DUMP_RETVAL -ne 0 ]; then
ab224c
    do_default_action
ab224c
fi
ab224c
ab224c
do_final_action