3866 lines
		
	
	
		
			111 KiB
		
	
	
	
		
			C
		
	
	
			
		
		
	
	
			3866 lines
		
	
	
		
			111 KiB
		
	
	
	
		
			C
		
	
	
| // elf.c:
 | |
| #include <errno.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <sys/stat.h>
 | |
| #include <sys/types.h>
 | |
| #include <unistd.h>
 | |
| 
 | |
| #ifdef HAVE_DL_ITERATE_PHDR
 | |
| #include <link.h>
 | |
| #endif
 | |
| 
 | |
| #ifndef S_ISLNK
 | |
| #ifndef S_IFLNK
 | |
| #define S_IFLNK 0120000
 | |
| #endif
 | |
| #ifndef S_IFMT
 | |
| #define S_IFMT 0170000
 | |
| #endif
 | |
| #define S_ISLNK(m) (((m)&S_IFMT) == S_IFLNK)
 | |
| #endif
 | |
| 
 | |
| #ifndef __GNUC__
 | |
| #define __builtin_prefetch(p, r, l)
 | |
| #define unlikely(x) (x)
 | |
| #else
 | |
| #define unlikely(x) __builtin_expect(!!(x), 0)
 | |
| #endif
 | |
| 
 | |
| #if !defined(HAVE_DECL_STRNLEN) || !HAVE_DECL_STRNLEN
 | |
| 
 | |
| static size_t xstrnlen(const char *s, size_t maxlen) {
 | |
|   size_t i;
 | |
| 
 | |
|   for (i = 0; i < maxlen; ++i)
 | |
|     if (s[i] == '\0') break;
 | |
|   return i;
 | |
| }
 | |
| 
 | |
| #define strnlen xstrnlen
 | |
| 
 | |
| #endif
 | |
| 
 | |
| #ifndef HAVE_LSTAT
 | |
| 
 | |
| static int xlstat(const char *path ATTRIBUTE_UNUSED,
 | |
|                   struct stat *st ATTRIBUTE_UNUSED) {
 | |
|   return -1;
 | |
| }
 | |
| 
 | |
| #define lstat xlstat
 | |
| 
 | |
| #endif
 | |
| 
 | |
| #ifndef HAVE_READLINK
 | |
| 
 | |
| static ssize_t xreadlink(const char *path ATTRIBUTE_UNUSED,
 | |
|                          char *buf ATTRIBUTE_UNUSED,
 | |
|                          size_t bufsz ATTRIBUTE_UNUSED) {
 | |
|   return -1;
 | |
| }
 | |
| 
 | |
| #define readlink xreadlink
 | |
| 
 | |
| #endif
 | |
| 
 | |
| #ifndef HAVE_DL_ITERATE_PHDR
 | |
| 
 | |
| #define dl_phdr_info x_dl_phdr_info
 | |
| #define dl_iterate_phdr x_dl_iterate_phdr
 | |
| 
 | |
| struct dl_phdr_info {
 | |
|   uintptr_t dlpi_addr;
 | |
|   const char *dlpi_name;
 | |
| };
 | |
| 
 | |
