080f64dafa
- Update to version 2.0.14 (bsc#1039937, FATE#320672, FATE#320671) Changelog: http://git.kernel.org/cgit/utils/kernel/kexec/kexec-tools.git/log/?id=refs/tags/v2.0.13..v2.0.14 - Backport upstream fixes (bsc#1039937, FATE#320672, FATE#320671) 0001-kexec-tools-2.0.14.git.patch 0002-ppc64-Reduce-number-of-ELF-LOAD-segments.patch 0003-kexec-Increase-the-upper-limit-for-RAM-segments.patch 0004-alpha-add-missing-__NR_kexec_load-definition.patch 0005-kexec-implemented-XEN-KEXEC-STATUS-to-determine-if-a.patch 0006-kexec-Remove-redundant-space-from-help-message.patch 0007-purgatory-Add-purgatory.map-and-purgatory.ro.sym-to-.patch 0008-kexec-Add-option-to-get-crash-kernel-region-size.patch 0009-crashdump-arm-Add-get_crash_kernel_load_range-functi.patch 0010-crashdump-arm64-Add-get_crash_kernel_load_range-func.patch 0011-crashdump-cris-Add-get_crash_kernel_load_range-funct.patch 0012-crashdump-ia64-Add-get_crash_kernel_load_range-funct.patch 0013-crashdump-m68k-Add-get_crash_kernel_load_range-funct.patch 0014-crashdump-mips-Add-get_crash_kernel_load_range-funct.patch 0015-crashdump-ppc-Add-get_crash_kernel_load_range-functi.patch 0016-crashdump-ppc64-Add-get_crash_kernel_load_range-func.patch 0017-crashdump-s390-Add-get_crash_kernel_load_range-funct.patch 0018-crashdump-sh-Add-get_crash_kernel_load_range-functio.patch 0019-gitignore-add-two-generated-files-in-purgatory.patch 0020-Only-print-debug-message-when-failed-to-serach-for-k.patch 0021-build_mem_phdrs-check-if-p_paddr-is-invalid.patch 0022-uImage-fix-realloc-pointer-confusion.patch 0023-uImage-Fix-uImage_load-for-little-endian-machines.patch 0024-uImage-Add-new-IH_ARCH_xxx-definitions.patch 0025-uImage-use-char-instead-of-unsigned-char-for-uImage_.patch 0026-uImage-use-char-instead-of-unsigned-char-for-uImage_.patch 0027-arm64-add-uImage-support.patch OBS-URL: https://build.opensuse.org/request/show/500203 OBS-URL: https://build.opensuse.org/package/show/Kernel:kdump/kexec-tools?expand=0&rev=83
283 lines
7.4 KiB
Diff
283 lines
7.4 KiB
Diff
From 5f955585c7c9166da5b8f33e3d8a4c43845ee70b Mon Sep 17 00:00:00 2001
|
|
From: AKASHI Takahiro <takahiro.akashi@linaro.org>
|
|
Date: Wed, 17 May 2017 14:51:49 +0900
|
|
Subject: [PATCH 43/45] arm64: kdump: add DT properties to crash dump kernel's
|
|
dtb
|
|
|
|
We pass the following properties to crash dump kernel:
|
|
linux,elfcorehdr: elf core header segment,
|
|
same as "elfcorehdr=" kernel parameter on other archs
|
|
linux,usable-memory-range: usable memory reserved for crash dump kernel
|
|
|
|
Signed-off-by: AKASHI Takahiro <takahiro.akashi@linaro.org>
|
|
Tested-by: David Woodhouse <dwmw@amazon.co.uk>
|
|
Tested-by: Pratyush Anand <panand@redhat.com>
|
|
Signed-off-by: Simon Horman <horms@verge.net.au>
|
|
---
|
|
kexec/arch/arm64/kexec-arm64.c | 197 ++++++++++++++++++++++++++++++++++++-
|
|
kexec/arch/arm64/kexec-elf-arm64.c | 5 -
|
|
2 files changed, 192 insertions(+), 10 deletions(-)
|
|
|
|
diff --git a/kexec/arch/arm64/kexec-arm64.c b/kexec/arch/arm64/kexec-arm64.c
|
|
index dfe16a626136..62f37585b788 100644
|
|
--- a/kexec/arch/arm64/kexec-arm64.c
|
|
+++ b/kexec/arch/arm64/kexec-arm64.c
|
|
@@ -25,6 +25,14 @@
|
|
#include "kexec-syscall.h"
|
|
#include "arch/options.h"
|
|
|
|
+#define ROOT_NODE_ADDR_CELLS_DEFAULT 1
|
|
+#define ROOT_NODE_SIZE_CELLS_DEFAULT 1
|
|
+
|
|
+#define PROP_ADDR_CELLS "#address-cells"
|
|
+#define PROP_SIZE_CELLS "#size-cells"
|
|
+#define PROP_ELFCOREHDR "linux,elfcorehdr"
|
|
+#define PROP_USABLE_MEM_RANGE "linux,usable-memory-range"
|
|
+
|
|
/* Global varables the core kexec routines expect. */
|
|
|
|
unsigned char reuse_initrd;
|
|
@@ -129,9 +137,6 @@ int arch_process_options(int argc, char **argv)
|
|
case OPT_INITRD:
|
|
arm64_opts.initrd = optarg;
|
|
break;
|
|
- case OPT_PANIC:
|
|
- die("load-panic (-p) not supported");
|
|
- break;
|
|
default:
|
|
break; /* Ignore core and unknown options. */
|
|
}
|
|
@@ -282,12 +287,115 @@ on_success:
|
|
return 0;
|
|
}
|
|
|
|
+static int get_cells_size(void *fdt, uint32_t *address_cells,
|
|
+ uint32_t *size_cells)
|
|
+{
|
|
+ int nodeoffset;
|
|
+ const uint32_t *prop = NULL;
|
|
+ int prop_len;
|
|
+
|
|
+ /* default values */
|
|
+ *address_cells = ROOT_NODE_ADDR_CELLS_DEFAULT;
|
|
+ *size_cells = ROOT_NODE_SIZE_CELLS_DEFAULT;
|
|
+
|
|
+ /* under root node */
|
|
+ nodeoffset = fdt_path_offset(fdt, "/");
|
|
+ if (nodeoffset < 0)
|
|
+ goto on_error;
|
|
+
|
|
+ prop = fdt_getprop(fdt, nodeoffset, PROP_ADDR_CELLS, &prop_len);
|
|
+ if (prop) {
|
|
+ if (prop_len == sizeof(*prop))
|
|
+ *address_cells = fdt32_to_cpu(*prop);
|
|
+ else
|
|
+ goto on_error;
|
|
+ }
|
|
+
|
|
+ prop = fdt_getprop(fdt, nodeoffset, PROP_SIZE_CELLS, &prop_len);
|
|
+ if (prop) {
|
|
+ if (prop_len == sizeof(*prop))
|
|
+ *size_cells = fdt32_to_cpu(*prop);
|
|
+ else
|
|
+ goto on_error;
|
|
+ }
|
|
+
|
|
+ dbgprintf("%s: #address-cells:%d #size-cells:%d\n", __func__,
|
|
+ *address_cells, *size_cells);
|
|
+ return 0;
|
|
+
|
|
+on_error:
|
|
+ return EFAILED;
|
|
+}
|
|
+
|
|
+static bool cells_size_fitted(uint32_t address_cells, uint32_t size_cells,
|
|
+ struct memory_range *range)
|
|
+{
|
|
+ dbgprintf("%s: %llx-%llx\n", __func__, range->start, range->end);
|
|
+
|
|
+ /* if *_cells >= 2, cells can hold 64-bit values anyway */
|
|
+ if ((address_cells == 1) && (range->start >= (1ULL << 32)))
|
|
+ return false;
|
|
+
|
|
+ if ((size_cells == 1) &&
|
|
+ ((range->end - range->start + 1) >= (1ULL << 32)))
|
|
+ return false;
|
|
+
|
|
+ return true;
|
|
+}
|
|
+
|
|
+static void fill_property(void *buf, uint64_t val, uint32_t cells)
|
|
+{
|
|
+ uint32_t val32;
|
|
+ int i;
|
|
+
|
|
+ if (cells == 1) {
|
|
+ val32 = cpu_to_fdt32((uint32_t)val);
|
|
+ memcpy(buf, &val32, sizeof(uint32_t));
|
|
+ } else {
|
|
+ for (i = 0;
|
|
+ i < (cells * sizeof(uint32_t) - sizeof(uint64_t)); i++)
|
|
+ *(char *)buf++ = 0;
|
|
+
|
|
+ val = cpu_to_fdt64(val);
|
|
+ memcpy(buf, &val, sizeof(uint64_t));
|
|
+ }
|
|
+}
|
|
+
|
|
+static int fdt_setprop_range(void *fdt, int nodeoffset,
|
|
+ const char *name, struct memory_range *range,
|
|
+ uint32_t address_cells, uint32_t size_cells)
|
|
+{
|
|
+ void *buf, *prop;
|
|
+ size_t buf_size;
|
|
+ int result;
|
|
+
|
|
+ buf_size = (address_cells + size_cells) * sizeof(uint32_t);
|
|
+ prop = buf = xmalloc(buf_size);
|
|
+
|
|
+ fill_property(prop, range->start, address_cells);
|
|
+ prop += address_cells * sizeof(uint32_t);
|
|
+
|
|
+ fill_property(prop, range->end - range->start + 1, size_cells);
|
|
+ prop += size_cells * sizeof(uint32_t);
|
|
+
|
|
+ result = fdt_setprop(fdt, nodeoffset, name, buf, buf_size);
|
|
+
|
|
+ free(buf);
|
|
+
|
|
+ return result;
|
|
+}
|
|
+
|
|
/**
|
|
* setup_2nd_dtb - Setup the 2nd stage kernel's dtb.
|
|
*/
|
|
|
|
-static int setup_2nd_dtb(struct dtb *dtb, char *command_line)
|
|
+static int setup_2nd_dtb(struct dtb *dtb, char *command_line, int on_crash)
|
|
{
|
|
+ uint32_t address_cells, size_cells;
|
|
+ int range_len;
|
|
+ int nodeoffset;
|
|
+ char *new_buf = NULL;
|
|
+ int new_size;
|
|
int result;
|
|
|
|
result = fdt_check_header(dtb->buf);
|
|
@@ -299,8 +407,86 @@ static int setup_2nd_dtb(struct dtb *dtb, char *command_line)
|
|
|
|
result = set_bootargs(dtb, command_line);
|
|
|
|
+ if (on_crash) {
|
|
+ /* determine #address-cells and #size-cells */
|
|
+ result = get_cells_size(dtb->buf, &address_cells, &size_cells);
|
|
+ if (result) {
|
|
+ fprintf(stderr,
|
|
+ "kexec: cannot determine cells-size.\n");
|
|
+ result = -EINVAL;
|
|
+ goto on_error;
|
|
+ }
|
|
+
|
|
+ if (!cells_size_fitted(address_cells, size_cells,
|
|
+ &elfcorehdr_mem)) {
|
|
+ fprintf(stderr,
|
|
+ "kexec: elfcorehdr doesn't fit cells-size.\n");
|
|
+ result = -EINVAL;
|
|
+ goto on_error;
|
|
+ }
|
|
+
|
|
+ if (!cells_size_fitted(address_cells, size_cells,
|
|
+ &crash_reserved_mem)) {
|
|
+ fprintf(stderr,
|
|
+ "kexec: usable memory range doesn't fit cells-size.\n");
|
|
+ result = -EINVAL;
|
|
+ goto on_error;
|
|
+ }
|
|
+
|
|
+ /* duplicate dt blob */
|
|
+ range_len = sizeof(uint32_t) * (address_cells + size_cells);
|
|
+ new_size = fdt_totalsize(dtb->buf)
|
|
+ + fdt_prop_len(PROP_ELFCOREHDR, range_len)
|
|
+ + fdt_prop_len(PROP_USABLE_MEM_RANGE, range_len);
|
|
+
|
|
+ new_buf = xmalloc(new_size);
|
|
+ result = fdt_open_into(dtb->buf, new_buf, new_size);
|
|
+ if (result) {
|
|
+ dbgprintf("%s: fdt_open_into failed: %s\n", __func__,
|
|
+ fdt_strerror(result));
|
|
+ result = -ENOSPC;
|
|
+ goto on_error;
|
|
+ }
|
|
+
|
|
+ /* add linux,elfcorehdr */
|
|
+ nodeoffset = fdt_path_offset(new_buf, "/chosen");
|
|
+ result = fdt_setprop_range(new_buf, nodeoffset,
|
|
+ PROP_ELFCOREHDR, &elfcorehdr_mem,
|
|
+ address_cells, size_cells);
|
|
+ if (result) {
|
|
+ dbgprintf("%s: fdt_setprop failed: %s\n", __func__,
|
|
+ fdt_strerror(result));
|
|
+ result = -EINVAL;
|
|
+ goto on_error;
|
|
+ }
|
|
+
|
|
+ /* add linux,usable-memory-range */
|
|
+ nodeoffset = fdt_path_offset(new_buf, "/chosen");
|
|
+ result = fdt_setprop_range(new_buf, nodeoffset,
|
|
+ PROP_USABLE_MEM_RANGE, &crash_reserved_mem,
|
|
+ address_cells, size_cells);
|
|
+ if (result) {
|
|
+ dbgprintf("%s: fdt_setprop failed: %s\n", __func__,
|
|
+ fdt_strerror(result));
|
|
+ result = -EINVAL;
|
|
+ goto on_error;
|
|
+ }
|
|
+
|
|
+ fdt_pack(new_buf);
|
|
+ dtb->buf = new_buf;
|
|
+ dtb->size = fdt_totalsize(new_buf);
|
|
+ }
|
|
+
|
|
dump_reservemap(dtb);
|
|
|
|
+
|
|
+ return result;
|
|
+
|
|
+on_error:
|
|
+ fprintf(stderr, "kexec: %s failed.\n", __func__);
|
|
+ if (new_buf)
|
|
+ free(new_buf);
|
|
+
|
|
return result;
|
|
}
|
|
|
|
@@ -368,7 +554,8 @@ int arm64_load_other_segments(struct kexec_info *info,
|
|
}
|
|
}
|
|
|
|
- result = setup_2nd_dtb(&dtb, command_line);
|
|
+ result = setup_2nd_dtb(&dtb, command_line,
|
|
+ info->kexec_flags & KEXEC_ON_CRASH);
|
|
|
|
if (result)
|
|
return EFAILED;
|
|
diff --git a/kexec/arch/arm64/kexec-elf-arm64.c b/kexec/arch/arm64/kexec-elf-arm64.c
|
|
index a9611474ef51..fc83b42c9d2b 100644
|
|
--- a/kexec/arch/arm64/kexec-elf-arm64.c
|
|
+++ b/kexec/arch/arm64/kexec-elf-arm64.c
|
|
@@ -47,11 +47,6 @@ int elf_arm64_load(int argc, char **argv, const char *kernel_buf,
|
|
int result;
|
|
int i;
|
|
|
|
- if (info->kexec_flags & KEXEC_ON_CRASH) {
|
|
- fprintf(stderr, "kexec: kdump not yet supported on arm64\n");
|
|
- return EFAILED;
|
|
- }
|
|
-
|
|
result = build_elf_exec_info(kernel_buf, kernel_size, &ehdr, 0);
|
|
|
|
if (result < 0) {
|
|
--
|
|
2.13.0
|
|
|