[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

[Xen-ia64-devel] [rfc 14/16] Kdump: Add /proc/iomem_machine



Add /proc/iomem_machine. This is basically the iomem regions
as the hypervisor sees them. As opposed to Linux's /proc/iomem
which is provides a somewhat limited and distorted view of the world.
Or in other words, /proc/iomem is for pseudo-phical memory and
/proc/iomem_machine is for machine memory.

This is needed for kdump to work on ia64 as else it can't place
the crashkernel region correctly, nor can it map out all physical
memory to be included in the vmcore file in the second kernel.

There is an acompanying patch to kexec-tools to allow it
to use /proc/iomem_machine instead of /proc/iomem as appropriate.

Signed-off-by: Simon Horman <horms@xxxxxxxxxxxx>

Index: linux-2.6.18-xen.hg/fs/Kconfig
===================================================================
--- linux-2.6.18-xen.hg.orig/fs/Kconfig 2007-07-09 11:53:16.000000000 +0900
+++ linux-2.6.18-xen.hg/fs/Kconfig      2007-07-09 15:35:20.000000000 +0900
@@ -826,6 +826,11 @@ config PROC_VMCORE
         help
         Exports the dump image of crashed kernel in ELF format.
 
+config PROC_IOMEM_MACHINE
+       bool
+       depends on PROC_FS && EXPERIMENTAL && KEXEC && XEN && IA64
+       default y
+
 config SYSFS
        bool "sysfs file system support" if EMBEDDED
        default y
Index: linux-2.6.18-xen.hg/arch/ia64/kernel/efi.c
===================================================================
--- linux-2.6.18-xen.hg.orig/arch/ia64/kernel/efi.c     2007-07-09 
15:37:34.000000000 +0900
+++ linux-2.6.18-xen.hg/arch/ia64/kernel/efi.c  2007-07-09 15:37:41.000000000 
+0900
@@ -35,6 +35,11 @@
 #include <asm/processor.h>
 #include <asm/mca.h>
 
+#ifdef CONFIG_PROC_IOMEM_MACHINE
+#include <xen/interface/memory.h>
+#include <asm/hypercall.h>
+#endif
+
 #define EFI_DEBUG      0
 
 extern efi_status_t efi_call_phys (void *, ...);
@@ -947,7 +952,7 @@ efi_memmap_init(unsigned long *s, unsign
                md = p;
                if (!efi_wb(md)) {
                        if (efi_uc(md) && (md->type == EFI_CONVENTIONAL_MEMORY 
||
-                                          md->type == EFI_BOOT_SERVICES_DATA)) 
{
+                                          md->type == EFI_BOOT_SERVICES_DATA)) 
{
                                k->attribute = EFI_MEMORY_UC;
                                k->start = md->phys_addr;
                                k->num_pages = md->num_pages;
@@ -1040,21 +1045,22 @@ efi_memmap_init(unsigned long *s, unsign
        *e = (u64)++k;
 }
 
-void
-efi_initialize_iomem_resources(struct resource *code_resource,
-                              struct resource *data_resource)
+#define EFI_INITIALISE_PHYS 0x1
+#define EFI_INITIALISE_MACH 0x2
+#define EFI_INITIALISE_ALL  (EFI_INITIALISE_PHYS|EFI_INITIALISE_MACH)
+
+static void
+efi_initialize_resources(void *efi_map_start, void *efi_map_end,
+                        u64 efi_desc_size, struct resource *root_resource,
+                        struct resource *code_resource,
+                        struct resource *data_resource, unsigned flag)
 {
        struct resource *res;
-       void *efi_map_start, *efi_map_end, *p;
+       void *p;
        efi_memory_desc_t *md;
-       u64 efi_desc_size;
        char *name;
        unsigned long flags;
 
-       efi_map_start = __va(ia64_boot_param->efi_memmap);
-       efi_map_end   = efi_map_start + ia64_boot_param->efi_memmap_size;
-       efi_desc_size = ia64_boot_param->efi_memdesc_size;
-
        res = NULL;
 
        for (p = efi_map_start; p < efi_map_end; p += efi_desc_size) {
@@ -1113,7 +1119,7 @@ efi_initialize_iomem_resources(struct re
                res->end = md->phys_addr + (md->num_pages << EFI_PAGE_SHIFT) - 
1;
                res->flags = flags;
 
-               if (insert_resource(&iomem_resource, res) < 0)
+               if (insert_resource(root_resource, res) < 0)
                        kfree(res);
                else {
                        /*
@@ -1121,21 +1127,87 @@ efi_initialize_iomem_resources(struct re
                         * kernel data so we try it repeatedly and
                         * let the resource manager test it.
                         */
-                       insert_resource(res, code_resource);
-                       insert_resource(res, data_resource);
+                       if (flag & EFI_INITIALISE_PHYS) {
+                               insert_resource(res, code_resource);
+                               insert_resource(res, data_resource);
+                       }
 #ifdef CONFIG_KEXEC
-                        insert_resource(res, &efi_memmap_res);
-                        insert_resource(res, &boot_param_res);
-                       if (crashk_res.end > crashk_res.start)
-                               insert_resource(res, &crashk_res);
+                       if (flag & EFI_INITIALISE_MACH) {
+                               insert_resource(res, &efi_memmap_res);
+                               insert_resource(res, &boot_param_res);
+                               if (crashk_res.end > crashk_res.start)
+                                       insert_resource(res, &crashk_res);
 #ifdef CONFIG_XEN
-                       xen_machine_kexec_register_resources(res);
+                               xen_machine_kexec_register_resources(res);
 #endif
+                       }
 #endif
                }
        }
 }
 
+#ifdef CONFIG_PROC_IOMEM_MACHINE
+static int
+efi_initialize_iomem_machine_resources(void)
+{
+       unsigned long size;
+       xen_memory_map_t memmap;
+       xen_ia64_memmap_info_t *memmap_info = NULL;
+       void *efi_map_start, *efi_map_end;
+       u64 efi_desc_size;
+       int ret;
+
+       /* It would be nice if it wasn't neccessary to loop like this */
+       for (size = 1024; 1; size += 1024) {
+               memmap_info = kmalloc(size, GFP_KERNEL);
+               if (memmap_info == NULL)
+                       return -ENOMEM;
+
+               memmap.nr_entries = size;
+               set_xen_guest_handle(memmap.buffer, memmap_info);
+               ret = HYPERVISOR_memory_op(XENMEM_machine_memory_map, &memmap);
+               if (!ret)
+                       break;
+
+               kfree(memmap_info);
+       }
+
+       efi_map_start = &memmap_info->memdesc;
+       efi_map_end = efi_map_start + memmap_info->efi_memmap_size;
+       efi_desc_size = memmap_info->efi_memdesc_size;
+       efi_initialize_resources(efi_map_start, efi_map_end, efi_desc_size,
+                                &iomem_machine_resource, NULL, NULL,
+                                EFI_INITIALISE_MACH);
+
+       kfree(memmap_info);
+}
+#endif
+
+void
+efi_initialize_iomem_resources(struct resource *code_resource,
+                              struct resource *data_resource)
+{
+       void *efi_map_start, *efi_map_end;
+       u64 efi_desc_size;
+
+       efi_map_start = __va(ia64_boot_param->efi_memmap);
+       efi_map_end   = efi_map_start + ia64_boot_param->efi_memmap_size;
+       efi_desc_size = ia64_boot_param->efi_memdesc_size;
+
+#ifdef CONFIG_PROC_IOMEM_MACHINE
+       efi_initialize_resources(efi_map_start, efi_map_end, efi_desc_size,
+                                &iomem_resource, code_resource,
+                                data_resource, EFI_INITIALISE_PHYS);
+       efi_initialize_iomem_machine_resources();
+#else
+       efi_initialize_resources(efi_map_start, efi_map_end, efi_desc_size,
+                                &iomem_resource, code_resource,
+                                data_resource, EFI_INITIALISE_ALL);
+#endif
+}
+
+
+
 #ifdef CONFIG_KEXEC
 /* find a block of memory aligned to 64M exclude reserved regions
    rsvd_regions are sorted
Index: linux-2.6.18-xen.hg/include/linux/ioport.h
===================================================================
--- linux-2.6.18-xen.hg.orig/include/linux/ioport.h     2007-07-09 
15:37:34.000000000 +0900
+++ linux-2.6.18-xen.hg/include/linux/ioport.h  2007-07-09 15:37:41.000000000 
+0900
@@ -93,6 +93,9 @@ struct resource_list {
 /* PC/ISA/whatever - the normal PC address spaces: IO and memory */
 extern struct resource ioport_resource;
 extern struct resource iomem_resource;
+#ifdef CONFIG_XEN
+extern struct resource iomem_machine_resource;
+#endif
 
 extern int request_resource(struct resource *root, struct resource *new);
 extern struct resource * ____request_resource(struct resource *root, struct 
resource *new);
Index: linux-2.6.18-xen.hg/kernel/resource.c
===================================================================
--- linux-2.6.18-xen.hg.orig/kernel/resource.c  2007-07-09 15:37:34.000000000 
+0900
+++ linux-2.6.18-xen.hg/kernel/resource.c       2007-07-09 15:37:41.000000000 
+0900
@@ -36,6 +36,16 @@ struct resource iomem_resource = {
 };
 EXPORT_SYMBOL(iomem_resource);
 
+#ifdef CONFIG_PROC_IOMEM_MACHINE
+struct resource iomem_machine_resource = {
+       .name   = "Machine PCI mem",
+       .start  = 0,
+       .end    = -1,
+       .flags  = IORESOURCE_MEM,
+};
+EXPORT_SYMBOL(iomem_machine_resource);
+#endif
+
 static DEFINE_RWLOCK(resource_lock);
 
 #ifdef CONFIG_PROC_FS
@@ -115,6 +125,18 @@ static int iomem_open(struct inode *inod
        return res;
 }
 
+#ifdef CONFIG_PROC_IOMEM_MACHINE
+static int iomem_machine_open(struct inode *inode, struct file *file)
+{
+       int res = seq_open(file, &resource_op);
+       if (!res) {
+               struct seq_file *m = file->private_data;
+               m->private = &iomem_machine_resource;
+       }
+       return res;
+}
+#endif
+
 static struct file_operations proc_ioports_operations = {
        .open           = ioports_open,
        .read           = seq_read,
@@ -129,6 +151,15 @@ static struct file_operations proc_iomem
        .release        = seq_release,
 };
 
+#ifdef CONFIG_PROC_IOMEM_MACHINE
+static struct file_operations proc_iomem_machine_operations = {
+       .open           = iomem_machine_open,
+       .read           = seq_read,
+       .llseek         = seq_lseek,
+       .release        = seq_release,
+};
+#endif
+
 static int __init ioresources_init(void)
 {
        struct proc_dir_entry *entry;
@@ -139,6 +170,11 @@ static int __init ioresources_init(void)
        entry = create_proc_entry("iomem", 0, NULL);
        if (entry)
                entry->proc_fops = &proc_iomem_operations;
+#ifdef CONFIG_PROC_IOMEM_MACHINE
+       entry = create_proc_entry("iomem_machine", 0, NULL);
+       if (entry)
+               entry->proc_fops = &proc_iomem_machine_operations;
+#endif
        return 0;
 }
 __initcall(ioresources_init);

-- 

-- 
Horms
  H: http://www.vergenet.net/~horms/
  W: http://www.valinux.co.jp/en/


_______________________________________________
Xen-ia64-devel mailing list
Xen-ia64-devel@xxxxxxxxxxxxxxxxxxx
http://lists.xensource.com/xen-ia64-devel


 


Rackspace

Lists.xenproject.org is hosted with RackSpace, monitoring our
servers 24x7x365 and backed by RackSpace's Fanatical Support®.