5e4bb476c0
split kexec-tools into two packages, kexec and kdump. * kexec to simply execute a new kernel * kdump is for loading and collecting debris of a crashed kernel with support for kdump forensics. In order to properly support booting into a crashkernel, an init script as well as UCI configuration has been added. As modifying the kernel cmdline is required for this to work in x86 platforms use an uci-defaults script to modify /boot/grub/grub.cfg. To test collecting crash information, use the 'c' sysrq-trigger, ie. echo c > /proc/sysrq-trigger This should result in the crash kernel being executed and (depending on the configution) dmesg and/or vmcore getting saved. To check if the crash kernel was loaded properly, use the 'status' command of the kdump init script. Signed-off-by: Daniel Golle <daniel@makrotopia.org>
182 lines
3.4 KiB
Bash
Executable file
182 lines
3.4 KiB
Bash
Executable file
#!/bin/sh /etc/rc.common
|
|
|
|
START=41
|
|
STOP=98
|
|
|
|
EXTRA_COMMANDS="status"
|
|
EXTRA_HELP=" status Print crashkernel status"
|
|
|
|
verify_kdump() {
|
|
local cfg="$1"
|
|
local enabled
|
|
local path
|
|
local save_vmcore
|
|
local save_dmesg
|
|
|
|
config_get_bool enabled "$cfg" enabled 1
|
|
config_get_bool save_dmesg "$cfg" save_dmesg 1
|
|
config_get_bool save_vmcore "$cfg" save_vmcore 0
|
|
|
|
[ "$enabled" -gt 0 ] || return 2
|
|
|
|
[ "$save_dmesg" -gt 0 ] || [ "$save_vmcore" -gt 0 ] || return 2
|
|
|
|
config_get path "$cfg" path "/"
|
|
|
|
[ -d "$path" ] || mkdir -p "$path" 2>/dev/null || return 1
|
|
}
|
|
|
|
run_kdump() {
|
|
local cfg="$1"
|
|
local enabled
|
|
local path
|
|
local save_vmcore
|
|
local save_dmesg
|
|
|
|
config_get_bool enabled "$cfg" enabled 1
|
|
[ "$enabled" -gt 0 ] || return
|
|
|
|
config_get_bool save_dmesg "$cfg" save_dmesg 1
|
|
config_get_bool save_vmcore "$cfg" save_vmcore 0
|
|
config_get path "$cfg" path "/"
|
|
|
|
timestamp=$(date "+%Y%m%dT%H%M%S")
|
|
|
|
if [ "$save_vmcore" -eq 1 ]; then
|
|
# would like 'sparse' but busybox doesn't support it
|
|
dd if=/proc/vmcore of="$path/vmcore-$timestamp" conv=fsync bs=1M
|
|
fi
|
|
|
|
if [ "$save_dmesg" -eq 1 ]; then
|
|
vmcore-dmesg /proc/vmcore > "$path/dmesg-$timestamp"
|
|
fi
|
|
|
|
sync
|
|
reboot -f
|
|
}
|
|
|
|
find_kernel() {
|
|
. /lib/functions.sh
|
|
local kernel
|
|
|
|
kernel="$BOOT_IMAGE"
|
|
if [ -r "$kernel" ]; then
|
|
echo $kernel
|
|
return 0
|
|
fi
|
|
|
|
kernel="$(find_mtd_part kernel)"
|
|
if [ -r "$kernel" ]; then
|
|
echo $kernel
|
|
return 0
|
|
fi
|
|
|
|
for voldir in /sys/class/ubi/ubi*_*; do
|
|
[ ! -e "$voldir" ] && continue
|
|
if [ "$(cat "${voldir}/name")" = "kernel" ]; then
|
|
kernel="/dev/$(basename "$voldir")"
|
|
echo $kernel
|
|
return 0
|
|
fi
|
|
done
|
|
|
|
return 1
|
|
}
|
|
|
|
load_crashkernel() {
|
|
local append_cmdline
|
|
local kernel
|
|
|
|
kernel="$(find_kernel)"
|
|
[ $? -gt 0 ] && return 1
|
|
|
|
case "$(uname -m)" in
|
|
i?86|x86_64)
|
|
grep -q "crashkernel=" /proc/cmdline || return 1
|
|
append_cmdline="1 irqpoll reset_devices maxcpus=1"
|
|
;;
|
|
arm*)
|
|
append_cmdline="1 maxcpus=1 reset_devices"
|
|
;;
|
|
esac
|
|
kexec -p "$kernel" --reuse-cmdline --append="$append_cmdline"
|
|
return $?
|
|
}
|
|
|
|
start() {
|
|
local retval
|
|
|
|
if [ ! -e /sys/kernel/kexec_crash_loaded ]; then
|
|
return 1
|
|
fi
|
|
|
|
if [ -e /proc/vmcore ]; then
|
|
config_load kdump
|
|
config_foreach run_kdump kdump
|
|
else
|
|
config_load kdump
|
|
config_foreach verify_kdump kdump
|
|
retval=$?
|
|
[ $retval = 1 ] && return 1
|
|
[ $retval = 0 ] && load_crashkernel
|
|
return $?
|
|
fi
|
|
}
|
|
|
|
stop() {
|
|
[ "$(cat /sys/kernel/kexec_crash_loaded)" = "1" ] || return
|
|
|
|
if [ -e "$BOOT_IMAGE" ]; then
|
|
kexec -p -u "$BOOT_IMAGE"
|
|
fi
|
|
}
|
|
|
|
status() {
|
|
local retval kernel
|
|
|
|
if [ ! -e /sys/kernel/kexec_crash_loaded ]; then
|
|
echo "crashdump not supported by kernel"
|
|
return
|
|
fi
|
|
|
|
if [ $(cat /sys/kernel/kexec_crash_size) -eq 0 ]; then
|
|
echo "memory for crashdump kernel not reserved!"
|
|
echo "check crashkernel= kernel cmdline parameter"
|
|
echo "(a reboot is required after installing kdump)"
|
|
return
|
|
fi
|
|
|
|
kernel="$(find_kernel)"
|
|
if [ $? -gt 0 ]; then
|
|
echo "cannot find kernel image"
|
|
return
|
|
else
|
|
echo "using kernel image $kernel"
|
|
fi
|
|
|
|
echo -n "kdump configuration is "
|
|
config_load kdump
|
|
retval=$?
|
|
if [ $retval = 0 ]; then
|
|
if [ "$(config_foreach echo kdump)" ]; then
|
|
config_foreach verify_kdump kdump
|
|
retval=$?
|
|
else
|
|
retval=1
|
|
fi
|
|
fi
|
|
|
|
if [ $retval = 0 ]; then
|
|
echo "valid"
|
|
elif [ $retval = 2 ]; then
|
|
echo "disabled"
|
|
else
|
|
echo "BROKEN"
|
|
fi
|
|
|
|
echo -n "kexec crash kernel "
|
|
if [ "$(cat /sys/kernel/kexec_crash_loaded)" = "0" ]; then
|
|
echo -n "not "
|
|
fi
|
|
echo "loaded"
|
|
}
|