From: Mahesh Salgaonkar <mahesh@xxxxxxxxxxxxxxxxxx> This patch implements support for inclusion of erase information in ELF formatted dumpfile. This patch introduces an additional ELF Note section that contains erase information. Signed-off-by: Mahesh Salgaonkar <mahesh at linux.vnet.ibm.com> --- makedumpfile.c | 121 ++++++++++++++++++++++++++++++++++++++++++++++++++++++++ makedumpfile.h | 17 ++++++++ 2 files changed, 137 insertions(+), 1 deletions(-) diff --git a/makedumpfile.c b/makedumpfile.c index b5ad5f7..f7842d2 100644 --- a/makedumpfile.c +++ b/makedumpfile.c @@ -3352,6 +3352,9 @@ get_pt_note_info(off_t off_note, unsigned long sz_note) return FALSE; } info->p2m_mfn = p2m_mfn; + } else if (n_type == NT_ERASE_INFO) { + info->offset_eraseinfo = offset_desc; + info->size_eraseinfo = size_desc; } offset += offset_next_note(note); } @@ -4647,6 +4650,15 @@ write_cache_bufsz(struct cache_data *cd) } int +write_cache_zero(struct cache_data *cd, size_t size) +{ + memset(cd->buf + cd->buf_size, 0, size); + cd->buf_size += size; + + return write_cache_bufsz(cd); +} + +int read_buf_from_stdin(void *buf, int buf_size) { int read_size = 0, tmp_read_size = 0; @@ -5799,10 +5811,12 @@ write_elf_header(struct cache_data *cd_header) { int i, num_loads_dumpfile, phnum; off_t offset_note_memory, offset_note_dumpfile; - size_t size_note; + size_t size_note, size_eraseinfo = 0; Elf64_Ehdr ehdr64; Elf32_Ehdr ehdr32; Elf64_Phdr note; + char size_str[MAX_SIZE_STR_LEN]; + struct filter_info *fl_info = filter_info; char *buf = NULL; const off_t failed = (off_t)-1; @@ -5855,6 +5869,36 @@ write_elf_header(struct cache_data *cd_header) } /* + * Pre-calculate the required size to store eraseinfo in ELF note + * section so that we can add enough space in ELF notes section and + * adjust the PT_LOAD offset accordingly. + */ + while (fl_info) { + struct erase_info *ei; + + if (!fl_info->erase_info_idx) + continue; + ei = &erase_info[fl_info->erase_info_idx]; + if (fl_info->nullify) + sprintf(size_str, "nullify\n"); + else + sprintf(size_str, "%ld\n", fl_info->size); + + size_eraseinfo += strlen("erase ") + + strlen(ei->symbol_expr) + 1 + + strlen(size_str); + fl_info = fl_info->next; + } + + /* + * Store the size_eraseinfo for later use in write_elf_eraseinfo() + * function. This will overwrite the size fetched during + * get elf_info() function but we are ok with that. + */ + info->size_eraseinfo = size_eraseinfo; + DEBUG_MSG("erase info size: %lu\n", info->size_eraseinfo); + + /* * Write a PT_NOTE header. */ if (!(phnum = get_phnum_memory())) @@ -5884,6 +5928,17 @@ write_elf_header(struct cache_data *cd_header) note.p_offset = offset_note_dumpfile; size_note = note.p_filesz; + /* + * Modify the note size in PT_NOTE header to accomodate eraseinfo data. + * Eraseinfo will be written later. + */ + if (info->flag_elf64_memory) + note.p_filesz += sizeof(Elf64_Nhdr); + else + note.p_filesz += sizeof(Elf32_Nhdr); + note.p_filesz += roundup(ERASEINFO_NOTE_NAME_BYTES, 4) + + roundup(size_eraseinfo, 4); + if (!write_elf_phdr(cd_header, ¬e)) goto out; @@ -5910,10 +5965,14 @@ write_elf_header(struct cache_data *cd_header) size_note, info->name_dumpfile)) goto out; + /* Set the size_note with new size. */ + size_note = note.p_filesz; + /* * Set an offset of PT_LOAD segment. */ info->offset_load_dumpfile = offset_note_dumpfile + size_note; + info->offset_note_dumpfile = offset_note_dumpfile; ret = TRUE; out: @@ -6728,6 +6787,64 @@ out: } int +write_elf_eraseinfo(struct cache_data *cd_header) +{ + char note[MAX_SIZE_NHDR]; + char buf[ERASEINFO_NOTE_NAME_BYTES + 4]; + unsigned long note_header_size, size_eraseinfo; + + DEBUG_MSG("Writing erase info...\n"); + + /* calculate the eraseinfo ELF note offset */ + cd_header->offset = info->offset_note_dumpfile + + roundup(info->size_note, 4); + + /* Write eraseinfo ELF note header. */ + memset(note, 0, sizeof(note)); + if (info->flag_elf64_memory) { + Elf64_Nhdr *nh = (Elf64_Nhdr *)note; + + note_header_size = sizeof(Elf64_Nhdr); + nh->n_namesz = ERASEINFO_NOTE_NAME_BYTES; + nh->n_descsz = info->size_eraseinfo; + nh->n_type = NT_ERASE_INFO; + } else { + Elf32_Nhdr *nh = (Elf32_Nhdr *)note; + + note_header_size = sizeof(Elf32_Nhdr); + nh->n_namesz = ERASEINFO_NOTE_NAME_BYTES; + nh->n_descsz = info->size_eraseinfo; + nh->n_type = NT_ERASE_INFO; + } + if (!write_cache(cd_header, note, note_header_size)) + return FALSE; + + /* Write eraseinfo Note name */ + memset(buf, 0, sizeof(buf)); + memcpy(buf, ERASEINFO_NOTE_NAME, ERASEINFO_NOTE_NAME_BYTES); + if (!write_cache(cd_header, buf, + roundup(ERASEINFO_NOTE_NAME_BYTES, 4))) + return FALSE; + + info->offset_eraseinfo = cd_header->offset; + if (!write_eraseinfo(cd_header, &size_eraseinfo)) + return FALSE; + + /* + * The actual eraseinfo written may be less than pre-calculated size. + * Hence fill up the rest of size with zero's. + */ + if (size_eraseinfo < info->size_eraseinfo) + write_cache_zero(cd_header, + info->size_eraseinfo - size_eraseinfo); + + DEBUG_MSG("offset_eraseinfo: %lx, size_eraseinfo: %ld\n", + info->offset_eraseinfo, info->size_eraseinfo); + + return TRUE; +} + +int write_kdump_eraseinfo(struct cache_data *cd_page) { off_t offset_eraseinfo; @@ -7581,6 +7698,8 @@ writeout_dumpfile(void) goto out; if (!write_elf_pages(&cd_header, &cd_page)) goto out; + if (!write_elf_eraseinfo(&cd_header)) + goto out; } else { if (!write_kdump_header()) goto out; diff --git a/makedumpfile.h b/makedumpfile.h index 2e4b902..662bc97 100644 --- a/makedumpfile.h +++ b/makedumpfile.h @@ -504,6 +504,22 @@ do { \ #define MAX_SIZE_STR_LEN (21) /* + * ELF note section for erase information + * + * According to elf.h the unused values are 0x15(21) through 0xff. The value + * range 0x1XX, 0x2XX and 0x3XX is been used for PPC, i386 and s390 + * respectively. + * + * Using 0xff to be on safer side so that any new Elf Note addition in elf.h + * after 0x15 value would not clash. + */ +#ifndef NT_ERASE_INFO +#define NT_ERASE_INFO (0xff) /* Contains erased information. */ +#endif +#define ERASEINFO_NOTE_NAME "ERASEINFO" +#define ERASEINFO_NOTE_NAME_BYTES (sizeof(ERASEINFO_NOTE_NAME)) + +/* * The value of dependence on machine */ #define PAGE_OFFSET (info->page_offset) @@ -977,6 +993,7 @@ struct DumpInfo { * ELF NOTE section in dump memory image info: */ off_t offset_note; + off_t offset_note_dumpfile; unsigned long size_note; /*