From 51e17d89d7c9bd0ed77eeff9b3ed4509b9213a24 Mon Sep 17 00:00:00 2001
From: Dave Anderson <anderson@redhat.com>
Date: Thu, 13 Nov 2014 14:40:54 -0500
Subject: [PATCH] Fix for the support of compressed kdump clones created with
 the KVM "virsh dump --memory-only --format <compression-type>" command, where
 the compression-type is either "kdump-zlib", "kdump-lzo" or "kdump-snappy". 
 Without the patch, if an x86_64 guest kernel was loaded with a non-zero
 "phys_base", the "--machdep phys_base=<offset>" command line option was
 required as a workaround or the crash session would fail with the warning
 message "WARNING: cannot read linux_banner string" followed by the fatal
 error message "crash: vmlinux and <dumpfile name> do not match!".
 (anderson@redhat.com)

---
 defs.h     | 3 ++-
 diskdump.c | 8 ++++++--
 help.c     | 2 +-
 main.c     | 6 ++++--
 netdump.c  | 8 ++++----
 task.c     | 2 +-
 x86_64.c   | 8 +++++++-
 7 files changed, 25 insertions(+), 12 deletions(-)

diff --git a/defs.h b/defs.h
index 049da4e..2e52bc4 100644
--- a/defs.h
+++ b/defs.h
@@ -499,7 +499,7 @@ struct program_context {
 #define FLAT_FORMAT() (pc->flags2 & FLAT)
 #define ELF_NOTES_VALID() (pc->flags2 & ELF_NOTES)
 #define RADIX_OVERRIDE (0x80ULL)
-#define QEMU_MEM_DUMP (0x100ULL)
+#define QEMU_MEM_DUMP_ELF (0x100ULL)
 #define GET_LOG       (0x200ULL)
 #define VMCOREINFO    (0x400ULL)
 #define ALLOW_FP      (0x800ULL)
@@ -509,6 +509,7 @@ struct program_context {
 #define OFFLINE_HIDE     (0x4000ULL)
 #define INCOMPLETE_DUMP  (0x8000ULL)
 #define is_incomplete_dump() (pc->flags2 & INCOMPLETE_DUMP)
+#define QEMU_MEM_DUMP_COMPRESSED (0x10000ULL)
 	char *cleanup;
 	char *namelist_orig;
 	char *namelist_debug_orig;
diff --git a/diskdump.c b/diskdump.c
index 603c325..3d33fdc 100644
--- a/diskdump.c
+++ b/diskdump.c
@@ -249,11 +249,13 @@ process_elf32_notes(void *note_buf, unsigned long size_note)
 	for (index = 0; index < size_note; index += len) {
 		nt = note_buf + index;
 
-		if(nt->n_type == NT_PRSTATUS) {
+		if (nt->n_type == NT_PRSTATUS) {
 			dd->nt_prstatus_percpu[num] = nt;
 			num++;
 		}
 		len = sizeof(Elf32_Nhdr);
+		if (STRNEQ((char *)nt + len, "QEMU"))
+			pc->flags2 |= QEMU_MEM_DUMP_COMPRESSED;
 		len = roundup(len + nt->n_namesz, 4);
 		len = roundup(len + nt->n_descsz, 4);
 	}
@@ -275,11 +277,13 @@ process_elf64_notes(void *note_buf, unsigned long size_note)
 	for (index = 0; index < size_note; index += len) {
 		nt = note_buf + index;
 
-		if(nt->n_type == NT_PRSTATUS) {
+		if (nt->n_type == NT_PRSTATUS) {
 			dd->nt_prstatus_percpu[num] = nt;
 			num++;
 		}
 		len = sizeof(Elf64_Nhdr);
+		if (STRNEQ((char *)nt + len, "QEMU"))
+			pc->flags2 |= QEMU_MEM_DUMP_COMPRESSED;
 		len = roundup(len + nt->n_namesz, 4);
 		len = roundup(len + nt->n_descsz, 4);
 	}
diff --git a/help.c b/help.c
index 48e6de6..6aa3e20 100644
--- a/help.c
+++ b/help.c
@@ -688,7 +688,7 @@ cmd_help(void)
 static void
 dump_registers(void)
 {
-	if (pc->flags2 & QEMU_MEM_DUMP) {
+	if (pc->flags2 & QEMU_MEM_DUMP_ELF) {
 		dump_registers_for_qemu_mem_dump();
 		return;
 	} else if (DISKDUMP_DUMPFILE()) {
diff --git a/main.c b/main.c
index cd8c37e..d16ef12 100644
--- a/main.c
+++ b/main.c
@@ -1401,8 +1401,10 @@ dump_program_context(void)
 		fprintf(fp, "%sLIVE_DUMP", others++ ? "|" : "");
 	if (pc->flags2 & RADIX_OVERRIDE)
 		fprintf(fp, "%sRADIX_OVERRIDE", others++ ? "|" : "");
-	if (pc->flags2 & QEMU_MEM_DUMP)
-		fprintf(fp, "%sQEMU_MEM_DUMP", others++ ? "|" : "");
+	if (pc->flags2 & QEMU_MEM_DUMP_ELF)
+		fprintf(fp, "%sQEMU_MEM_DUMP_ELF", others++ ? "|" : "");
+	if (pc->flags2 & QEMU_MEM_DUMP_COMPRESSED)
+		fprintf(fp, "%sQEMU_MEM_DUMP_COMPRESSED", others++ ? "|" : "");
 	if (pc->flags2 & GET_LOG)
 		fprintf(fp, "%sGET_LOG", others++ ? "|" : "");
 	if (pc->flags2 & VMCOREINFO)
diff --git a/netdump.c b/netdump.c
index 726c3f1..903faa0 100644
--- a/netdump.c
+++ b/netdump.c
@@ -1883,7 +1883,7 @@ dump_Elf32_Nhdr(Elf32_Off offset, int store)
 			netdump_print("(?)\n");
 
 		if (qemuinfo)
-			pc->flags2 |= QEMU_MEM_DUMP;
+			pc->flags2 |= QEMU_MEM_DUMP_ELF;
 		break;
 
 	case NT_XEN_KDUMP_CR3: 
@@ -2163,7 +2163,7 @@ dump_Elf64_Nhdr(Elf64_Off offset, int store)
                         netdump_print("(?)\n");
 
 		if (qemuinfo)
-			pc->flags2 |= QEMU_MEM_DUMP;
+			pc->flags2 |= QEMU_MEM_DUMP_ELF;
                 break;
 
 	case NT_XEN_KDUMP_CR3: 
@@ -3939,7 +3939,7 @@ kdump_backup_region_init(void)
 		sd = get_sadump_data();
 		is_32_bit = FALSE;
 		sprintf(typename, "sadump");
-	} else if (pc->flags2 & QEMU_MEM_DUMP) {
+	} else if (pc->flags2 & QEMU_MEM_DUMP_ELF) {
 		vd = get_kdump_vmcore_data();
 		if (vd->flags & KDUMP_ELF32)
 			is_32_bit = TRUE;
@@ -4122,7 +4122,7 @@ kdump_backup_region_init(void)
 					sd->backup_src_start = backup_src_start;
 					sd->backup_src_size = backup_src_size;
 					sd->backup_offset = backup_offset;
-				} else if (pc->flags2 & QEMU_MEM_DUMP) {
+				} else if (pc->flags2 & QEMU_MEM_DUMP_ELF) {
 					vd->flags |= QEMU_MEM_DUMP_KDUMP_BACKUP;
 					vd->backup_src_start = backup_src_start;
 					vd->backup_src_size = backup_src_size;
diff --git a/task.c b/task.c
index fe45921..f5bbe64 100644
--- a/task.c
+++ b/task.c
@@ -483,7 +483,7 @@ task_init(void)
 		tt->this_task = pid_to_task(active_pid);
 	}
 	else {
-		if (KDUMP_DUMPFILE() && !(pc->flags2 & QEMU_MEM_DUMP))
+		if (KDUMP_DUMPFILE() && !(pc->flags2 & QEMU_MEM_DUMP_ELF))
 			map_cpus_to_prstatus();
 		else if (ELF_NOTES_VALID() && DISKDUMP_DUMPFILE())
 			map_cpus_to_prstatus_kdump_cmprs();
diff --git a/x86_64.c b/x86_64.c
index 03adc85..bbf1326 100644
--- a/x86_64.c
+++ b/x86_64.c
@@ -6039,6 +6039,12 @@ x86_64_calc_phys_base(void)
 	if (DISKDUMP_DUMPFILE()) {
 		if (diskdump_phys_base(&phys_base)) {
 			machdep->machspec->phys_base = phys_base;
+			if ((pc->flags2 & QEMU_MEM_DUMP_COMPRESSED) && 
+			    !x86_64_virt_phys_base())
+				error(WARNING,
+				    "cannot determine physical base address:"
+				    " defaulting to %lx\n\n",
+					machdep->machspec->phys_base);
 			if (CRASHDEBUG(1))
 				fprintf(fp, "compressed kdump: phys_base: %lx\n",
 					phys_base);
@@ -6099,7 +6105,7 @@ x86_64_calc_phys_base(void)
 			}
 		}
 
-		if ((pc->flags2 & QEMU_MEM_DUMP) && !x86_64_virt_phys_base())
+		if ((pc->flags2 & QEMU_MEM_DUMP_ELF) && !x86_64_virt_phys_base())
 			error(WARNING,
 			    "cannot determine physical base address:"
 			    " defaulting to %lx\n\n",