add mips64 support to kexec-tools, counterpart of r17768.
SVN-Revision: 17769
This commit is contained in:
		| @@ -9,7 +9,7 @@ include $(TOPDIR)/rules.mk | ||||
|  | ||||
| PKG_NAME:=kexec-tools | ||||
| PKG_VERSION:=2.0.1 | ||||
| PKG_RELEASE:=2 | ||||
| PKG_RELEASE:=3 | ||||
|  | ||||
| PKG_SOURCE:=$(PKG_NAME)-$(PKG_VERSION).tar.bz2 | ||||
| PKG_SOURCE_URL:=@KERNEL/linux/kernel/people/horms/kexec-tools | ||||
|   | ||||
							
								
								
									
										732
									
								
								package/kexec-tools/patches/0005-mips64_support.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										732
									
								
								package/kexec-tools/patches/0005-mips64_support.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,732 @@ | ||||
| Index: kexec-tools-2.0.1/kexec/arch/mips/Makefile | ||||
| =================================================================== | ||||
| --- kexec-tools-2.0.1.orig/kexec/arch/mips/Makefile	2008-07-15 02:46:43.000000000 +0200 | ||||
| +++ kexec-tools-2.0.1/kexec/arch/mips/Makefile	2009-09-27 19:07:26.000000000 +0200 | ||||
| @@ -4,7 +4,7 @@ | ||||
|  mips_KEXEC_SRCS =  kexec/arch/mips/kexec-mips.c | ||||
|  mips_KEXEC_SRCS += kexec/arch/mips/kexec-elf-mips.c | ||||
|  mips_KEXEC_SRCS += kexec/arch/mips/kexec-elf-rel-mips.c | ||||
| -mips_KEXEC_SRCS += kexec/arch/mips/mips-setup-simple.S | ||||
| +mips_KEXEC_SRCS += kexec/arch/mips/crashdump-mips.c | ||||
|   | ||||
|  mips_ADD_BUFFER = | ||||
|  mips_ADD_SEGMENT = | ||||
| Index: kexec-tools-2.0.1/kexec/arch/mips/crashdump-mips.c | ||||
| =================================================================== | ||||
| --- /dev/null	1970-01-01 00:00:00.000000000 +0000 | ||||
| +++ kexec-tools-2.0.1/kexec/arch/mips/crashdump-mips.c	2009-09-27 19:07:26.000000000 +0200 | ||||
| @@ -0,0 +1,371 @@ | ||||
| +/* | ||||
| + * kexec: Linux boots Linux | ||||
| + * | ||||
| + * 2005 (C) IBM Corporation. | ||||
| + * 2008 (C) MontaVista Software, Inc. | ||||
| + * | ||||
| + * This program is free software; you can redistribute it and/or modify | ||||
| + * it under the terms of the GNU General Public License as published by | ||||
| + * the Free Software Foundation (version 2 of the License). | ||||
| + * | ||||
| + * This program is distributed in the hope that it will be useful, | ||||
| + * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||||
| + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | ||||
| + * GNU General Public License for more details. | ||||
| + * | ||||
| + * You should have received a copy of the GNU General Public License | ||||
| + * along with this program; if not, write to the Free Software | ||||
| + * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | ||||
| + */ | ||||
| +#include <stdio.h> | ||||
| +#include <string.h> | ||||
| +#include <stdlib.h> | ||||
| +#include <errno.h> | ||||
| +#include <elf.h> | ||||
| +#include <sys/types.h> | ||||
| +#include <sys/stat.h> | ||||
| +#include <unistd.h> | ||||
| +#include "../../kexec.h" | ||||
| +#include "../../kexec-elf.h" | ||||
| +#include "../../kexec-syscall.h" | ||||
| +#include "../../crashdump.h" | ||||
| +#include "kexec-mips.h" | ||||
| +#include "crashdump-mips.h" | ||||
| + | ||||
| +extern struct arch_options_t arch_options; | ||||
| + | ||||
| +/* Stores a sorted list of RAM memory ranges for which to create elf headers. | ||||
| + * A separate program header is created for backup region */ | ||||
| +static struct memory_range crash_memory_range[CRASH_MAX_MEMORY_RANGES]; | ||||
| + | ||||
| +/* Memory region reserved for storing panic kernel and other data. */ | ||||
| +static struct memory_range crash_reserved_mem; | ||||
| + | ||||
| +/* | ||||
| + * To store the memory size of the first kernel and this value will be | ||||
| + * passed to the second kernel as command line (savemaxmem=xM). | ||||
| + * The second kernel will be calculated saved_max_pfn based on this | ||||
| + * variable. | ||||
| + */ | ||||
| +unsigned long long saved_max_mem = 0; | ||||
| + | ||||
| +/* Removes crash reserve region from list of memory chunks for whom elf program | ||||
| + * headers have to be created. Assuming crash reserve region to be a single | ||||
| + * continuous area fully contained inside one of the memory chunks */ | ||||
| +static int exclude_crash_reserve_region(int *nr_ranges) | ||||
| +{ | ||||
| +    int i, j, tidx = -1; | ||||
| +    unsigned long long cstart, cend; | ||||
| +    struct memory_range temp_region; | ||||
| + | ||||
| +    /* Crash reserved region. */ | ||||
| +    cstart = crash_reserved_mem.start; | ||||
| +    cend = crash_reserved_mem.end; | ||||
| + | ||||
| +    for (i = 0; i < (*nr_ranges); i++) { | ||||
| +        unsigned long long mstart, mend; | ||||
| +        mstart = crash_memory_range[i].start; | ||||
| +        mend = crash_memory_range[i].end; | ||||
| +        if (cstart < mend && cend > mstart) { | ||||
| +            if (cstart != mstart && cend != mend) { | ||||
| +                /* Split memory region */ | ||||
| +                crash_memory_range[i].end = cstart - 1; | ||||
| +                temp_region.start = cend + 1; | ||||
| +                temp_region.end = mend; | ||||
| +                temp_region.type = RANGE_RAM; | ||||
| +                tidx = i+1; | ||||
| +            } else if (cstart != mstart) | ||||
| +                crash_memory_range[i].end = cstart - 1; | ||||
| +            else | ||||
| +                crash_memory_range[i].start = cend + 1; | ||||
| +        } | ||||
| +    } | ||||
| +    /* Insert split memory region, if any. */ | ||||
| +    if (tidx >= 0) { | ||||
| +        if (*nr_ranges == CRASH_MAX_MEMORY_RANGES) { | ||||
| +            /* No space to insert another element. */ | ||||
| +            fprintf(stderr, "Error: Number of crash memory ranges" | ||||
| +                    " excedeed the max limit\n"); | ||||
| +            return -1; | ||||
| +        } | ||||
| +        for (j = (*nr_ranges - 1); j >= tidx; j--) | ||||
| +            crash_memory_range[j+1] = crash_memory_range[j]; | ||||
| +        crash_memory_range[tidx].start = temp_region.start; | ||||
| +        crash_memory_range[tidx].end = temp_region.end; | ||||
| +        crash_memory_range[tidx].type = temp_region.type; | ||||
| +        (*nr_ranges)++; | ||||
| +    } | ||||
| +    return 0; | ||||
| +} | ||||
| +/* Reads the appropriate file and retrieves the SYSTEM RAM regions for whom to | ||||
| + * create Elf headers. Keeping it separate from get_memory_ranges() as | ||||
| + * requirements are different in the case of normal kexec and crashdumps. | ||||
| + * | ||||
| + * Normal kexec needs to look at all of available physical memory irrespective | ||||
| + * of the fact how much of it is being used by currently running kernel. | ||||
| + * Crashdumps need to have access to memory regions actually being used by | ||||
| + * running  kernel. Expecting a different file/data structure than /proc/iomem | ||||
| + * to look into down the line. May be something like /proc/kernelmem or may | ||||
| + * be zone data structures exported from kernel. | ||||
| + */ | ||||
| +static int get_crash_memory_ranges(struct memory_range **range, int *ranges) | ||||
| +{ | ||||
| +    const char iomem[]= "/proc/iomem"; | ||||
| +    int i, memory_ranges = 0; | ||||
| +    char line[MAX_LINE]; | ||||
| +    FILE *fp; | ||||
| +    unsigned long long start, end; | ||||
| + | ||||
| +    fp = fopen(iomem, "r"); | ||||
| +    if (!fp) { | ||||
| +        fprintf(stderr, "Cannot open %s: %s\n", | ||||
| +            iomem, strerror(errno)); | ||||
| +        return -1; | ||||
| +    } | ||||
| + | ||||
| +    /* Separate segment for backup region */ | ||||
| +    crash_memory_range[0].start = BACKUP_SRC_START; | ||||
| +    crash_memory_range[0].end = BACKUP_SRC_END; | ||||
| +    crash_memory_range[0].type = RANGE_RAM; | ||||
| +    memory_ranges++; | ||||
| + | ||||
| +    while(fgets(line, sizeof(line), fp) != 0) { | ||||
| +        char *str; | ||||
| +        int type, consumed, count; | ||||
| +        if (memory_ranges >= CRASH_MAX_MEMORY_RANGES) | ||||
| +            break; | ||||
| +        count = sscanf(line, "%Lx-%Lx : %n", | ||||
| +            &start, &end, &consumed); | ||||
| +        if (count != 2) | ||||
| +            continue; | ||||
| +        str = line + consumed; | ||||
| + | ||||
| +        /* Only Dumping memory of type System RAM. */ | ||||
| +        if (memcmp(str, "System RAM\n", 11) == 0) { | ||||
| +            type = RANGE_RAM; | ||||
| +        } else if (memcmp(str, "Crash kernel\n", 13) == 0) { | ||||
| +                /* Reserved memory region. New kernel can | ||||
| +                 * use this region to boot into. */ | ||||
| +                crash_reserved_mem.start = start; | ||||
| +                crash_reserved_mem.end = end; | ||||
| +                crash_reserved_mem.type = RANGE_RAM; | ||||
| +                continue; | ||||
| +        } else { | ||||
| +            continue; | ||||
| +        } | ||||
| + | ||||
| +        if (start == BACKUP_SRC_START && end >= (BACKUP_SRC_END + 1)) | ||||
| +            start = BACKUP_SRC_END + 1; | ||||
| + | ||||
| +        crash_memory_range[memory_ranges].start = start; | ||||
| +        crash_memory_range[memory_ranges].end = end; | ||||
| +        crash_memory_range[memory_ranges].type = type; | ||||
| +        memory_ranges++; | ||||
| + | ||||
| +        /* Segregate linearly mapped region. */ | ||||
| +        if ((MAXMEM - 1) >= start && (MAXMEM - 1) <= end) { | ||||
| +            crash_memory_range[memory_ranges-1].end = MAXMEM -1; | ||||
| + | ||||
| +            /* Add segregated region. */ | ||||
| +            crash_memory_range[memory_ranges].start = MAXMEM; | ||||
| +            crash_memory_range[memory_ranges].end = end; | ||||
| +            crash_memory_range[memory_ranges].type = type; | ||||
| +            memory_ranges++; | ||||
| +        } | ||||
| +    } | ||||
| +    fclose(fp); | ||||
| + | ||||
| +    if (exclude_crash_reserve_region(&memory_ranges) < 0) | ||||
| +        return -1; | ||||
| + | ||||
| +    for (i = 0; i < memory_ranges; i++) | ||||
| +        if (saved_max_mem < crash_memory_range[i].end) | ||||
| +            saved_max_mem = crash_memory_range[i].end + 1; | ||||
| + | ||||
| +    *range = crash_memory_range; | ||||
| +    *ranges = memory_ranges; | ||||
| +    return 0; | ||||
| +} | ||||
| + | ||||
| +/* Converts unsigned long to ascii string. */ | ||||
| +static void ultoa(unsigned long i, char *str) | ||||
| +{ | ||||
| +    int j = 0, k; | ||||
| +    char tmp; | ||||
| + | ||||
| +    do { | ||||
| +        str[j++] = i % 10 + '0'; | ||||
| +    } while ((i /=10) > 0); | ||||
| +    str[j] = '\0'; | ||||
| + | ||||
| +    /* Reverse the string. */ | ||||
| +    for (j = 0, k = strlen(str) - 1; j < k; j++, k--) { | ||||
| +        tmp = str[k]; | ||||
| +        str[k] = str[j]; | ||||
| +        str[j] = tmp; | ||||
| +    } | ||||
| +} | ||||
| + | ||||
| +/* Adds the appropriate mem= options to command line, indicating the | ||||
| + * memory region the new kernel can use to boot into. */ | ||||
| +static int cmdline_add_mem(char *cmdline, unsigned long addr, unsigned long size) | ||||
| +{ | ||||
| +    int cmdlen, len; | ||||
| +    char str[50], *ptr; | ||||
| + | ||||
| +    addr = addr/1024; | ||||
| +    size = size/1024; | ||||
| +    ptr = str; | ||||
| +    strcpy (str, " mem="); | ||||
| +    ptr += strlen(str); | ||||
| +    ultoa(size, ptr); | ||||
| +    strcat (str, "K@"); | ||||
| +    ptr = str + strlen(str); | ||||
| +    ultoa(addr, ptr); | ||||
| +    strcat (str, "K"); | ||||
| +    len = strlen(str); | ||||
| +    cmdlen = strlen(cmdline) + len; | ||||
| +    if (cmdlen > (COMMAND_LINE_SIZE - 1)) | ||||
| +        die("Command line overflow\n"); | ||||
| +    strcat(cmdline, str); | ||||
| + | ||||
| +    return 0; | ||||
| +} | ||||
| + | ||||
| +/* Adds the elfcorehdr= command line parameter to command line. */ | ||||
| +static int cmdline_add_elfcorehdr(char *cmdline, unsigned long addr) | ||||
| +{ | ||||
| +    int cmdlen, len, align = 1024; | ||||
| +    char str[30], *ptr; | ||||
| + | ||||
| +    /* Passing in elfcorehdr=xxxK format. Saves space required in cmdline. | ||||
| +     * Ensure 1K alignment*/ | ||||
| +    if (addr%align) | ||||
| +        return -1; | ||||
| +    addr = addr/align; | ||||
| +    ptr = str; | ||||
| +    strcpy(str, " elfcorehdr="); | ||||
| +    ptr += strlen(str); | ||||
| +    ultoa(addr, ptr); | ||||
| +    strcat(str, "K"); | ||||
| +    len = strlen(str); | ||||
| +    cmdlen = strlen(cmdline) + len; | ||||
| +    if (cmdlen > (COMMAND_LINE_SIZE - 1)) | ||||
| +        die("Command line overflow\n"); | ||||
| +    strcat(cmdline, str); | ||||
| +    return 0; | ||||
| +} | ||||
| + | ||||
| +/* Adds the elfcorehdr= command line parameter to command line. */ | ||||
| +static int cmdline_add_savemaxmem(char *cmdline, unsigned long addr) | ||||
| +{ | ||||
| +    int cmdlen, len, align = 1024; | ||||
| +    char str[30], *ptr; | ||||
| + | ||||
| +    /* Passing in savemaxmem=xxxM format. Saves space required in cmdline.*/ | ||||
| +    addr = addr/(align*align); | ||||
| +    ptr = str; | ||||
| +    strcpy(str, " savemaxmem="); | ||||
| +    ptr += strlen(str); | ||||
| +    ultoa(addr, ptr); | ||||
| +    strcat(str, "M"); | ||||
| +    len = strlen(str); | ||||
| +    cmdlen = strlen(cmdline) + len; | ||||
| +    if (cmdlen > (COMMAND_LINE_SIZE - 1)) | ||||
| +        die("Command line overflow\n"); | ||||
| +    strcat(cmdline, str); | ||||
| +    return 0; | ||||
| +} | ||||
| + | ||||
| +#ifdef __mips64 | ||||
| +static struct crash_elf_info elf_info64 = | ||||
| +{ | ||||
| +    class: ELFCLASS64, | ||||
| +    data: ELFDATA2MSB, | ||||
| +    machine: EM_MIPS, | ||||
| +    backup_src_start: BACKUP_SRC_START, | ||||
| +    backup_src_end: BACKUP_SRC_END, | ||||
| +    page_offset: PAGE_OFFSET, | ||||
| +    lowmem_limit: MAXMEM, | ||||
| +}; | ||||
| +#endif | ||||
| +static struct crash_elf_info elf_info32 = | ||||
| +{ | ||||
| +    class: ELFCLASS32, | ||||
| +    data: ELFDATA2MSB, | ||||
| +    machine: EM_MIPS, | ||||
| +    backup_src_start: BACKUP_SRC_START, | ||||
| +    backup_src_end: BACKUP_SRC_END, | ||||
| +    page_offset: PAGE_OFFSET, | ||||
| +    lowmem_limit: MAXMEM, | ||||
| +}; | ||||
| + | ||||
| +/* Loads additional segments in case of a panic kernel is being loaded. | ||||
| + * One segment for backup region, another segment for storing elf headers | ||||
| + * for crash memory image. | ||||
| + */ | ||||
| +int load_crashdump_segments(struct kexec_info *info, char* mod_cmdline, | ||||
| +                unsigned long max_addr, unsigned long min_base) | ||||
| +{ | ||||
| +    void *tmp; | ||||
| +    unsigned long sz, elfcorehdr; | ||||
| +    int nr_ranges, align = 1024; | ||||
| +    struct memory_range *mem_range; | ||||
| + | ||||
| +    if (get_crash_memory_ranges(&mem_range, &nr_ranges) < 0) | ||||
| +        return -1; | ||||
| + | ||||
| +    /* Create a backup region segment to store backup data*/ | ||||
| +    sz = (BACKUP_SRC_SIZE + align - 1) & ~(align - 1); | ||||
| +    tmp = xmalloc(sz); | ||||
| +    memset(tmp, 0, sz); | ||||
| +    info->backup_start = add_buffer(info, tmp, sz, sz, align, | ||||
| +                crash_reserved_mem.start, | ||||
| +                crash_reserved_mem.end,-1); | ||||
| + | ||||
| +#ifdef __mips64 | ||||
| +    /* Create elf header segment and store crash image data. */ | ||||
| +    if (arch_options.core_header_type == CORE_TYPE_ELF64) { | ||||
| +        if (crash_create_elf64_headers(info, &elf_info64, | ||||
| +                           crash_memory_range, nr_ranges, | ||||
| +                           &tmp, &sz, | ||||
| +                           ELF_CORE_HEADER_ALIGN) < 0) | ||||
| +            return -1; | ||||
| +    } | ||||
| +    else { | ||||
| +        if (crash_create_elf32_headers(info, &elf_info32, | ||||
| +                           crash_memory_range, nr_ranges, | ||||
| +                           &tmp, &sz, | ||||
| +                           ELF_CORE_HEADER_ALIGN) < 0) | ||||
| +            return -1; | ||||
| +    } | ||||
| +#else | ||||
| +    if (crash_create_elf32_headers(info, &elf_info32, | ||||
| +                   crash_memory_range, nr_ranges, | ||||
| +                   &tmp, &sz, | ||||
| +                   ELF_CORE_HEADER_ALIGN) < 0) | ||||
| +        return -1; | ||||
| +#endif | ||||
| +    elfcorehdr = add_buffer(info, tmp, sz, sz, align, | ||||
| +                crash_reserved_mem.start, | ||||
| +                crash_reserved_mem.end, -1); | ||||
| + | ||||
| +    /* | ||||
| +     * backup segment is after elfcorehdr, so use elfcorehdr as top of | ||||
| +     * kernel's available memory | ||||
| +     */ | ||||
| +    cmdline_add_mem(mod_cmdline, crash_reserved_mem.start, | ||||
| +        elfcorehdr - crash_reserved_mem.start); | ||||
| +    cmdline_add_elfcorehdr(mod_cmdline, elfcorehdr); | ||||
| +    cmdline_add_savemaxmem(mod_cmdline, saved_max_mem); | ||||
| +    return 0; | ||||
| +} | ||||
| + | ||||
| +int is_crashkernel_mem_reserved(void) | ||||
| +{ | ||||
| +    uint64_t start, end; | ||||
| + | ||||
| +    return parse_iomem_single("Crash kernel\n", &start, &end) == 0 ? | ||||
| +      (start != end) : 0; | ||||
| +} | ||||
| + | ||||
| Index: kexec-tools-2.0.1/kexec/arch/mips/crashdump-mips.h | ||||
| =================================================================== | ||||
| --- /dev/null	1970-01-01 00:00:00.000000000 +0000 | ||||
| +++ kexec-tools-2.0.1/kexec/arch/mips/crashdump-mips.h	2009-09-27 19:07:26.000000000 +0200 | ||||
| @@ -0,0 +1,26 @@ | ||||
| +#ifndef CRASHDUMP_MIPS_H | ||||
| +#define CRASHDUMP_MIPS_H | ||||
| + | ||||
| +struct kexec_info; | ||||
| +int load_crashdump_segments(struct kexec_info *info, char *mod_cmdline, | ||||
| +                unsigned long max_addr, unsigned long min_base); | ||||
| +#ifdef __mips64 | ||||
| +#define PAGE_OFFSET    0xa800000000000000ULL | ||||
| +#else | ||||
| +#define PAGE_OFFSET    0x80000000 | ||||
| +#endif | ||||
| +#define __pa(x)        ((unsigned long)(X)& 0x7fffffff) | ||||
| + | ||||
| +#define MAXMEM        0x80000000 | ||||
| + | ||||
| +#define CRASH_MAX_MEMMAP_NR    (KEXEC_MAX_SEGMENTS + 1) | ||||
| +#define CRASH_MAX_MEMORY_RANGES    (MAX_MEMORY_RANGES + 2) | ||||
| + | ||||
| +#define COMMAND_LINE_SIZE    512 | ||||
| + | ||||
| +/* Backup Region, First 1M of System RAM. */ | ||||
| +#define BACKUP_SRC_START    0x00000000 | ||||
| +#define BACKUP_SRC_END        0x000fffff | ||||
| +#define BACKUP_SRC_SIZE    (BACKUP_SRC_END - BACKUP_SRC_START + 1) | ||||
| + | ||||
| +#endif /* CRASHDUMP_MIPS_H */ | ||||
| Index: kexec-tools-2.0.1/kexec/arch/mips/include/arch/options.h | ||||
| =================================================================== | ||||
| --- kexec-tools-2.0.1.orig/kexec/arch/mips/include/arch/options.h	2008-07-15 02:46:43.000000000 +0200 | ||||
| +++ kexec-tools-2.0.1/kexec/arch/mips/include/arch/options.h	2009-09-27 19:18:21.000000000 +0200 | ||||
| @@ -2,10 +2,21 @@ | ||||
|  #define KEXEC_ARCH_MIPS_OPTIONS_H | ||||
|   | ||||
|  #define OPT_ARCH_MAX   (OPT_MAX+0) | ||||
| +#define KEXEC_ARCH_OPT_STR KEXEC_OPT_STR "" | ||||
|   | ||||
| +#ifdef __mips64 | ||||
| +#define OPT_ELF64_CORE	(OPT_MAX+1) | ||||
|  #define KEXEC_ARCH_OPTIONS \ | ||||
|  	KEXEC_OPTIONS \ | ||||
| +	{ "elf64-core-headers", 0, 0, OPT_ELF64_CORE }, \ | ||||
|   | ||||
|  #define KEXEC_ARCH_OPT_STR KEXEC_OPT_STR "" | ||||
| +#define OPT_ARCH_MAX       (OPT_MAX+2) | ||||
| +#else | ||||
| +#define KEXEC_ARCH_OPTIONS \ | ||||
| +	KEXEC_OPTIONS \ | ||||
| + | ||||
| +#define OPT_ARCH_MAX   (OPT_MAX+0) | ||||
| +#endif | ||||
|   | ||||
|  #endif /* KEXEC_ARCH_MIPS_OPTIONS_H */ | ||||
| Index: kexec-tools-2.0.1/kexec/arch/mips/kexec-elf-mips.c | ||||
| =================================================================== | ||||
| --- kexec-tools-2.0.1.orig/kexec/arch/mips/kexec-elf-mips.c	2008-07-15 02:46:43.000000000 +0200 | ||||
| +++ kexec-tools-2.0.1/kexec/arch/mips/kexec-elf-mips.c	2009-09-27 19:16:39.000000000 +0200 | ||||
| @@ -25,51 +25,18 @@ | ||||
|  #include <ip_checksum.h> | ||||
|  #include "../../kexec.h" | ||||
|  #include "../../kexec-elf.h" | ||||
| +#include "../../kexec-syscall.h" | ||||
|  #include "kexec-mips.h" | ||||
|  #include <arch/options.h> | ||||
| +#include "crashdump-mips.h" | ||||
|   | ||||
|  static const int probe_debug = 0; | ||||
|   | ||||
|  #define BOOTLOADER         "kexec" | ||||
|  #define MAX_COMMAND_LINE   256 | ||||
|   | ||||
| -#define UPSZ(X) ((sizeof(X) + 3) & ~3) | ||||
| -static struct boot_notes { | ||||
| -	Elf_Bhdr hdr; | ||||
| -	Elf_Nhdr bl_hdr; | ||||
| -	unsigned char bl_desc[UPSZ(BOOTLOADER)]; | ||||
| -	Elf_Nhdr blv_hdr; | ||||
| -	unsigned char blv_desc[UPSZ(BOOTLOADER_VERSION)]; | ||||
| -	Elf_Nhdr cmd_hdr; | ||||
| -	unsigned char command_line[0]; | ||||
| -} elf_boot_notes = { | ||||
| -	.hdr = { | ||||
| -		.b_signature = 0x0E1FB007, | ||||
| -		.b_size = sizeof(elf_boot_notes), | ||||
| -		.b_checksum = 0, | ||||
| -		.b_records = 3, | ||||
| -	}, | ||||
| -	.bl_hdr = { | ||||
| -		.n_namesz = 0, | ||||
| -		.n_descsz = sizeof(BOOTLOADER), | ||||
| -		.n_type = EBN_BOOTLOADER_NAME, | ||||
| -	}, | ||||
| -	.bl_desc = BOOTLOADER, | ||||
| -	.blv_hdr = { | ||||
| -		.n_namesz = 0, | ||||
| -		.n_descsz = sizeof(BOOTLOADER_VERSION), | ||||
| -		.n_type = EBN_BOOTLOADER_VERSION, | ||||
| -	}, | ||||
| -	.blv_desc = BOOTLOADER_VERSION, | ||||
| -	.cmd_hdr = { | ||||
| -		.n_namesz = 0, | ||||
| -		.n_descsz = 0, | ||||
| -		.n_type = EBN_COMMAND_LINE, | ||||
| -	}, | ||||
| -}; | ||||
| - | ||||
| - | ||||
| -#define OPT_APPEND	(OPT_ARCH_MAX+0) | ||||
| +/* 'kexec' in cmdline is used to find cmdline buffer by kernel */ | ||||
| +static char cmdline_buf[256] = "kexec "; | ||||
|   | ||||
|  int elf_mips_probe(const char *buf, off_t len) | ||||
|  { | ||||
| @@ -108,16 +75,14 @@ | ||||
|  	struct kexec_info *info) | ||||
|  { | ||||
|  	struct mem_ehdr ehdr; | ||||
| -	char *arg_buf; | ||||
| -	size_t arg_bytes; | ||||
| -	unsigned long arg_base; | ||||
| -	struct boot_notes *notes; | ||||
| -	size_t note_bytes; | ||||
| -	const char *command_line; | ||||
| -	int command_line_len; | ||||
| -	unsigned char *setup_start; | ||||
| -	uint32_t setup_size; | ||||
| +	unsigned long bss_start, bss_size = 0; | ||||
| +	const char *command_line = NULL; | ||||
| +	char *modified_cmdline; | ||||
| +	int modified_cmdline_len; | ||||
| +	unsigned long cmdline_addr; | ||||
| +	int result,i; | ||||
|  	int opt; | ||||
| +#define OPT_APPEND     (OPT_ARCH_MAX+0) | ||||
|  	static const struct option options[] = { | ||||
|  		KEXEC_ARCH_OPTIONS | ||||
|  		{"command-line", 1, 0, OPT_APPEND}, | ||||
| @@ -144,38 +109,81 @@ | ||||
|  			break; | ||||
|  		} | ||||
|  	} | ||||
| -	command_line_len = 0; | ||||
| -	setup_simple_regs.spr9 = 0; | ||||
| -	if (command_line) { | ||||
| -		command_line_len = strlen(command_line) + 1; | ||||
| -		setup_simple_regs.spr9 = 2; | ||||
| +	/* Need to append some command line parameters internally in case of | ||||
| +	 * taking crash dumps. | ||||
| +	 */ | ||||
| +	if (info->kexec_flags & KEXEC_ON_CRASH) { | ||||
| +		modified_cmdline = xmalloc(COMMAND_LINE_SIZE); | ||||
| +		memset((void *)modified_cmdline, 0, COMMAND_LINE_SIZE); | ||||
| +		if (command_line) { | ||||
| +			strncpy(modified_cmdline, command_line, COMMAND_LINE_SIZE); | ||||
| +			modified_cmdline[COMMAND_LINE_SIZE - 1] = '\0'; | ||||
| +		} | ||||
| +		modified_cmdline_len = strlen(modified_cmdline); | ||||
|  	} | ||||
|   | ||||
| -	/* Load the ELF executable */ | ||||
| -	elf_exec_build_load(info, &ehdr, buf, len, 0); | ||||
| - | ||||
| -	setup_start = setup_simple_start; | ||||
| -	setup_size = setup_simple_size; | ||||
| -	setup_simple_regs.spr8 = ehdr.e_entry; | ||||
| - | ||||
| -	note_bytes = sizeof(elf_boot_notes) + ((command_line_len + 3) & ~3); | ||||
| -	arg_bytes = note_bytes + ((setup_size + 3) & ~3); | ||||
| - | ||||
| -	arg_buf = xmalloc(arg_bytes); | ||||
| -	arg_base = add_buffer_virt(info, | ||||
| -		 arg_buf, arg_bytes, arg_bytes, 4, 0, elf_max_addr(&ehdr), 1); | ||||
| +	/* Parse the Elf file */ | ||||
| +	result = build_elf_exec_info(buf, len, &ehdr, 0); | ||||
| +	if (result < 0) { | ||||
| +		die("ELF exec parse failed\n"); | ||||
| +	} | ||||
|   | ||||
| -	notes = (struct boot_notes *)(arg_buf + ((setup_size + 3) & ~3)); | ||||
| +	/* Read in the PT_LOAD segments and remove CKSEG0 mask from address*/ | ||||
| +	for(i = 0; i < ehdr.e_phnum; i++) { | ||||
| +		struct mem_phdr *phdr; | ||||
| +		phdr = &ehdr.e_phdr[i]; | ||||
| +		if (phdr->p_type == PT_LOAD) { | ||||
| +			phdr->p_paddr = virt_to_phys(phdr->p_paddr); | ||||
| +		} | ||||
| +	} | ||||
|   | ||||
| -	memcpy(arg_buf, setup_start, setup_size); | ||||
| -	memcpy(notes, &elf_boot_notes, sizeof(elf_boot_notes)); | ||||
| -	memcpy(notes->command_line, command_line, command_line_len); | ||||
| +	for(i = 0; i < ehdr.e_shnum; i++) { | ||||
| +		struct mem_shdr *shdr; | ||||
| +		unsigned char *strtab; | ||||
| +		strtab = (unsigned char *)ehdr.e_shdr[ehdr.e_shstrndx].sh_data; | ||||
| + | ||||
| +		shdr = &ehdr.e_shdr[i]; | ||||
| +		if ( shdr->sh_size && | ||||
| +				strcmp((char *)&strtab[shdr->sh_name], | ||||
| +					".bss") == 0) { | ||||
| +			bss_start = virt_to_phys(shdr->sh_addr); | ||||
| +			bss_size = shdr->sh_size; | ||||
| +			break; | ||||
| +		} | ||||
| +	} | ||||
|   | ||||
| -	notes->hdr.b_size = note_bytes; | ||||
| -	notes->cmd_hdr.n_descsz = command_line_len; | ||||
| -	notes->hdr.b_checksum = compute_ip_checksum(notes, note_bytes); | ||||
| +	/* Load the Elf data */ | ||||
| +	result = elf_exec_load(&ehdr, info); | ||||
| +	if (result < 0) { | ||||
| +		die("ELF exec load failed\n"); | ||||
| +	} | ||||
| +	info->entry = (void *)virt_to_phys(ehdr.e_entry); | ||||
| +	if(!bss_size) | ||||
| +		die("No .bss segment present\n"); | ||||
| + | ||||
| +	/* Put cmdline right after bss */ | ||||
| +	cmdline_addr = bss_start + bss_size; | ||||
| + | ||||
| +	/* If panic kernel is being loaded, additional segments need | ||||
| +	 * to be created. | ||||
| +	 */ | ||||
| +	if (info->kexec_flags & KEXEC_ON_CRASH) { | ||||
| +		result = load_crashdump_segments(info, modified_cmdline, | ||||
| +								0, 0); | ||||
| +		if (result < 0) | ||||
| +			return -1; | ||||
| +		/* Use new command line. */ | ||||
| +		command_line = modified_cmdline; | ||||
| +	} | ||||
|   | ||||
| -	info->entry = (void *)arg_base; | ||||
| +	if (command_line) | ||||
| +	{ | ||||
| +		strncat(cmdline_buf,command_line, | ||||
| +			sizeof(cmdline_buf) - strlen(cmdline_buf) - 1); | ||||
| +		add_buffer(info, cmdline_buf, sizeof(cmdline_buf), | ||||
| +			sizeof(cmdline_buf), sizeof(void*), | ||||
| +			cmdline_addr, 0x0fffffff, 1); | ||||
| +	} | ||||
|   | ||||
|  	return 0; | ||||
|  } | ||||
| Index: kexec-tools-2.0.1/kexec/arch/mips/kexec-mips.c | ||||
| =================================================================== | ||||
| --- kexec-tools-2.0.1.orig/kexec/arch/mips/kexec-mips.c	2008-07-15 02:46:43.000000000 +0200 | ||||
| +++ kexec-tools-2.0.1/kexec/arch/mips/kexec-mips.c	2009-09-27 19:20:25.000000000 +0200 | ||||
| @@ -97,8 +97,18 @@ | ||||
|   | ||||
|  void arch_usage(void) | ||||
|  { | ||||
| +#ifdef __mips64 | ||||
| +       fprintf(stderr, "        --elf32-core-headers Prepare core headers in " | ||||
| +                       "ELF32 format\n"); | ||||
| +#endif | ||||
|  } | ||||
|   | ||||
| +#ifdef __mips64 | ||||
| +struct arch_options_t arch_options = { | ||||
| +       .core_header_type = CORE_TYPE_ELF64 | ||||
| +}; | ||||
| +#endif | ||||
| + | ||||
|  int arch_process_options(int argc, char **argv) | ||||
|  { | ||||
|  	static const struct option options[] = { | ||||
| @@ -113,6 +123,11 @@ | ||||
|  		switch(opt) { | ||||
|  		default: | ||||
|  			break; | ||||
| +#ifdef __mips64 | ||||
| +		case OPT_ELF64_CORE: | ||||
| +			arch_options.core_header_type = CORE_TYPE_ELF64; | ||||
| +			break; | ||||
| +#endif | ||||
|  		} | ||||
|  	} | ||||
|  	/* Reset getopt for the next pass; called in other source modules */ | ||||
| @@ -126,6 +141,10 @@ | ||||
|  	 * use KEXEC_ARCH_DEFAULT instead of KEXEC_ARCH_MIPS here. | ||||
|  	 */ | ||||
|  	{ "mips", KEXEC_ARCH_DEFAULT }, | ||||
| +	/* Not using KEXEC_ARCH_DEFAULT because that will fail | ||||
| +	 * in the kernel's compat_sys_kexec_load() routine. | ||||
| +	 */ | ||||
| +	{ "mips64", KEXEC_ARCH_MIPS }, | ||||
|  	{ 0 }, | ||||
|  }; | ||||
|   | ||||
| @@ -138,18 +157,9 @@ | ||||
|  { | ||||
|  } | ||||
|   | ||||
| -/* | ||||
| - * Adding a dummy function, so that build on mips will not break. | ||||
| - * Need to implement the actual checking code | ||||
| - */ | ||||
| -int is_crashkernel_mem_reserved(void) | ||||
| -{ | ||||
| -	return 1; | ||||
| -} | ||||
| - | ||||
|  unsigned long virt_to_phys(unsigned long addr) | ||||
|  { | ||||
| -	return addr - 0x80000000; | ||||
| +	return ((addr)& 0x7fffffff); | ||||
|  } | ||||
|   | ||||
|  /* | ||||
| Index: kexec-tools-2.0.1/kexec/arch/mips/kexec-mips.h | ||||
| =================================================================== | ||||
| --- kexec-tools-2.0.1.orig/kexec/arch/mips/kexec-mips.h	2008-07-15 02:46:43.000000000 +0200 | ||||
| +++ kexec-tools-2.0.1/kexec/arch/mips/kexec-mips.h	2009-09-27 19:21:32.000000000 +0200 | ||||
| @@ -1,17 +1,16 @@ | ||||
|  #ifndef KEXEC_MIPS_H | ||||
|  #define KEXEC_MIPS_H | ||||
|   | ||||
| -extern unsigned char setup_simple_start[]; | ||||
| -extern uint32_t setup_simple_size; | ||||
| - | ||||
| -extern struct { | ||||
| -	uint32_t spr8; | ||||
| -	uint32_t spr9; | ||||
| -} setup_simple_regs; | ||||
| +#define MAX_MEMORY_RANGES 64 | ||||
| +#define CORE_TYPE_ELF32 1 | ||||
| +#define CORE_TYPE_ELF64 2 | ||||
|   | ||||
|  int elf_mips_probe(const char *buf, off_t len); | ||||
|  int elf_mips_load(int argc, char **argv, const char *buf, off_t len, | ||||
|  	struct kexec_info *info); | ||||
|  void elf_mips_usage(void); | ||||
|   | ||||
| +struct arch_options_t { | ||||
| +	int core_header_type; | ||||
| +}; | ||||
|  #endif /* KEXEC_MIPS_H */ | ||||
		Reference in New Issue
	
	Block a user
	 Florian Fainelli
					Florian Fainelli