18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 28c2ecf20Sopenharmony_ci#include <linux/elf.h> 38c2ecf20Sopenharmony_ci#include <linux/coredump.h> 48c2ecf20Sopenharmony_ci#include <linux/fs.h> 58c2ecf20Sopenharmony_ci#include <linux/mm.h> 68c2ecf20Sopenharmony_ci 78c2ecf20Sopenharmony_ci#include <asm/elf.h> 88c2ecf20Sopenharmony_ci 98c2ecf20Sopenharmony_ci 108c2ecf20Sopenharmony_ciElf32_Half elf_core_extra_phdrs(void) 118c2ecf20Sopenharmony_ci{ 128c2ecf20Sopenharmony_ci return vsyscall_ehdr ? (((struct elfhdr *)vsyscall_ehdr)->e_phnum) : 0; 138c2ecf20Sopenharmony_ci} 148c2ecf20Sopenharmony_ci 158c2ecf20Sopenharmony_ciint elf_core_write_extra_phdrs(struct coredump_params *cprm, loff_t offset) 168c2ecf20Sopenharmony_ci{ 178c2ecf20Sopenharmony_ci if ( vsyscall_ehdr ) { 188c2ecf20Sopenharmony_ci const struct elfhdr *const ehdrp = 198c2ecf20Sopenharmony_ci (struct elfhdr *) vsyscall_ehdr; 208c2ecf20Sopenharmony_ci const struct elf_phdr *const phdrp = 218c2ecf20Sopenharmony_ci (const struct elf_phdr *) (vsyscall_ehdr + ehdrp->e_phoff); 228c2ecf20Sopenharmony_ci int i; 238c2ecf20Sopenharmony_ci Elf32_Off ofs = 0; 248c2ecf20Sopenharmony_ci 258c2ecf20Sopenharmony_ci for (i = 0; i < ehdrp->e_phnum; ++i) { 268c2ecf20Sopenharmony_ci struct elf_phdr phdr = phdrp[i]; 278c2ecf20Sopenharmony_ci 288c2ecf20Sopenharmony_ci if (phdr.p_type == PT_LOAD) { 298c2ecf20Sopenharmony_ci ofs = phdr.p_offset = offset; 308c2ecf20Sopenharmony_ci offset += phdr.p_filesz; 318c2ecf20Sopenharmony_ci } else { 328c2ecf20Sopenharmony_ci phdr.p_offset += ofs; 338c2ecf20Sopenharmony_ci } 348c2ecf20Sopenharmony_ci phdr.p_paddr = 0; /* match other core phdrs */ 358c2ecf20Sopenharmony_ci if (!dump_emit(cprm, &phdr, sizeof(phdr))) 368c2ecf20Sopenharmony_ci return 0; 378c2ecf20Sopenharmony_ci } 388c2ecf20Sopenharmony_ci } 398c2ecf20Sopenharmony_ci return 1; 408c2ecf20Sopenharmony_ci} 418c2ecf20Sopenharmony_ci 428c2ecf20Sopenharmony_ciint elf_core_write_extra_data(struct coredump_params *cprm) 438c2ecf20Sopenharmony_ci{ 448c2ecf20Sopenharmony_ci if ( vsyscall_ehdr ) { 458c2ecf20Sopenharmony_ci const struct elfhdr *const ehdrp = 468c2ecf20Sopenharmony_ci (struct elfhdr *) vsyscall_ehdr; 478c2ecf20Sopenharmony_ci const struct elf_phdr *const phdrp = 488c2ecf20Sopenharmony_ci (const struct elf_phdr *) (vsyscall_ehdr + ehdrp->e_phoff); 498c2ecf20Sopenharmony_ci int i; 508c2ecf20Sopenharmony_ci 518c2ecf20Sopenharmony_ci for (i = 0; i < ehdrp->e_phnum; ++i) { 528c2ecf20Sopenharmony_ci if (phdrp[i].p_type == PT_LOAD) { 538c2ecf20Sopenharmony_ci void *addr = (void *) phdrp[i].p_vaddr; 548c2ecf20Sopenharmony_ci size_t filesz = phdrp[i].p_filesz; 558c2ecf20Sopenharmony_ci if (!dump_emit(cprm, addr, filesz)) 568c2ecf20Sopenharmony_ci return 0; 578c2ecf20Sopenharmony_ci } 588c2ecf20Sopenharmony_ci } 598c2ecf20Sopenharmony_ci } 608c2ecf20Sopenharmony_ci return 1; 618c2ecf20Sopenharmony_ci} 628c2ecf20Sopenharmony_ci 638c2ecf20Sopenharmony_cisize_t elf_core_extra_data_size(void) 648c2ecf20Sopenharmony_ci{ 658c2ecf20Sopenharmony_ci if ( vsyscall_ehdr ) { 668c2ecf20Sopenharmony_ci const struct elfhdr *const ehdrp = 678c2ecf20Sopenharmony_ci (struct elfhdr *)vsyscall_ehdr; 688c2ecf20Sopenharmony_ci const struct elf_phdr *const phdrp = 698c2ecf20Sopenharmony_ci (const struct elf_phdr *) (vsyscall_ehdr + ehdrp->e_phoff); 708c2ecf20Sopenharmony_ci int i; 718c2ecf20Sopenharmony_ci 728c2ecf20Sopenharmony_ci for (i = 0; i < ehdrp->e_phnum; ++i) 738c2ecf20Sopenharmony_ci if (phdrp[i].p_type == PT_LOAD) 748c2ecf20Sopenharmony_ci return (size_t) phdrp[i].p_filesz; 758c2ecf20Sopenharmony_ci } 768c2ecf20Sopenharmony_ci return 0; 778c2ecf20Sopenharmony_ci} 78