| #!/bin/sh /etc/rc.common |
| |
| START=41 |
| STOP=90 |
| |
| BOOT_IMAGE=/boot/vmlinuz |
| |
| 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 |
| echo -n "Saving vmcore (this may take a while)..." |
| # would like 'sparse' but busybox doesn't support it |
| dd if=/proc/vmcore of="$path/vmcore-$timestamp" conv=fsync bs=1M |
| echo " done" |
| 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" |
| } |