162306a36Sopenharmony_ci/* SPDX-License-Identifier: GPL-2.0-only */ 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * sorttable.h 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Added ORC unwind tables sort support and other updates: 662306a36Sopenharmony_ci * Copyright (C) 1999-2019 Alibaba Group Holding Limited. by: 762306a36Sopenharmony_ci * Shile Zhang <shile.zhang@linux.alibaba.com> 862306a36Sopenharmony_ci * 962306a36Sopenharmony_ci * Copyright 2011 - 2012 Cavium, Inc. 1062306a36Sopenharmony_ci * 1162306a36Sopenharmony_ci * Some of code was taken out of arch/x86/kernel/unwind_orc.c, written by: 1262306a36Sopenharmony_ci * Copyright (C) 2017 Josh Poimboeuf <jpoimboe@redhat.com> 1362306a36Sopenharmony_ci * 1462306a36Sopenharmony_ci * Some of this code was taken out of recordmcount.h written by: 1562306a36Sopenharmony_ci * 1662306a36Sopenharmony_ci * Copyright 2009 John F. Reiser <jreiser@BitWagon.com>. All rights reserved. 1762306a36Sopenharmony_ci * Copyright 2010 Steven Rostedt <srostedt@redhat.com>, Red Hat Inc. 1862306a36Sopenharmony_ci */ 1962306a36Sopenharmony_ci 2062306a36Sopenharmony_ci#undef extable_ent_size 2162306a36Sopenharmony_ci#undef compare_extable 2262306a36Sopenharmony_ci#undef get_mcount_loc 2362306a36Sopenharmony_ci#undef sort_mcount_loc 2462306a36Sopenharmony_ci#undef elf_mcount_loc 2562306a36Sopenharmony_ci#undef do_sort 2662306a36Sopenharmony_ci#undef Elf_Addr 2762306a36Sopenharmony_ci#undef Elf_Ehdr 2862306a36Sopenharmony_ci#undef Elf_Shdr 2962306a36Sopenharmony_ci#undef Elf_Rel 3062306a36Sopenharmony_ci#undef Elf_Rela 3162306a36Sopenharmony_ci#undef Elf_Sym 3262306a36Sopenharmony_ci#undef ELF_R_SYM 3362306a36Sopenharmony_ci#undef Elf_r_sym 3462306a36Sopenharmony_ci#undef ELF_R_INFO 3562306a36Sopenharmony_ci#undef Elf_r_info 3662306a36Sopenharmony_ci#undef ELF_ST_BIND 3762306a36Sopenharmony_ci#undef ELF_ST_TYPE 3862306a36Sopenharmony_ci#undef fn_ELF_R_SYM 3962306a36Sopenharmony_ci#undef fn_ELF_R_INFO 4062306a36Sopenharmony_ci#undef uint_t 4162306a36Sopenharmony_ci#undef _r 4262306a36Sopenharmony_ci#undef _w 4362306a36Sopenharmony_ci 4462306a36Sopenharmony_ci#ifdef SORTTABLE_64 4562306a36Sopenharmony_ci# define extable_ent_size 16 4662306a36Sopenharmony_ci# define compare_extable compare_extable_64 4762306a36Sopenharmony_ci# define get_mcount_loc get_mcount_loc_64 4862306a36Sopenharmony_ci# define sort_mcount_loc sort_mcount_loc_64 4962306a36Sopenharmony_ci# define elf_mcount_loc elf_mcount_loc_64 5062306a36Sopenharmony_ci# define do_sort do_sort_64 5162306a36Sopenharmony_ci# define Elf_Addr Elf64_Addr 5262306a36Sopenharmony_ci# define Elf_Ehdr Elf64_Ehdr 5362306a36Sopenharmony_ci# define Elf_Shdr Elf64_Shdr 5462306a36Sopenharmony_ci# define Elf_Rel Elf64_Rel 5562306a36Sopenharmony_ci# define Elf_Rela Elf64_Rela 5662306a36Sopenharmony_ci# define Elf_Sym Elf64_Sym 5762306a36Sopenharmony_ci# define ELF_R_SYM ELF64_R_SYM 5862306a36Sopenharmony_ci# define Elf_r_sym Elf64_r_sym 5962306a36Sopenharmony_ci# define ELF_R_INFO ELF64_R_INFO 6062306a36Sopenharmony_ci# define Elf_r_info Elf64_r_info 6162306a36Sopenharmony_ci# define ELF_ST_BIND ELF64_ST_BIND 6262306a36Sopenharmony_ci# define ELF_ST_TYPE ELF64_ST_TYPE 6362306a36Sopenharmony_ci# define fn_ELF_R_SYM fn_ELF64_R_SYM 6462306a36Sopenharmony_ci# define fn_ELF_R_INFO fn_ELF64_R_INFO 6562306a36Sopenharmony_ci# define uint_t uint64_t 6662306a36Sopenharmony_ci# define _r r8 6762306a36Sopenharmony_ci# define _w w8 6862306a36Sopenharmony_ci#else 6962306a36Sopenharmony_ci# define extable_ent_size 8 7062306a36Sopenharmony_ci# define compare_extable compare_extable_32 7162306a36Sopenharmony_ci# define get_mcount_loc get_mcount_loc_32 7262306a36Sopenharmony_ci# define sort_mcount_loc sort_mcount_loc_32 7362306a36Sopenharmony_ci# define elf_mcount_loc elf_mcount_loc_32 7462306a36Sopenharmony_ci# define do_sort do_sort_32 7562306a36Sopenharmony_ci# define Elf_Addr Elf32_Addr 7662306a36Sopenharmony_ci# define Elf_Ehdr Elf32_Ehdr 7762306a36Sopenharmony_ci# define Elf_Shdr Elf32_Shdr 7862306a36Sopenharmony_ci# define Elf_Rel Elf32_Rel 7962306a36Sopenharmony_ci# define Elf_Rela Elf32_Rela 8062306a36Sopenharmony_ci# define Elf_Sym Elf32_Sym 8162306a36Sopenharmony_ci# define ELF_R_SYM ELF32_R_SYM 8262306a36Sopenharmony_ci# define Elf_r_sym Elf32_r_sym 8362306a36Sopenharmony_ci# define ELF_R_INFO ELF32_R_INFO 8462306a36Sopenharmony_ci# define Elf_r_info Elf32_r_info 8562306a36Sopenharmony_ci# define ELF_ST_BIND ELF32_ST_BIND 8662306a36Sopenharmony_ci# define ELF_ST_TYPE ELF32_ST_TYPE 8762306a36Sopenharmony_ci# define fn_ELF_R_SYM fn_ELF32_R_SYM 8862306a36Sopenharmony_ci# define fn_ELF_R_INFO fn_ELF32_R_INFO 8962306a36Sopenharmony_ci# define uint_t uint32_t 9062306a36Sopenharmony_ci# define _r r 9162306a36Sopenharmony_ci# define _w w 9262306a36Sopenharmony_ci#endif 9362306a36Sopenharmony_ci 9462306a36Sopenharmony_ci#if defined(SORTTABLE_64) && defined(UNWINDER_ORC_ENABLED) 9562306a36Sopenharmony_ci/* ORC unwinder only support X86_64 */ 9662306a36Sopenharmony_ci#include <asm/orc_types.h> 9762306a36Sopenharmony_ci 9862306a36Sopenharmony_ci#define ERRSTR_MAXSZ 256 9962306a36Sopenharmony_ci 10062306a36Sopenharmony_cichar g_err[ERRSTR_MAXSZ]; 10162306a36Sopenharmony_ciint *g_orc_ip_table; 10262306a36Sopenharmony_cistruct orc_entry *g_orc_table; 10362306a36Sopenharmony_ci 10462306a36Sopenharmony_cipthread_t orc_sort_thread; 10562306a36Sopenharmony_ci 10662306a36Sopenharmony_cistatic inline unsigned long orc_ip(const int *ip) 10762306a36Sopenharmony_ci{ 10862306a36Sopenharmony_ci return (unsigned long)ip + *ip; 10962306a36Sopenharmony_ci} 11062306a36Sopenharmony_ci 11162306a36Sopenharmony_cistatic int orc_sort_cmp(const void *_a, const void *_b) 11262306a36Sopenharmony_ci{ 11362306a36Sopenharmony_ci struct orc_entry *orc_a; 11462306a36Sopenharmony_ci const int *a = g_orc_ip_table + *(int *)_a; 11562306a36Sopenharmony_ci const int *b = g_orc_ip_table + *(int *)_b; 11662306a36Sopenharmony_ci unsigned long a_val = orc_ip(a); 11762306a36Sopenharmony_ci unsigned long b_val = orc_ip(b); 11862306a36Sopenharmony_ci 11962306a36Sopenharmony_ci if (a_val > b_val) 12062306a36Sopenharmony_ci return 1; 12162306a36Sopenharmony_ci if (a_val < b_val) 12262306a36Sopenharmony_ci return -1; 12362306a36Sopenharmony_ci 12462306a36Sopenharmony_ci /* 12562306a36Sopenharmony_ci * The "weak" section terminator entries need to always be on the left 12662306a36Sopenharmony_ci * to ensure the lookup code skips them in favor of real entries. 12762306a36Sopenharmony_ci * These terminator entries exist to handle any gaps created by 12862306a36Sopenharmony_ci * whitelisted .o files which didn't get objtool generation. 12962306a36Sopenharmony_ci */ 13062306a36Sopenharmony_ci orc_a = g_orc_table + (a - g_orc_ip_table); 13162306a36Sopenharmony_ci return orc_a->type == ORC_TYPE_UNDEFINED ? -1 : 1; 13262306a36Sopenharmony_ci} 13362306a36Sopenharmony_ci 13462306a36Sopenharmony_cistatic void *sort_orctable(void *arg) 13562306a36Sopenharmony_ci{ 13662306a36Sopenharmony_ci int i; 13762306a36Sopenharmony_ci int *idxs = NULL; 13862306a36Sopenharmony_ci int *tmp_orc_ip_table = NULL; 13962306a36Sopenharmony_ci struct orc_entry *tmp_orc_table = NULL; 14062306a36Sopenharmony_ci unsigned int *orc_ip_size = (unsigned int *)arg; 14162306a36Sopenharmony_ci unsigned int num_entries = *orc_ip_size / sizeof(int); 14262306a36Sopenharmony_ci unsigned int orc_size = num_entries * sizeof(struct orc_entry); 14362306a36Sopenharmony_ci 14462306a36Sopenharmony_ci idxs = (int *)malloc(*orc_ip_size); 14562306a36Sopenharmony_ci if (!idxs) { 14662306a36Sopenharmony_ci snprintf(g_err, ERRSTR_MAXSZ, "malloc idxs: %s", 14762306a36Sopenharmony_ci strerror(errno)); 14862306a36Sopenharmony_ci pthread_exit(g_err); 14962306a36Sopenharmony_ci } 15062306a36Sopenharmony_ci 15162306a36Sopenharmony_ci tmp_orc_ip_table = (int *)malloc(*orc_ip_size); 15262306a36Sopenharmony_ci if (!tmp_orc_ip_table) { 15362306a36Sopenharmony_ci snprintf(g_err, ERRSTR_MAXSZ, "malloc tmp_orc_ip_table: %s", 15462306a36Sopenharmony_ci strerror(errno)); 15562306a36Sopenharmony_ci pthread_exit(g_err); 15662306a36Sopenharmony_ci } 15762306a36Sopenharmony_ci 15862306a36Sopenharmony_ci tmp_orc_table = (struct orc_entry *)malloc(orc_size); 15962306a36Sopenharmony_ci if (!tmp_orc_table) { 16062306a36Sopenharmony_ci snprintf(g_err, ERRSTR_MAXSZ, "malloc tmp_orc_table: %s", 16162306a36Sopenharmony_ci strerror(errno)); 16262306a36Sopenharmony_ci pthread_exit(g_err); 16362306a36Sopenharmony_ci } 16462306a36Sopenharmony_ci 16562306a36Sopenharmony_ci /* initialize indices array, convert ip_table to absolute address */ 16662306a36Sopenharmony_ci for (i = 0; i < num_entries; i++) { 16762306a36Sopenharmony_ci idxs[i] = i; 16862306a36Sopenharmony_ci tmp_orc_ip_table[i] = g_orc_ip_table[i] + i * sizeof(int); 16962306a36Sopenharmony_ci } 17062306a36Sopenharmony_ci memcpy(tmp_orc_table, g_orc_table, orc_size); 17162306a36Sopenharmony_ci 17262306a36Sopenharmony_ci qsort(idxs, num_entries, sizeof(int), orc_sort_cmp); 17362306a36Sopenharmony_ci 17462306a36Sopenharmony_ci for (i = 0; i < num_entries; i++) { 17562306a36Sopenharmony_ci if (idxs[i] == i) 17662306a36Sopenharmony_ci continue; 17762306a36Sopenharmony_ci 17862306a36Sopenharmony_ci /* convert back to relative address */ 17962306a36Sopenharmony_ci g_orc_ip_table[i] = tmp_orc_ip_table[idxs[i]] - i * sizeof(int); 18062306a36Sopenharmony_ci g_orc_table[i] = tmp_orc_table[idxs[i]]; 18162306a36Sopenharmony_ci } 18262306a36Sopenharmony_ci 18362306a36Sopenharmony_ci free(idxs); 18462306a36Sopenharmony_ci free(tmp_orc_ip_table); 18562306a36Sopenharmony_ci free(tmp_orc_table); 18662306a36Sopenharmony_ci pthread_exit(NULL); 18762306a36Sopenharmony_ci} 18862306a36Sopenharmony_ci#endif 18962306a36Sopenharmony_ci 19062306a36Sopenharmony_cistatic int compare_extable(const void *a, const void *b) 19162306a36Sopenharmony_ci{ 19262306a36Sopenharmony_ci Elf_Addr av = _r(a); 19362306a36Sopenharmony_ci Elf_Addr bv = _r(b); 19462306a36Sopenharmony_ci 19562306a36Sopenharmony_ci if (av < bv) 19662306a36Sopenharmony_ci return -1; 19762306a36Sopenharmony_ci if (av > bv) 19862306a36Sopenharmony_ci return 1; 19962306a36Sopenharmony_ci return 0; 20062306a36Sopenharmony_ci} 20162306a36Sopenharmony_ci#ifdef MCOUNT_SORT_ENABLED 20262306a36Sopenharmony_cipthread_t mcount_sort_thread; 20362306a36Sopenharmony_ci 20462306a36Sopenharmony_cistruct elf_mcount_loc { 20562306a36Sopenharmony_ci Elf_Ehdr *ehdr; 20662306a36Sopenharmony_ci Elf_Shdr *init_data_sec; 20762306a36Sopenharmony_ci uint_t start_mcount_loc; 20862306a36Sopenharmony_ci uint_t stop_mcount_loc; 20962306a36Sopenharmony_ci}; 21062306a36Sopenharmony_ci 21162306a36Sopenharmony_ci/* Sort the addresses stored between __start_mcount_loc to __stop_mcount_loc in vmlinux */ 21262306a36Sopenharmony_cistatic void *sort_mcount_loc(void *arg) 21362306a36Sopenharmony_ci{ 21462306a36Sopenharmony_ci struct elf_mcount_loc *emloc = (struct elf_mcount_loc *)arg; 21562306a36Sopenharmony_ci uint_t offset = emloc->start_mcount_loc - _r(&(emloc->init_data_sec)->sh_addr) 21662306a36Sopenharmony_ci + _r(&(emloc->init_data_sec)->sh_offset); 21762306a36Sopenharmony_ci uint_t count = emloc->stop_mcount_loc - emloc->start_mcount_loc; 21862306a36Sopenharmony_ci unsigned char *start_loc = (void *)emloc->ehdr + offset; 21962306a36Sopenharmony_ci 22062306a36Sopenharmony_ci qsort(start_loc, count/sizeof(uint_t), sizeof(uint_t), compare_extable); 22162306a36Sopenharmony_ci return NULL; 22262306a36Sopenharmony_ci} 22362306a36Sopenharmony_ci 22462306a36Sopenharmony_ci/* Get the address of __start_mcount_loc and __stop_mcount_loc in System.map */ 22562306a36Sopenharmony_cistatic void get_mcount_loc(uint_t *_start, uint_t *_stop) 22662306a36Sopenharmony_ci{ 22762306a36Sopenharmony_ci FILE *file_start, *file_stop; 22862306a36Sopenharmony_ci char start_buff[20]; 22962306a36Sopenharmony_ci char stop_buff[20]; 23062306a36Sopenharmony_ci int len = 0; 23162306a36Sopenharmony_ci 23262306a36Sopenharmony_ci file_start = popen(" grep start_mcount System.map | awk '{print $1}' ", "r"); 23362306a36Sopenharmony_ci if (!file_start) { 23462306a36Sopenharmony_ci fprintf(stderr, "get start_mcount_loc error!"); 23562306a36Sopenharmony_ci return; 23662306a36Sopenharmony_ci } 23762306a36Sopenharmony_ci 23862306a36Sopenharmony_ci file_stop = popen(" grep stop_mcount System.map | awk '{print $1}' ", "r"); 23962306a36Sopenharmony_ci if (!file_stop) { 24062306a36Sopenharmony_ci fprintf(stderr, "get stop_mcount_loc error!"); 24162306a36Sopenharmony_ci pclose(file_start); 24262306a36Sopenharmony_ci return; 24362306a36Sopenharmony_ci } 24462306a36Sopenharmony_ci 24562306a36Sopenharmony_ci while (fgets(start_buff, sizeof(start_buff), file_start) != NULL) { 24662306a36Sopenharmony_ci len = strlen(start_buff); 24762306a36Sopenharmony_ci start_buff[len - 1] = '\0'; 24862306a36Sopenharmony_ci } 24962306a36Sopenharmony_ci *_start = strtoul(start_buff, NULL, 16); 25062306a36Sopenharmony_ci 25162306a36Sopenharmony_ci while (fgets(stop_buff, sizeof(stop_buff), file_stop) != NULL) { 25262306a36Sopenharmony_ci len = strlen(stop_buff); 25362306a36Sopenharmony_ci stop_buff[len - 1] = '\0'; 25462306a36Sopenharmony_ci } 25562306a36Sopenharmony_ci *_stop = strtoul(stop_buff, NULL, 16); 25662306a36Sopenharmony_ci 25762306a36Sopenharmony_ci pclose(file_start); 25862306a36Sopenharmony_ci pclose(file_stop); 25962306a36Sopenharmony_ci} 26062306a36Sopenharmony_ci#endif 26162306a36Sopenharmony_cistatic int do_sort(Elf_Ehdr *ehdr, 26262306a36Sopenharmony_ci char const *const fname, 26362306a36Sopenharmony_ci table_sort_t custom_sort) 26462306a36Sopenharmony_ci{ 26562306a36Sopenharmony_ci int rc = -1; 26662306a36Sopenharmony_ci Elf_Shdr *s, *shdr = (Elf_Shdr *)((char *)ehdr + _r(&ehdr->e_shoff)); 26762306a36Sopenharmony_ci Elf_Shdr *strtab_sec = NULL; 26862306a36Sopenharmony_ci Elf_Shdr *symtab_sec = NULL; 26962306a36Sopenharmony_ci Elf_Shdr *extab_sec = NULL; 27062306a36Sopenharmony_ci Elf_Sym *sym; 27162306a36Sopenharmony_ci const Elf_Sym *symtab; 27262306a36Sopenharmony_ci Elf32_Word *symtab_shndx = NULL; 27362306a36Sopenharmony_ci Elf_Sym *sort_needed_sym = NULL; 27462306a36Sopenharmony_ci Elf_Shdr *sort_needed_sec; 27562306a36Sopenharmony_ci Elf_Rel *relocs = NULL; 27662306a36Sopenharmony_ci int relocs_size = 0; 27762306a36Sopenharmony_ci uint32_t *sort_needed_loc; 27862306a36Sopenharmony_ci const char *secstrings; 27962306a36Sopenharmony_ci const char *strtab; 28062306a36Sopenharmony_ci char *extab_image; 28162306a36Sopenharmony_ci int extab_index = 0; 28262306a36Sopenharmony_ci int i; 28362306a36Sopenharmony_ci int idx; 28462306a36Sopenharmony_ci unsigned int shnum; 28562306a36Sopenharmony_ci unsigned int shstrndx; 28662306a36Sopenharmony_ci#ifdef MCOUNT_SORT_ENABLED 28762306a36Sopenharmony_ci struct elf_mcount_loc mstruct = {0}; 28862306a36Sopenharmony_ci uint_t _start_mcount_loc = 0; 28962306a36Sopenharmony_ci uint_t _stop_mcount_loc = 0; 29062306a36Sopenharmony_ci#endif 29162306a36Sopenharmony_ci#if defined(SORTTABLE_64) && defined(UNWINDER_ORC_ENABLED) 29262306a36Sopenharmony_ci unsigned int orc_ip_size = 0; 29362306a36Sopenharmony_ci unsigned int orc_size = 0; 29462306a36Sopenharmony_ci unsigned int orc_num_entries = 0; 29562306a36Sopenharmony_ci#endif 29662306a36Sopenharmony_ci 29762306a36Sopenharmony_ci shstrndx = r2(&ehdr->e_shstrndx); 29862306a36Sopenharmony_ci if (shstrndx == SHN_XINDEX) 29962306a36Sopenharmony_ci shstrndx = r(&shdr[0].sh_link); 30062306a36Sopenharmony_ci secstrings = (const char *)ehdr + _r(&shdr[shstrndx].sh_offset); 30162306a36Sopenharmony_ci 30262306a36Sopenharmony_ci shnum = r2(&ehdr->e_shnum); 30362306a36Sopenharmony_ci if (shnum == SHN_UNDEF) 30462306a36Sopenharmony_ci shnum = _r(&shdr[0].sh_size); 30562306a36Sopenharmony_ci 30662306a36Sopenharmony_ci for (i = 0, s = shdr; s < shdr + shnum; i++, s++) { 30762306a36Sopenharmony_ci idx = r(&s->sh_name); 30862306a36Sopenharmony_ci if (!strcmp(secstrings + idx, "__ex_table")) { 30962306a36Sopenharmony_ci extab_sec = s; 31062306a36Sopenharmony_ci extab_index = i; 31162306a36Sopenharmony_ci } 31262306a36Sopenharmony_ci if (!strcmp(secstrings + idx, ".symtab")) 31362306a36Sopenharmony_ci symtab_sec = s; 31462306a36Sopenharmony_ci if (!strcmp(secstrings + idx, ".strtab")) 31562306a36Sopenharmony_ci strtab_sec = s; 31662306a36Sopenharmony_ci 31762306a36Sopenharmony_ci if ((r(&s->sh_type) == SHT_REL || 31862306a36Sopenharmony_ci r(&s->sh_type) == SHT_RELA) && 31962306a36Sopenharmony_ci r(&s->sh_info) == extab_index) { 32062306a36Sopenharmony_ci relocs = (void *)ehdr + _r(&s->sh_offset); 32162306a36Sopenharmony_ci relocs_size = _r(&s->sh_size); 32262306a36Sopenharmony_ci } 32362306a36Sopenharmony_ci if (r(&s->sh_type) == SHT_SYMTAB_SHNDX) 32462306a36Sopenharmony_ci symtab_shndx = (Elf32_Word *)((const char *)ehdr + 32562306a36Sopenharmony_ci _r(&s->sh_offset)); 32662306a36Sopenharmony_ci 32762306a36Sopenharmony_ci#ifdef MCOUNT_SORT_ENABLED 32862306a36Sopenharmony_ci /* locate the .init.data section in vmlinux */ 32962306a36Sopenharmony_ci if (!strcmp(secstrings + idx, ".init.data")) { 33062306a36Sopenharmony_ci get_mcount_loc(&_start_mcount_loc, &_stop_mcount_loc); 33162306a36Sopenharmony_ci mstruct.ehdr = ehdr; 33262306a36Sopenharmony_ci mstruct.init_data_sec = s; 33362306a36Sopenharmony_ci mstruct.start_mcount_loc = _start_mcount_loc; 33462306a36Sopenharmony_ci mstruct.stop_mcount_loc = _stop_mcount_loc; 33562306a36Sopenharmony_ci } 33662306a36Sopenharmony_ci#endif 33762306a36Sopenharmony_ci 33862306a36Sopenharmony_ci#if defined(SORTTABLE_64) && defined(UNWINDER_ORC_ENABLED) 33962306a36Sopenharmony_ci /* locate the ORC unwind tables */ 34062306a36Sopenharmony_ci if (!strcmp(secstrings + idx, ".orc_unwind_ip")) { 34162306a36Sopenharmony_ci orc_ip_size = s->sh_size; 34262306a36Sopenharmony_ci g_orc_ip_table = (int *)((void *)ehdr + 34362306a36Sopenharmony_ci s->sh_offset); 34462306a36Sopenharmony_ci } 34562306a36Sopenharmony_ci if (!strcmp(secstrings + idx, ".orc_unwind")) { 34662306a36Sopenharmony_ci orc_size = s->sh_size; 34762306a36Sopenharmony_ci g_orc_table = (struct orc_entry *)((void *)ehdr + 34862306a36Sopenharmony_ci s->sh_offset); 34962306a36Sopenharmony_ci } 35062306a36Sopenharmony_ci#endif 35162306a36Sopenharmony_ci } /* for loop */ 35262306a36Sopenharmony_ci 35362306a36Sopenharmony_ci#if defined(SORTTABLE_64) && defined(UNWINDER_ORC_ENABLED) 35462306a36Sopenharmony_ci if (!g_orc_ip_table || !g_orc_table) { 35562306a36Sopenharmony_ci fprintf(stderr, 35662306a36Sopenharmony_ci "incomplete ORC unwind tables in file: %s\n", fname); 35762306a36Sopenharmony_ci goto out; 35862306a36Sopenharmony_ci } 35962306a36Sopenharmony_ci 36062306a36Sopenharmony_ci orc_num_entries = orc_ip_size / sizeof(int); 36162306a36Sopenharmony_ci if (orc_ip_size % sizeof(int) != 0 || 36262306a36Sopenharmony_ci orc_size % sizeof(struct orc_entry) != 0 || 36362306a36Sopenharmony_ci orc_num_entries != orc_size / sizeof(struct orc_entry)) { 36462306a36Sopenharmony_ci fprintf(stderr, 36562306a36Sopenharmony_ci "inconsistent ORC unwind table entries in file: %s\n", 36662306a36Sopenharmony_ci fname); 36762306a36Sopenharmony_ci goto out; 36862306a36Sopenharmony_ci } 36962306a36Sopenharmony_ci 37062306a36Sopenharmony_ci /* create thread to sort ORC unwind tables concurrently */ 37162306a36Sopenharmony_ci if (pthread_create(&orc_sort_thread, NULL, 37262306a36Sopenharmony_ci sort_orctable, &orc_ip_size)) { 37362306a36Sopenharmony_ci fprintf(stderr, 37462306a36Sopenharmony_ci "pthread_create orc_sort_thread failed '%s': %s\n", 37562306a36Sopenharmony_ci strerror(errno), fname); 37662306a36Sopenharmony_ci goto out; 37762306a36Sopenharmony_ci } 37862306a36Sopenharmony_ci#endif 37962306a36Sopenharmony_ci 38062306a36Sopenharmony_ci#ifdef MCOUNT_SORT_ENABLED 38162306a36Sopenharmony_ci if (!mstruct.init_data_sec || !_start_mcount_loc || !_stop_mcount_loc) { 38262306a36Sopenharmony_ci fprintf(stderr, 38362306a36Sopenharmony_ci "incomplete mcount's sort in file: %s\n", 38462306a36Sopenharmony_ci fname); 38562306a36Sopenharmony_ci goto out; 38662306a36Sopenharmony_ci } 38762306a36Sopenharmony_ci 38862306a36Sopenharmony_ci /* create thread to sort mcount_loc concurrently */ 38962306a36Sopenharmony_ci if (pthread_create(&mcount_sort_thread, NULL, &sort_mcount_loc, &mstruct)) { 39062306a36Sopenharmony_ci fprintf(stderr, 39162306a36Sopenharmony_ci "pthread_create mcount_sort_thread failed '%s': %s\n", 39262306a36Sopenharmony_ci strerror(errno), fname); 39362306a36Sopenharmony_ci goto out; 39462306a36Sopenharmony_ci } 39562306a36Sopenharmony_ci#endif 39662306a36Sopenharmony_ci if (!extab_sec) { 39762306a36Sopenharmony_ci fprintf(stderr, "no __ex_table in file: %s\n", fname); 39862306a36Sopenharmony_ci goto out; 39962306a36Sopenharmony_ci } 40062306a36Sopenharmony_ci 40162306a36Sopenharmony_ci if (!symtab_sec) { 40262306a36Sopenharmony_ci fprintf(stderr, "no .symtab in file: %s\n", fname); 40362306a36Sopenharmony_ci goto out; 40462306a36Sopenharmony_ci } 40562306a36Sopenharmony_ci 40662306a36Sopenharmony_ci if (!strtab_sec) { 40762306a36Sopenharmony_ci fprintf(stderr, "no .strtab in file: %s\n", fname); 40862306a36Sopenharmony_ci goto out; 40962306a36Sopenharmony_ci } 41062306a36Sopenharmony_ci 41162306a36Sopenharmony_ci extab_image = (void *)ehdr + _r(&extab_sec->sh_offset); 41262306a36Sopenharmony_ci strtab = (const char *)ehdr + _r(&strtab_sec->sh_offset); 41362306a36Sopenharmony_ci symtab = (const Elf_Sym *)((const char *)ehdr + 41462306a36Sopenharmony_ci _r(&symtab_sec->sh_offset)); 41562306a36Sopenharmony_ci 41662306a36Sopenharmony_ci if (custom_sort) { 41762306a36Sopenharmony_ci custom_sort(extab_image, _r(&extab_sec->sh_size)); 41862306a36Sopenharmony_ci } else { 41962306a36Sopenharmony_ci int num_entries = _r(&extab_sec->sh_size) / extable_ent_size; 42062306a36Sopenharmony_ci qsort(extab_image, num_entries, 42162306a36Sopenharmony_ci extable_ent_size, compare_extable); 42262306a36Sopenharmony_ci } 42362306a36Sopenharmony_ci 42462306a36Sopenharmony_ci /* If there were relocations, we no longer need them. */ 42562306a36Sopenharmony_ci if (relocs) 42662306a36Sopenharmony_ci memset(relocs, 0, relocs_size); 42762306a36Sopenharmony_ci 42862306a36Sopenharmony_ci /* find the flag main_extable_sort_needed */ 42962306a36Sopenharmony_ci for (sym = (void *)ehdr + _r(&symtab_sec->sh_offset); 43062306a36Sopenharmony_ci sym < sym + _r(&symtab_sec->sh_size) / sizeof(Elf_Sym); 43162306a36Sopenharmony_ci sym++) { 43262306a36Sopenharmony_ci if (ELF_ST_TYPE(sym->st_info) != STT_OBJECT) 43362306a36Sopenharmony_ci continue; 43462306a36Sopenharmony_ci if (!strcmp(strtab + r(&sym->st_name), 43562306a36Sopenharmony_ci "main_extable_sort_needed")) { 43662306a36Sopenharmony_ci sort_needed_sym = sym; 43762306a36Sopenharmony_ci break; 43862306a36Sopenharmony_ci } 43962306a36Sopenharmony_ci } 44062306a36Sopenharmony_ci 44162306a36Sopenharmony_ci if (!sort_needed_sym) { 44262306a36Sopenharmony_ci fprintf(stderr, 44362306a36Sopenharmony_ci "no main_extable_sort_needed symbol in file: %s\n", 44462306a36Sopenharmony_ci fname); 44562306a36Sopenharmony_ci goto out; 44662306a36Sopenharmony_ci } 44762306a36Sopenharmony_ci 44862306a36Sopenharmony_ci sort_needed_sec = &shdr[get_secindex(r2(&sym->st_shndx), 44962306a36Sopenharmony_ci sort_needed_sym - symtab, 45062306a36Sopenharmony_ci symtab_shndx)]; 45162306a36Sopenharmony_ci sort_needed_loc = (void *)ehdr + 45262306a36Sopenharmony_ci _r(&sort_needed_sec->sh_offset) + 45362306a36Sopenharmony_ci _r(&sort_needed_sym->st_value) - 45462306a36Sopenharmony_ci _r(&sort_needed_sec->sh_addr); 45562306a36Sopenharmony_ci 45662306a36Sopenharmony_ci /* extable has been sorted, clear the flag */ 45762306a36Sopenharmony_ci w(0, sort_needed_loc); 45862306a36Sopenharmony_ci rc = 0; 45962306a36Sopenharmony_ci 46062306a36Sopenharmony_ciout: 46162306a36Sopenharmony_ci#if defined(SORTTABLE_64) && defined(UNWINDER_ORC_ENABLED) 46262306a36Sopenharmony_ci if (orc_sort_thread) { 46362306a36Sopenharmony_ci void *retval = NULL; 46462306a36Sopenharmony_ci /* wait for ORC tables sort done */ 46562306a36Sopenharmony_ci rc = pthread_join(orc_sort_thread, &retval); 46662306a36Sopenharmony_ci if (rc) { 46762306a36Sopenharmony_ci fprintf(stderr, 46862306a36Sopenharmony_ci "pthread_join failed '%s': %s\n", 46962306a36Sopenharmony_ci strerror(errno), fname); 47062306a36Sopenharmony_ci } else if (retval) { 47162306a36Sopenharmony_ci rc = -1; 47262306a36Sopenharmony_ci fprintf(stderr, 47362306a36Sopenharmony_ci "failed to sort ORC tables '%s': %s\n", 47462306a36Sopenharmony_ci (char *)retval, fname); 47562306a36Sopenharmony_ci } 47662306a36Sopenharmony_ci } 47762306a36Sopenharmony_ci#endif 47862306a36Sopenharmony_ci 47962306a36Sopenharmony_ci#ifdef MCOUNT_SORT_ENABLED 48062306a36Sopenharmony_ci if (mcount_sort_thread) { 48162306a36Sopenharmony_ci void *retval = NULL; 48262306a36Sopenharmony_ci /* wait for mcount sort done */ 48362306a36Sopenharmony_ci rc = pthread_join(mcount_sort_thread, &retval); 48462306a36Sopenharmony_ci if (rc) { 48562306a36Sopenharmony_ci fprintf(stderr, 48662306a36Sopenharmony_ci "pthread_join failed '%s': %s\n", 48762306a36Sopenharmony_ci strerror(errno), fname); 48862306a36Sopenharmony_ci } else if (retval) { 48962306a36Sopenharmony_ci rc = -1; 49062306a36Sopenharmony_ci fprintf(stderr, 49162306a36Sopenharmony_ci "failed to sort mcount '%s': %s\n", 49262306a36Sopenharmony_ci (char *)retval, fname); 49362306a36Sopenharmony_ci } 49462306a36Sopenharmony_ci } 49562306a36Sopenharmony_ci#endif 49662306a36Sopenharmony_ci return rc; 49762306a36Sopenharmony_ci} 498