| static int dl_iterate_phdr(int (*callback)(struct dl_phdr_info *, size_t,
 | |
|                                            void *) ATTRIBUTE_UNUSED,
 | |
|                            void *data ATTRIBUTE_UNUSED) {
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| #endif
 | |
| 
 | |
| #if BACKTRACE_ELF_SIZE != 32 && BACKTRACE_ELF_SIZE != 64
 | |
| #error "Unknown BACKTRACE_ELF_SIZE"
 | |
| #endif
 | |
| 
 | |
| #undef EI_NIDENT
 | |
| #undef EI_MAG0
 | |
| #undef EI_MAG1
 | |
| #undef EI_MAG2
 | |
| #undef EI_MAG3
 | |
| #undef EI_CLASS
 | |
| #undef EI_DATA
 | |
| #undef EI_VERSION
 | |
| #undef ELF_MAG0
 | |
| #undef ELF_MAG1
 | |
| #undef ELF_MAG2
 | |
| #undef ELF_MAG3
 | |
| #undef ELFCLASS32
 | |
| #undef ELFCLASS64
 | |
| #undef ELFDATA2LSB
 | |
| #undef ELFDATA2MSB
 | |
| #undef EV_CURRENT
 | |
| #undef ET_DYN
 | |
| #undef EM_PPC64
 | |
| #undef EF_PPC64_ABI
 | |
| #undef SHN_LORESERVE
 | |
| #undef SHN_XINDEX
 | |
| #undef SHN_UNDEF
 | |
| #undef SHT_PROGBITS
 | |
| #undef SHT_SYMTAB
 | |
| #undef SHT_STRTAB
 | |
| #undef SHT_DYNSYM
 | |
| #undef SHF_COMPRESSED
 | |
| #undef STT_OBJECT
 | |
| #undef STT_FUNC
 | |
| #undef NT_GNU_BUILD_ID
 | |
| #undef ELFCOMPRESS_ZLIB
 | |
| 
 | |
| typedef uint16_t b_elf_half;
 | |
| typedef uint32_t b_elf_word;
 | |
| typedef int32_t b_elf_sword;
 | |
| #if BACKTRACE_ELF_SIZE == 32
 | |
| 
 | |
| typedef uint32_t b_elf_addr;
 | |
| typedef uint32_t b_elf_off;
 | |
| typedef uint32_t b_elf_wxword;
 | |
| #else
 | |
| 
 | |
| typedef uint64_t b_elf_addr;
 | |
| typedef uint64_t b_elf_off;
 | |
| typedef uint64_t b_elf_xword;
 | |
| typedef int64_t b_elf_sxword;
 | |
| typedef uint64_t b_elf_wxword;
 | |
| #endif
 | |
| 
 | |
| #define EI_NIDENT 16
 | |
| 
 | |
| typedef struct {
 | |
|   unsigned char e_ident[EI_NIDENT];
 | |
|   b_elf_half e_type;
 | |
|   b_elf_half e_machine;
 | |
|   b_elf_word e_version;
 | |
|   b_elf_addr e_entry;
 | |
|   b_elf_off e_phoff;
 | |
|   b_elf_off e_shoff;
 | |
|   b_elf_word e_flags;
 | |
|   b_elf_half e_ehsize;
 | |
|   b_elf_half e_phentsize;
 | |
|   b_elf_half e_phnum;
 | |
|   b_elf_half e_shentsize;
 | |
|   b_elf_half e_shnum;
 | |
|   b_elf_half e_shstrndx;
 | |
| } b_elf_ehdr;
 | |
| #define EI_MAG0 0
 | |
| #define EI_MAG1 1
 | |
| #define EI_MAG2 2
 | |
| #define EI_MAG3 3
 | |
| #define EI_CLASS 4
 | |
| #define EI_DATA 5
 | |
| #define EI_VERSION 6
 | |
| 
 | |
| #define ELFMAG0 0x7f
 | |
| #define ELFMAG1 'E'
 | |
| #define ELFMAG2 'L'
 | |
| #define ELFMAG3 'F'
 | |
| 
 | |
| #define ELFCLASS32 1
 | |
| #define ELFCLASS64 2
 | |
| 
 | |
| #define ELFDATA2LSB 1
 | |
| #define ELFDATA2MSB 2
 | |
| 
 | |
| #define EV_CURRENT 1
 | |
| 
 | |
| #define ET_DYN 3
 | |
| 
 | |
| #define EM_PPC64 21
 | |
| #define EF_PPC64_ABI 3
 | |
| 
 | |
| typedef struct {
 | |
|   b_elf_word sh_name;
 | |
|   b_elf_word sh_type;
 | |
|   b_elf_wxword sh_flags;
 | |
|   b_elf_addr sh_addr;
 | |
|   b_elf_off sh_offset;
 | |
|   b_elf_wxword sh_size;
 | |
|   b_elf_word sh_link;
 | |
|   b_elf_word sh_info;
 | |
|   b_elf_wxword sh_addralign;
 | |
|   b_elf_wxword sh_entsize;
 | |
| } b_elf_shdr;
 | |
| #define SHN_UNDEF 0x0000
 | |
| #define SHN_LORESERVE 0xFF00
 | |
| #define SHN_XINDEX 0xFFFF
 | |
| #define SHT_PROGBITS 1
 | |
| #define SHT_SYMTAB 2
 | |
| #define SHT_STRTAB 3
 | |
| #define SHT_DYNSYM 11
 | |
| 
 | |
| #define SHF_COMPRESSED 0x800
 | |
| 
 | |
| #if BACKTRACE_ELF_SIZE == 32
 | |
| 
 | |
| typedef struct {
 | |
|   b_elf_word st_name;
 | |
|   b_elf_addr st_value;
 | |
|   b_elf_word st_size;
 | |
|   unsigned char st_info;
 | |
|   unsigned char st_other;
 | |
|   b_elf_half st_shndx;
 | |
| } b_elf_sym;
 | |
| #else
 | |
| typedef struct {
 | |
|   b_elf_word st_name;
 | |
|   unsigned char st_info;
 | |
|   unsigned char st_other;
 | |
|   b_elf_half st_shndx;
 | |
|   b_elf_addr st_value;
 | |
|   b_elf_xword st_size;
 | |
| } b_elf_sym;
 | |
| #endif
 | |
| #define STT_OBJECT 1
 | |
| #define STT_FUNC 2
 | |
| 
 | |
| typedef struct {
 | |
|   uint32_t namesz;
 | |
|   uint32_t descsz;
 | |
|   uint32_t type;
 | |
|   char name[1];
 | |
| } b_elf_note;
 | |
| 
 | |
| #define NT_GNU_BUILD_ID 3
 | |
| 
 | |
| #if BACKTRACE_ELF_SIZE == 32
 | |
| 
 | |
| typedef struct {
 | |
|   b_elf_word ch_type;
 | |
|   b_elf_word ch_size;
 | |
|   b_elf_word ch_addralign;
 | |
| } b_elf_chdr;
 | |
| #else
 | |
| typedef struct {
 | |
|   b_elf_word ch_type;
 | |
|   b_elf_word ch_reserved;
 | |
|   b_elf_xword ch_size;
 | |
|   b_elf_xword ch_addralign;
 | |
| } b_elf_chdr;
 | |
| #endif
 | |
| #define ELFCOMPRESS_ZLIB 1
 | |
| 
 | |
| static const char *const dwarf_section_names[DEBUG_MAX] = {
 | |
|     ".debug_info",        ".debug_line",     ".debug_abbrev",
 | |
|     ".debug_ranges",      ".debug_str",      ".debug_addr",
 | |
|     ".debug_str_offsets", ".debug_line_str", ".debug_rnglists"};
 | |
| 
 | |
| struct debug_section_info {
 | |
|   off_t offset;
 | |
| 
 | |
|   size_t size;
 | |
| 
 | |
|   const unsigned char *data;
 | |
| 
 | |
|   int compressed;
 | |
| };
 | |
| 
 | |
| struct elf_symbol {
 | |
|   const char *name;
 | |
| 
 | |
|   uintptr_t address;
 | |
| 
 | |
|   size_t size;
 | |
| };
 | |
| 
 | |
| struct elf_syminfo_data {
 | |
|   struct elf_syminfo_data *next;
 | |
| 
 | |
|   struct elf_symbol *symbols;
 | |
| 
 | |
|   size_t count;
 | |
| };
 | |
| 
 | |
| struct elf_view {
 | |
|   struct backtrace_view view;
 | |
|   int release;
 | |
| };
 | |
| 
 | |
| struct elf_ppc64_opd_data {
 | |
|   b_elf_addr addr;
 | |
| 
 | |
|   const char *data;
 | |
| 
 | |
|   size_t size;
 | |
| 
 | |
|   struct elf_view view;
 | |
| };
 | |
| 
 | |
| static int elf_get_view(struct backtrace_state *state, int descriptor,
 | |
|                         const unsigned char *memory, size_t memory_size,
 | |
|                         off_t offset, uint64_t size,
 | |
|                         backtrace_error_callback error_callback, void *data,
 | |
|                         struct elf_view *view) {
 | |
|   if (memory == NULL) {
 | |
|     view->release = 1;
 | |
|     return backtrace_get_view(state, descriptor, offset, size, error_callback,
 | |
|                               data, &view->view);
 | |
|   } else {
 | |
|     if ((uint64_t)offset + size > (uint64_t)memory_size) {
 | |
|       error_callback(data, "out of range for in-memory file", 0);
 | |
|       return 0;
 | |
|     }
 | |
|     view->view.data = (const void *)(memory + offset);
 | |
|     view->view.base = NULL;
 | |
|     view->view.len = size;
 | |
|     view->release = 0;
 | |
|     return 1;
 | |
|   }
 | |
| }
 | |
| 
 | |
| static void elf_release_view(struct backtrace_state *state,
 | |
|                              struct elf_view *view,
 | |
|                              backtrace_error_callback error_callback,
 | |
|                              void *data) {
 | |
|   if (view->release)
 | |
|     backtrace_release_view(state, &view->view, error_callback, data);
 | |
| }
 | |
| 
 | |
| static uint32_t elf_crc32(uint32_t crc, const unsigned char *buf, size_t len) {
 | |
|   static const uint32_t crc32_table[256] = {
 | |
|       0x00000000, 0x77073096, 0xee0e612c, 0x990951ba, 0x076dc419, 0x706af48f,
 | |
|       0xe963a535, 0x9e6495a3, 0x0edb8832, 0x79dcb8a4, 0xe0d5e91e, 0x97d2d988,
 | |
|       0x09b64c2b, 0x7eb17cbd, 0xe7b82d07, 0x90bf1d91, 0x1db71064, 0x6ab020f2,
 | |
|       0xf3b97148, 0x84be41de, 0x1adad47d, 0x6ddde4eb, 0xf4d4b551, 0x83d385c7,
 | |
|       0x136c9856, 0x646ba8c0, 0xfd62f97a, 0x8a65c9ec, 0x14015c4f, 0x63066cd9,
 | |
|       0xfa0f3d63, 0x8d080df5, 0x3b6e20c8, 0x4c69105e, 0xd56041e4, 0xa2677172,
 | |
|       0x3c03e4d1, 0x4b04d447, 0xd20d85fd, 0xa50ab56b, 0x35b5a8fa, 0x42b2986c,
 | |
|       0xdbbbc9d6, 0xacbcf940, 0x32d86ce3, 0x45df5c75, 0xdcd60dcf, 0xabd13d59,
 | |
|       0x26d930ac, 0x51de003a, 0xc8d75180, 0xbfd06116, 0x21b4f4b5, 0x56b3c423,
 | |
|       0xcfba9599, 0xb8bda50f, 0x2802b89e, 0x5f058808, 0xc60cd9b2, 0xb10be924,
 | |
|       0x2f6f7c87, 0x58684c11, 0xc1611dab, 0xb6662d3d, 0x76dc4190, 0x01db7106,
 | |
|       0x98d220bc, 0xefd5102a, 0x71b18589, 0x06b6b51f, 0x9fbfe4a5, 0xe8b8d433,
 | |
|       0x7807c9a2, 0x0f00f934, 0x9609a88e, 0xe10e9818, 0x7f6a0dbb, 0x086d3d2d,
 | |
|       0x91646c97, 0xe6635c01, 0x6b6b51f4, 0x1c6c6162, 0x856530d8, 0xf262004e,
 | |
|       0x6c0695ed, 0x1b01a57b, 0x8208f4c1, 0xf50fc457, 0x65b0d9c6, 0x12b7e950,
 | |
|       0x8bbeb8ea, 0xfcb9887c, 0x62dd1ddf, 0x15da2d49, 0x8cd37cf3, 0xfbd44c65,
 | |
|       0x4db26158, 0x3ab551ce, 0xa3bc0074, 0xd4bb30e2, 0x4adfa541, 0x3dd895d7,
 | |
|       0xa4d1c46d, 0xd3d6f4fb, 0x4369e96a, 0x346ed9fc, 0xad678846, 0xda60b8d0,
 | |
|       0x44042d73, 0x33031de5, 0xaa0a4c5f, 0xdd0d7cc9, 0x5005713c, 0x270241aa,
 | |
|       0xbe0b1010, 0xc90c2086, 0x5768b525, 0x206f85b3, 0xb966d409, 0xce61e49f,
 | |
|       0x5edef90e, 0x29d9c998, 0xb0d09822, 0xc7d7a8b4, 0x59b33d17, 0x2eb40d81,
 | |
|       0xb7bd5c3b, 0xc0ba6cad, 0xedb88320, 0x9abfb3b6, 0x03b6e20c, 0x74b1d29a,
 | |
|       0xead54739, 0x9dd277af, 0x04db2615, 0x73dc1683, 0xe3630b12, 0x94643b84,
 | |
|       0x0d6d6a3e, 0x7a6a5aa8, 0xe40ecf0b, 0x9309ff9d, 0x0a00ae27, 0x7d079eb1,
 | |
|       0xf00f9344, 0x8708a3d2, 0x1e01f268, 0x6906c2fe, 0xf762575d, 0x806567cb,
 | |
|       0x196c3671, 0x6e6b06e7, 0xfed41b76, 0x89d32be0, 0x10da7a5a, 0x67dd4acc,
 | |
|       0xf9b9df6f, 0x8ebeeff9, 0x17b7be43, 0x60b08ed5, 0xd6d6a3e8, 0xa1d1937e,
 | |
|       0x38d8c2c4, 0x4fdff252, 0xd1bb67f1, 0xa6bc5767, 0x3fb506dd, 0x48b2364b,
 | |
|       0xd80d2bda, 0xaf0a1b4c, 0x36034af6, 0x41047a60, 0xdf60efc3, 0xa867df55,
 | |
|       0x316e8eef, 0x4669be79, 0xcb61b38c, 0xbc66831a, 0x256fd2a0, 0x5268e236,
 | |
|       0xcc0c7795, 0xbb0b4703, 0x220216b9, 0x5505262f, 0xc5ba3bbe, 0xb2bd0b28,
 | |
|       0x2bb45a92, 0x5cb36a04, 0xc2d7ffa7, 0xb5d0cf31, 0x2cd99e8b, 0x5bdeae1d,
 | |
|       0x9b64c2b0, 0xec63f226, 0x756aa39c, 0x026d930a, 0x9c0906a9, 0xeb0e363f,
 | |
|       0x72076785, 0x05005713, 0x95bf4a82, 0xe2b87a14, 0x7bb12bae, 0x0cb61b38,
 | |
|       0x92d28e9b, 0xe5d5be0d, 0x7cdcefb7, 0x0bdbdf21, 0x86d3d2d4, 0xf1d4e242,
 | |
|       0x68ddb3f8, 0x1fda836e, 0x81be16cd, 0xf6b9265b, 0x6fb077e1, 0x18b74777,
 | |
|       0x88085ae6, 0xff0f6a70, 0x66063bca, 0x11010b5c, 0x8f659eff, 0xf862ae69,
 | |
|       0x616bffd3, 0x166ccf45, 0xa00ae278, 0xd70dd2ee, 0x4e048354, 0x3903b3c2,
 | |
|       0xa7672661, 0xd06016f7, 0x4969474d, 0x3e6e77db, 0xaed16a4a, 0xd9d65adc,
 | |
|       0x40df0b66, 0x37d83bf0, 0xa9bcae53, 0xdebb9ec5, 0x47b2cf7f, 0x30b5ffe9,
 | |
|       0xbdbdf21c, 0xcabac28a, 0x53b39330, 0x24b4a3a6, 0xbad03605, 0xcdd70693,
 | |
|       0x54de5729, 0x23d967bf, 0xb3667a2e, 0xc4614ab8, 0x5d681b02, 0x2a6f2b94,
 | |
|       0xb40bbe37, 0xc30c8ea1, 0x5a05df1b, 0x2d02ef8d};
 | |
|   const unsigned char *end;
 | |
| 
 | |
|   crc = ~crc;
 | |
|   for (end = buf + len; buf < end; ++buf)
 | |
|     crc = crc32_table[(crc ^ *buf) & 0xff] ^ (crc >> 8);
 | |
|   return ~crc;
 | |
| }
 | |
| 
 | |
| static uint32_t elf_crc32_file(struct backtrace_state *state, int descriptor,
 | |
|                                backtrace_error_callback error_callback,
 | |
|                                void *data) {
 | |
|   struct stat st;
 | |
|   struct backtrace_view file_view;
 | |
|   uint32_t ret;
 | |
| 
 | |
|   if (fstat(descriptor, &st) < 0) {
 | |
|     error_callback(data, "fstat", errno);
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   if (!backtrace_get_view(state, descriptor, 0, st.st_size, error_callback,
 | |
|                           data, &file_view))
 | |
|     return 0;
 | |
| 
 | |
|   ret = elf_crc32(0, (const unsigned char *)file_view.data, st.st_size);
 | |
| 
 | |
|   backtrace_release_view(state, &file_view, error_callback, data);
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| static void elf_nosyms(struct backtrace_state *state ATTRIBUTE_UNUSED,
 | |
|                        uintptr_t addr ATTRIBUTE_UNUSED,
 | |
|                        backtrace_syminfo_callback callback ATTRIBUTE_UNUSED,
 | |
|                        backtrace_error_callback error_callback, void *data) {
 | |
|   error_callback(data, "no symbol table in ELF executable", -1);
 | |
| }
 | |
| 
 | |
| static int elf_nodebug(struct backtrace_state *state, uintptr_t pc,
 | |
|                        backtrace_full_callback callback,
 | |
|                        backtrace_error_callback error_callback, void *data) {
 | |
|   if (state->syminfo_fn != NULL && state->syminfo_fn != elf_nosyms) {
 | |
|     struct backtrace_call_full bdata;
 | |
| 
 | |
|     bdata.full_callback = callback;
 | |
|     bdata.full_error_callback = error_callback;
 | |
|     bdata.full_data = data;
 | |
|     bdata.ret = 0;
 | |
|     state->syminfo_fn(state, pc, backtrace_syminfo_to_full_callback,
 | |
|                       backtrace_syminfo_to_full_error_callback, &bdata);
 | |
|     return bdata.ret;
 | |
|   }
 | |
| 
 | |
|   error_callback(data, "no debug info in ELF executable", -1);
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| static int elf_symbol_compare(const void *v1, const void *v2) {
 | |
|   const struct elf_symbol *e1 = (const struct elf_symbol *)v1;
 | |
|   const struct elf_symbol *e2 = (const struct elf_symbol *)v2;
 | |
| 
 | |
|   if (e1->address < e2->address)
 | |
|     return -1;
 | |
|   else if (e1->address > e2->address)
 | |
|     return 1;
 | |
|   else
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int elf_symbol_search(const void *vkey, const void *ventry) {
 | |
|   const uintptr_t *key = (const uintptr_t *)vkey;
 | |
|   const struct elf_symbol *entry = (const struct elf_symbol *)ventry;
 | |
|   uintptr_t addr;
 | |
| 
 | |
|   addr = *key;
 | |
|   if (addr < entry->address)
 | |
|     return -1;
 | |
|   else if (addr >= entry->address + entry->size)
 | |
|     return 1;
 | |
|   else
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static int elf_initialize_syminfo(
 | |
|     struct backtrace_state *state, uintptr_t base_address,
 | |
|     const unsigned char *symtab_data, size_t symtab_size,
 | |
|     const unsigned char *strtab, size_t strtab_size,
 | |
|     backtrace_error_callback error_callback, void *data,
 | |
|     struct elf_syminfo_data *sdata, struct elf_ppc64_opd_data *opd) {
 | |
|   size_t sym_count;
 | |
|   const b_elf_sym *sym;
 | |
|   size_t elf_symbol_count;
 | |
|   size_t elf_symbol_size;
 | |
|   struct elf_symbol *elf_symbols;
 | |
|   size_t i;
 | |
|   unsigned int j;
 | |
| 
 | |
|   sym_count = symtab_size / sizeof(b_elf_sym);
 | |
| 
 | |
|   sym = (const b_elf_sym *)symtab_data;
 | |
|   elf_symbol_count = 0;
 | |
|   for (i = 0; i < sym_count; ++i, ++sym) {
 | |
|     int info;
 | |
| 
 | |
|     info = sym->st_info & 0xf;
 | |
|     if ((info == STT_FUNC || info == STT_OBJECT) && sym->st_shndx != SHN_UNDEF)
 | |
|       ++elf_symbol_count;
 | |
|   }
 | |
| 
 | |
|   elf_symbol_size = elf_symbol_count * sizeof(struct elf_symbol);
 | |
|   elf_symbols = ((struct elf_symbol *)backtrace_alloc(state, elf_symbol_size,
 | |
|                                                       error_callback, data));
 | |
|   if (elf_symbols == NULL) return 0;
 | |
| 
 | |
|   sym = (const b_elf_sym *)symtab_data;
 | |
|   j = 0;
 | |
|   for (i = 0; i < sym_count; ++i, ++sym) {
 | |
|     int info;
 | |
| 
 | |
|     info = sym->st_info & 0xf;
 | |
|     if (info != STT_FUNC && info != STT_OBJECT) continue;
 | |
|     if (sym->st_shndx == SHN_UNDEF) continue;
 | |
|     if (sym->st_name >= strtab_size) {
 | |
|       error_callback(data, "symbol string index out of range", 0);
 | |
|       backtrace_free(state, elf_symbols, elf_symbol_size, error_callback, data);
 | |
|       return 0;
 | |
|     }
 | |
|     elf_symbols[j].name = (const char *)strtab + sym->st_name;
 | |
| 
 | |
|     if (opd && sym->st_value >= opd->addr &&
 | |
|         sym->st_value < opd->addr + opd->size)
 | |
|       elf_symbols[j].address =
 | |
|           *(const b_elf_addr *)(opd->data + (sym->st_value - opd->addr));
 | |
|     else
 | |
|       elf_symbols[j].address = sym->st_value;
 | |
|     elf_symbols[j].address += base_address;
 | |
|     elf_symbols[j].size = sym->st_size;
 | |
|     ++j;
 | |
|   }
 | |
| 
 | |
|   backtrace_qsort(elf_symbols, elf_symbol_count, sizeof(struct elf_symbol),
 | |
|                   elf_symbol_compare);
 | |
| 
 | |
|   sdata->next = NULL;
 | |
|   sdata->symbols = elf_symbols;
 | |
|   sdata->count = elf_symbol_count;
 | |
| 
 | |
|   return 1;
 | |
| }
 | |
| 
 | |
| static void elf_add_syminfo_data(struct backtrace_state *state,
 | |
|                                  struct elf_syminfo_data *edata) {
 | |
|   if (!state->threaded) {
 | |
|     struct elf_syminfo_data **pp;
 | |
| 
 | |
|     for (pp = (struct elf_syminfo_data **)(void *)&state->syminfo_data;
 | |
|          *pp != NULL; pp = &(*pp)->next)
 | |
|       ;
 | |
|     *pp = edata;
 | |
|   } else {
 | |
|     while (1) {
 | |
|       struct elf_syminfo_data **pp;
 | |
| 
 | |
|       pp = (struct elf_syminfo_data **)(void *)&state->syminfo_data;
 | |
| 
 | |
|       while (1) {
 | |
|         struct elf_syminfo_data *p;
 | |
| 
 | |
|         p = backtrace_atomic_load_pointer(pp);
 | |
| 
 | |
|         if (p == NULL) break;
 | |
| 
 | |
|         pp = &p->next;
 | |
|       }
 | |
| 
 | |
|       if (__sync_bool_compare_and_swap(pp, NULL, edata)) break;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| static void elf_syminfo(
 | |
|     struct backtrace_state *state, uintptr_t addr,
 | |
|     backtrace_syminfo_callback callback,
 | |
|     backtrace_error_callback error_callback ATTRIBUTE_UNUSED, void *data) {
 | |
|   struct elf_syminfo_data *edata;
 | |
|   struct elf_symbol *sym = NULL;
 | |
| 
 | |
|   if (!state->threaded) {
 | |
|     for (edata = (struct elf_syminfo_data *)state->syminfo_data; edata != NULL;
 | |
|          edata = edata->next) {
 | |
|       sym = ((struct elf_symbol *)bsearch(&addr, edata->symbols, edata->count,
 | |
|                                           sizeof(struct elf_symbol),
 | |
|                                           elf_symbol_search));
 | |
|       if (sym != NULL) break;
 | |
|     }
 | |
|   } else {
 | |
|     struct elf_syminfo_data **pp;
 | |
| 
 | |
|     pp = (struct elf_syminfo_data **)(void *)&state->syminfo_data;
 | |
|     while (1) {
 | |
|       edata = backtrace_atomic_load_pointer(pp);
 | |
|       if (edata == NULL) break;
 | |
| 
 | |
|       sym = ((struct elf_symbol *)bsearch(&addr, edata->symbols, edata->count,
 | |
|                                           sizeof(struct elf_symbol),
 | |
|                                           elf_symbol_search));
 | |
|       if (sym != NULL) break;
 | |
| 
 | |
|       pp = &edata->next;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (sym == NULL)
 | |
|     callback(data, addr, NULL, 0, 0);
 | |
|   else
 | |
|     callback(data, addr, sym->name, sym->address, sym->size);
 | |
| }
 | |
| 
 | |
| static int elf_is_symlink(const char *filename) {
 | |
|   struct stat st;
 | |
| 
 | |
|   if (lstat(filename, &st) < 0) return 0;
 | |
|   return S_ISLNK(st.st_mode);
 | |
| }
 | |
| 
 | |
| static char *elf_readlink(struct backtrace_state *state, const char *filename,
 | |
|                           backtrace_error_callback error_callback, void *data,
 | |
|                           size_t *plen) {
 | |
|   size_t len;
 | |
|   char *buf;
 | |
| 
 | |
|   len = 128;
 | |
|   while (1) {
 | |
|     ssize_t rl;
 | |
| 
 | |
|     buf = backtrace_alloc(state, len, error_callback, data);
 | |
|     if (buf == NULL) return NULL;
 | |
|     rl = readlink(filename, buf, len);
 | |
|     if (rl < 0) {
 | |
|       backtrace_free(state, buf, len, error_callback, data);
 | |
|       return NULL;
 | |
|     }
 | |
|     if ((size_t)rl < len - 1) {
 | |
|       buf[rl] = '\0';
 | |
|       *plen = len;
 | |
|       return buf;
 | |
|     }
 | |
|     backtrace_free(state, buf, len, error_callback, data);
 | |
|     len *= 2;
 | |
|   }
 | |
| }
 | |
| 
 | |
| #define SYSTEM_BUILD_ID_DIR "/usr/lib/debug/.build-id/"
 | |
| 
 | |
| static int elf_open_debugfile_by_buildid(
 | |
|     struct backtrace_state *state, const char *buildid_data,
 | |
|     size_t buildid_size, backtrace_error_callback error_callback, void *data) {
 | |
|   const char *const prefix = SYSTEM_BUILD_ID_DIR;
 | |
|   const size_t prefix_len = strlen(prefix);
 | |
|   const char *const suffix = ".debug";
 | |
|   const size_t suffix_len = strlen(suffix);
 | |
|   size_t len;
 | |
|   char *bd_filename;
 | |
|   char *t;
 | |
|   size_t i;
 | |
|   int ret;
 | |
|   int does_not_exist;
 | |
| 
 | |
|   len = prefix_len + buildid_size * 2 + suffix_len + 2;
 | |
|   bd_filename = backtrace_alloc(state, len, error_callback, data);
 | |
|   if (bd_filename == NULL) return -1;
 | |
| 
 | |
|   t = bd_filename;
 | |
|   memcpy(t, prefix, prefix_len);
 | |
|   t += prefix_len;
 | |
|   for (i = 0; i < buildid_size; i++) {
 | |
|     unsigned char b;
 | |
|     unsigned char nib;
 | |
| 
 | |
|     b = (unsigned char)buildid_data[i];
 | |
|     nib = (b & 0xf0) >> 4;
 | |
|     *t++ = nib < 10 ? '0' + nib : 'a' + nib - 10;
 | |
|     nib = b & 0x0f;
 | |
|     *t++ = nib < 10 ? '0' + nib : 'a' + nib - 10;
 | |
|     if (i == 0) *t++ = '/';
 | |
|   }
 | |
|   memcpy(t, suffix, suffix_len);
 | |
|   t[suffix_len] = '\0';
 | |
| 
 | |
|   ret = backtrace_open(bd_filename, error_callback, data, &does_not_exist);
 | |
| 
 | |
|   backtrace_free(state, bd_filename, len, error_callback, data);
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| static int elf_try_debugfile(struct backtrace_state *state, const char *prefix,
 | |
|                              size_t prefix_len, const char *prefix2,
 | |
|                              size_t prefix2_len, const char *debuglink_name,
 | |
|                              backtrace_error_callback error_callback,
 | |
|                              void *data) {
 | |
|   size_t debuglink_len;
 | |
|   size_t try_len;
 | |
|   char *try;
 | |
|   int does_not_exist;
 | |
|   int ret;
 | |
| 
 | |
|   debuglink_len = strlen(debuglink_name);
 | |
|   try_len = prefix_len + prefix2_len + debuglink_len + 1;
 | |
|   try = backtrace_alloc(state, try_len, error_callback, data);
 | |
|   if (try == NULL) return -1;
 | |
| 
 | |
|   memcpy(try, prefix, prefix_len);
 | |
|   memcpy(try + prefix_len, prefix2, prefix2_len);
 | |
|   memcpy(try + prefix_len + prefix2_len, debuglink_name, debuglink_len);
 | |
|   try[prefix_len + prefix2_len + debuglink_len] = '\0';
 | |
| 
 | |
|   ret = backtrace_open(try, error_callback, data, &does_not_exist);
 | |
| 
 | |
|   backtrace_free(state, try, try_len, error_callback, data);
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| static int elf_find_debugfile_by_debuglink(
 | |
|     struct backtrace_state *state, const char *filename,
 | |
|     const char *debuglink_name, backtrace_error_callback error_callback,
 | |
|     void *data) {
 | |
|   int ret;
 | |
|   char *alc;
 | |
|   size_t alc_len;
 | |
|   const char *slash;
 | |
|   int ddescriptor;
 | |
|   const char *prefix;
 | |
|   size_t prefix_len;
 | |
| 
 | |
|   ret = -1;
 | |
|   alc = NULL;
 | |
|   alc_len = 0;
 | |
|   while (elf_is_symlink(filename)) {
 | |
|     char *new_buf;
 | |
|     size_t new_len;
 | |
| 
 | |
|     new_buf = elf_readlink(state, filename, error_callback, data, &new_len);
 | |
|     if (new_buf == NULL) break;
 | |
| 
 | |
|     if (new_buf[0] == '/')
 | |
|       filename = new_buf;
 | |
|     else {
 | |
|       slash = strrchr(filename, '/');
 | |
|       if (slash == NULL)
 | |
|         filename = new_buf;
 | |
|       else {
 | |
|         size_t clen;
 | |
|         char *c;
 | |
| 
 | |
|         slash++;
 | |
|         clen = slash - filename + strlen(new_buf) + 1;
 | |
|         c = backtrace_alloc(state, clen, error_callback, data);
 | |
|         if (c == NULL) goto done;
 | |
| 
 | |
|         memcpy(c, filename, slash - filename);
 | |
|         memcpy(c + (slash - filename), new_buf, strlen(new_buf));
 | |
|         c[slash - filename + strlen(new_buf)] = '\0';
 | |
|         backtrace_free(state, new_buf, new_len, error_callback, data);
 | |
|         filename = c;
 | |
|         new_buf = c;
 | |
|         new_len = clen;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (alc != NULL) backtrace_free(state, alc, alc_len, error_callback, data);
 | |
|     alc = new_buf;
 | |
|     alc_len = new_len;
 | |
|   }
 | |
| 
 | |
|   slash = strrchr(filename, '/');
 | |
|   if (slash == NULL) {
 | |
|     prefix = "";
 | |
|     prefix_len = 0;
 | |
|   } else {
 | |
|     slash++;
 | |
|     prefix = filename;
 | |
|     prefix_len = slash - filename;
 | |
|   }
 | |
| 
 | |
|   ddescriptor = elf_try_debugfile(state, prefix, prefix_len, "", 0,
 | |
|                                   debuglink_name, error_callback, data);
 | |
|   if (ddescriptor >= 0) {
 | |
|     ret = ddescriptor;
 | |
|     goto done;
 | |
|   }
 | |
| 
 | |
|   ddescriptor =
 | |
|       elf_try_debugfile(state, prefix, prefix_len, ".debug/", strlen(".debug/"),
 | |
|                         debuglink_name, error_callback, data);
 | |
|   if (ddescriptor >= 0) {
 | |
|     ret = ddescriptor;
 | |
|     goto done;
 | |
|   }
 | |
| 
 | |
|   ddescriptor = elf_try_debugfile(state, "/usr/lib/debug/",
 | |
|                                   strlen("/usr/lib/debug/"), prefix, prefix_len,
 | |
|                                   debuglink_name, error_callback, data);
 | |
|   if (ddescriptor >= 0) ret = ddescriptor;
 | |
| 
 | |
| done:
 | |
|   if (alc != NULL && alc_len > 0)
 | |
|     backtrace_free(state, alc, alc_len, error_callback, data);
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| static int elf_open_debugfile_by_debuglink(
 | |
|     struct backtrace_state *state, const char *filename,
 | |
|     const char *debuglink_name, uint32_t debuglink_crc,
 | |
|     backtrace_error_callback error_callback, void *data) {
 | |
|   int ddescriptor;
 | |
| 
 | |
|   ddescriptor = elf_find_debugfile_by_debuglink(state, filename, debuglink_name,
 | |
|                                                 error_callback, data);
 | |
|   if (ddescriptor < 0) return -1;
 | |
| 
 | |
|   if (debuglink_crc != 0) {
 | |
|     uint32_t got_crc;
 | |
| 
 | |
|     got_crc = elf_crc32_file(state, ddescriptor, error_callback, data);
 | |
|     if (got_crc != debuglink_crc) {
 | |
|       backtrace_close(ddescriptor, error_callback, data);
 | |
|       return -1;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return ddescriptor;
 | |
| }
 | |
| 
 | |
| static void elf_uncompress_failed(void) {}
 | |
| 
 | |
| static int elf_zlib_fetch(const unsigned char **ppin,
 | |
|                           const unsigned char *pinend, uint64_t *pval,
 | |
|                           unsigned int *pbits) {
 | |
|   unsigned int bits;
 | |
|   const unsigned char *pin;
 | |
|   uint64_t val;
 | |
|   uint32_t next;
 | |
| 
 | |
|   bits = *pbits;
 | |
|   if (bits >= 15) return 1;
 | |
|   pin = *ppin;
 | |
|   val = *pval;
 | |
| 
 | |
|   if (unlikely(pinend - pin < 4)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
| #if defined(__BYTE_ORDER__) && defined(__ORDER_LITTLE_ENDIAN__) && \
 | |
|     defined(__ORDER_BIG_ENDIAN__) &&                               \
 | |
|     (__BYTE_ORDER__ == __ORDER_BIG_ENDIAN__ ||                     \
 | |
|      __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__)
 | |
| 
 | |
|   next = *(const uint32_t *)pin;
 | |
| 
 | |
| #if __BYTE_ORDER__ == __ORDER_BIG_ENDIAN__
 | |
|   next = __builtin_bswap32(next);
 | |
| #endif
 | |
| #else
 | |
|   next = pin[0] | (pin[1] << 8) | (pin[2] << 16) | (pin[3] << 24);
 | |
| #endif
 | |
| 
 | |
|   val |= (uint64_t)next << bits;
 | |
|   bits += 32;
 | |
|   pin += 4;
 | |
| 
 | |
|   __builtin_prefetch(pin, 0, 0);
 | |
| 
 | |
|   *ppin = pin;
 | |
|   *pval = val;
 | |
|   *pbits = bits;
 | |
|   return 1;
 | |
| }
 | |
| 
 | |
| #define HUFFMAN_TABLE_SIZE (1024)
 | |
| 
 | |
| #define HUFFMAN_VALUE_MASK 0x01ff
 | |
| #define HUFFMAN_BITS_SHIFT 9
 | |
| #define HUFFMAN_BITS_MASK 0x7
 | |
| #define HUFFMAN_SECONDARY_SHIFT 12
 | |
| 
 | |
| #define ZDEBUG_TABLE_SIZE                                                      \
 | |
|   (2 * HUFFMAN_TABLE_SIZE * sizeof(uint16_t) + (286 + 30) * sizeof(uint16_t) + \
 | |
|    (286 + 30) * sizeof(unsigned char))
 | |
| 
 | |
| #define ZDEBUG_TABLE_CODELEN_OFFSET \
 | |
|   (2 * HUFFMAN_TABLE_SIZE * sizeof(uint16_t) + (286 + 30) * sizeof(uint16_t))
 | |
| 
 | |
| #define ZDEBUG_TABLE_WORK_OFFSET (2 * HUFFMAN_TABLE_SIZE * sizeof(uint16_t))
 | |
| 
 | |
| #ifdef BACKTRACE_GENERATE_FIXED_HUFFMAN_TABLE
 | |
| 
 | |
| static size_t final_next_secondary;
 | |
| 
 | |
| #endif
 | |
| 
 | |
| static int elf_zlib_inflate_table(unsigned char *codes, size_t codes_len,
 | |
|                                   uint16_t *zdebug_table, uint16_t *table) {
 | |
|   uint16_t count[16];
 | |
|   uint16_t start[16];
 | |
|   uint16_t prev[16];
 | |
|   uint16_t firstcode[7];
 | |
|   uint16_t *next;
 | |
|   size_t i;
 | |
|   size_t j;
 | |
|   unsigned int code;
 | |
|   size_t next_secondary;
 | |
| 
 | |
|   next =
 | |
|       (uint16_t *)(((unsigned char *)zdebug_table) + ZDEBUG_TABLE_WORK_OFFSET);
 | |
| 
 | |
|   memset(&count[0], 0, 16 * sizeof(uint16_t));
 | |
|   for (i = 0; i < codes_len; ++i) {
 | |
|     if (unlikely(codes[i] >= 16)) {
 | |
|       elf_uncompress_failed();
 | |
|       return 0;
 | |
|     }
 | |
| 
 | |
|     if (count[codes[i]] == 0) {
 | |
|       start[codes[i]] = i;
 | |
|       prev[codes[i]] = i;
 | |
|     } else {
 | |
|       next[prev[codes[i]]] = i;
 | |
|       prev[codes[i]] = i;
 | |
|     }
 | |
| 
 | |
|     ++count[codes[i]];
 | |
|   }
 | |
| 
 | |
|   memset(table, 0, HUFFMAN_TABLE_SIZE * sizeof(uint16_t));
 | |
| 
 | |
|   code = 0;
 | |
|   for (j = 1; j <= 8; ++j) {
 | |
|     unsigned int jcnt;
 | |
|     unsigned int val;
 | |
| 
 | |
|     jcnt = count[j];
 | |
|     if (jcnt == 0) continue;
 | |
| 
 | |
|     if (unlikely(jcnt > (1U << j))) {
 | |
|       elf_uncompress_failed();
 | |
|       return 0;
 | |
|     }
 | |
| 
 | |
|     val = start[j];
 | |
|     for (i = 0; i < jcnt; ++i) {
 | |
|       uint16_t tval;
 | |
|       size_t ind;
 | |
|       unsigned int incr;
 | |
| 
 | |
|       if (unlikely((val & ~HUFFMAN_VALUE_MASK) != 0)) {
 | |
|         elf_uncompress_failed();
 | |
|         return 0;
 | |
|       }
 | |
| 
 | |
|       tval = val | ((j - 1) << HUFFMAN_BITS_SHIFT);
 | |
| 
 | |
|       for (ind = code; ind < 0x100; ind += 1 << j) {
 | |
|         if (unlikely(table[ind] != 0)) {
 | |
|           elf_uncompress_failed();
 | |
|           return 0;
 | |
|         }
 | |
|         table[ind] = tval;
 | |
|       }
 | |
| 
 | |
|       if (i + 1 < jcnt) val = next[val];
 | |
| 
 | |
|       incr = 1U << (j - 1);
 | |
|       while ((code & incr) != 0) incr >>= 1;
 | |
|       if (incr == 0)
 | |
|         code = 0;
 | |
|       else {
 | |
|         code &= incr - 1;
 | |
|         code += incr;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   for (j = 9; j < 16; j++) {
 | |
|     unsigned int jcnt;
 | |
|     unsigned int k;
 | |
| 
 | |
|     jcnt = count[j];
 | |
|     if (jcnt == 0) continue;
 | |
| 
 | |
|     firstcode[j - 9] = code;
 | |
| 
 | |
|     for (k = 0; k < j; ++k) {
 | |
|       if ((jcnt & (1U << k)) != 0) {
 | |
|         unsigned int m;
 | |
|         unsigned int bit;
 | |
| 
 | |
|         bit = 1U << (j - k - 1);
 | |
|         for (m = 0; m < j - k; ++m, bit >>= 1) {
 | |
|           if ((code & bit) == 0) {
 | |
|             code += bit;
 | |
|             break;
 | |
|           }
 | |
|           code &= ~bit;
 | |
|         }
 | |
|         jcnt &= ~(1U << k);
 | |
|       }
 | |
|     }
 | |
|     if (unlikely(jcnt != 0)) {
 | |
|       elf_uncompress_failed();
 | |
|       return 0;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   next_secondary = 0;
 | |
|   for (j = 15; j >= 9; j--) {
 | |
|     unsigned int jcnt;
 | |
|     unsigned int val;
 | |
|     size_t primary;
 | |
|     size_t secondary;
 | |
|     size_t secondary_bits;
 | |
|     jcnt = count[j];
 | |
|     if (jcnt == 0) continue;
 | |
| 
 | |
|     val = start[j];
 | |
|     code = firstcode[j - 9];
 | |
|     primary = 0x100;
 | |
|     secondary = 0;
 | |
|     secondary_bits = 0;
 | |
|     for (i = 0; i < jcnt; ++i) {
 | |
|       uint16_t tval;
 | |
|       size_t ind;
 | |
|       unsigned int incr;
 | |
| 
 | |
|       if ((code & 0xff) != primary) {
 | |
|         uint16_t tprimary;
 | |
| 
 | |
|         primary = code & 0xff;
 | |
| 
 | |
|         tprimary = table[primary];
 | |
|         if (tprimary == 0) {
 | |
|           if (unlikely((next_secondary & HUFFMAN_VALUE_MASK) !=
 | |
|                        next_secondary)) {
 | |
|             elf_uncompress_failed();
 | |
|             return 0;
 | |
|           }
 | |
| 
 | |
|           secondary = next_secondary;
 | |
|           secondary_bits = j - 8;
 | |
|           next_secondary += 1 << secondary_bits;
 | |
|           table[primary] = (secondary + ((j - 8) << HUFFMAN_BITS_SHIFT) +
 | |
|                             (1U << HUFFMAN_SECONDARY_SHIFT));
 | |
|         } else {
 | |
|           if (unlikely((tprimary & (1U << HUFFMAN_SECONDARY_SHIFT)) == 0)) {
 | |
|             elf_uncompress_failed();
 | |
|             return 0;
 | |
|           }
 | |
|           secondary = tprimary & HUFFMAN_VALUE_MASK;
 | |
|           secondary_bits =
 | |
|               ((tprimary >> HUFFMAN_BITS_SHIFT) & HUFFMAN_BITS_MASK);
 | |
|           if (unlikely(secondary_bits < j - 8)) {
 | |
|             elf_uncompress_failed();
 | |
|             return 0;
 | |
|           }
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       tval = val | ((j - 8) << HUFFMAN_BITS_SHIFT);
 | |
| 
 | |
|       for (ind = code >> 8; ind < (1U << secondary_bits);
 | |
|            ind += 1U << (j - 8)) {
 | |
|         if (unlikely(table[secondary + 0x100 + ind] != 0)) {
 | |
|           elf_uncompress_failed();
 | |
|           return 0;
 | |
|         }
 | |
|         table[secondary + 0x100 + ind] = tval;
 | |
|       }
 | |
| 
 | |
|       if (i + 1 < jcnt) val = next[val];
 | |
| 
 | |
|       incr = 1U << (j - 1);
 | |
|       while ((code & incr) != 0) incr >>= 1;
 | |
|       if (incr == 0)
 | |
|         code = 0;
 | |
|       else {
 | |
|         code &= incr - 1;
 | |
|         code += incr;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
| #ifdef BACKTRACE_GENERATE_FIXED_HUFFMAN_TABLE
 | |
|   final_next_secondary = next_secondary;
 | |
| #endif
 | |
| 
 | |
|   return 1;
 | |
| }
 | |
| 
 | |
| static const uint16_t elf_zlib_default_table[0x170] = {
 | |
|     0xd00, 0xe50,  0xe10, 0xf18,  0xd10, 0xe70,  0xe30, 0x1230, 0xd08, 0xe60,
 | |
|     0xe20, 0x1210, 0xe00, 0xe80,  0xe40, 0x1250, 0xd04, 0xe58,  0xe18, 0x1200,
 | |
|     0xd14, 0xe78,  0xe38, 0x1240, 0xd0c, 0xe68,  0xe28, 0x1220, 0xe08, 0xe88,
 | |
|     0xe48, 0x1260, 0xd02, 0xe54,  0xe14, 0xf1c,  0xd12, 0xe74,  0xe34, 0x1238,
 | |
|     0xd0a, 0xe64,  0xe24, 0x1218, 0xe04, 0xe84,  0xe44, 0x1258, 0xd06, 0xe5c,
 | |
|     0xe1c, 0x1208, 0xd16, 0xe7c,  0xe3c, 0x1248, 0xd0e, 0xe6c,  0xe2c, 0x1228,
 | |
|     0xe0c, 0xe8c,  0xe4c, 0x1268, 0xd01, 0xe52,  0xe12, 0xf1a,  0xd11, 0xe72,
 | |
|     0xe32, 0x1234, 0xd09, 0xe62,  0xe22, 0x1214, 0xe02, 0xe82,  0xe42, 0x1254,
 | |
|     0xd05, 0xe5a,  0xe1a, 0x1204, 0xd15, 0xe7a,  0xe3a, 0x1244, 0xd0d, 0xe6a,
 | |
|     0xe2a, 0x1224, 0xe0a, 0xe8a,  0xe4a, 0x1264, 0xd03, 0xe56,  0xe16, 0xf1e,
 | |
|     0xd13, 0xe76,  0xe36, 0x123c, 0xd0b, 0xe66,  0xe26, 0x121c, 0xe06, 0xe86,
 | |
|     0xe46, 0x125c, 0xd07, 0xe5e,  0xe1e, 0x120c, 0xd17, 0xe7e,  0xe3e, 0x124c,
 | |
|     0xd0f, 0xe6e,  0xe2e, 0x122c, 0xe0e, 0xe8e,  0xe4e, 0x126c, 0xd00, 0xe51,
 | |
|     0xe11, 0xf19,  0xd10, 0xe71,  0xe31, 0x1232, 0xd08, 0xe61,  0xe21, 0x1212,
 | |
|     0xe01, 0xe81,  0xe41, 0x1252, 0xd04, 0xe59,  0xe19, 0x1202, 0xd14, 0xe79,
 | |
|     0xe39, 0x1242, 0xd0c, 0xe69,  0xe29, 0x1222, 0xe09, 0xe89,  0xe49, 0x1262,
 | |
|     0xd02, 0xe55,  0xe15, 0xf1d,  0xd12, 0xe75,  0xe35, 0x123a, 0xd0a, 0xe65,
 | |
|     0xe25, 0x121a, 0xe05, 0xe85,  0xe45, 0x125a, 0xd06, 0xe5d,  0xe1d, 0x120a,
 | |
|     0xd16, 0xe7d,  0xe3d, 0x124a, 0xd0e, 0xe6d,  0xe2d, 0x122a, 0xe0d, 0xe8d,
 | |
|     0xe4d, 0x126a, 0xd01, 0xe53,  0xe13, 0xf1b,  0xd11, 0xe73,  0xe33, 0x1236,
 | |
|     0xd09, 0xe63,  0xe23, 0x1216, 0xe03, 0xe83,  0xe43, 0x1256, 0xd05, 0xe5b,
 | |
|     0xe1b, 0x1206, 0xd15, 0xe7b,  0xe3b, 0x1246, 0xd0d, 0xe6b,  0xe2b, 0x1226,
 | |
|     0xe0b, 0xe8b,  0xe4b, 0x1266, 0xd03, 0xe57,  0xe17, 0xf1f,  0xd13, 0xe77,
 | |
|     0xe37, 0x123e, 0xd0b, 0xe67,  0xe27, 0x121e, 0xe07, 0xe87,  0xe47, 0x125e,
 | |
|     0xd07, 0xe5f,  0xe1f, 0x120e, 0xd17, 0xe7f,  0xe3f, 0x124e, 0xd0f, 0xe6f,
 | |
|     0xe2f, 0x122e, 0xe0f, 0xe8f,  0xe4f, 0x126e, 0x290, 0x291,  0x292, 0x293,
 | |
|     0x294, 0x295,  0x296, 0x297,  0x298, 0x299,  0x29a, 0x29b,  0x29c, 0x29d,
 | |
|     0x29e, 0x29f,  0x2a0, 0x2a1,  0x2a2, 0x2a3,  0x2a4, 0x2a5,  0x2a6, 0x2a7,
 | |
|     0x2a8, 0x2a9,  0x2aa, 0x2ab,  0x2ac, 0x2ad,  0x2ae, 0x2af,  0x2b0, 0x2b1,
 | |
|     0x2b2, 0x2b3,  0x2b4, 0x2b5,  0x2b6, 0x2b7,  0x2b8, 0x2b9,  0x2ba, 0x2bb,
 | |
|     0x2bc, 0x2bd,  0x2be, 0x2bf,  0x2c0, 0x2c1,  0x2c2, 0x2c3,  0x2c4, 0x2c5,
 | |
|     0x2c6, 0x2c7,  0x2c8, 0x2c9,  0x2ca, 0x2cb,  0x2cc, 0x2cd,  0x2ce, 0x2cf,
 | |
|     0x2d0, 0x2d1,  0x2d2, 0x2d3,  0x2d4, 0x2d5,  0x2d6, 0x2d7,  0x2d8, 0x2d9,
 | |
|     0x2da, 0x2db,  0x2dc, 0x2dd,  0x2de, 0x2df,  0x2e0, 0x2e1,  0x2e2, 0x2e3,
 | |
|     0x2e4, 0x2e5,  0x2e6, 0x2e7,  0x2e8, 0x2e9,  0x2ea, 0x2eb,  0x2ec, 0x2ed,
 | |
|     0x2ee, 0x2ef,  0x2f0, 0x2f1,  0x2f2, 0x2f3,  0x2f4, 0x2f5,  0x2f6, 0x2f7,
 | |
|     0x2f8, 0x2f9,  0x2fa, 0x2fb,  0x2fc, 0x2fd,  0x2fe, 0x2ff,
 | |
| };
 | |
| 
 | |
| static const uint16_t elf_zlib_default_dist_table[0x100] = {
 | |
|     0x800, 0x810, 0x808, 0x818, 0x804, 0x814, 0x80c, 0x81c, 0x802, 0x812, 0x80a,
 | |
|     0x81a, 0x806, 0x816, 0x80e, 0x81e, 0x801, 0x811, 0x809, 0x819, 0x805, 0x815,
 | |
|     0x80d, 0x81d, 0x803, 0x813, 0x80b, 0x81b, 0x807, 0x817, 0x80f, 0x81f, 0x800,
 | |
|     0x810, 0x808, 0x818, 0x804, 0x814, 0x80c, 0x81c, 0x802, 0x812, 0x80a, 0x81a,
 | |
|     0x806, 0x816, 0x80e, 0x81e, 0x801, 0x811, 0x809, 0x819, 0x805, 0x815, 0x80d,
 | |
|     0x81d, 0x803, 0x813, 0x80b, 0x81b, 0x807, 0x817, 0x80f, 0x81f, 0x800, 0x810,
 | |
|     0x808, 0x818, 0x804, 0x814, 0x80c, 0x81c, 0x802, 0x812, 0x80a, 0x81a, 0x806,
 | |
|     0x816, 0x80e, 0x81e, 0x801, 0x811, 0x809, 0x819, 0x805, 0x815, 0x80d, 0x81d,
 | |
|     0x803, 0x813, 0x80b, 0x81b, 0x807, 0x817, 0x80f, 0x81f, 0x800, 0x810, 0x808,
 | |
|     0x818, 0x804, 0x814, 0x80c, 0x81c, 0x802, 0x812, 0x80a, 0x81a, 0x806, 0x816,
 | |
|     0x80e, 0x81e, 0x801, 0x811, 0x809, 0x819, 0x805, 0x815, 0x80d, 0x81d, 0x803,
 | |
|     0x813, 0x80b, 0x81b, 0x807, 0x817, 0x80f, 0x81f, 0x800, 0x810, 0x808, 0x818,
 | |
|     0x804, 0x814, 0x80c, 0x81c, 0x802, 0x812, 0x80a, 0x81a, 0x806, 0x816, 0x80e,
 | |
|     0x81e, 0x801, 0x811, 0x809, 0x819, 0x805, 0x815, 0x80d, 0x81d, 0x803, 0x813,
 | |
|     0x80b, 0x81b, 0x807, 0x817, 0x80f, 0x81f, 0x800, 0x810, 0x808, 0x818, 0x804,
 | |
|     0x814, 0x80c, 0x81c, 0x802, 0x812, 0x80a, 0x81a, 0x806, 0x816, 0x80e, 0x81e,
 | |
|     0x801, 0x811, 0x809, 0x819, 0x805, 0x815, 0x80d, 0x81d, 0x803, 0x813, 0x80b,
 | |
|     0x81b, 0x807, 0x817, 0x80f, 0x81f, 0x800, 0x810, 0x808, 0x818, 0x804, 0x814,
 | |
|     0x80c, 0x81c, 0x802, 0x812, 0x80a, 0x81a, 0x806, 0x816, 0x80e, 0x81e, 0x801,
 | |
|     0x811, 0x809, 0x819, 0x805, 0x815, 0x80d, 0x81d, 0x803, 0x813, 0x80b, 0x81b,
 | |
|     0x807, 0x817, 0x80f, 0x81f, 0x800, 0x810, 0x808, 0x818, 0x804, 0x814, 0x80c,
 | |
|     0x81c, 0x802, 0x812, 0x80a, 0x81a, 0x806, 0x816, 0x80e, 0x81e, 0x801, 0x811,
 | |
|     0x809, 0x819, 0x805, 0x815, 0x80d, 0x81d, 0x803, 0x813, 0x80b, 0x81b, 0x807,
 | |
|     0x817, 0x80f, 0x81f,
 | |
| };
 | |
| 
 | |
| static int elf_zlib_inflate(const unsigned char *pin, size_t sin,
 | |
|                             uint16_t *zdebug_table, unsigned char *pout,
 | |
|                             size_t sout) {
 | |
|   unsigned char *porigout;
 | |
|   const unsigned char *pinend;
 | |
|   unsigned char *poutend;
 | |
| 
 | |
|   porigout = pout;
 | |
|   pinend = pin + sin;
 | |
|   poutend = pout + sout;
 | |
|   while ((pinend - pin) > 4) {
 | |
|     uint64_t val;
 | |
|     unsigned int bits;
 | |
|     int last;
 | |
| 
 | |
|     if (unlikely((pin[0] & 0xf) != 8)) {
 | |
|       elf_uncompress_failed();
 | |
|       return 0;
 | |
|     }
 | |
|     if (unlikely((pin[0] >> 4) > 7)) {
 | |
|       elf_uncompress_failed();
 | |
|       return 0;
 | |
|     }
 | |
|     if (unlikely((pin[1] & 0x20) != 0)) {
 | |
|       elf_uncompress_failed();
 | |
|       return 0;
 | |
|     }
 | |
|     val = (pin[0] << 8) | pin[1];
 | |
|     if (unlikely(val % 31 != 0)) {
 | |
|       elf_uncompress_failed();
 | |
|       return 0;
 | |
|     }
 | |
|     pin += 2;
 | |
| 
 | |
|     val = 0;
 | |
|     bits = 0;
 | |
|     while ((((uintptr_t)pin) & 3) != 0) {
 | |
|       val |= (uint64_t)*pin << bits;
 | |
|       bits += 8;
 | |
|       ++pin;
 | |
|     }
 | |
| 
 | |
|     last = 0;
 | |
| 
 | |
|     while (!last) {
 | |
|       unsigned int type;
 | |
|       const uint16_t *tlit;
 | |
|       const uint16_t *tdist;
 | |
| 
 | |
|       if (!elf_zlib_fetch(&pin, pinend, &val, &bits)) return 0;
 | |
| 
 | |
|       last = val & 1;
 | |
|       type = (val >> 1) & 3;
 | |
|       val >>= 3;
 | |
|       bits -= 3;
 | |
| 
 | |
|       if (unlikely(type == 3)) {
 | |
|         elf_uncompress_failed();
 | |
|         return 0;
 | |
|       }
 | |
| 
 | |
|       if (type == 0) {
 | |
|         uint16_t len;
 | |
|         uint16_t lenc;
 | |
| 
 | |
|         while (bits >= 8) {
 | |
|           --pin;
 | |
|           bits -= 8;
 | |
|         }
 | |
| 
 | |
|         val = 0;
 | |
|         bits = 0;
 | |
|         if (unlikely((pinend - pin) < 4)) {
 | |
|           elf_uncompress_failed();
 | |
|           return 0;
 | |
|         }
 | |
|         len = pin[0] | (pin[1] << 8);
 | |
|         lenc = pin[2] | (pin[3] << 8);
 | |
|         pin += 4;
 | |
|         lenc = ~lenc;
 | |
|         if (unlikely(len != lenc)) {
 | |
|           elf_uncompress_failed();
 | |
|           return 0;
 | |
|         }
 | |
|         if (unlikely(len > (unsigned int)(pinend - pin) ||
 | |
|                      len > (unsigned int)(poutend - pout))) {
 | |
|           elf_uncompress_failed();
 | |
|           return 0;
 | |
|         }
 | |
|         memcpy(pout, pin, len);
 | |
|         pout += len;
 | |
|         pin += len;
 | |
| 
 | |
|         while ((((uintptr_t)pin) & 3) != 0) {
 | |
|           val |= (uint64_t)*pin << bits;
 | |
|           bits += 8;
 | |
|           ++pin;
 | |
|         }
 | |
| 
 | |
|         continue;
 | |
|       }
 | |
| 
 | |
|       if (type == 1) {
 | |
|         tlit = elf_zlib_default_table;
 | |
|         tdist = elf_zlib_default_dist_table;
 | |
|       } else {
 | |
|         unsigned int nlit;
 | |
|         unsigned int ndist;
 | |
|         unsigned int nclen;
 | |
|         unsigned char codebits[19];
 | |
|         unsigned char *plenbase;
 | |
|         unsigned char *plen;
 | |
|         unsigned char *plenend;
 | |
| 
 | |
|         if (!elf_zlib_fetch(&pin, pinend, &val, &bits)) return 0;
 | |
| 
 | |
|         nlit = (val & 0x1f) + 257;
 | |
|         val >>= 5;
 | |
|         ndist = (val & 0x1f) + 1;
 | |
|         val >>= 5;
 | |
|         nclen = (val & 0xf) + 4;
 | |
|         val >>= 4;
 | |
|         bits -= 14;
 | |
|         if (unlikely(nlit > 286 || ndist > 30)) {
 | |
|           elf_uncompress_failed();
 | |
|           return 0;
 | |
|         }
 | |
| 
 | |
|         memset(&codebits[0], 0, 19);
 | |
| 
 | |
|         if (!elf_zlib_fetch(&pin, pinend, &val, &bits)) return 0;
 | |
| 
 | |
|         codebits[16] = val & 7;
 | |
|         codebits[17] = (val >> 3) & 7;
 | |
|         codebits[18] = (val >> 6) & 7;
 | |
|         codebits[0] = (val >> 9) & 7;
 | |
|         val >>= 12;
 | |
|         bits -= 12;
 | |
| 
 | |
|         if (nclen == 4) goto codebitsdone;
 | |
| 
 | |
|         codebits[8] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 5) goto codebitsdone;
 | |
| 
 | |
|         if (!elf_zlib_fetch(&pin, pinend, &val, &bits)) return 0;
 | |
| 
 | |
|         codebits[7] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 6) goto codebitsdone;
 | |
| 
 | |
|         codebits[9] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 7) goto codebitsdone;
 | |
| 
 | |
|         codebits[6] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 8) goto codebitsdone;
 | |
| 
 | |
|         codebits[10] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 9) goto codebitsdone;
 | |
| 
 | |
|         codebits[5] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 10) goto codebitsdone;
 | |
| 
 | |
|         if (!elf_zlib_fetch(&pin, pinend, &val, &bits)) return 0;
 | |
| 
 | |
|         codebits[11] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 11) goto codebitsdone;
 | |
| 
 | |
|         codebits[4] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 12) goto codebitsdone;
 | |
| 
 | |
|         codebits[12] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 13) goto codebitsdone;
 | |
| 
 | |
|         codebits[3] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 14) goto codebitsdone;
 | |
| 
 | |
|         codebits[13] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 15) goto codebitsdone;
 | |
| 
 | |
|         if (!elf_zlib_fetch(&pin, pinend, &val, &bits)) return 0;
 | |
| 
 | |
|         codebits[2] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 16) goto codebitsdone;
 | |
| 
 | |
|         codebits[14] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 17) goto codebitsdone;
 | |
| 
 | |
|         codebits[1] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|         if (nclen == 18) goto codebitsdone;
 | |
| 
 | |
|         codebits[15] = val & 7;
 | |
|         val >>= 3;
 | |
|         bits -= 3;
 | |
| 
 | |
|       codebitsdone:
 | |
| 
 | |
|         if (!elf_zlib_inflate_table(codebits, 19, zdebug_table, zdebug_table))
 | |
|           return 0;
 | |
| 
 | |
|         plenbase =
 | |
|             (((unsigned char *)zdebug_table) + ZDEBUG_TABLE_CODELEN_OFFSET);
 | |
|         plen = plenbase;
 | |
|         plenend = plen + nlit + ndist;
 | |
|         while (plen < plenend) {
 | |
|           uint16_t t;
 | |
|           unsigned int b;
 | |
|           uint16_t v;
 | |
| 
 | |
|           if (!elf_zlib_fetch(&pin, pinend, &val, &bits)) return 0;
 | |
| 
 | |
|           t = zdebug_table[val & 0xff];
 | |
| 
 | |
|           if (unlikely((t & (1U << HUFFMAN_SECONDARY_SHIFT)) != 0)) {
 | |
|             elf_uncompress_failed();
 | |
|             return 0;
 | |
|           }
 | |
| 
 | |
|           b = (t >> HUFFMAN_BITS_SHIFT) & HUFFMAN_BITS_MASK;
 | |
|           val >>= b + 1;
 | |
|           bits -= b + 1;
 | |
| 
 | |
|           v = t & HUFFMAN_VALUE_MASK;
 | |
|           if (v < 16)
 | |
|             *plen++ = v;
 | |
|           else if (v == 16) {
 | |
|             unsigned int c;
 | |
|             unsigned int prev;
 | |
| 
 | |
|             if (unlikely(plen == plenbase)) {
 | |
|               elf_uncompress_failed();
 | |
|               return 0;
 | |
|             }
 | |
| 
 | |
|             c = 3 + (val & 0x3);
 | |
|             val >>= 2;
 | |
|             bits -= 2;
 | |
|             if (unlikely((unsigned int)(plenend - plen) < c)) {
 | |
|               elf_uncompress_failed();
 | |
|               return 0;
 | |
|             }
 | |
| 
 | |
|             prev = plen[-1];
 | |
|             switch (c) {
 | |
|               case 6:
 | |
|                 *plen++ = prev;
 | |
|                 ATTRIBUTE_FALLTHROUGH;
 | |
|               case 5:
 | |
|                 *plen++ = prev;
 | |
|                 ATTRIBUTE_FALLTHROUGH;
 | |
|               case 4:
 | |
|                 *plen++ = prev;
 | |
|             }
 | |
|             *plen++ = prev;
 | |
|             *plen++ = prev;
 | |
|             *plen++ = prev;
 | |
|           } else if (v == 17) {
 | |
|             unsigned int c;
 | |
| 
 | |
|             c = 3 + (val & 0x7);
 | |
|             val >>= 3;
 | |
|             bits -= 3;
 | |
|             if (unlikely((unsigned int)(plenend - plen) < c)) {
 | |
|               elf_uncompress_failed();
 | |
|               return 0;
 | |
|             }
 | |
| 
 | |
|             switch (c) {
 | |
|               case 10:
 | |
|                 *plen++ = 0;
 | |
|                 ATTRIBUTE_FALLTHROUGH;
 | |
|               case 9:
 | |
|                 *plen++ = 0;
 | |
|                 ATTRIBUTE_FALLTHROUGH;
 | |
|               case 8:
 | |
|                 *plen++ = 0;
 | |
|                 ATTRIBUTE_FALLTHROUGH;
 | |
|               case 7:
 | |
|                 *plen++ = 0;
 | |
|                 ATTRIBUTE_FALLTHROUGH;
 | |
|               case 6:
 | |
|                 *plen++ = 0;
 | |
|                 ATTRIBUTE_FALLTHROUGH;
 | |
|               case 5:
 | |
|                 *plen++ = 0;
 | |
|                 ATTRIBUTE_FALLTHROUGH;
 | |
|               case 4:
 | |
|                 *plen++ = 0;
 | |
|             }
 | |
|             *plen++ = 0;
 | |
|             *plen++ = 0;
 | |
|             *plen++ = 0;
 | |
|           } else if (v == 18) {
 | |
|             unsigned int c;
 | |
| 
 | |
|             c = 11 + (val & 0x7f);
 | |
|             val >>= 7;
 | |
|             bits -= 7;
 | |
|             if (unlikely((unsigned int)(plenend - plen) < c)) {
 | |
|               elf_uncompress_failed();
 | |
|               return 0;
 | |
|             }
 | |
| 
 | |
|             memset(plen, 0, c);
 | |
|             plen += c;
 | |
|           } else {
 | |
|             elf_uncompress_failed();
 | |
|             return 0;
 | |
|           }
 | |
|         }
 | |
| 
 | |
|         plen = plenbase;
 | |
|         if (unlikely(plen[256] == 0)) {
 | |
|           elf_uncompress_failed();
 | |
|           return 0;
 | |
|         }
 | |
| 
 | |
|         if (!elf_zlib_inflate_table(plen, nlit, zdebug_table, zdebug_table))
 | |
|           return 0;
 | |
|         if (!elf_zlib_inflate_table(plen + nlit, ndist, zdebug_table,
 | |
|                                     zdebug_table + HUFFMAN_TABLE_SIZE))
 | |
|           return 0;
 | |
|         tlit = zdebug_table;
 | |
|         tdist = zdebug_table + HUFFMAN_TABLE_SIZE;
 | |
|       }
 | |
| 
 | |
|       while (1) {
 | |
|         uint16_t t;
 | |
|         unsigned int b;
 | |
|         uint16_t v;
 | |
|         unsigned int lit;
 | |
| 
 | |
|         if (!elf_zlib_fetch(&pin, pinend, &val, &bits)) return 0;
 | |
| 
 | |
|         t = tlit[val & 0xff];
 | |
|         b = (t >> HUFFMAN_BITS_SHIFT) & HUFFMAN_BITS_MASK;
 | |
|         v = t & HUFFMAN_VALUE_MASK;
 | |
| 
 | |
|         if ((t & (1U << HUFFMAN_SECONDARY_SHIFT)) == 0) {
 | |
|           lit = v;
 | |
|           val >>= b + 1;
 | |
|           bits -= b + 1;
 | |
|         } else {
 | |
|           t = tlit[v + 0x100 + ((val >> 8) & ((1U << b) - 1))];
 | |
|           b = (t >> HUFFMAN_BITS_SHIFT) & HUFFMAN_BITS_MASK;
 | |
|           lit = t & HUFFMAN_VALUE_MASK;
 | |
|           val >>= b + 8;
 | |
|           bits -= b + 8;
 | |
|         }
 | |
| 
 | |
|         if (lit < 256) {
 | |
|           if (unlikely(pout == poutend)) {
 | |
|             elf_uncompress_failed();
 | |
|             return 0;
 | |
|           }
 | |
| 
 | |
|           *pout++ = lit;
 | |
| 
 | |
|           __builtin_prefetch(pout, 1, 3);
 | |
|         } else if (lit == 256) {
 | |
|           break;
 | |
|         } else {
 | |
|           unsigned int dist;
 | |
|           unsigned int len;
 | |
| 
 | |
|           if (lit < 265)
 | |
|             len = lit - 257 + 3;
 | |
|           else if (lit == 285)
 | |
|             len = 258;
 | |
|           else if (unlikely(lit > 285)) {
 | |
|             elf_uncompress_failed();
 | |
|             return 0;
 | |
|           } else {
 | |
|             unsigned int extra;
 | |
| 
 | |
|             if (!elf_zlib_fetch(&pin, pinend, &val, &bits)) return 0;
 | |
| 
 | |
|             lit -= 265;
 | |
|             extra = (lit >> 2) + 1;
 | |
|             len = (lit & 3) << extra;
 | |
|             len += 11;
 | |
|             len += ((1U << (extra - 1)) - 1) << 3;
 | |
|             len += val & ((1U << extra) - 1);
 | |
|             val >>= extra;
 | |
|             bits -= extra;
 | |
|           }
 | |
| 
 | |
|           if (!elf_zlib_fetch(&pin, pinend, &val, &bits)) return 0;
 | |
| 
 | |
|           t = tdist[val & 0xff];
 | |
|           b = (t >> HUFFMAN_BITS_SHIFT) & HUFFMAN_BITS_MASK;
 | |
|           v = t & HUFFMAN_VALUE_MASK;
 | |
| 
 | |
|           if ((t & (1U << HUFFMAN_SECONDARY_SHIFT)) == 0) {
 | |
|             dist = v;
 | |
|             val >>= b + 1;
 | |
|             bits -= b + 1;
 | |
|           } else {
 | |
|             t = tdist[v + 0x100 + ((val >> 8) & ((1U << b) - 1))];
 | |
|             b = (t >> HUFFMAN_BITS_SHIFT) & HUFFMAN_BITS_MASK;
 | |
|             dist = t & HUFFMAN_VALUE_MASK;
 | |
|             val >>= b + 8;
 | |
|             bits -= b + 8;
 | |
|           }
 | |
| 
 | |
|           if (dist == 0) {
 | |
|             if (unlikely(pout == porigout)) {
 | |
|               elf_uncompress_failed();
 | |
|               return 0;
 | |
|             }
 | |
| 
 | |
|             if (unlikely((unsigned int)(poutend - pout) < len)) {
 | |
|               elf_uncompress_failed();
 | |
|               return 0;
 | |
|             }
 | |
| 
 | |
|             memset(pout, pout[-1], len);
 | |
|             pout += len;
 | |
|           } else if (unlikely(dist > 29)) {
 | |
|             elf_uncompress_failed();
 | |
|             return 0;
 | |
|           } else {
 | |
|             if (dist < 4)
 | |
|               dist = dist + 1;
 | |
|             else {
 | |
|               unsigned int extra;
 | |
| 
 | |
|               if (!elf_zlib_fetch(&pin, pinend, &val, &bits)) return 0;
 | |
| 
 | |
|               dist -= 4;
 | |
|               extra = (dist >> 1) + 1;
 | |
|               dist = (dist & 1) << extra;
 | |
|               dist += 5;
 | |
|               dist += ((1U << (extra - 1)) - 1) << 2;
 | |
|               dist += val & ((1U << extra) - 1);
 | |
|               val >>= extra;
 | |
|               bits -= extra;
 | |
|             }
 | |
| 
 | |
|             if (unlikely((unsigned int)(pout - porigout) < dist)) {
 | |
|               elf_uncompress_failed();
 | |
|               return 0;
 | |
|             }
 | |
| 
 | |
|             if (unlikely((unsigned int)(poutend - pout) < len)) {
 | |
|               elf_uncompress_failed();
 | |
|               return 0;
 | |
|             }
 | |
| 
 | |
|             if (dist >= len) {
 | |
|               memcpy(pout, pout - dist, len);
 | |
|               pout += len;
 | |
|             } else {
 | |
|               while (len > 0) {
 | |
|                 unsigned int copy;
 | |
| 
 | |
|                 copy = len < dist ? len : dist;
 | |
|                 memcpy(pout, pout - dist, copy);
 | |
|                 len -= copy;
 | |
|                 pout += copy;
 | |
|               }
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (unlikely(pout != poutend)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   return 1;
 | |
| }
 | |
| 
 | |
| static int elf_zlib_verify_checksum(const unsigned char *checkbytes,
 | |
|                                     const unsigned char *uncompressed,
 | |
|                                     size_t uncompressed_size) {
 | |
|   unsigned int i;
 | |
|   unsigned int cksum;
 | |
|   const unsigned char *p;
 | |
|   uint32_t s1;
 | |
|   uint32_t s2;
 | |
|   size_t hsz;
 | |
| 
 | |
|   cksum = 0;
 | |
|   for (i = 0; i < 4; i++) cksum = (cksum << 8) | checkbytes[i];
 | |
| 
 | |
|   s1 = 1;
 | |
|   s2 = 0;
 | |
| 
 | |
|   p = uncompressed;
 | |
|   hsz = uncompressed_size;
 | |
|   while (hsz >= 5552) {
 | |
|     for (i = 0; i < 5552; i += 16) {
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|       s1 = s1 + *p++;
 | |
|       s2 = s2 + s1;
 | |
|     }
 | |
|     hsz -= 5552;
 | |
|     s1 %= 65521;
 | |
|     s2 %= 65521;
 | |
|   }
 | |
| 
 | |
|   while (hsz >= 16) {
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
| 
 | |
|     hsz -= 16;
 | |
|   }
 | |
| 
 | |
|   for (i = 0; i < hsz; ++i) {
 | |
|     s1 = s1 + *p++;
 | |
|     s2 = s2 + s1;
 | |
|   }
 | |
| 
 | |
|   s1 %= 65521;
 | |
|   s2 %= 65521;
 | |
| 
 | |
|   if (unlikely((s2 << 16) + s1 != cksum)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   return 1;
 | |
| }
 | |
| 
 | |
| static int elf_zlib_inflate_and_verify(const unsigned char *pin, size_t sin,
 | |
|                                        uint16_t *zdebug_table,
 | |
|                                        unsigned char *pout, size_t sout) {
 | |
|   if (!elf_zlib_inflate(pin, sin, zdebug_table, pout, sout)) return 0;
 | |
|   if (!elf_zlib_verify_checksum(pin + sin - 4, pout, sout)) return 0;
 | |
|   return 1;
 | |
| }
 | |
| 
 | |
| static int elf_uncompress_zdebug(struct backtrace_state *state,
 | |
|                                  const unsigned char *compressed,
 | |
|                                  size_t compressed_size, uint16_t *zdebug_table,
 | |
|                                  backtrace_error_callback error_callback,
 | |
|                                  void *data, unsigned char **uncompressed,
 | |
|                                  size_t *uncompressed_size) {
 | |
|   size_t sz;
 | |
|   size_t i;
 | |
|   unsigned char *po;
 | |
| 
 | |
|   *uncompressed = NULL;
 | |
|   *uncompressed_size = 0;
 | |
| 
 | |
|   if (compressed_size < 12 || memcmp(compressed, "ZLIB", 4) != 0) return 1;
 | |
| 
 | |
|   sz = 0;
 | |
|   for (i = 0; i < 8; i++) sz = (sz << 8) | compressed[i + 4];
 | |
| 
 | |
|   if (*uncompressed != NULL && *uncompressed_size >= sz)
 | |
|     po = *uncompressed;
 | |
|   else {
 | |
|     po = (unsigned char *)backtrace_alloc(state, sz, error_callback, data);
 | |
|     if (po == NULL) return 0;
 | |
|   }
 | |
| 
 | |
|   if (!elf_zlib_inflate_and_verify(compressed + 12, compressed_size - 12,
 | |
|                                    zdebug_table, po, sz))
 | |
|     return 1;
 | |
| 
 | |
|   *uncompressed = po;
 | |
|   *uncompressed_size = sz;
 | |
| 
 | |
|   return 1;
 | |
| }
 | |
| 
 | |
| static int elf_uncompress_chdr(struct backtrace_state *state,
 | |
|                                const unsigned char *compressed,
 | |
|                                size_t compressed_size, uint16_t *zdebug_table,
 | |
|                                backtrace_error_callback error_callback,
 | |
|                                void *data, unsigned char **uncompressed,
 | |
|                                size_t *uncompressed_size) {
 | |
|   const b_elf_chdr *chdr;
 | |
|   unsigned char *po;
 | |
| 
 | |
|   *uncompressed = NULL;
 | |
|   *uncompressed_size = 0;
 | |
| 
 | |
|   if (compressed_size < sizeof(b_elf_chdr)) return 1;
 | |
| 
 | |
|   chdr = (const b_elf_chdr *)compressed;
 | |
| 
 | |
|   if (chdr->ch_type != ELFCOMPRESS_ZLIB) {
 | |
|     return 1;
 | |
|   }
 | |
| 
 | |
|   if (*uncompressed != NULL && *uncompressed_size >= chdr->ch_size)
 | |
|     po = *uncompressed;
 | |
|   else {
 | |
|     po = (unsigned char *)backtrace_alloc(state, chdr->ch_size, error_callback,
 | |
|                                           data);
 | |
|     if (po == NULL) return 0;
 | |
|   }
 | |
| 
 | |
|   if (!elf_zlib_inflate_and_verify(compressed + sizeof(b_elf_chdr),
 | |
|                                    compressed_size - sizeof(b_elf_chdr),
 | |
|                                    zdebug_table, po, chdr->ch_size))
 | |
|     return 1;
 | |
| 
 | |
|   *uncompressed = po;
 | |
|   *uncompressed_size = chdr->ch_size;
 | |
| 
 | |
|   return 1;
 | |
| }
 | |
| 
 | |
| int backtrace_uncompress_zdebug(struct backtrace_state *state,
 | |
|                                 const unsigned char *compressed,
 | |
|                                 size_t compressed_size,
 | |
|                                 backtrace_error_callback error_callback,
 | |
|                                 void *data, unsigned char **uncompressed,
 | |
|                                 size_t *uncompressed_size) {
 | |
|   uint16_t *zdebug_table;
 | |
|   int ret;
 | |
| 
 | |
|   zdebug_table = ((uint16_t *)backtrace_alloc(state, ZDEBUG_TABLE_SIZE,
 | |
|                                               error_callback, data));
 | |
|   if (zdebug_table == NULL) return 0;
 | |
|   ret = elf_uncompress_zdebug(state, compressed, compressed_size, zdebug_table,
 | |
|                               error_callback, data, uncompressed,
 | |
|                               uncompressed_size);
 | |
|   backtrace_free(state, zdebug_table, ZDEBUG_TABLE_SIZE, error_callback, data);
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| #define LZMA_STATES (12)
 | |
| 
 | |
| #define LZMA_POS_STATES (16)
 | |
| 
 | |
| #define LZMA_DIST_STATES (4)
 | |
| 
 | |
| #define LZMA_DIST_SLOTS (64)
 | |
| 
 | |
| #define LZMA_DIST_MODEL_START (4)
 | |
| 
 | |
| #define LZMA_DIST_MODEL_END (14)
 | |
| 
 | |
| #define LZMA_FULL_DISTANCES (128)
 | |
| 
 | |
| #define LZMA_ALIGN_SIZE (16)
 | |
| 
 | |
| #define LZMA_LEN_LOW_SYMBOLS (8)
 | |
| #define LZMA_LEN_MID_SYMBOLS (8)
 | |
| #define LZMA_LEN_HIGH_SYMBOLS (256)
 | |
| 
 | |
| #define LZMA_LITERAL_CODERS_MAX (16)
 | |
| #define LZMA_LITERAL_CODER_SIZE (0x300)
 | |
| 
 | |
| #define LZMA_PROB_IS_MATCH_LEN (LZMA_STATES * LZMA_POS_STATES)
 | |
| #define LZMA_PROB_IS_REP_LEN LZMA_STATES
 | |
| #define LZMA_PROB_IS_REP0_LEN LZMA_STATES
 | |
| #define LZMA_PROB_IS_REP1_LEN LZMA_STATES
 | |
| #define LZMA_PROB_IS_REP2_LEN LZMA_STATES
 | |
| #define LZMA_PROB_IS_REP0_LONG_LEN (LZMA_STATES * LZMA_POS_STATES)
 | |
| #define LZMA_PROB_DIST_SLOT_LEN (LZMA_DIST_STATES * LZMA_DIST_SLOTS)
 | |
| #define LZMA_PROB_DIST_SPECIAL_LEN (LZMA_FULL_DISTANCES - LZMA_DIST_MODEL_END)
 | |
| #define LZMA_PROB_DIST_ALIGN_LEN LZMA_ALIGN_SIZE
 | |
| #define LZMA_PROB_MATCH_LEN_CHOICE_LEN 1
 | |
| #define LZMA_PROB_MATCH_LEN_CHOICE2_LEN 1
 | |
| #define LZMA_PROB_MATCH_LEN_LOW_LEN (LZMA_POS_STATES * LZMA_LEN_LOW_SYMBOLS)
 | |
| #define LZMA_PROB_MATCH_LEN_MID_LEN (LZMA_POS_STATES * LZMA_LEN_MID_SYMBOLS)
 | |
| #define LZMA_PROB_MATCH_LEN_HIGH_LEN LZMA_LEN_HIGH_SYMBOLS
 | |
| #define LZMA_PROB_REP_LEN_CHOICE_LEN 1
 | |
| #define LZMA_PROB_REP_LEN_CHOICE2_LEN 1
 | |
| #define LZMA_PROB_REP_LEN_LOW_LEN (LZMA_POS_STATES * LZMA_LEN_LOW_SYMBOLS)
 | |
| #define LZMA_PROB_REP_LEN_MID_LEN (LZMA_POS_STATES * LZMA_LEN_MID_SYMBOLS)
 | |
| #define LZMA_PROB_REP_LEN_HIGH_LEN LZMA_LEN_HIGH_SYMBOLS
 | |
| #define LZMA_PROB_LITERAL_LEN \
 | |
|   (LZMA_LITERAL_CODERS_MAX * LZMA_LITERAL_CODER_SIZE)
 | |
| 
 | |
| #define LZMA_PROB_IS_MATCH_OFFSET 0
 | |
| #define LZMA_PROB_IS_REP_OFFSET \
 | |
|   (LZMA_PROB_IS_MATCH_OFFSET + LZMA_PROB_IS_MATCH_LEN)
 | |
| #define LZMA_PROB_IS_REP0_OFFSET \
 | |
|   (LZMA_PROB_IS_REP_OFFSET + LZMA_PROB_IS_REP_LEN)
 | |
| #define LZMA_PROB_IS_REP1_OFFSET \
 | |
|   (LZMA_PROB_IS_REP0_OFFSET + LZMA_PROB_IS_REP0_LEN)
 | |
| #define LZMA_PROB_IS_REP2_OFFSET \
 | |
|   (LZMA_PROB_IS_REP1_OFFSET + LZMA_PROB_IS_REP1_LEN)
 | |
| #define LZMA_PROB_IS_REP0_LONG_OFFSET \
 | |
|   (LZMA_PROB_IS_REP2_OFFSET + LZMA_PROB_IS_REP2_LEN)
 | |
| #define LZMA_PROB_DIST_SLOT_OFFSET \
 | |
|   (LZMA_PROB_IS_REP0_LONG_OFFSET + LZMA_PROB_IS_REP0_LONG_LEN)
 | |
| #define LZMA_PROB_DIST_SPECIAL_OFFSET \
 | |
|   (LZMA_PROB_DIST_SLOT_OFFSET + LZMA_PROB_DIST_SLOT_LEN)
 | |
| #define LZMA_PROB_DIST_ALIGN_OFFSET \
 | |
|   (LZMA_PROB_DIST_SPECIAL_OFFSET + LZMA_PROB_DIST_SPECIAL_LEN)
 | |
| #define LZMA_PROB_MATCH_LEN_CHOICE_OFFSET \
 | |
|   (LZMA_PROB_DIST_ALIGN_OFFSET + LZMA_PROB_DIST_ALIGN_LEN)
 | |
| #define LZMA_PROB_MATCH_LEN_CHOICE2_OFFSET \
 | |
|   (LZMA_PROB_MATCH_LEN_CHOICE_OFFSET + LZMA_PROB_MATCH_LEN_CHOICE_LEN)
 | |
| #define LZMA_PROB_MATCH_LEN_LOW_OFFSET \
 | |
|   (LZMA_PROB_MATCH_LEN_CHOICE2_OFFSET + LZMA_PROB_MATCH_LEN_CHOICE2_LEN)
 | |
| #define LZMA_PROB_MATCH_LEN_MID_OFFSET \
 | |
|   (LZMA_PROB_MATCH_LEN_LOW_OFFSET + LZMA_PROB_MATCH_LEN_LOW_LEN)
 | |
| #define LZMA_PROB_MATCH_LEN_HIGH_OFFSET \
 | |
|   (LZMA_PROB_MATCH_LEN_MID_OFFSET + LZMA_PROB_MATCH_LEN_MID_LEN)
 | |
| #define LZMA_PROB_REP_LEN_CHOICE_OFFSET \
 | |
|   (LZMA_PROB_MATCH_LEN_HIGH_OFFSET + LZMA_PROB_MATCH_LEN_HIGH_LEN)
 | |
| #define LZMA_PROB_REP_LEN_CHOICE2_OFFSET \
 | |
|   (LZMA_PROB_REP_LEN_CHOICE_OFFSET + LZMA_PROB_REP_LEN_CHOICE_LEN)
 | |
| #define LZMA_PROB_REP_LEN_LOW_OFFSET \
 | |
|   (LZMA_PROB_REP_LEN_CHOICE2_OFFSET + LZMA_PROB_REP_LEN_CHOICE2_LEN)
 | |
| #define LZMA_PROB_REP_LEN_MID_OFFSET \
 | |
|   (LZMA_PROB_REP_LEN_LOW_OFFSET + LZMA_PROB_REP_LEN_LOW_LEN)
 | |
| #define LZMA_PROB_REP_LEN_HIGH_OFFSET \
 | |
|   (LZMA_PROB_REP_LEN_MID_OFFSET + LZMA_PROB_REP_LEN_MID_LEN)
 | |
| #define LZMA_PROB_LITERAL_OFFSET \
 | |
|   (LZMA_PROB_REP_LEN_HIGH_OFFSET + LZMA_PROB_REP_LEN_HIGH_LEN)
 | |
| 
 | |
| #define LZMA_PROB_TOTAL_COUNT (LZMA_PROB_LITERAL_OFFSET + LZMA_PROB_LITERAL_LEN)
 | |
| 
 | |
| #if LZMA_PROB_TOTAL_COUNT != 1846 + (1 << 4) * 0x300
 | |
| #error Wrong number of LZMA probabilities
 | |
| #endif
 | |
| 
 | |
| #define LZMA_IS_MATCH(state, pos) \
 | |
|   (LZMA_PROB_IS_MATCH_OFFSET + (state)*LZMA_POS_STATES + (pos))
 | |
| #define LZMA_IS_REP(state) (LZMA_PROB_IS_REP_OFFSET + (state))
 | |
| #define LZMA_IS_REP0(state) (LZMA_PROB_IS_REP0_OFFSET + (state))
 | |
| #define LZMA_IS_REP1(state) (LZMA_PROB_IS_REP1_OFFSET + (state))
 | |
| #define LZMA_IS_REP2(state) (LZMA_PROB_IS_REP2_OFFSET + (state))
 | |
| #define LZMA_IS_REP0_LONG(state, pos) \
 | |
|   (LZMA_PROB_IS_REP0_LONG_OFFSET + (state)*LZMA_POS_STATES + (pos))
 | |
| #define LZMA_DIST_SLOT(dist, slot) \
 | |
|   (LZMA_PROB_DIST_SLOT_OFFSET + (dist)*LZMA_DIST_SLOTS + (slot))
 | |
| #define LZMA_DIST_SPECIAL(dist) (LZMA_PROB_DIST_SPECIAL_OFFSET + (dist))
 | |
| #define LZMA_DIST_ALIGN(dist) (LZMA_PROB_DIST_ALIGN_OFFSET + (dist))
 | |
| #define LZMA_MATCH_LEN_CHOICE LZMA_PROB_MATCH_LEN_CHOICE_OFFSET
 | |
| #define LZMA_MATCH_LEN_CHOICE2 LZMA_PROB_MATCH_LEN_CHOICE2_OFFSET
 | |
| #define LZMA_MATCH_LEN_LOW(pos, sym) \
 | |
|   (LZMA_PROB_MATCH_LEN_LOW_OFFSET + (pos)*LZMA_LEN_LOW_SYMBOLS + (sym))
 | |
| #define LZMA_MATCH_LEN_MID(pos, sym) \
 | |
|   (LZMA_PROB_MATCH_LEN_MID_OFFSET + (pos)*LZMA_LEN_MID_SYMBOLS + (sym))
 | |
| #define LZMA_MATCH_LEN_HIGH(sym) (LZMA_PROB_MATCH_LEN_HIGH_OFFSET + (sym))
 | |
| #define LZMA_REP_LEN_CHOICE LZMA_PROB_REP_LEN_CHOICE_OFFSET
 | |
| #define LZMA_REP_LEN_CHOICE2 LZMA_PROB_REP_LEN_CHOICE2_OFFSET
 | |
| #define LZMA_REP_LEN_LOW(pos, sym) \
 | |
|   (LZMA_PROB_REP_LEN_LOW_OFFSET + (pos)*LZMA_LEN_LOW_SYMBOLS + (sym))
 | |
| #define LZMA_REP_LEN_MID(pos, sym) \
 | |
|   (LZMA_PROB_REP_LEN_MID_OFFSET + (pos)*LZMA_LEN_MID_SYMBOLS + (sym))
 | |
| #define LZMA_REP_LEN_HIGH(sym) (LZMA_PROB_REP_LEN_HIGH_OFFSET + (sym))
 | |
| #define LZMA_LITERAL(code, size) \
 | |
|   (LZMA_PROB_LITERAL_OFFSET + (code)*LZMA_LITERAL_CODER_SIZE + (size))
 | |
| 
 | |
| static int elf_lzma_varint(const unsigned char *compressed,
 | |
|                            size_t compressed_size, size_t *poffset,
 | |
|                            uint64_t *val) {
 | |
|   size_t off;
 | |
|   int i;
 | |
|   uint64_t v;
 | |
|   unsigned char b;
 | |
| 
 | |
|   off = *poffset;
 | |
|   i = 0;
 | |
|   v = 0;
 | |
|   while (1) {
 | |
|     if (unlikely(off >= compressed_size)) {
 | |
|       elf_uncompress_failed();
 | |
|       return 0;
 | |
|     }
 | |
|     b = compressed[off];
 | |
|     v |= (b & 0x7f) << (i * 7);
 | |
|     ++off;
 | |
|     if ((b & 0x80) == 0) {
 | |
|       *poffset = off;
 | |
|       *val = v;
 | |
|       return 1;
 | |
|     }
 | |
|     ++i;
 | |
|     if (unlikely(i >= 9)) {
 | |
|       elf_uncompress_failed();
 | |
|       return 0;
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| static void elf_lzma_range_normalize(const unsigned char *compressed,
 | |
|                                      size_t compressed_size, size_t *poffset,
 | |
|                                      uint32_t *prange, uint32_t *pcode) {
 | |
|   if (*prange < (1U << 24)) {
 | |
|     if (unlikely(*poffset >= compressed_size)) {
 | |
|       elf_uncompress_failed();
 | |
|       return;
 | |
|     }
 | |
|     *prange <<= 8;
 | |
|     *pcode <<= 8;
 | |
|     *pcode += compressed[*poffset];
 | |
|     ++*poffset;
 | |
|   }
 | |
| }
 | |
| 
 | |
| static int elf_lzma_bit(const unsigned char *compressed, size_t compressed_size,
 | |
|                         uint16_t *prob, size_t *poffset, uint32_t *prange,
 | |
|                         uint32_t *pcode) {
 | |
|   uint32_t bound;
 | |
| 
 | |
|   elf_lzma_range_normalize(compressed, compressed_size, poffset, prange, pcode);
 | |
|   bound = (*prange >> 11) * (uint32_t)*prob;
 | |
|   if (*pcode < bound) {
 | |
|     *prange = bound;
 | |
|     *prob += ((1U << 11) - *prob) >> 5;
 | |
|     return 0;
 | |
|   } else {
 | |
|     *prange -= bound;
 | |
|     *pcode -= bound;
 | |
|     *prob -= *prob >> 5;
 | |
|     return 1;
 | |
|   }
 | |
| }
 | |
| 
 | |
| static uint32_t elf_lzma_integer(const unsigned char *compressed,
 | |
|                                  size_t compressed_size, uint16_t *probs,
 | |
|                                  uint32_t bits, size_t *poffset,
 | |
|                                  uint32_t *prange, uint32_t *pcode) {
 | |
|   uint32_t sym;
 | |
|   uint32_t i;
 | |
| 
 | |
|   sym = 1;
 | |
|   for (i = 0; i < bits; i++) {
 | |
|     int bit;
 | |
| 
 | |
|     bit = elf_lzma_bit(compressed, compressed_size, probs + sym, poffset,
 | |
|                        prange, pcode);
 | |
|     sym <<= 1;
 | |
|     sym += bit;
 | |
|   }
 | |
|   return sym - (1 << bits);
 | |
| }
 | |
| 
 | |
| static uint32_t elf_lzma_reverse_integer(const unsigned char *compressed,
 | |
|                                          size_t compressed_size,
 | |
|                                          uint16_t *probs, uint32_t bits,
 | |
|                                          size_t *poffset, uint32_t *prange,
 | |
|                                          uint32_t *pcode) {
 | |
|   uint32_t sym;
 | |
|   uint32_t val;
 | |
|   uint32_t i;
 | |
| 
 | |
|   sym = 1;
 | |
|   val = 0;
 | |
|   for (i = 0; i < bits; i++) {
 | |
|     int bit;
 | |
| 
 | |
|     bit = elf_lzma_bit(compressed, compressed_size, probs + sym, poffset,
 | |
|                        prange, pcode);
 | |
|     sym <<= 1;
 | |
|     sym += bit;
 | |
|     val += bit << i;
 | |
|   }
 | |
|   return val;
 | |
| }
 | |
| 
 | |
| static uint32_t elf_lzma_len(const unsigned char *compressed,
 | |
|                              size_t compressed_size, uint16_t *probs,
 | |
|                              int is_rep, unsigned int pos_state,
 | |
|                              size_t *poffset, uint32_t *prange,
 | |
|                              uint32_t *pcode) {
 | |
|   uint16_t *probs_choice;
 | |
|   uint16_t *probs_sym;
 | |
|   uint32_t bits;
 | |
|   uint32_t len;
 | |
| 
 | |
|   probs_choice = probs + (is_rep ? LZMA_REP_LEN_CHOICE : LZMA_MATCH_LEN_CHOICE);
 | |
|   if (elf_lzma_bit(compressed, compressed_size, probs_choice, poffset, prange,
 | |
|                    pcode)) {
 | |
|     probs_choice =
 | |
|         probs + (is_rep ? LZMA_REP_LEN_CHOICE2 : LZMA_MATCH_LEN_CHOICE2);
 | |
|     if (elf_lzma_bit(compressed, compressed_size, probs_choice, poffset, prange,
 | |
|                      pcode)) {
 | |
|       probs_sym =
 | |
|           probs + (is_rep ? LZMA_REP_LEN_HIGH(0) : LZMA_MATCH_LEN_HIGH(0));
 | |
|       bits = 8;
 | |
|       len = 2 + 8 + 8;
 | |
|     } else {
 | |
|       probs_sym = probs + (is_rep ? LZMA_REP_LEN_MID(pos_state, 0)
 | |
|                                   : LZMA_MATCH_LEN_MID(pos_state, 0));
 | |
|       bits = 3;
 | |
|       len = 2 + 8;
 | |
|     }
 | |
|   } else {
 | |
|     probs_sym = probs + (is_rep ? LZMA_REP_LEN_LOW(pos_state, 0)
 | |
|                                 : LZMA_MATCH_LEN_LOW(pos_state, 0));
 | |
|     bits = 3;
 | |
|     len = 2;
 | |
|   }
 | |
| 
 | |
|   len += elf_lzma_integer(compressed, compressed_size, probs_sym, bits, poffset,
 | |
|                           prange, pcode);
 | |
|   return len;
 | |
| }
 | |
| 
 | |
| static int elf_uncompress_lzma_block(const unsigned char *compressed,
 | |
|                                      size_t compressed_size,
 | |
|                                      unsigned char check, uint16_t *probs,
 | |
|                                      unsigned char *uncompressed,
 | |
|                                      size_t uncompressed_size,
 | |
|                                      size_t *poffset) {
 | |
|   size_t off;
 | |
|   size_t block_header_offset;
 | |
|   size_t block_header_size;
 | |
|   unsigned char block_flags;
 | |
|   uint64_t header_compressed_size;
 | |
|   uint64_t header_uncompressed_size;
 | |
|   unsigned char lzma2_properties;
 | |
|   uint32_t computed_crc;
 | |
|   uint32_t stream_crc;
 | |
|   size_t uncompressed_offset;
 | |
|   size_t dict_start_offset;
 | |
|   unsigned int lc;
 | |
|   unsigned int lp;
 | |
|   unsigned int pb;
 | |
|   uint32_t range;
 | |
|   uint32_t code;
 | |
|   uint32_t lstate;
 | |
|   uint32_t dist[4];
 | |
| 
 | |
|   off = *poffset;
 | |
|   block_header_offset = off;
 | |
| 
 | |
|   if (unlikely(off >= compressed_size)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
|   block_header_size = (compressed[off] + 1) * 4;
 | |
|   if (unlikely(off + block_header_size > compressed_size)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   block_flags = compressed[off + 1];
 | |
|   if (unlikely((block_flags & 0x3c) != 0)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   off += 2;
 | |
| 
 | |
|   header_compressed_size = 0;
 | |
|   if ((block_flags & 0x40) != 0) {
 | |
|     *poffset = off;
 | |
|     if (!elf_lzma_varint(compressed, compressed_size, poffset,
 | |
|                          &header_compressed_size))
 | |
|       return 0;
 | |
|     off = *poffset;
 | |
|   }
 | |
| 
 | |
|   header_uncompressed_size = 0;
 | |
|   if ((block_flags & 0x80) != 0) {
 | |
|     *poffset = off;
 | |
|     if (!elf_lzma_varint(compressed, compressed_size, poffset,
 | |
|                          &header_uncompressed_size))
 | |
|       return 0;
 | |
|     off = *poffset;
 | |
|   }
 | |
| 
 | |
|   if (unlikely((block_flags & 0x3) != 0)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   if (unlikely(off + 2 >= block_header_offset + block_header_size)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   if (unlikely(compressed[off] != 0x21)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
|   ++off;
 | |
| 
 | |
|   if (unlikely(compressed[off] != 1)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
|   ++off;
 | |
| 
 | |
|   lzma2_properties = compressed[off];
 | |
|   ++off;
 | |
| 
 | |
|   if (unlikely(lzma2_properties > 40)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   if (unlikely(off + 4 > compressed_size)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   off = (off + 3) & ~(size_t)3;
 | |
| 
 | |
|   if (unlikely(off + 4 > compressed_size)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   computed_crc =
 | |
|       elf_crc32(0, compressed + block_header_offset, block_header_size - 4);
 | |
|   stream_crc = (compressed[off] | (compressed[off + 1] << 8) |
 | |
|                 (compressed[off + 2] << 16) | (compressed[off + 3] << 24));
 | |
|   if (unlikely(computed_crc != stream_crc)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
|   off += 4;
 | |
| 
 | |
|   uncompressed_offset = 0;
 | |
|   dict_start_offset = 0;
 | |
|   lc = 0;
 | |
|   lp = 0;
 | |
|   pb = 0;
 | |
|   lstate = 0;
 | |
|   while (off < compressed_size) {
 | |
|     unsigned char control;
 | |
| 
 | |
|     range = 0xffffffff;
 | |
|     code = 0;
 | |
| 
 | |
|     control = compressed[off];
 | |
|     ++off;
 | |
|     if (unlikely(control == 0)) {
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     if (control == 1 || control >= 0xe0) {
 | |
|       dict_start_offset = uncompressed_offset;
 | |
|     }
 | |
| 
 | |
|     if (control < 0x80) {
 | |
|       size_t chunk_size;
 | |
| 
 | |
|       if (unlikely(control > 2)) {
 | |
|         elf_uncompress_failed();
 | |
|         return 0;
 | |
|       }
 | |
| 
 | |
|       if (unlikely(off + 2 > compressed_size)) {
 | |
|         elf_uncompress_failed();
 | |
|         return 0;
 | |
|       }
 | |
| 
 | |
|       chunk_size = compressed[off] << 8;
 | |
|       chunk_size += compressed[off + 1];
 | |
|       ++chunk_size;
 | |
| 
 | |
|       off += 2;
 | |
| 
 | |
|       if (unlikely(off + chunk_size > compressed_size)) {
 | |
|         elf_uncompress_failed();
 | |
|         return 0;
 | |
|       }
 | |
|       if (unlikely(uncompressed_offset + chunk_size > uncompressed_size)) {
 | |
|         elf_uncompress_failed();
 | |
|         return 0;
 | |
|       }
 | |
| 
 | |
|       memcpy(uncompressed + uncompressed_offset, compressed + off, chunk_size);
 | |
|       uncompressed_offset += chunk_size;
 | |
|       off += chunk_size;
 | |
|     } else {
 | |
|       size_t uncompressed_chunk_start;
 | |
|       size_t uncompressed_chunk_size;
 | |
|       size_t compressed_chunk_size;
 | |
|       size_t limit;
 | |
| 
 | |
|       if (unlikely(off + 4 >= compressed_size)) {
 | |
|         elf_uncompress_failed();
 | |
|         return 0;
 | |
|       }
 | |
| 
 | |
|       uncompressed_chunk_start = uncompressed_offset;
 | |
| 
 | |
|       uncompressed_chunk_size = (control & 0x1f) << 16;
 | |
|       uncompressed_chunk_size += compressed[off] << 8;
 | |
|       uncompressed_chunk_size += compressed[off + 1];
 | |
|       ++uncompressed_chunk_size;
 | |
| 
 | |
|       compressed_chunk_size = compressed[off + 2] << 8;
 | |
|       compressed_chunk_size += compressed[off + 3];
 | |
|       ++compressed_chunk_size;
 | |
| 
 | |
|       off += 4;
 | |
| 
 | |
|       if (control >= 0xc0) {
 | |
|         unsigned char props;
 | |
| 
 | |
|         if (unlikely(off >= compressed_size)) {
 | |
|           elf_uncompress_failed();
 | |
|           return 0;
 | |
|         }
 | |
|         props = compressed[off];
 | |
|         ++off;
 | |
|         if (unlikely(props > (4 * 5 + 4) * 9 + 8)) {
 | |
|           elf_uncompress_failed();
 | |
|           return 0;
 | |
|         }
 | |
|         pb = 0;
 | |
|         while (props >= 9 * 5) {
 | |
|           props -= 9 * 5;
 | |
|           ++pb;
 | |
|         }
 | |
|         lp = 0;
 | |
|         while (props > 9) {
 | |
|           props -= 9;
 | |
|           ++lp;
 | |
|         }
 | |
|         lc = props;
 | |
|         if (unlikely(lc + lp > 4)) {
 | |
|           elf_uncompress_failed();
 | |
|           return 0;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       if (control >= 0xa0) {
 | |
|         size_t i;
 | |
| 
 | |
|         lstate = 0;
 | |
|         memset(&dist, 0, sizeof dist);
 | |
|         for (i = 0; i < LZMA_PROB_TOTAL_COUNT; i++) probs[i] = 1 << 10;
 | |
|         range = 0xffffffff;
 | |
|         code = 0;
 | |
|       }
 | |
| 
 | |
|       if (unlikely(off + 5 > compressed_size)) {
 | |
|         elf_uncompress_failed();
 | |
|         return 0;
 | |
|       }
 | |
| 
 | |
|       code = ((compressed[off + 1] << 24) + (compressed[off + 2] << 16) +
 | |
|               (compressed[off + 3] << 8) + compressed[off + 4]);
 | |
|       off += 5;
 | |
| 
 | |
|       limit = off + compressed_chunk_size;
 | |
|       *poffset = off;
 | |
|       while (*poffset < limit) {
 | |
|         unsigned int pos_state;
 | |
| 
 | |
|         if (unlikely(uncompressed_offset ==
 | |
|                      (uncompressed_chunk_start + uncompressed_chunk_size))) {
 | |
|           break;
 | |
|         }
 | |
| 
 | |
|         pos_state =
 | |
|             ((uncompressed_offset - dict_start_offset) & ((1 << pb) - 1));
 | |
| 
 | |
|         if (elf_lzma_bit(compressed, compressed_size,
 | |
|                          probs + LZMA_IS_MATCH(lstate, pos_state), poffset,
 | |
|                          &range, &code)) {
 | |
|           uint32_t len;
 | |
| 
 | |
|           if (elf_lzma_bit(compressed, compressed_size,
 | |
|                            probs + LZMA_IS_REP(lstate), poffset, &range,
 | |
|                            &code)) {
 | |
|             int short_rep;
 | |
|             uint32_t next_dist;
 | |
| 
 | |
|             short_rep = 0;
 | |
|             if (elf_lzma_bit(compressed, compressed_size,
 | |
|                              probs + LZMA_IS_REP0(lstate), poffset, &range,
 | |
|                              &code)) {
 | |
|               if (elf_lzma_bit(compressed, compressed_size,
 | |
|                                probs + LZMA_IS_REP1(lstate), poffset, &range,
 | |
|                                &code)) {
 | |
|                 if (elf_lzma_bit(compressed, compressed_size,
 | |
|                                  probs + LZMA_IS_REP2(lstate), poffset, &range,
 | |
|                                  &code)) {
 | |
|                   next_dist = dist[3];
 | |
|                   dist[3] = dist[2];
 | |
|                 } else {
 | |
|                   next_dist = dist[2];
 | |
|                 }
 | |
|                 dist[2] = dist[1];
 | |
|               } else {
 | |
|                 next_dist = dist[1];
 | |
|               }
 | |
| 
 | |
|               dist[1] = dist[0];
 | |
|               dist[0] = next_dist;
 | |
|             } else {
 | |
|               if (!elf_lzma_bit(compressed, compressed_size,
 | |
|                                 (probs + LZMA_IS_REP0_LONG(lstate, pos_state)),
 | |
|                                 poffset, &range, &code))
 | |
|                 short_rep = 1;
 | |
|             }
 | |
| 
 | |
|             if (lstate < 7)
 | |
|               lstate = short_rep ? 9 : 8;
 | |
|             else
 | |
|               lstate = 11;
 | |
| 
 | |
|             if (short_rep)
 | |
|               len = 1;
 | |
|             else
 | |
|               len = elf_lzma_len(compressed, compressed_size, probs, 1,
 | |
|                                  pos_state, poffset, &range, &code);
 | |
|           } else {
 | |
|             uint32_t dist_state;
 | |
|             uint32_t dist_slot;
 | |
|             uint16_t *probs_dist;
 | |
| 
 | |
|             if (lstate < 7)
 | |
|               lstate = 7;
 | |
|             else
 | |
|               lstate = 10;
 | |
|             dist[3] = dist[2];
 | |
|             dist[2] = dist[1];
 | |
|             dist[1] = dist[0];
 | |
|             len = elf_lzma_len(compressed, compressed_size, probs, 0, pos_state,
 | |
|                                poffset, &range, &code);
 | |
| 
 | |
|             if (len < 4 + 2)
 | |
|               dist_state = len - 2;
 | |
|             else
 | |
|               dist_state = 3;
 | |
|             probs_dist = probs + LZMA_DIST_SLOT(dist_state, 0);
 | |
|             dist_slot = elf_lzma_integer(compressed, compressed_size,
 | |
|                                          probs_dist, 6, poffset, &range, &code);
 | |
|             if (dist_slot < LZMA_DIST_MODEL_START)
 | |
|               dist[0] = dist_slot;
 | |
|             else {
 | |
|               uint32_t limit;
 | |
| 
 | |
|               limit = (dist_slot >> 1) - 1;
 | |
|               dist[0] = 2 + (dist_slot & 1);
 | |
|               if (dist_slot < LZMA_DIST_MODEL_END) {
 | |
|                 dist[0] <<= limit;
 | |
|                 probs_dist =
 | |
|                     (probs + LZMA_DIST_SPECIAL(dist[0] - dist_slot - 1));
 | |
|                 dist[0] += elf_lzma_reverse_integer(compressed, compressed_size,
 | |
|                                                     probs_dist, limit, poffset,
 | |
|                                                     &range, &code);
 | |
|               } else {
 | |
|                 uint32_t dist0;
 | |
|                 uint32_t i;
 | |
| 
 | |
|                 dist0 = dist[0];
 | |
|                 for (i = 0; i < limit - 4; i++) {
 | |
|                   uint32_t mask;
 | |
| 
 | |
|                   elf_lzma_range_normalize(compressed, compressed_size, poffset,
 | |
|                                            &range, &code);
 | |
|                   range >>= 1;
 | |
|                   code -= range;
 | |
|                   mask = -(code >> 31);
 | |
|                   code += range & mask;
 | |
|                   dist0 <<= 1;
 | |
|                   dist0 += mask + 1;
 | |
|                 }
 | |
|                 dist0 <<= 4;
 | |
|                 probs_dist = probs + LZMA_DIST_ALIGN(0);
 | |
|                 dist0 += elf_lzma_reverse_integer(compressed, compressed_size,
 | |
|                                                   probs_dist, 4, poffset,
 | |
|                                                   &range, &code);
 | |
|                 dist[0] = dist0;
 | |
|               }
 | |
|             }
 | |
|           }
 | |
| 
 | |
|           if (unlikely(uncompressed_offset - dict_start_offset < dist[0] + 1)) {
 | |
|             elf_uncompress_failed();
 | |
|             return 0;
 | |
|           }
 | |
|           if (unlikely(uncompressed_offset + len > uncompressed_size)) {
 | |
|             elf_uncompress_failed();
 | |
|             return 0;
 | |
|           }
 | |
| 
 | |
|           if (dist[0] == 0) {
 | |
|             memset(uncompressed + uncompressed_offset,
 | |
|                    uncompressed[uncompressed_offset - 1], len);
 | |
|             uncompressed_offset += len;
 | |
|           } else if (dist[0] + 1 >= len) {
 | |
|             memcpy(uncompressed + uncompressed_offset,
 | |
|                    uncompressed + uncompressed_offset - dist[0] - 1, len);
 | |
|             uncompressed_offset += len;
 | |
|           } else {
 | |
|             while (len > 0) {
 | |
|               uint32_t copy;
 | |
| 
 | |
|               copy = len < dist[0] + 1 ? len : dist[0] + 1;
 | |
|               memcpy(uncompressed + uncompressed_offset,
 | |
|                      (uncompressed + uncompressed_offset - dist[0] - 1), copy);
 | |
|               len -= copy;
 | |
|               uncompressed_offset += copy;
 | |
|             }
 | |
|           }
 | |
|         } else {
 | |
|           unsigned char prev;
 | |
|           unsigned char low;
 | |
|           size_t high;
 | |
|           uint16_t *lit_probs;
 | |
|           unsigned int sym;
 | |
| 
 | |
|           if (uncompressed_offset > 0)
 | |
|             prev = uncompressed[uncompressed_offset - 1];
 | |
|           else
 | |
|             prev = 0;
 | |
|           low = prev >> (8 - lc);
 | |
|           high = (((uncompressed_offset - dict_start_offset) & ((1 << lp) - 1))
 | |
|                   << lc);
 | |
|           lit_probs = probs + LZMA_LITERAL(low + high, 0);
 | |
|           if (lstate < 7)
 | |
|             sym = elf_lzma_integer(compressed, compressed_size, lit_probs, 8,
 | |
|                                    poffset, &range, &code);
 | |
|           else {
 | |
|             unsigned int match;
 | |
|             unsigned int bit;
 | |
|             unsigned int match_bit;
 | |
|             unsigned int idx;
 | |
| 
 | |
|             sym = 1;
 | |
|             if (uncompressed_offset >= dist[0] + 1)
 | |
|               match = uncompressed[uncompressed_offset - dist[0] - 1];
 | |
|             else
 | |
|               match = 0;
 | |
|             match <<= 1;
 | |
|             bit = 0x100;
 | |
|             do {
 | |
|               match_bit = match & bit;
 | |
|               match <<= 1;
 | |
|               idx = bit + match_bit + sym;
 | |
|               sym <<= 1;
 | |
|               if (elf_lzma_bit(compressed, compressed_size, lit_probs + idx,
 | |
|                                poffset, &range, &code)) {
 | |
|                 ++sym;
 | |
|                 bit &= match_bit;
 | |
|               } else {
 | |
|                 bit &= ~match_bit;
 | |
|               }
 | |
|             } while (sym < 0x100);
 | |
|           }
 | |
| 
 | |
|           if (unlikely(uncompressed_offset >= uncompressed_size)) {
 | |
|             elf_uncompress_failed();
 | |
|             return 0;
 | |
|           }
 | |
| 
 | |
|           uncompressed[uncompressed_offset] = (unsigned char)sym;
 | |
|           ++uncompressed_offset;
 | |
|           if (lstate <= 3)
 | |
|             lstate = 0;
 | |
|           else if (lstate <= 9)
 | |
|             lstate -= 3;
 | |
|           else
 | |
|             lstate -= 6;
 | |
|         }
 | |
|       }
 | |
| 
 | |
|       elf_lzma_range_normalize(compressed, compressed_size, poffset, &range,
 | |
|                                &code);
 | |
| 
 | |
|       off = *poffset;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   off = (off + 3) & ~(size_t)3;
 | |
|   if (unlikely(off > compressed_size)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   switch (check) {
 | |
|     case 0:
 | |
| 
 | |
|       break;
 | |
| 
 | |
|     case 1:
 | |
| 
 | |
|       if (unlikely(off + 4 > compressed_size)) {
 | |
|         elf_uncompress_failed();
 | |
|         return 0;
 | |
|       }
 | |
|       computed_crc = elf_crc32(0, uncompressed, uncompressed_offset);
 | |
|       stream_crc = (compressed[off] | (compressed[off + 1] << 8) |
 | |
|                     (compressed[off + 2] << 16) | (compressed[off + 3] << 24));
 | |
|       if (computed_crc != stream_crc) {
 | |
|         elf_uncompress_failed();
 | |
|         return 0;
 | |
|       }
 | |
|       off += 4;
 | |
|       break;
 | |
| 
 | |
|     case 4:
 | |
| 
 | |
|       if (unlikely(off + 8 > compressed_size)) {
 | |
|         elf_uncompress_failed();
 | |
|         return 0;
 | |
|       }
 | |
|       off += 8;
 | |
|       break;
 | |
| 
 | |
|     case 10:
 | |
| 
 | |
|       if (unlikely(off + 32 > compressed_size)) {
 | |
|         elf_uncompress_failed();
 | |
|         return 0;
 | |
|       }
 | |
|       off += 32;
 | |
|       break;
 | |
| 
 | |
|     default:
 | |
|       elf_uncompress_failed();
 | |
|       return 0;
 | |
|   }
 | |
| 
 | |
|   *poffset = off;
 | |
| 
 | |
|   return 1;
 | |
| }
 | |
| 
 | |
| static int elf_uncompress_lzma(struct backtrace_state *state,
 | |
|                                const unsigned char *compressed,
 | |
|                                size_t compressed_size,
 | |
|                                backtrace_error_callback error_callback,
 | |
|                                void *data, unsigned char **uncompressed,
 | |
|                                size_t *uncompressed_size) {
 | |
|   size_t header_size;
 | |
|   size_t footer_size;
 | |
|   unsigned char check;
 | |
|   uint32_t computed_crc;
 | |
|   uint32_t stream_crc;
 | |
|   size_t offset;
 | |
|   size_t index_size;
 | |
|   size_t footer_offset;
 | |
|   size_t index_offset;
 | |
|   uint64_t index_compressed_size;
 | |
|   uint64_t index_uncompressed_size;
 | |
|   unsigned char *mem;
 | |
|   uint16_t *probs;
 | |
|   size_t compressed_block_size;
 | |
| 
 | |
|   header_size = 12;
 | |
|   footer_size = 12;
 | |
|   if (unlikely(compressed_size < header_size + footer_size)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   if (unlikely(memcmp(compressed,
 | |
|                       "\375"
 | |
|                       "7zXZ\0",
 | |
|                       6) != 0)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   if (unlikely(compressed[6] != 0)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
|   check = compressed[7];
 | |
|   if (unlikely((check & 0xf8) != 0)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   computed_crc = elf_crc32(0, compressed + 6, 2);
 | |
|   stream_crc = (compressed[8] | (compressed[9] << 8) | (compressed[10] << 16) |
 | |
|                 (compressed[11] << 24));
 | |
|   if (unlikely(computed_crc != stream_crc)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   offset = compressed_size;
 | |
|   if (unlikely(memcmp(compressed + offset - 2, "YZ", 2) != 0)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
|   offset -= 2;
 | |
| 
 | |
|   if (unlikely(compressed[offset - 2] != 0 ||
 | |
|                compressed[offset - 1] != check)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
|   offset -= 2;
 | |
| 
 | |
|   index_size =
 | |
|       (compressed[offset - 4] | (compressed[offset - 3] << 8) |
 | |
|        (compressed[offset - 2] << 16) | (compressed[offset - 1] << 24));
 | |
|   index_size = (index_size + 1) * 4;
 | |
|   offset -= 4;
 | |
| 
 | |
|   computed_crc = elf_crc32(0, compressed + offset, 6);
 | |
|   stream_crc =
 | |
|       (compressed[offset - 4] | (compressed[offset - 3] << 8) |
 | |
|        (compressed[offset - 2] << 16) | (compressed[offset - 1] << 24));
 | |
|   if (unlikely(computed_crc != stream_crc)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
|   offset -= 4;
 | |
| 
 | |
|   if (unlikely(offset < index_size + header_size)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   footer_offset = offset;
 | |
|   offset -= index_size;
 | |
|   index_offset = offset;
 | |
| 
 | |
|   if (unlikely(compressed[offset] != 0)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
|   ++offset;
 | |
| 
 | |
|   if (unlikely(compressed[offset] == 0)) {
 | |
|     *uncompressed = NULL;
 | |
|     *uncompressed_size = 0;
 | |
|     return 1;
 | |
|   }
 | |
|   if (unlikely(compressed[offset] != 1)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
|   ++offset;
 | |
| 
 | |
|   if (!elf_lzma_varint(compressed, compressed_size, &offset,
 | |
|                        &index_compressed_size))
 | |
|     return 0;
 | |
|   if (!elf_lzma_varint(compressed, compressed_size, &offset,
 | |
|                        &index_uncompressed_size))
 | |
|     return 0;
 | |
| 
 | |
|   offset = (offset + 3) & ~(size_t)3;
 | |
| 
 | |
|   computed_crc = elf_crc32(0, compressed + index_offset, offset - index_offset);
 | |
|   stream_crc =
 | |
|       (compressed[offset] | (compressed[offset + 1] << 8) |
 | |
|        (compressed[offset + 2] << 16) | (compressed[offset + 3] << 24));
 | |
|   if (unlikely(computed_crc != stream_crc)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
|   offset += 4;
 | |
| 
 | |
|   if (unlikely(offset != footer_offset)) {
 | |
|     elf_uncompress_failed();
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   mem = (unsigned char *)backtrace_alloc(state, index_uncompressed_size,
 | |
|                                          error_callback, data);
 | |
|   if (unlikely(mem == NULL)) return 0;
 | |
|   *uncompressed = mem;
 | |
|   *uncompressed_size = index_uncompressed_size;
 | |
| 
 | |
|   probs = ((uint16_t *)backtrace_alloc(
 | |
|       state, LZMA_PROB_TOTAL_COUNT * sizeof(uint16_t), error_callback, data));
 | |
|   if (unlikely(probs == NULL)) {
 | |
|     backtrace_free(state, mem, index_uncompressed_size, error_callback, data);
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   offset = 12;
 | |
|   if (!elf_uncompress_lzma_block(compressed, compressed_size, check, probs, mem,
 | |
|                                  index_uncompressed_size, &offset)) {
 | |
|     backtrace_free(state, mem, index_uncompressed_size, error_callback, data);
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   compressed_block_size = offset - 12;
 | |
|   if (unlikely(compressed_block_size !=
 | |
|                ((index_compressed_size + 3) & ~(size_t)3))) {
 | |
|     elf_uncompress_failed();
 | |
|     backtrace_free(state, mem, index_uncompressed_size, error_callback, data);
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   offset = (offset + 3) & ~(size_t)3;
 | |
|   if (unlikely(offset != index_offset)) {
 | |
|     elf_uncompress_failed();
 | |
|     backtrace_free(state, mem, index_uncompressed_size, error_callback, data);
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   return 1;
 | |
| }
 | |
| 
 | |
| int backtrace_uncompress_lzma(struct backtrace_state *state,
 | |
|                               const unsigned char *compressed,
 | |
|                               size_t compressed_size,
 | |
|                               backtrace_error_callback error_callback,
 | |
|                               void *data, unsigned char **uncompressed,
 | |
|                               size_t *uncompressed_size) {
 | |
|   return elf_uncompress_lzma(state, compressed, compressed_size, error_callback,
 | |
|                              data, uncompressed, uncompressed_size);
 | |
| }
 | |
| 
 | |
| static int elf_add(struct backtrace_state *state, const char *filename,
 | |
|                    int descriptor, const unsigned char *memory,
 | |
|                    size_t memory_size, uintptr_t base_address,
 | |
|                    backtrace_error_callback error_callback, void *data,
 | |
|                    fileline *fileline_fn, int *found_sym, int *found_dwarf,
 | |
|                    struct dwarf_data **fileline_entry, int exe, int debuginfo,
 | |
|                    const char *with_buildid_data, uint32_t with_buildid_size) {
 | |
|   struct elf_view ehdr_view;
 | |
|   b_elf_ehdr ehdr;
 | |
|   off_t shoff;
 | |
|   unsigned int shnum;
 | |
|   unsigned int shstrndx;
 | |
|   struct elf_view shdrs_view;
 | |
|   int shdrs_view_valid;
 | |
|   const b_elf_shdr *shdrs;
 | |
|   const b_elf_shdr *shstrhdr;
 | |
|   size_t shstr_size;
 | |
|   off_t shstr_off;
 | |
|   struct elf_view names_view;
 | |
|   int names_view_valid;
 | |
|   const char *names;
 | |
|   unsigned int symtab_shndx;
 | |
|   unsigned int dynsym_shndx;
 | |
|   unsigned int i;
 | |
|   struct debug_section_info sections[DEBUG_MAX];
 | |
|   struct debug_section_info zsections[DEBUG_MAX];
 | |
|   struct elf_view symtab_view;
 | |
|   int symtab_view_valid;
 | |
|   struct elf_view strtab_view;
 | |
|   int strtab_view_valid;
 | |
|   struct elf_view buildid_view;
 | |
|   int buildid_view_valid;
 | |
|   const char *buildid_data;
 | |
|   uint32_t buildid_size;
 | |
|   struct elf_view debuglink_view;
 | |
|   int debuglink_view_valid;
 | |
|   const char *debuglink_name;
 | |
|   uint32_t debuglink_crc;
 | |
|   struct elf_view debugaltlink_view;
 | |
|   int debugaltlink_view_valid;
 | |
|   const char *debugaltlink_name;
 | |
|   const char *debugaltlink_buildid_data;
 | |
|   uint32_t debugaltlink_buildid_size;
 | |
|   struct elf_view gnu_debugdata_view;
 | |
|   int gnu_debugdata_view_valid;
 | |
|   size_t gnu_debugdata_size;
 | |
|   unsigned char *gnu_debugdata_uncompressed;
 | |
|   size_t gnu_debugdata_uncompressed_size;
 | |
|   off_t min_offset;
 | |
|   off_t max_offset;
 | |
|   off_t debug_size;
 | |
|   struct elf_view debug_view;
 | |
|   int debug_view_valid;
 | |
|   unsigned int using_debug_view;
 | |
|   uint16_t *zdebug_table;
 | |
|   struct elf_view split_debug_view[DEBUG_MAX];
 | |
|   unsigned char split_debug_view_valid[DEBUG_MAX];
 | |
|   struct elf_ppc64_opd_data opd_data, *opd;
 | |
|   struct dwarf_sections dwarf_sections;
 | |
| 
 | |
|   if (!debuginfo) {
 | |
|     *found_sym = 0;
 | |
|     *found_dwarf = 0;
 | |
|   }
 | |
| 
 | |
|   shdrs_view_valid = 0;
 | |
|   names_view_valid = 0;
 | |
|   symtab_view_valid = 0;
 | |
|   strtab_view_valid = 0;
 | |
|   buildid_view_valid = 0;
 | |
|   buildid_data = NULL;
 | |
|   buildid_size = 0;
 | |
|   debuglink_view_valid = 0;
 | |
|   debuglink_name = NULL;
 | |
|   debuglink_crc = 0;
 | |
|   debugaltlink_view_valid = 0;
 | |
|   debugaltlink_name = NULL;
 | |
|   debugaltlink_buildid_data = NULL;
 | |
|   debugaltlink_buildid_size = 0;
 | |
|   gnu_debugdata_view_valid = 0;
 | |
|   gnu_debugdata_size = 0;
 | |
|   debug_view_valid = 0;
 | |
|   memset(&split_debug_view_valid[0], 0, sizeof split_debug_view_valid);
 | |
|   opd = NULL;
 | |
| 
 | |
|   if (!elf_get_view(state, descriptor, memory, memory_size, 0, sizeof ehdr,
 | |
|                     error_callback, data, &ehdr_view))
 | |
|     goto fail;
 | |
| 
 | |
|   memcpy(&ehdr, ehdr_view.view.data, sizeof ehdr);
 | |
| 
 | |
|   elf_release_view(state, &ehdr_view, error_callback, data);
 | |
| 
 | |
|   if (ehdr.e_ident[EI_MAG0] != ELFMAG0 || ehdr.e_ident[EI_MAG1] != ELFMAG1 ||
 | |
|       ehdr.e_ident[EI_MAG2] != ELFMAG2 || ehdr.e_ident[EI_MAG3] != ELFMAG3) {
 | |
|     error_callback(data, "executable file is not ELF", 0);
 | |
|     goto fail;
 | |
|   }
 | |
|   if (ehdr.e_ident[EI_VERSION] != EV_CURRENT) {
 | |
|     error_callback(data, "executable file is unrecognized ELF version", 0);
 | |
|     goto fail;
 | |
|   }
 | |
| 
 | |
| #if BACKTRACE_ELF_SIZE == 32
 | |
| #define BACKTRACE_ELFCLASS ELFCLASS32
 | |
| #else
 | |
| #define BACKTRACE_ELFCLASS ELFCLASS64
 | |
| #endif
 | |
| 
 | |
|   if (ehdr.e_ident[EI_CLASS] != BACKTRACE_ELFCLASS) {
 | |
|     error_callback(data, "executable file is unexpected ELF class", 0);
 | |
|     goto fail;
 | |
|   }
 | |
| 
 | |
|   if (ehdr.e_ident[EI_DATA] != ELFDATA2LSB &&
 | |
|       ehdr.e_ident[EI_DATA] != ELFDATA2MSB) {
 | |
|     error_callback(data, "executable file has unknown endianness", 0);
 | |
|     goto fail;
 | |
|   }
 | |
| 
 | |
|   if (exe && ehdr.e_type == ET_DYN) return -1;
 | |
| 
 | |
|   shoff = ehdr.e_shoff;
 | |
|   shnum = ehdr.e_shnum;
 | |
|   shstrndx = ehdr.e_shstrndx;
 | |
| 
 | |
|   if ((shnum == 0 || shstrndx == SHN_XINDEX) && shoff != 0) {
 | |
|     struct elf_view shdr_view;
 | |
|     const b_elf_shdr *shdr;
 | |
| 
 | |
|     if (!elf_get_view(state, descriptor, memory, memory_size, shoff,
 | |
|                       sizeof shdr, error_callback, data, &shdr_view))
 | |
|       goto fail;
 | |
| 
 | |
|     shdr = (const b_elf_shdr *)shdr_view.view.data;
 | |
| 
 | |
|     if (shnum == 0) shnum = shdr->sh_size;
 | |
| 
 | |
|     if (shstrndx == SHN_XINDEX) {
 | |
|       shstrndx = shdr->sh_link;
 | |
| 
 | |
|       if (shstrndx >= shnum && shstrndx >= SHN_LORESERVE + 0x100)
 | |
|         shstrndx -= 0x100;
 | |
|     }
 | |
| 
 | |
|     elf_release_view(state, &shdr_view, error_callback, data);
 | |
|   }
 | |
| 
 | |
|   if (shnum == 0 || shstrndx == 0) goto fail;
 | |
| 
 | |
|   if (!elf_get_view(
 | |
|           state, descriptor, memory, memory_size, shoff + sizeof(b_elf_shdr),
 | |
|           (shnum - 1) * sizeof(b_elf_shdr), error_callback, data, &shdrs_view))
 | |
|     goto fail;
 | |
|   shdrs_view_valid = 1;
 | |
|   shdrs = (const b_elf_shdr *)shdrs_view.view.data;
 | |
| 
 | |
|   shstrhdr = &shdrs[shstrndx - 1];
 | |
|   shstr_size = shstrhdr->sh_size;
 | |
|   shstr_off = shstrhdr->sh_offset;
 | |
| 
 | |
|   if (!elf_get_view(state, descriptor, memory, memory_size, shstr_off,
 | |
|                     shstrhdr->sh_size, error_callback, data, &names_view))
 | |
|     goto fail;
 | |
|   names_view_valid = 1;
 | |
|   names = (const char *)names_view.view.data;
 | |
| 
 | |
|   symtab_shndx = 0;
 | |
|   dynsym_shndx = 0;
 | |
| 
 | |
|   memset(sections, 0, sizeof sections);
 | |
|   memset(zsections, 0, sizeof zsections);
 | |
| 
 | |
|   for (i = 1; i < shnum; ++i) {
 | |
|     const b_elf_shdr *shdr;
 | |
|     unsigned int sh_name;
 | |
|     const char *name;
 | |
|     int j;
 | |
| 
 | |
|     shdr = &shdrs[i - 1];
 | |
| 
 | |
|     if (shdr->sh_type == SHT_SYMTAB)
 | |
|       symtab_shndx = i;
 | |
|     else if (shdr->sh_type == SHT_DYNSYM)
 | |
|       dynsym_shndx = i;
 | |
| 
 | |
|     sh_name = shdr->sh_name;
 | |
|     if (sh_name >= shstr_size) {
 | |
|       error_callback(data, "ELF section name out of range", 0);
 | |
|       goto fail;
 | |
|     }
 | |
| 
 | |
|     name = names + sh_name;
 | |
| 
 | |
|     for (j = 0; j < (int)DEBUG_MAX; ++j) {
 | |
|       if (strcmp(name, dwarf_section_names[j]) == 0) {
 | |
|         sections[j].offset = shdr->sh_offset;
 | |
|         sections[j].size = shdr->sh_size;
 | |
|         sections[j].compressed = (shdr->sh_flags & SHF_COMPRESSED) != 0;
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (name[0] == '.' && name[1] == 'z') {
 | |
|       for (j = 0; j < (int)DEBUG_MAX; ++j) {
 | |
|         if (strcmp(name + 2, dwarf_section_names[j] + 1) == 0) {
 | |
|           zsections[j].offset = shdr->sh_offset;
 | |
|           zsections[j].size = shdr->sh_size;
 | |
|           break;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if ((!debuginfo || with_buildid_data != NULL) && !buildid_view_valid &&
 | |
|         strcmp(name, ".note.gnu.build-id") == 0) {
 | |
|       const b_elf_note *note;
 | |
| 
 | |
|       if (!elf_get_view(state, descriptor, memory, memory_size, shdr->sh_offset,
 | |
|                         shdr->sh_size, error_callback, data, &buildid_view))
 | |
|         goto fail;
 | |
| 
 | |
|       buildid_view_valid = 1;
 | |
|       note = (const b_elf_note *)buildid_view.view.data;
 | |
|       if (note->type == NT_GNU_BUILD_ID && note->namesz == 4 &&
 | |
|           strncmp(note->name, "GNU", 4) == 0 &&
 | |
|           shdr->sh_size <= 12 + ((note->namesz + 3) & ~3) + note->descsz) {
 | |
|         buildid_data = ¬e->name[0] + ((note->namesz + 3) & ~3);
 | |
|         buildid_size = note->descsz;
 | |
|       }
 | |
| 
 | |
|       if (with_buildid_size != 0) {
 | |
|         if (buildid_size != with_buildid_size) goto fail;
 | |
| 
 | |
|         if (memcmp(buildid_data, with_buildid_data, buildid_size) != 0)
 | |
|           goto fail;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (!debuginfo && !debuglink_view_valid &&
 | |
|         strcmp(name, ".gnu_debuglink") == 0) {
 | |
|       const char *debuglink_data;
 | |
|       size_t crc_offset;
 | |
| 
 | |
|       if (!elf_get_view(state, descriptor, memory, memory_size, shdr->sh_offset,
 | |
|                         shdr->sh_size, error_callback, data, &debuglink_view))
 | |
|         goto fail;
 | |
| 
 | |
|       debuglink_view_valid = 1;
 | |
|       debuglink_data = (const char *)debuglink_view.view.data;
 | |
|       crc_offset = strnlen(debuglink_data, shdr->sh_size);
 | |
|       crc_offset = (crc_offset + 3) & ~3;
 | |
|       if (crc_offset + 4 <= shdr->sh_size) {
 | |
|         debuglink_name = debuglink_data;
 | |
|         debuglink_crc = *(const uint32_t *)(debuglink_data + crc_offset);
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (!debugaltlink_view_valid && strcmp(name, ".gnu_debugaltlink") == 0) {
 | |
|       const char *debugaltlink_data;
 | |
|       size_t debugaltlink_name_len;
 | |
| 
 | |
|       if (!elf_get_view(state, descriptor, memory, memory_size, shdr->sh_offset,
 | |
|                         shdr->sh_size, error_callback, data,
 | |
|                         &debugaltlink_view))
 | |
|         goto fail;
 | |
| 
 | |
|       debugaltlink_view_valid = 1;
 | |
|       debugaltlink_data = (const char *)debugaltlink_view.view.data;
 | |
|       debugaltlink_name = debugaltlink_data;
 | |
|       debugaltlink_name_len = strnlen(debugaltlink_data, shdr->sh_size);
 | |
|       if (debugaltlink_name_len < shdr->sh_size) {
 | |
|         debugaltlink_name_len += 1;
 | |
| 
 | |
|         debugaltlink_buildid_data = debugaltlink_data + debugaltlink_name_len;
 | |
|         debugaltlink_buildid_size = shdr->sh_size - debugaltlink_name_len;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (!gnu_debugdata_view_valid && strcmp(name, ".gnu_debugdata") == 0) {
 | |
|       if (!elf_get_view(state, descriptor, memory, memory_size, shdr->sh_offset,
 | |
|                         shdr->sh_size, error_callback, data,
 | |
|                         &gnu_debugdata_view))
 | |
|         goto fail;
 | |
| 
 | |
|       gnu_debugdata_size = shdr->sh_size;
 | |
|       gnu_debugdata_view_valid = 1;
 | |
|     }
 | |
| 
 | |
|     if (ehdr.e_machine == EM_PPC64 && (ehdr.e_flags & EF_PPC64_ABI) < 2 &&
 | |
|         shdr->sh_type == SHT_PROGBITS && strcmp(name, ".opd") == 0) {
 | |
|       if (!elf_get_view(state, descriptor, memory, memory_size, shdr->sh_offset,
 | |
|                         shdr->sh_size, error_callback, data, &opd_data.view))
 | |
|         goto fail;
 | |
| 
 | |
|       opd = &opd_data;
 | |
|       opd->addr = shdr->sh_addr;
 | |
|       opd->data = (const char *)opd_data.view.view.data;
 | |
|       opd->size = shdr->sh_size;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (symtab_shndx == 0) symtab_shndx = dynsym_shndx;
 | |
|   if (symtab_shndx != 0 && !debuginfo) {
 | |
|     const b_elf_shdr *symtab_shdr;
 | |
|     unsigned int strtab_shndx;
 | |
|     const b_elf_shdr *strtab_shdr;
 | |
|     struct elf_syminfo_data *sdata;
 | |
| 
 | |
|     symtab_shdr = &shdrs[symtab_shndx - 1];
 | |
|     strtab_shndx = symtab_shdr->sh_link;
 | |
|     if (strtab_shndx >= shnum) {
 | |
|       error_callback(data, "ELF symbol table strtab link out of range", 0);
 | |
|       goto fail;
 | |
|     }
 | |
|     strtab_shdr = &shdrs[strtab_shndx - 1];
 | |
| 
 | |
|     if (!elf_get_view(state, descriptor, memory, memory_size,
 | |
|                       symtab_shdr->sh_offset, symtab_shdr->sh_size,
 | |
|                       error_callback, data, &symtab_view))
 | |
|       goto fail;
 | |
|     symtab_view_valid = 1;
 | |
| 
 | |
|     if (!elf_get_view(state, descriptor, memory, memory_size,
 | |
|                       strtab_shdr->sh_offset, strtab_shdr->sh_size,
 | |
|                       error_callback, data, &strtab_view))
 | |
|       goto fail;
 | |
|     strtab_view_valid = 1;
 | |
| 
 | |
|     sdata = ((struct elf_syminfo_data *)backtrace_alloc(state, sizeof *sdata,
 | |
|                                                         error_callback, data));
 | |
|     if (sdata == NULL) goto fail;
 | |
| 
 | |
|     if (!elf_initialize_syminfo(state, base_address, symtab_view.view.data,
 | |
|                                 symtab_shdr->sh_size, strtab_view.view.data,
 | |
|                                 strtab_shdr->sh_size, error_callback, data,
 | |
|                                 sdata, opd)) {
 | |
|       backtrace_free(state, sdata, sizeof *sdata, error_callback, data);
 | |
|       goto fail;
 | |
|     }
 | |
| 
 | |
|     elf_release_view(state, &symtab_view, error_callback, data);
 | |
|     symtab_view_valid = 0;
 | |
|     strtab_view_valid = 0;
 | |
| 
 | |
|     *found_sym = 1;
 | |
| 
 | |
|     elf_add_syminfo_data(state, sdata);
 | |
|   }
 | |
| 
 | |
|   elf_release_view(state, &shdrs_view, error_callback, data);
 | |
|   shdrs_view_valid = 0;
 | |
|   elf_release_view(state, &names_view, error_callback, data);
 | |
|   names_view_valid = 0;
 | |
| 
 | |
|   if (buildid_data != NULL) {
 | |
|     int d;
 | |
| 
 | |
|     d = elf_open_debugfile_by_buildid(state, buildid_data, buildid_size,
 | |
|                                       error_callback, data);
 | |
|     if (d >= 0) {
 | |
|       int ret;
 | |
| 
 | |
|       elf_release_view(state, &buildid_view, error_callback, data);
 | |
|       if (debuglink_view_valid)
 | |
|         elf_release_view(state, &debuglink_view, error_callback, data);
 | |
|       if (debugaltlink_view_valid)
 | |
|         elf_release_view(state, &debugaltlink_view, error_callback, data);
 | |
|       ret = elf_add(state, "", d, NULL, 0, base_address, error_callback, data,
 | |
|                     fileline_fn, found_sym, found_dwarf, NULL, 0, 1, NULL, 0);
 | |
|       if (ret < 0)
 | |
|         backtrace_close(d, error_callback, data);
 | |
|       else if (descriptor >= 0)
 | |
|         backtrace_close(descriptor, error_callback, data);
 | |
|       return ret;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (buildid_view_valid) {
 | |
|     elf_release_view(state, &buildid_view, error_callback, data);
 | |
|     buildid_view_valid = 0;
 | |
|   }
 | |
| 
 | |
|   if (opd) {
 | |
|     elf_release_view(state, &opd->view, error_callback, data);
 | |
|     opd = NULL;
 | |
|   }
 | |
| 
 | |
|   if (debuglink_name != NULL) {
 | |
|     int d;
 | |
| 
 | |
|     d = elf_open_debugfile_by_debuglink(state, filename, debuglink_name,
 | |
|                                         debuglink_crc, error_callback, data);
 | |
|     if (d >= 0) {
 | |
|       int ret;
 | |
| 
 | |
|       elf_release_view(state, &debuglink_view, error_callback, data);
 | |
|       if (debugaltlink_view_valid)
 | |
|         elf_release_view(state, &debugaltlink_view, error_callback, data);
 | |
|       ret = elf_add(state, "", d, NULL, 0, base_address, error_callback, data,
 | |
|                     fileline_fn, found_sym, found_dwarf, NULL, 0, 1, NULL, 0);
 | |
|       if (ret < 0)
 | |
|         backtrace_close(d, error_callback, data);
 | |
|       else if (descriptor >= 0)
 | |
|         backtrace_close(descriptor, error_callback, data);
 | |
|       return ret;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (debuglink_view_valid) {
 | |
|     elf_release_view(state, &debuglink_view, error_callback, data);
 | |
|     debuglink_view_valid = 0;
 | |
|   }
 | |
| 
 | |
|   struct dwarf_data *fileline_altlink = NULL;
 | |
|   if (debugaltlink_name != NULL) {
 | |
|     int d;
 | |
| 
 | |
|     d = elf_open_debugfile_by_debuglink(state, filename, debugaltlink_name, 0,
 | |
|                                         error_callback, data);
 | |
|     if (d >= 0) {
 | |
|       int ret;
 | |
| 
 | |
|       ret =
 | |
|           elf_add(state, filename, d, NULL, 0, base_address, error_callback,
 | |
|                   data, fileline_fn, found_sym, found_dwarf, &fileline_altlink,
 | |
|                   0, 1, debugaltlink_buildid_data, debugaltlink_buildid_size);
 | |
|       elf_release_view(state, &debugaltlink_view, error_callback, data);
 | |
|       debugaltlink_view_valid = 0;
 | |
|       if (ret < 0) {
 | |
|         backtrace_close(d, error_callback, data);
 | |
|         return ret;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (debugaltlink_view_valid) {
 | |
|     elf_release_view(state, &debugaltlink_view, error_callback, data);
 | |
|     debugaltlink_view_valid = 0;
 | |
|   }
 | |
| 
 | |
|   if (gnu_debugdata_view_valid) {
 | |
|     int ret;
 | |
| 
 | |
|     ret = elf_uncompress_lzma(
 | |
|         state, ((const unsigned char *)gnu_debugdata_view.view.data),
 | |
|         gnu_debugdata_size, error_callback, data, &gnu_debugdata_uncompressed,
 | |
|         &gnu_debugdata_uncompressed_size);
 | |
| 
 | |
|     elf_release_view(state, &gnu_debugdata_view, error_callback, data);
 | |
|     gnu_debugdata_view_valid = 0;
 | |
| 
 | |
|     if (ret) {
 | |
|       ret = elf_add(state, filename, -1, gnu_debugdata_uncompressed,
 | |
|                     gnu_debugdata_uncompressed_size, base_address,
 | |
|                     error_callback, data, fileline_fn, found_sym, found_dwarf,
 | |
|                     NULL, 0, 0, NULL, 0);
 | |
|       if (ret >= 0 && descriptor >= 0)
 | |
|         backtrace_close(descriptor, error_callback, data);
 | |
|       return ret;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   min_offset = 0;
 | |
|   max_offset = 0;
 | |
|   debug_size = 0;
 | |
|   for (i = 0; i < (int)DEBUG_MAX; ++i) {
 | |
|     off_t end;
 | |
| 
 | |
|     if (sections[i].size != 0) {
 | |
|       if (min_offset == 0 || sections[i].offset < min_offset)
 | |
|         min_offset = sections[i].offset;
 | |
|       end = sections[i].offset + sections[i].size;
 | |
|       if (end > max_offset) max_offset = end;
 | |
|       debug_size += sections[i].size;
 | |
|     }
 | |
|     if (zsections[i].size != 0) {
 | |
|       if (min_offset == 0 || zsections[i].offset < min_offset)
 | |
|         min_offset = zsections[i].offset;
 | |
|       end = zsections[i].offset + zsections[i].size;
 | |
|       if (end > max_offset) max_offset = end;
 | |
|       debug_size += zsections[i].size;
 | |
|     }
 | |
|   }
 | |
|   if (min_offset == 0 || max_offset == 0) {
 | |
|     if (descriptor >= 0) {
 | |
|       if (!backtrace_close(descriptor, error_callback, data)) goto fail;
 | |
|     }
 | |
|     return 1;
 | |
|   }
 | |
| 
 | |
|   if (max_offset - min_offset < 0x20000000 ||
 | |
|       max_offset - min_offset < debug_size + 0x10000) {
 | |
|     if (!elf_get_view(state, descriptor, memory, memory_size, min_offset,
 | |
|                       max_offset - min_offset, error_callback, data,
 | |
|                       &debug_view))
 | |
|       goto fail;
 | |
|     debug_view_valid = 1;
 | |
|   } else {
 | |
|     memset(&split_debug_view[0], 0, sizeof split_debug_view);
 | |
|     for (i = 0; i < (int)DEBUG_MAX; ++i) {
 | |
|       struct debug_section_info *dsec;
 | |
| 
 | |
|       if (sections[i].size != 0)
 | |
|         dsec = §ions[i];
 | |
|       else if (zsections[i].size != 0)
 | |
|         dsec = &zsections[i];
 | |
|       else
 | |
|         continue;
 | |
| 
 | |
|       if (!elf_get_view(state, descriptor, memory, memory_size, dsec->offset,
 | |
|                         dsec->size, error_callback, data, &split_debug_view[i]))
 | |
|         goto fail;
 | |
|       split_debug_view_valid[i] = 1;
 | |
| 
 | |
|       if (sections[i].size != 0)
 | |
|         sections[i].data =
 | |
|             ((const unsigned char *)split_debug_view[i].view.data);
 | |
|       else
 | |
|         zsections[i].data =
 | |
|             ((const unsigned char *)split_debug_view[i].view.data);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (descriptor >= 0) {
 | |
|     if (!backtrace_close(descriptor, error_callback, data)) goto fail;
 | |
|     descriptor = -1;
 | |
|   }
 | |
| 
 | |
|   using_debug_view = 0;
 | |
|   if (debug_view_valid) {
 | |
|     for (i = 0; i < (int)DEBUG_MAX; ++i) {
 | |
|       if (sections[i].size == 0)
 | |
|         sections[i].data = NULL;
 | |
|       else {
 | |
|         sections[i].data = ((const unsigned char *)debug_view.view.data +
 | |
|                             (sections[i].offset - min_offset));
 | |
|         ++using_debug_view;
 | |
|       }
 | |
| 
 | |
|       if (zsections[i].size == 0)
 | |
|         zsections[i].data = NULL;
 | |
|       else
 | |
|         zsections[i].data = ((const unsigned char *)debug_view.view.data +
 | |
|                              (zsections[i].offset - min_offset));
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   zdebug_table = NULL;
 | |
|   for (i = 0; i < (int)DEBUG_MAX; ++i) {
 | |
|     if (sections[i].size == 0 && zsections[i].size > 0) {
 | |
|       unsigned char *uncompressed_data;
 | |
|       size_t uncompressed_size;
 | |
| 
 | |
|       if (zdebug_table == NULL) {
 | |
|         zdebug_table = ((uint16_t *)backtrace_alloc(state, ZDEBUG_TABLE_SIZE,
 | |
|                                                     error_callback, data));
 | |
|         if (zdebug_table == NULL) goto fail;
 | |
|       }
 | |
| 
 | |
|       uncompressed_data = NULL;
 | |
|       uncompressed_size = 0;
 | |
|       if (!elf_uncompress_zdebug(state, zsections[i].data, zsections[i].size,
 | |
|                                  zdebug_table, error_callback, data,
 | |
|                                  &uncompressed_data, &uncompressed_size))
 | |
|         goto fail;
 | |
|       sections[i].data = uncompressed_data;
 | |
|       sections[i].size = uncompressed_size;
 | |
|       sections[i].compressed = 0;
 | |
| 
 | |
|       if (split_debug_view_valid[i]) {
 | |
|         elf_release_view(state, &split_debug_view[i], error_callback, data);
 | |
|         split_debug_view_valid[i] = 0;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   for (i = 0; i < (int)DEBUG_MAX; ++i) {
 | |
|     unsigned char *uncompressed_data;
 | |
|     size_t uncompressed_size;
 | |
| 
 | |
|     if (sections[i].size == 0 || !sections[i].compressed) continue;
 | |
| 
 | |
|     if (zdebug_table == NULL) {
 | |
|       zdebug_table = ((uint16_t *)backtrace_alloc(state, ZDEBUG_TABLE_SIZE,
 | |
|                                                   error_callback, data));
 | |
|       if (zdebug_table == NULL) goto fail;
 | |
|     }
 | |
| 
 | |
|     uncompressed_data = NULL;
 | |
|     uncompressed_size = 0;
 | |
|     if (!elf_uncompress_chdr(state, sections[i].data, sections[i].size,
 | |
|                              zdebug_table, error_callback, data,
 | |
|                              &uncompressed_data, &uncompressed_size))
 | |
|       goto fail;
 | |
|     sections[i].data = uncompressed_data;
 | |
|     sections[i].size = uncompressed_size;
 | |
|     sections[i].compressed = 0;
 | |
| 
 | |
|     if (debug_view_valid)
 | |
|       --using_debug_view;
 | |
|     else if (split_debug_view_valid[i]) {
 | |
|       elf_release_view(state, &split_debug_view[i], error_callback, data);
 | |
|       split_debug_view_valid[i] = 0;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (zdebug_table != NULL)
 | |
|     backtrace_free(state, zdebug_table, ZDEBUG_TABLE_SIZE, error_callback,
 | |
|                    data);
 | |
| 
 | |
|   if (debug_view_valid && using_debug_view == 0) {
 | |
|     elf_release_view(state, &debug_view, error_callback, data);
 | |
|     debug_view_valid = 0;
 | |
|   }
 | |
| 
 | |
|   for (i = 0; i < (int)DEBUG_MAX; ++i) {
 | |
|     dwarf_sections.data[i] = sections[i].data;
 | |
|     dwarf_sections.size[i] = sections[i].size;
 | |
|   }
 | |
| 
 | |
|   if (!backtrace_dwarf_add(state, base_address, &dwarf_sections,
 | |
|                            ehdr.e_ident[EI_DATA] == ELFDATA2MSB,
 | |
|                            fileline_altlink, error_callback, data, fileline_fn,
 | |
|                            fileline_entry))
 | |
|     goto fail;
 | |
| 
 | |
|   *found_dwarf = 1;
 | |
| 
 | |
|   return 1;
 | |
| 
 | |
| fail:
 | |
|   if (shdrs_view_valid)
 | |
|     elf_release_view(state, &shdrs_view, error_callback, data);
 | |
|   if (names_view_valid)
 | |
|     elf_release_view(state, &names_view, error_callback, data);
 | |
|   if (symtab_view_valid)
 | |
|     elf_release_view(state, &symtab_view, error_callback, data);
 | |
|   if (strtab_view_valid)
 | |
|     elf_release_view(state, &strtab_view, error_callback, data);
 | |
|   if (debuglink_view_valid)
 | |
|     elf_release_view(state, &debuglink_view, error_callback, data);
 | |
|   if (debugaltlink_view_valid)
 | |
|     elf_release_view(state, &debugaltlink_view, error_callback, data);
 | |
|   if (gnu_debugdata_view_valid)
 | |
|     elf_release_view(state, &gnu_debugdata_view, error_callback, data);
 | |
|   if (buildid_view_valid)
 | |
|     elf_release_view(state, &buildid_view, error_callback, data);
 | |
|   if (debug_view_valid)
 | |
|     elf_release_view(state, &debug_view, error_callback, data);
 | |
|   for (i = 0; i < (int)DEBUG_MAX; ++i) {
 | |
|     if (split_debug_view_valid[i])
 | |
|       elf_release_view(state, &split_debug_view[i], error_callback, data);
 | |
|   }
 | |
|   if (opd) elf_release_view(state, &opd->view, error_callback, data);
 | |
|   if (descriptor >= 0) backtrace_close(descriptor, error_callback, data);
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| struct phdr_data {
 | |
|   struct backtrace_state *state;
 | |
|   backtrace_error_callback error_callback;
 | |
|   void *data;
 | |
|   fileline *fileline_fn;
 | |
|   int *found_sym;
 | |
|   int *found_dwarf;
 | |
|   const char *exe_filename;
 | |
|   int exe_descriptor;
 | |
| };
 | |
| 
 | |
| static int
 | |
| #ifdef __i386__
 | |
|     __attribute__((__force_align_arg_pointer__))
 | |
| #endif
 | |
|     phdr_callback(struct dl_phdr_info *info, size_t size ATTRIBUTE_UNUSED,
 | |
|                   void *pdata) {
 | |
|   struct phdr_data *pd = (struct phdr_data *)pdata;
 | |
|   const char *filename;
 | |
|   int descriptor;
 | |
|   int does_not_exist;
 | |
|   fileline elf_fileline_fn;
 | |
|   int found_dwarf;
 | |
| 
 | |
|   if (info->dlpi_name == NULL || info->dlpi_name[0] == '\0') {
 | |
|     if (pd->exe_descriptor == -1) return 0;
 | |
|     filename = pd->exe_filename;
 | |
|     descriptor = pd->exe_descriptor;
 | |
|     pd->exe_descriptor = -1;
 | |
|   } else {
 | |
|     if (pd->exe_descriptor != -1) {
 | |
|       backtrace_close(pd->exe_descriptor, pd->error_callback, pd->data);
 | |
|       pd->exe_descriptor = -1;
 | |
|     }
 | |
| 
 | |
|     filename = info->dlpi_name;
 | |
|     descriptor = backtrace_open(info->dlpi_name, pd->error_callback, pd->data,
 | |
|                                 &does_not_exist);
 | |
|     if (descriptor < 0) return 0;
 | |
|   }
 | |
| 
 | |
|   if (elf_add(pd->state, filename, descriptor, NULL, 0, info->dlpi_addr,
 | |
|               pd->error_callback, pd->data, &elf_fileline_fn, pd->found_sym,
 | |
|               &found_dwarf, NULL, 0, 0, NULL, 0)) {
 | |
|     if (found_dwarf) {
 | |
|       *pd->found_dwarf = 1;
 | |
|       *pd->fileline_fn = elf_fileline_fn;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| int backtrace_initialize(struct backtrace_state *state, const char *filename,
 | |
|                          int descriptor,
 | |
|                          backtrace_error_callback error_callback, void *data,
 | |
|                          fileline *fileline_fn) {
 | |
|   int ret;
 | |
|   int found_sym;
 | |
|   int found_dwarf;
 | |
|   fileline elf_fileline_fn = elf_nodebug;
 | |
|   struct phdr_data pd;
 | |
| 
 | |
|   ret =
 | |
|       elf_add(state, filename, descriptor, NULL, 0, 0, error_callback, data,
 | |
|               &elf_fileline_fn, &found_sym, &found_dwarf, NULL, 1, 0, NULL, 0);
 | |
|   if (!ret) return 0;
 | |
| 
 | |
|   pd.state = state;
 | |
|   pd.error_callback = error_callback;
 | |
|   pd.data = data;
 | |
|   pd.fileline_fn = &elf_fileline_fn;
 | |
|   pd.found_sym = &found_sym;
 | |
|   pd.found_dwarf = &found_dwarf;
 | |
|   pd.exe_filename = filename;
 | |
|   pd.exe_descriptor = ret < 0 ? descriptor : -1;
 | |
| 
 | |
|   dl_iterate_phdr(phdr_callback, (void *)&pd);
 | |
| 
 | |
|   if (!state->threaded) {
 | |
|     if (found_sym)
 | |
|       state->syminfo_fn = elf_syminfo;
 | |
|     else if (state->syminfo_fn == NULL)
 | |
|       state->syminfo_fn = elf_nosyms;
 | |
|   } else {
 | |
|     if (found_sym)
 | |
|       backtrace_atomic_store_pointer(&state->syminfo_fn, elf_syminfo);
 | |
|     else
 | |
|       (void)__sync_bool_compare_and_swap(&state->syminfo_fn, NULL, elf_nosyms);
 | |
|   }
 | |
| 
 | |
|   if (!state->threaded)
 | |
|     *fileline_fn = state->fileline_fn;
 | |
|   else
 | |
|     *fileline_fn = backtrace_atomic_load_pointer(&state->fileline_fn);
 | |
| 
 | |
|   if (*fileline_fn == NULL || *fileline_fn == elf_nodebug)
 | |
|     *fileline_fn = elf_fileline_fn;
 | |
| 
 | |
|   return 1;
 | |
| }
 | |
| 
 | |
| // mmapio.c:
 | |
| #include <errno.h>
 | |
| #include <sys/mman.h>
 | |
| #include <sys/types.h>
 | |
| #include <unistd.h>
 | |
| 
 | |
| #ifndef HAVE_DECL_GETPAGESIZE
 | |
| extern int getpagesize(void);
 | |
| #endif
 | |
| 
 | |
| #ifndef MAP_FAILED
 | |
| #define MAP_FAILED ((void *)-1)
 | |
| #endif
 | |
| 
 | |
| int backtrace_get_view(struct backtrace_state *state ATTRIBUTE_UNUSED,
 | |
|                        int descriptor, off_t offset, uint64_t size,
 | |
|                        backtrace_error_callback error_callback, void *data,
 | |
|                        struct backtrace_view *view) {
 | |
|   size_t pagesize;
 | |
|   unsigned int inpage;
 | |
|   off_t pageoff;
 | |
|   void *map;
 | |
| 
 | |
|   if ((uint64_t)(size_t)size != size) {
 | |
|     error_callback(data, "file size too large", 0);
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   pagesize = getpagesize();
 | |
|   inpage = offset % pagesize;
 | |
|   pageoff = offset - inpage;
 | |
| 
 | |
|   size += inpage;
 | |
|   size = (size + (pagesize - 1)) & ~(pagesize - 1);
 | |
| 
 | |
|   map = mmap(NULL, size, PROT_READ, MAP_PRIVATE, descriptor, pageoff);
 | |
|   if (map == MAP_FAILED) {
 | |
|     error_callback(data, "mmap", errno);
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   view->data = (char *)map + inpage;
 | |
|   view->base = map;
 | |
|   view->len = size;
 | |
| 
 | |
|   return 1;
 | |
| }
 | |
| 
 | |
| void backtrace_release_view(struct backtrace_state *state ATTRIBUTE_UNUSED,
 | |
|                             struct backtrace_view *view,
 | |
|                             backtrace_error_callback error_callback,
 | |
|                             void *data) {
 | |
|   union {
 | |
|     const void *cv;
 | |
|     void *v;
 | |
|   } const_cast;
 | |
| 
 | |
|   const_cast.cv = view->base;
 | |
|   if (munmap(const_cast.v, view->len) < 0)
 | |
|     error_callback(data, "munmap", errno);
 | |
| }
 | |
| 
 | |
| // mmap.c:
 | |
| #include <errno.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <sys/mman.h>
 | |
| #include <sys/types.h>
 | |
| #include <unistd.h>
 | |
| 
 | |
| #ifndef HAVE_DECL_GETPAGESIZE
 | |
| extern int getpagesize(void);
 | |
| #endif
 | |
| 
 | |
| #ifndef MAP_ANONYMOUS
 | |
| #define MAP_ANONYMOUS MAP_ANON
 | |
| #endif
 | |
| 
 | |
| #ifndef MAP_FAILED
 | |
| #define MAP_FAILED ((void *)-1)
 | |
| #endif
 | |
| 
 | |
| struct backtrace_freelist_struct {
 | |
|   struct backtrace_freelist_struct *next;
 | |
| 
 | |
|   size_t size;
 | |
| };
 | |
| 
 | |
| static void backtrace_free_locked(struct backtrace_state *state, void *addr,
 | |
|                                   size_t size) {
 | |
|   if (size >= sizeof(struct backtrace_freelist_struct)) {
 | |
|     size_t c;
 | |
|     struct backtrace_freelist_struct **ppsmall;
 | |
|     struct backtrace_freelist_struct **pp;
 | |
|     struct backtrace_freelist_struct *p;
 | |
| 
 | |
|     c = 0;
 | |
|     ppsmall = NULL;
 | |
|     for (pp = &state->freelist; *pp != NULL; pp = &(*pp)->next) {
 | |
|       if (ppsmall == NULL || (*pp)->size < (*ppsmall)->size) ppsmall = pp;
 | |
|       ++c;
 | |
|     }
 | |
|     if (c >= 16) {
 | |
|       if (size <= (*ppsmall)->size) return;
 | |
|       *ppsmall = (*ppsmall)->next;
 | |
|     }
 | |
| 
 | |
|     p = (struct backtrace_freelist_struct *)addr;
 | |
|     p->next = state->freelist;
 | |
|     p->size = size;
 | |
|     state->freelist = p;
 | |
|   }
 | |
| }
 | |
| 
 | |
| void *backtrace_alloc(struct backtrace_state *state, size_t size,
 | |
|                       backtrace_error_callback error_callback, void *data) {
 | |
|   void *ret;
 | |
|   int locked;
 | |
|   struct backtrace_freelist_struct **pp;
 | |
|   size_t pagesize;
 | |
|   size_t asksize;
 | |
|   void *page;
 | |
| 
 | |
|   ret = NULL;
 | |
| 
 | |
|   if (!state->threaded)
 | |
|     locked = 1;
 | |
|   else
 | |
|     locked = __sync_lock_test_and_set(&state->lock_alloc, 1) == 0;
 | |
| 
 | |
|   if (locked) {
 | |
|     for (pp = &state->freelist; *pp != NULL; pp = &(*pp)->next) {
 | |
|       if ((*pp)->size >= size) {
 | |
|         struct backtrace_freelist_struct *p;
 | |
| 
 | |
|         p = *pp;
 | |
|         *pp = p->next;
 | |
| 
 | |
|         size = (size + 7) & ~(size_t)7;
 | |
|         if (size < p->size)
 | |
|           backtrace_free_locked(state, (char *)p + size, p->size - size);
 | |
| 
 | |
|         ret = (void *)p;
 | |
| 
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     if (state->threaded) __sync_lock_release(&state->lock_alloc);
 | |
|   }
 | |
| 
 | |
|   if (ret == NULL) {
 | |
|     pagesize = getpagesize();
 | |
|     asksize = (size + pagesize - 1) & ~(pagesize - 1);
 | |
|     page = mmap(NULL, asksize, PROT_READ | PROT_WRITE,
 | |
|                 MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
 | |
|     if (page == MAP_FAILED) {
 | |
|       if (error_callback) error_callback(data, "mmap", errno);
 | |
|     } else {
 | |
|       size = (size + 7) & ~(size_t)7;
 | |
|       if (size < asksize)
 | |
|         backtrace_free(state, (char *)page + size, asksize - size,
 | |
|                        error_callback, data);
 | |
| 
 | |
|       ret = page;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| void backtrace_free(struct backtrace_state *state, void *addr, size_t size,
 | |
|                     backtrace_error_callback error_callback ATTRIBUTE_UNUSED,
 | |
|                     void *data ATTRIBUTE_UNUSED) {
 | |
|   int locked;
 | |
| 
 | |
|   if (size >= 16 * 4096) {
 | |
|     size_t pagesize;
 | |
| 
 | |
|     pagesize = getpagesize();
 | |
|     if (((uintptr_t)addr & (pagesize - 1)) == 0 &&
 | |
|         (size & (pagesize - 1)) == 0) {
 | |
|       if (munmap(addr, size) == 0) return;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!state->threaded)
 | |
|     locked = 1;
 | |
|   else
 | |
|     locked = __sync_lock_test_and_set(&state->lock_alloc, 1) == 0;
 | |
| 
 | |
|   if (locked) {
 | |
|     backtrace_free_locked(state, addr, size);
 | |
| 
 | |
|     if (state->threaded) __sync_lock_release(&state->lock_alloc);
 | |
|   }
 | |
| }
 | |
| 
 | |
| void *backtrace_vector_grow(struct backtrace_state *state, size_t size,
 | |
|                             backtrace_error_callback error_callback, void *data,
 | |
|                             struct backtrace_vector *vec) {
 | |
|   void *ret;
 | |
| 
 | |
|   if (size > vec->alc) {
 | |
|     size_t pagesize;
 | |
|     size_t alc;
 | |
|     void *base;
 | |
| 
 | |
|     pagesize = getpagesize();
 | |
|     alc = vec->size + size;
 | |
|     if (vec->size == 0)
 | |
|       alc = 16 * size;
 | |
|     else if (alc < pagesize) {
 | |
|       alc *= 2;
 | |
|       if (alc > pagesize) alc = pagesize;
 | |
|     } else {
 | |
|       alc *= 2;
 | |
|       alc = (alc + pagesize - 1) & ~(pagesize - 1);
 | |
|     }
 | |
|     base = backtrace_alloc(state, alc, error_callback, data);
 | |
|     if (base == NULL) return NULL;
 | |
|     if (vec->base != NULL) {
 | |
|       memcpy(base, vec->base, vec->size);
 | |
|       backtrace_free(state, vec->base, vec->size + vec->alc, error_callback,
 | |
|                      data);
 | |
|     }
 | |
|     vec->base = base;
 | |
|     vec->alc = alc - vec->size;
 | |
|   }
 | |
| 
 | |
|   ret = (char *)vec->base + vec->size;
 | |
|   vec->size += size;
 | |
|   vec->alc -= size;
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| void *backtrace_vector_finish(struct backtrace_state *state ATTRIBUTE_UNUSED,
 | |
|                               struct backtrace_vector *vec,
 | |
|                               backtrace_error_callback error_callback
 | |
|                                   ATTRIBUTE_UNUSED,
 | |
|                               void *data ATTRIBUTE_UNUSED) {
 | |
|   void *ret;
 | |
| 
 | |
|   ret = vec->base;
 | |
|   vec->base = (char *)vec->base + vec->size;
 | |
|   vec->size = 0;
 | |
|   return ret;
 | |
| }
 | |
| 
 | |
| int backtrace_vector_release(struct backtrace_state *state,
 | |
|                              struct backtrace_vector *vec,
 | |
|                              backtrace_error_callback error_callback,
 | |
|                              void *data) {
 | |
|   size_t size;
 | |
|   size_t alc;
 | |
|   size_t aligned;
 | |
| 
 | |
|   size = vec->size;
 | |
|   alc = vec->alc;
 | |
|   aligned = (size + 7) & ~(size_t)7;
 | |
|   alc -= aligned - size;
 | |
| 
 | |
|   backtrace_free(state, (char *)vec->base + aligned, alc, error_callback, data);
 | |
|   vec->alc = 0;
 | |
|   if (vec->size == 0) vec->base = NULL;
 | |
|   return 1;
 | |
| }
 |