// NOTE: Portions of the code have been modified in order to fix compilation in TCC - Ned

// backtrace.h:
#ifndef BACKTRACE_H
#define BACKTRACE_H

#include <stddef.h>
#include <stdint.h>
#include <stdio.h>

#ifdef __cplusplus
extern "C" {
#endif

struct backtrace_state;

typedef void (*backtrace_error_callback)(void *data, const char *msg,
                                         int errnum);

extern struct backtrace_state *backtrace_create_state(
    const char *filename, int threaded, backtrace_error_callback error_callback,
    void *data);

typedef int (*backtrace_full_callback)(void *data, uintptr_t pc,
                                       const char *filename, int lineno,
                                       const char *function);

extern int backtrace_full(struct backtrace_state *state, int skip,
                          backtrace_full_callback callback,
                          backtrace_error_callback error_callback, void *data);

typedef int (*backtrace_simple_callback)(void *data, uintptr_t pc);

extern int backtrace_simple(struct backtrace_state *state, int skip,
                            backtrace_simple_callback callback,
                            backtrace_error_callback error_callback,
                            void *data);

extern void backtrace_print(struct backtrace_state *state, int skip, FILE *);

extern int backtrace_pcinfo(struct backtrace_state *state, uintptr_t pc,
                            backtrace_full_callback callback,
                            backtrace_error_callback error_callback,
                            void *data);

typedef void (*backtrace_syminfo_callback)(void *data, uintptr_t pc,
                                           const char *symname,
                                           uintptr_t symval, uintptr_t symsize);

extern int backtrace_syminfo(struct backtrace_state *state, uintptr_t addr,
                             backtrace_syminfo_callback callback,
                             backtrace_error_callback error_callback,
                             void *data);

#ifdef __cplusplus
}
#endif

#endif

// internal.h:
#ifndef BACKTRACE_INTERNAL_H
#define BACKTRACE_INTERNAL_H

#ifndef GCC_VERSION
#define GCC_VERSION (__GNUC__ * 1000 + __GNUC_MINOR__)
#endif

#if (GCC_VERSION < 2007)
#define __attribute__(x)
#endif

#ifndef ATTRIBUTE_UNUSED
#define ATTRIBUTE_UNUSED __attribute__((__unused__))
#endif

#ifndef ATTRIBUTE_MALLOC
#if (GCC_VERSION >= 2096)
#define ATTRIBUTE_MALLOC __attribute__((__malloc__))
#else
#define ATTRIBUTE_MALLOC
#endif
#endif

#ifndef ATTRIBUTE_FALLTHROUGH
#if (GCC_VERSION >= 7000)
#define ATTRIBUTE_FALLTHROUGH __attribute__((__fallthrough__))
#else
#define ATTRIBUTE_FALLTHROUGH
#endif
#endif

#ifndef HAVE_SYNC_FUNCTIONS

#define __sync_bool_compare_and_swap(A, B, C) (abort(), 1)
#define __sync_lock_test_and_set(A, B) (abort(), 0)
#define __sync_lock_release(A) abort()

#endif
#ifdef HAVE_ATOMIC_FUNCTIONS

#define backtrace_atomic_load_pointer(p) __atomic_load_n((p), __ATOMIC_ACQUIRE)
#define backtrace_atomic_load_int(p) __atomic_load_n((p), __ATOMIC_ACQUIRE)
#define backtrace_atomic_store_pointer(p, v) \
  __atomic_store_n((p), (v), __ATOMIC_RELEASE)
#define backtrace_atomic_store_size_t(p, v) \
  __atomic_store_n((p), (v), __ATOMIC_RELEASE)
#define backtrace_atomic_store_int(p, v) \
  __atomic_store_n((p), (v), __ATOMIC_RELEASE)

#else
#ifdef HAVE_SYNC_FUNCTIONS

extern void *backtrace_atomic_load_pointer(void *);
extern int backtrace_atomic_load_int(int *);
extern void backtrace_atomic_store_pointer(void *, void *);
extern void backtrace_atomic_store_size_t(size_t *, size_t);
extern void backtrace_atomic_store_int(int *, int);

#else

#define backtrace_atomic_load_pointer(p) (abort(), (void *)NULL)
#define backtrace_atomic_load_int(p) (abort(), 0)
#define backtrace_atomic_store_pointer(p, v) abort()
#define backtrace_atomic_store_size_t(p, v) abort()
#define backtrace_atomic_store_int(p, v) abort()

#endif
#endif

typedef int (*fileline)(struct backtrace_state *state, uintptr_t pc,
                        backtrace_full_callback callback,
                        backtrace_error_callback error_callback, void *data);

typedef void (*syminfo)(struct backtrace_state *state, uintptr_t pc,
                        backtrace_syminfo_callback callback,
                        backtrace_error_callback error_callback, void *data);

struct backtrace_state {
  const char *filename;

  int threaded;

  void *lock;

  fileline fileline_fn;

  void *fileline_data;

  syminfo syminfo_fn;

  void *syminfo_data;

  int fileline_initialization_failed;

  int lock_alloc;

  struct backtrace_freelist_struct *freelist;
};

extern int backtrace_open(const char *filename,
                          backtrace_error_callback error_callback, void *data,
                          int *does_not_exist);

struct backtrace_view {
  const void *data;

  void *base;

  size_t len;
};

extern int backtrace_get_view(struct backtrace_state *state, int descriptor,
                              off_t offset, uint64_t size,
                              backtrace_error_callback error_callback,
                              void *data, struct backtrace_view *view);

extern void backtrace_release_view(struct backtrace_state *state,
                                   struct backtrace_view *view,
                                   backtrace_error_callback error_callback,
                                   void *data);

extern int backtrace_close(int descriptor,
                           backtrace_error_callback error_callback, void *data);

extern void backtrace_qsort(void *base, size_t count, size_t size,
                            int (*compar)(const void *, const void *));

extern void *backtrace_alloc(struct backtrace_state *state, size_t size,
                             backtrace_error_callback error_callback,
                             void *data) ATTRIBUTE_MALLOC;

extern void backtrace_free(struct backtrace_state *state, void *mem,
                           size_t size, backtrace_error_callback error_callback,
                           void *data);

struct backtrace_vector {
  void *base;

  size_t size;

  size_t alc;
};

extern void *backtrace_vector_grow(struct backtrace_state *state, size_t size,
                                   backtrace_error_callback error_callback,
                                   void *data, struct backtrace_vector *vec);

extern void *backtrace_vector_finish(struct backtrace_state *state,
                                     struct backtrace_vector *vec,
                                     backtrace_error_callback error_callback,
                                     void *data);

extern int backtrace_vector_release(struct backtrace_state *state,
                                    struct backtrace_vector *vec,
                                    backtrace_error_callback error_callback,
                                    void *data);

static inline void backtrace_vector_free(
    struct backtrace_state *state, struct backtrace_vector *vec,
    backtrace_error_callback error_callback, void *data) {
  vec->alc += vec->size;
  vec->size = 0;
  backtrace_vector_release(state, vec, error_callback, data);
}

extern int backtrace_initialize(struct backtrace_state *state,
                                const char *filename, int descriptor,
                                backtrace_error_callback error_callback,
                                void *data, fileline *fileline_fn);

enum dwarf_section {
  DEBUG_INFO,
  DEBUG_LINE,
  DEBUG_ABBREV,
  DEBUG_RANGES,
  DEBUG_STR,
  DEBUG_ADDR,
  DEBUG_STR_OFFSETS,
  DEBUG_LINE_STR,
  DEBUG_RNGLISTS,

  DEBUG_MAX
};

struct dwarf_sections {
  const unsigned char *data[DEBUG_MAX];
  size_t size[DEBUG_MAX];
};

struct dwarf_data;

extern int backtrace_dwarf_add(struct backtrace_state *state,
                               uintptr_t base_address,
                               const struct dwarf_sections *dwarf_sections,
                               int is_bigendian,
                               struct dwarf_data *fileline_altlink,
                               backtrace_error_callback error_callback,
                               void *data, fileline *fileline_fn,
                               struct dwarf_data **fileline_entry);

struct backtrace_call_full {
  backtrace_full_callback full_callback;
  backtrace_error_callback full_error_callback;
  void *full_data;
  int ret;
};

extern void backtrace_syminfo_to_full_callback(void *data, uintptr_t pc,
                                               const char *symname,
                                               uintptr_t symval,
                                               uintptr_t symsize);

extern void backtrace_syminfo_to_full_error_callback(void *, const char *, int);

extern int backtrace_uncompress_zdebug(struct backtrace_state *,
                                       const unsigned char *compressed,
                                       size_t compressed_size,
                                       backtrace_error_callback, void *data,
                                       unsigned char **uncompressed,
                                       size_t *uncompressed_size);

extern int backtrace_uncompress_lzma(struct backtrace_state *,
                                     const unsigned char *compressed,
                                     size_t compressed_size,
                                     backtrace_error_callback, void *data,
                                     unsigned char **uncompressed,
                                     size_t *uncompressed_size);

#endif

// filenames.h:
#ifndef GCC_VERSION
#define GCC_VERSION (__GNUC__ * 1000 + __GNUC_MINOR__)
#endif

#if (GCC_VERSION < 2007)
#define __attribute__(x)
#endif

#ifndef ATTRIBUTE_UNUSED
#define ATTRIBUTE_UNUSED __attribute__((__unused__))
#endif

#if defined(__MSDOS__) || defined(_WIN32) || defined(__OS2__) || \
    defined(__CYGWIN__)
#define IS_DIR_SEPARATOR(c) ((c) == '/' || (c) == '\\')
#define HAS_DRIVE_SPEC(f) ((f)[0] != '\0' && (f)[1] == ':')
#define IS_ABSOLUTE_PATH(f) (IS_DIR_SEPARATOR((f)[0]) || HAS_DRIVE_SPEC(f))
#else
#define IS_DIR_SEPARATOR(c) ((c) == '/')
#define IS_ABSOLUTE_PATH(f) (IS_DIR_SEPARATOR((f)[0]))
#endif

// atomic.c:
#include <sys/types.h>

// dwarf.c:
#include <errno.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>

enum dwarf_tag {
  DW_TAG_entry_point = 0x3,
  DW_TAG_compile_unit = 0x11,
  DW_TAG_inlined_subroutine = 0x1d,
  DW_TAG_subprogram = 0x2e,
  DW_TAG_skeleton_unit = 0x4a,
};

enum dwarf_form {
  DW_FORM_addr = 0x01,
  DW_FORM_block2 = 0x03,
  DW_FORM_block4 = 0x04,
  DW_FORM_data2 = 0x05,
  DW_FORM_data4 = 0x06,
  DW_FORM_data8 = 0x07,
  DW_FORM_string = 0x08,
  DW_FORM_block = 0x09,
  DW_FORM_block1 = 0x0a,
  DW_FORM_data1 = 0x0b,
  DW_FORM_flag = 0x0c,
  DW_FORM_sdata = 0x0d,
  DW_FORM_strp = 0x0e,
  DW_FORM_udata = 0x0f,
  DW_FORM_ref_addr = 0x10,
  DW_FORM_ref1 = 0x11,
  DW_FORM_ref2 = 0x12,
  DW_FORM_ref4 = 0x13,
  DW_FORM_ref8 = 0x14,
  DW_FORM_ref_udata = 0x15,
  DW_FORM_indirect = 0x16,
  DW_FORM_sec_offset = 0x17,
  DW_FORM_exprloc = 0x18,
  DW_FORM_flag_present = 0x19,
  DW_FORM_ref_sig8 = 0x20,
  DW_FORM_strx = 0x1a,
  DW_FORM_addrx = 0x1b,
  DW_FORM_ref_sup4 = 0x1c,
  DW_FORM_strp_sup = 0x1d,
  DW_FORM_data16 = 0x1e,
  DW_FORM_line_strp = 0x1f,
  DW_FORM_implicit_const = 0x21,
  DW_FORM_loclistx = 0x22,
  DW_FORM_rnglistx = 0x23,
  DW_FORM_ref_sup8 = 0x24,
  DW_FORM_strx1 = 0x25,
  DW_FORM_strx2 = 0x26,
  DW_FORM_strx3 = 0x27,
  DW_FORM_strx4 = 0x28,
  DW_FORM_addrx1 = 0x29,
  DW_FORM_addrx2 = 0x2a,
  DW_FORM_addrx3 = 0x2b,
  DW_FORM_addrx4 = 0x2c,
  DW_FORM_GNU_addr_index = 0x1f01,
  DW_FORM_GNU_str_index = 0x1f02,
  DW_FORM_GNU_ref_alt = 0x1f20,
  DW_FORM_GNU_strp_alt = 0x1f21
};

enum dwarf_attribute {
  DW_AT_sibling = 0x01,
  DW_AT_location = 0x02,
  DW_AT_name = 0x03,
  DW_AT_ordering = 0x09,
  DW_AT_subscr_data = 0x0a,
  DW_AT_byte_size = 0x0b,
  DW_AT_bit_offset = 0x0c,
  DW_AT_bit_size = 0x0d,
  DW_AT_element_list = 0x0f,
  DW_AT_stmt_list = 0x10,
  DW_AT_low_pc = 0x11,
  DW_AT_high_pc = 0x12,
  DW_AT_language = 0x13,
  DW_AT_member = 0x14,
  DW_AT_discr = 0x15,
  DW_AT_discr_value = 0x16,
  DW_AT_visibility = 0x17,
  DW_AT_import = 0x18,
  DW_AT_string_length = 0x19,
  DW_AT_common_reference = 0x1a,
  DW_AT_comp_dir = 0x1b,
  DW_AT_const_value = 0x1c,
  DW_AT_containing_type = 0x1d,
  DW_AT_default_value = 0x1e,
  DW_AT_inline = 0x20,
  DW_AT_is_optional = 0x21,
  DW_AT_lower_bound = 0x22,
  DW_AT_producer = 0x25,
  DW_AT_prototyped = 0x27,
  DW_AT_return_addr = 0x2a,
  DW_AT_start_scope = 0x2c,
  DW_AT_bit_stride = 0x2e,
  DW_AT_upper_bound = 0x2f,
  DW_AT_abstract_origin = 0x31,
  DW_AT_accessibility = 0x32,
  DW_AT_address_class = 0x33,
  DW_AT_artificial = 0x34,
  DW_AT_base_types = 0x35,
  DW_AT_calling_convention = 0x36,
  DW_AT_count = 0x37,
  DW_AT_data_member_location = 0x38,
  DW_AT_decl_column = 0x39,
  DW_AT_decl_file = 0x3a,
  DW_AT_decl_line = 0x3b,
  DW_AT_declaration = 0x3c,
  DW_AT_discr_list = 0x3d,
  DW_AT_encoding = 0x3e,
  DW_AT_external = 0x3f,
  DW_AT_frame_base = 0x40,
  DW_AT_friend = 0x41,
  DW_AT_identifier_case = 0x42,
  DW_AT_macro_info = 0x43,
  DW_AT_namelist_items = 0x44,
  DW_AT_priority = 0x45,
  DW_AT_segment = 0x46,
  DW_AT_specification = 0x47,
  DW_AT_static_link = 0x48,
  DW_AT_type = 0x49,
  DW_AT_use_location = 0x4a,
  DW_AT_variable_parameter = 0x4b,
  DW_AT_virtuality = 0x4c,
  DW_AT_vtable_elem_location = 0x4d,
  DW_AT_allocated = 0x4e,
  DW_AT_associated = 0x4f,
  DW_AT_data_location = 0x50,
  DW_AT_byte_stride = 0x51,
  DW_AT_entry_pc = 0x52,
  DW_AT_use_UTF8 = 0x53,
  DW_AT_extension = 0x54,
  DW_AT_ranges = 0x55,
  DW_AT_trampoline = 0x56,
  DW_AT_call_column = 0x57,
  DW_AT_call_file = 0x58,
  DW_AT_call_line = 0x59,
  DW_AT_description = 0x5a,
  DW_AT_binary_scale = 0x5b,
  DW_AT_decimal_scale = 0x5c,
  DW_AT_small = 0x5d,
  DW_AT_decimal_sign = 0x5e,
  DW_AT_digit_count = 0x5f,
  DW_AT_picture_string = 0x60,
  DW_AT_mutable = 0x61,
  DW_AT_threads_scaled = 0x62,
  DW_AT_explicit = 0x63,
  DW_AT_object_pointer = 0x64,
  DW_AT_endianity = 0x65,
  DW_AT_elemental = 0x66,
  DW_AT_pure = 0x67,
  DW_AT_recursive = 0x68,
  DW_AT_signature = 0x69,
  DW_AT_main_subprogram = 0x6a,
  DW_AT_data_bit_offset = 0x6b,
  DW_AT_const_expr = 0x6c,
  DW_AT_enum_class = 0x6d,
  DW_AT_linkage_name = 0x6e,
  DW_AT_string_length_bit_size = 0x6f,
  DW_AT_string_length_byte_size = 0x70,
  DW_AT_rank = 0x71,
  DW_AT_str_offsets_base = 0x72,
  DW_AT_addr_base = 0x73,
  DW_AT_rnglists_base = 0x74,
  DW_AT_dwo_name = 0x76,
  DW_AT_reference = 0x77,
  DW_AT_rvalue_reference = 0x78,
  DW_AT_macros = 0x79,
  DW_AT_call_all_calls = 0x7a,
  DW_AT_call_all_source_calls = 0x7b,
  DW_AT_call_all_tail_calls = 0x7c,
  DW_AT_call_return_pc = 0x7d,
  DW_AT_call_value = 0x7e,
  DW_AT_call_origin = 0x7f,
  DW_AT_call_parameter = 0x80,
  DW_AT_call_pc = 0x81,
  DW_AT_call_tail_call = 0x82,
  DW_AT_call_target = 0x83,
  DW_AT_call_target_clobbered = 0x84,
  DW_AT_call_data_location = 0x85,
  DW_AT_call_data_value = 0x86,
  DW_AT_noreturn = 0x87,
  DW_AT_alignment = 0x88,
  DW_AT_export_symbols = 0x89,
  DW_AT_deleted = 0x8a,
  DW_AT_defaulted = 0x8b,
  DW_AT_loclists_base = 0x8c,
  DW_AT_lo_user = 0x2000,
  DW_AT_hi_user = 0x3fff,
  DW_AT_MIPS_fde = 0x2001,
  DW_AT_MIPS_loop_begin = 0x2002,
  DW_AT_MIPS_tail_loop_begin = 0x2003,
  DW_AT_MIPS_epilog_begin = 0x2004,
  DW_AT_MIPS_loop_unroll_factor = 0x2005,
  DW_AT_MIPS_software_pipeline_depth = 0x2006,
  DW_AT_MIPS_linkage_name = 0x2007,
  DW_AT_MIPS_stride = 0x2008,
  DW_AT_MIPS_abstract_name = 0x2009,
  DW_AT_MIPS_clone_origin = 0x200a,
  DW_AT_MIPS_has_inlines = 0x200b,
  DW_AT_HP_block_index = 0x2000,
  DW_AT_HP_unmodifiable = 0x2001,
  DW_AT_HP_prologue = 0x2005,
  DW_AT_HP_epilogue = 0x2008,
  DW_AT_HP_actuals_stmt_list = 0x2010,
  DW_AT_HP_proc_per_section = 0x2011,
  DW_AT_HP_raw_data_ptr = 0x2012,
  DW_AT_HP_pass_by_reference = 0x2013,
  DW_AT_HP_opt_level = 0x2014,
  DW_AT_HP_prof_version_id = 0x2015,
  DW_AT_HP_opt_flags = 0x2016,
  DW_AT_HP_cold_region_low_pc = 0x2017,
  DW_AT_HP_cold_region_high_pc = 0x2018,
  DW_AT_HP_all_variables_modifiable = 0x2019,
  DW_AT_HP_linkage_name = 0x201a,
  DW_AT_HP_prof_flags = 0x201b,
  DW_AT_HP_unit_name = 0x201f,
  DW_AT_HP_unit_size = 0x2020,
  DW_AT_HP_widened_byte_size = 0x2021,
  DW_AT_HP_definition_points = 0x2022,
  DW_AT_HP_default_location = 0x2023,
  DW_AT_HP_is_result_param = 0x2029,
  DW_AT_sf_names = 0x2101,
  DW_AT_src_info = 0x2102,
  DW_AT_mac_info = 0x2103,
  DW_AT_src_coords = 0x2104,
  DW_AT_body_begin = 0x2105,
  DW_AT_body_end = 0x2106,
  DW_AT_GNU_vector = 0x2107,
  DW_AT_GNU_guarded_by = 0x2108,
  DW_AT_GNU_pt_guarded_by = 0x2109,
  DW_AT_GNU_guarded = 0x210a,
  DW_AT_GNU_pt_guarded = 0x210b,
  DW_AT_GNU_locks_excluded = 0x210c,
  DW_AT_GNU_exclusive_locks_required = 0x210d,
  DW_AT_GNU_shared_locks_required = 0x210e,
  DW_AT_GNU_odr_signature = 0x210f,
  DW_AT_GNU_template_name = 0x2110,
  DW_AT_GNU_call_site_value = 0x2111,
  DW_AT_GNU_call_site_data_value = 0x2112,
  DW_AT_GNU_call_site_target = 0x2113,
  DW_AT_GNU_call_site_target_clobbered = 0x2114,
  DW_AT_GNU_tail_call = 0x2115,
  DW_AT_GNU_all_tail_call_sites = 0x2116,
  DW_AT_GNU_all_call_sites = 0x2117,
  DW_AT_GNU_all_source_call_sites = 0x2118,
  DW_AT_GNU_macros = 0x2119,
  DW_AT_GNU_deleted = 0x211a,
  DW_AT_GNU_dwo_name = 0x2130,
  DW_AT_GNU_dwo_id = 0x2131,
  DW_AT_GNU_ranges_base = 0x2132,
  DW_AT_GNU_addr_base = 0x2133,
  DW_AT_GNU_pubnames = 0x2134,
  DW_AT_GNU_pubtypes = 0x2135,
  DW_AT_GNU_discriminator = 0x2136,
  DW_AT_GNU_locviews = 0x2137,
  DW_AT_GNU_entry_view = 0x2138,
  DW_AT_VMS_rtnbeg_pd_address = 0x2201,
  DW_AT_use_GNAT_descriptive_type = 0x2301,
  DW_AT_GNAT_descriptive_type = 0x2302,
  DW_AT_GNU_numerator = 0x2303,
  DW_AT_GNU_denominator = 0x2304,
  DW_AT_GNU_bias = 0x2305,
  DW_AT_upc_threads_scaled = 0x3210,
  DW_AT_PGI_lbase = 0x3a00,
  DW_AT_PGI_soffset = 0x3a01,
  DW_AT_PGI_lstride = 0x3a02,
  DW_AT_APPLE_optimized = 0x3fe1,
  DW_AT_APPLE_flags = 0x3fe2,
  DW_AT_APPLE_isa = 0x3fe3,
  DW_AT_APPLE_block = 0x3fe4,
  DW_AT_APPLE_major_runtime_vers = 0x3fe5,
  DW_AT_APPLE_runtime_class = 0x3fe6,
  DW_AT_APPLE_omit_frame_ptr = 0x3fe7,
  DW_AT_APPLE_property_name = 0x3fe8,
  DW_AT_APPLE_property_getter = 0x3fe9,
  DW_AT_APPLE_property_setter = 0x3fea,
  DW_AT_APPLE_property_attribute = 0x3feb,
  DW_AT_APPLE_objc_complete_type = 0x3fec,
  DW_AT_APPLE_property = 0x3fed
};

enum dwarf_line_number_op {
  DW_LNS_extended_op = 0x0,
  DW_LNS_copy = 0x1,
  DW_LNS_advance_pc = 0x2,
  DW_LNS_advance_line = 0x3,
  DW_LNS_set_file = 0x4,
  DW_LNS_set_column = 0x5,
  DW_LNS_negate_stmt = 0x6,
  DW_LNS_set_basic_block = 0x7,
  DW_LNS_const_add_pc = 0x8,
  DW_LNS_fixed_advance_pc = 0x9,
  DW_LNS_set_prologue_end = 0xa,
  DW_LNS_set_epilogue_begin = 0xb,
  DW_LNS_set_isa = 0xc,
};

enum dwarf_extended_line_number_op {
  DW_LNE_end_sequence = 0x1,
  DW_LNE_set_address = 0x2,
  DW_LNE_define_file = 0x3,
  DW_LNE_set_discriminator = 0x4,
};

enum dwarf_line_number_content_type {
  DW_LNCT_path = 0x1,
  DW_LNCT_directory_index = 0x2,
  DW_LNCT_timestamp = 0x3,
  DW_LNCT_size = 0x4,
  DW_LNCT_MD5 = 0x5,
  DW_LNCT_lo_user = 0x2000,
  DW_LNCT_hi_user = 0x3fff
};

enum dwarf_range_list_entry {
  DW_RLE_end_of_list = 0x00,
  DW_RLE_base_addressx = 0x01,
  DW_RLE_startx_endx = 0x02,
  DW_RLE_startx_length = 0x03,
  DW_RLE_offset_pair = 0x04,
  DW_RLE_base_address = 0x05,
  DW_RLE_start_end = 0x06,
  DW_RLE_start_length = 0x07
};

enum dwarf_unit_type {
  DW_UT_compile = 0x01,
  DW_UT_type = 0x02,
  DW_UT_partial = 0x03,
  DW_UT_skeleton = 0x04,
  DW_UT_split_compile = 0x05,
  DW_UT_split_type = 0x06,
  DW_UT_lo_user = 0x80,
  DW_UT_hi_user = 0xff
};

#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

struct dwarf_buf {
  const char *name;

  const unsigned char *start;

  const unsigned char *buf;

  size_t left;

  int is_bigendian;

  backtrace_error_callback error_callback;

  void *data;

  int reported_underflow;
};

struct attr {
  enum dwarf_attribute name;

  enum dwarf_form form;

  int64_t val;
};

struct abbrev {
  uint64_t code;

  enum dwarf_tag tag;

  int has_children;

  size_t num_attrs;

  struct attr *attrs;
};

struct abbrevs {
  size_t num_abbrevs;

  struct abbrev *abbrevs;
};

enum attr_val_encoding {

  ATTR_VAL_NONE,

  ATTR_VAL_ADDRESS,

  ATTR_VAL_ADDRESS_INDEX,

  ATTR_VAL_UINT,

  ATTR_VAL_SINT,

  ATTR_VAL_STRING,

  ATTR_VAL_STRING_INDEX,

  ATTR_VAL_REF_UNIT,

  ATTR_VAL_REF_INFO,

  ATTR_VAL_REF_ALT_INFO,

  ATTR_VAL_REF_SECTION,

  ATTR_VAL_REF_TYPE,

  ATTR_VAL_RNGLISTS_INDEX,

  ATTR_VAL_BLOCK,

  ATTR_VAL_EXPR,
};

struct attr_val {
  enum attr_val_encoding encoding;
  union {
    uint64_t uint;

    int64_t sint;

    const char *string;

  } u;
};

struct line_header {
  int version;

  int addrsize;

  unsigned int min_insn_len;

  unsigned int max_ops_per_insn;

  int line_base;

  unsigned int line_range;

  unsigned int opcode_base;

  const unsigned char *opcode_lengths;

  size_t dirs_count;

  const char **dirs;

  size_t filenames_count;

  const char **filenames;
};

struct line_header_format {
  int lnct;
  enum dwarf_form form;
};

struct line {
  uintptr_t pc;

  const char *filename;

  int lineno;

  int idx;
};

struct line_vector {
  struct backtrace_vector vec;

  size_t count;
};

struct function {
  const char *name;

  const char *caller_filename;

  int caller_lineno;

  struct function_addrs *function_addrs;
  size_t function_addrs_count;
};

struct function_addrs {
  uint64_t low;
  uint64_t high;

  struct function *function;
};

struct function_vector {
  struct backtrace_vector vec;

  size_t count;
};

struct unit {
  const unsigned char *unit_data;

  size_t unit_data_len;

  size_t unit_data_offset;

  size_t low_offset;

  size_t high_offset;

  int version;

  int is_dwarf64;

  int addrsize;

  off_t lineoff;

  uint64_t str_offsets_base;

  uint64_t addr_base;

  uint64_t rnglists_base;

  const char *filename;

  const char *comp_dir;

  const char *abs_filename;

  struct abbrevs abbrevs;

  struct line *lines;

  size_t lines_count;

  struct function_addrs *function_addrs;
  size_t function_addrs_count;
};

struct unit_addrs {
  uint64_t low;
  uint64_t high;

  struct unit *u;
};

struct unit_addrs_vector {
  struct backtrace_vector vec;

  size_t count;
};

struct unit_vector {
  struct backtrace_vector vec;
  size_t count;
};

struct dwarf_data {
  struct dwarf_data *next;

  struct dwarf_data *altlink;

  uintptr_t base_address;

  struct unit_addrs *addrs;

  size_t addrs_count;

  struct unit **units;

  size_t units_count;

  struct dwarf_sections dwarf_sections;

  int is_bigendian;

  struct function_vector fvec;
};

static void dwarf_buf_error(struct dwarf_buf *buf, const char *msg,
                            int errnum) {
  char b[200];

  snprintf(b, sizeof b, "%s in %s at %d", msg, buf->name,
           (int)(buf->buf - buf->start));
  buf->error_callback(buf->data, b, errnum);
}

static int require(struct dwarf_buf *buf, size_t count) {
  if (buf->left >= count) return 1;

  if (!buf->reported_underflow) {
    dwarf_buf_error(buf, "DWARF underflow", 0);
    buf->reported_underflow = 1;
  }

  return 0;
}

static int advance(struct dwarf_buf *buf, size_t count) {
  if (!require(buf, count)) return 0;
  buf->buf += count;
  buf->left -= count;
  return 1;
}

static const char *read_string(struct dwarf_buf *buf) {
  const char *p = (const char *)buf->buf;
  size_t len = strnlen(p, buf->left);

  size_t count = len + 1;

  if (!advance(buf, count)) return NULL;

  return p;
}

static unsigned char read_byte(struct dwarf_buf *buf) {
  const unsigned char *p = buf->buf;

  if (!advance(buf, 1)) return 0;
  return p[0];
}

static signed char read_sbyte(struct dwarf_buf *buf) {
  const unsigned char *p = buf->buf;

  if (!advance(buf, 1)) return 0;
  return (*p ^ 0x80) - 0x80;
}

static uint16_t read_uint16(struct dwarf_buf *buf) {
  const unsigned char *p = buf->buf;

  if (!advance(buf, 2)) return 0;
  if (buf->is_bigendian)
    return ((uint16_t)p[0] << 8) | (uint16_t)p[1];
  else
    return ((uint16_t)p[1] << 8) | (uint16_t)p[0];
}

static uint32_t read_uint24(struct dwarf_buf *buf) {
  const unsigned char *p = buf->buf;

  if (!advance(buf, 3)) return 0;
  if (buf->is_bigendian)
    return (((uint32_t)p[0] << 16) | ((uint32_t)p[1] << 8) | (uint32_t)p[2]);
  else
    return (((uint32_t)p[2] << 16) | ((uint32_t)p[1] << 8) | (uint32_t)p[0]);
}

static uint32_t read_uint32(struct dwarf_buf *buf) {
  const unsigned char *p = buf->buf;

  if (!advance(buf, 4)) return 0;
  if (buf->is_bigendian)
    return (((uint32_t)p[0] << 24) | ((uint32_t)p[1] << 16) |
            ((uint32_t)p[2] << 8) | (uint32_t)p[3]);
  else
    return (((uint32_t)p[3] << 24) | ((uint32_t)p[2] << 16) |
            ((uint32_t)p[1] << 8) | (uint32_t)p[0]);
}

static uint64_t read_uint64(struct dwarf_buf *buf) {
  const unsigned char *p = buf->buf;

  if (!advance(buf, 8)) return 0;
  if (buf->is_bigendian)
    return (((uint64_t)p[0] << 56) | ((uint64_t)p[1] << 48) |
            ((uint64_t)p[2] << 40) | ((uint64_t)p[3] << 32) |
            ((uint64_t)p[4] << 24) | ((uint64_t)p[5] << 16) |
            ((uint64_t)p[6] << 8) | (uint64_t)p[7]);
  else
    return (((uint64_t)p[7] << 56) | ((uint64_t)p[6] << 48) |
            ((uint64_t)p[5] << 40) | ((uint64_t)p[4] << 32) |
            ((uint64_t)p[3] << 24) | ((uint64_t)p[2] << 16) |
            ((uint64_t)p[1] << 8) | (uint64_t)p[0]);
}

static uint64_t read_offset(struct dwarf_buf *buf, int is_dwarf64) {
  if (is_dwarf64)
    return read_uint64(buf);
  else
    return read_uint32(buf);
}

static uint64_t read_address(struct dwarf_buf *buf, int addrsize) {
  switch (addrsize) {
    case 1:
      return read_byte(buf);
    case 2:
      return read_uint16(buf);
    case 4:
      return read_uint32(buf);
    case 8:
      return read_uint64(buf);
    default:
      dwarf_buf_error(buf, "unrecognized address size", 0);
      return 0;
  }
}

static int is_highest_address(uint64_t address, int addrsize) {
  switch (addrsize) {
    case 1:
      return address == (unsigned char)-1;
    case 2:
      return address == (uint16_t)-1;
    case 4:
      return address == (uint32_t)-1;
    case 8:
      return address == (uint64_t)-1;
    default:
      return 0;
  }
}

static uint64_t read_uleb128(struct dwarf_buf *buf) {
  uint64_t ret;
  unsigned int shift;
  int overflow;
  unsigned char b;

  ret = 0;
  shift = 0;
  overflow = 0;
  do {
    const unsigned char *p;

    p = buf->buf;
    if (!advance(buf, 1)) return 0;
    b = *p;
    if (shift < 64)
      ret |= ((uint64_t)(b & 0x7f)) << shift;
    else if (!overflow) {
      dwarf_buf_error(buf, "LEB128 overflows uint64_t", 0);
      overflow = 1;
    }
    shift += 7;
  } while ((b & 0x80) != 0);

  return ret;
}

static int64_t read_sleb128(struct dwarf_buf *buf) {
  uint64_t val;
  unsigned int shift;
  int overflow;
  unsigned char b;

  val = 0;
  shift = 0;
  overflow = 0;
  do {
    const unsigned char *p;

    p = buf->buf;
    if (!advance(buf, 1)) return 0;
    b = *p;
    if (shift < 64)
      val |= ((uint64_t)(b & 0x7f)) << shift;
    else if (!overflow) {
      dwarf_buf_error(buf, "signed LEB128 overflows uint64_t", 0);
      overflow = 1;
    }
    shift += 7;
  } while ((b & 0x80) != 0);

  if ((b & 0x40) != 0 && shift < 64) val |= ((uint64_t)-1) << shift;

  return (int64_t)val;
}

static size_t leb128_len(const unsigned char *p) {
  size_t ret;

  ret = 1;
  while ((*p & 0x80) != 0) {
    ++p;
    ++ret;
  }
  return ret;
}

static uint64_t read_initial_length(struct dwarf_buf *buf, int *is_dwarf64) {
  uint64_t len;

  len = read_uint32(buf);
  if (len == 0xffffffff) {
    len = read_uint64(buf);
    *is_dwarf64 = 1;
  } else
    *is_dwarf64 = 0;

  return len;
}

static void free_abbrevs(struct backtrace_state *state, struct abbrevs *abbrevs,
                         backtrace_error_callback error_callback, void *data) {
  size_t i;

  for (i = 0; i < abbrevs->num_abbrevs; ++i)
    backtrace_free(state, abbrevs->abbrevs[i].attrs,
                   abbrevs->abbrevs[i].num_attrs * sizeof(struct attr),
                   error_callback, data);
  backtrace_free(state, abbrevs->abbrevs,
                 abbrevs->num_abbrevs * sizeof(struct abbrev), error_callback,
                 data);
  abbrevs->num_abbrevs = 0;
  abbrevs->abbrevs = NULL;
}

static int read_attribute(enum dwarf_form form, uint64_t implicit_val,
                          struct dwarf_buf *buf, int is_dwarf64, int version,
                          int addrsize,
                          const struct dwarf_sections *dwarf_sections,
                          struct dwarf_data *altlink, struct attr_val *val) {
  memset(val, 0, sizeof *val);

  switch (form) {
    case DW_FORM_addr:
      val->encoding = ATTR_VAL_ADDRESS;
      val->u.uint = read_address(buf, addrsize);
      return 1;
    case DW_FORM_block2:
      val->encoding = ATTR_VAL_BLOCK;
      return advance(buf, read_uint16(buf));
    case DW_FORM_block4:
      val->encoding = ATTR_VAL_BLOCK;
      return advance(buf, read_uint32(buf));
    case DW_FORM_data2:
      val->encoding = ATTR_VAL_UINT;
      val->u.uint = read_uint16(buf);
      return 1;
    case DW_FORM_data4:
      val->encoding = ATTR_VAL_UINT;
      val->u.uint = read_uint32(buf);
      return 1;
    case DW_FORM_data8:
      val->encoding = ATTR_VAL_UINT;
      val->u.uint = read_uint64(buf);
      return 1;
    case DW_FORM_data16:
      val->encoding = ATTR_VAL_BLOCK;
      return advance(buf, 16);
    case DW_FORM_string:
      val->encoding = ATTR_VAL_STRING;
      val->u.string = read_string(buf);
      return val->u.string == NULL ? 0 : 1;
    case DW_FORM_block:
      val->encoding = ATTR_VAL_BLOCK;
      return advance(buf, read_uleb128(buf));
    case DW_FORM_block1:
      val->encoding = ATTR_VAL_BLOCK;
      return advance(buf, read_byte(buf));
    case DW_FORM_data1:
      val->encoding = ATTR_VAL_UINT;
      val->u.uint = read_byte(buf);
      return 1;
    case DW_FORM_flag:
      val->encoding = ATTR_VAL_UINT;
      val->u.uint = read_byte(buf);
      return 1;
    case DW_FORM_sdata:
      val->encoding = ATTR_VAL_SINT;
      val->u.sint = read_sleb128(buf);
      return 1;
    case DW_FORM_strp: {
      uint64_t offset;

      offset = read_offset(buf, is_dwarf64);
      if (offset >= dwarf_sections->size[DEBUG_STR]) {
        dwarf_buf_error(buf, "DW_FORM_strp out of range", 0);
        return 0;
      }
      val->encoding = ATTR_VAL_STRING;
      val->u.string = (const char *)dwarf_sections->data[DEBUG_STR] + offset;
      return 1;
    }
    case DW_FORM_line_strp: {
      uint64_t offset;

      offset = read_offset(buf, is_dwarf64);
      if (offset >= dwarf_sections->size[DEBUG_LINE_STR]) {
        dwarf_buf_error(buf, "DW_FORM_line_strp out of range", 0);
        return 0;
      }
      val->encoding = ATTR_VAL_STRING;
      val->u.string =
          (const char *)dwarf_sections->data[DEBUG_LINE_STR] + offset;
      return 1;
    }
    case DW_FORM_udata:
      val->encoding = ATTR_VAL_UINT;
      val->u.uint = read_uleb128(buf);
      return 1;
    case DW_FORM_ref_addr:
      val->encoding = ATTR_VAL_REF_INFO;
      if (version == 2)
        val->u.uint = read_address(buf, addrsize);
      else
        val->u.uint = read_offset(buf, is_dwarf64);
      return 1;
    case DW_FORM_ref1:
      val->encoding = ATTR_VAL_REF_UNIT;
      val->u.uint = read_byte(buf);
      return 1;
    case DW_FORM_ref2:
      val->encoding = ATTR_VAL_REF_UNIT;
      val->u.uint = read_uint16(buf);
      return 1;
    case DW_FORM_ref4:
      val->encoding = ATTR_VAL_REF_UNIT;
      val->u.uint = read_uint32(buf);
      return 1;
    case DW_FORM_ref8:
      val->encoding = ATTR_VAL_REF_UNIT;
      val->u.uint = read_uint64(buf);
      return 1;
    case DW_FORM_ref_udata:
      val->encoding = ATTR_VAL_REF_UNIT;
      val->u.uint = read_uleb128(buf);
      return 1;
    case DW_FORM_indirect: {
      uint64_t form;

      form = read_uleb128(buf);
      if (form == DW_FORM_implicit_const) {
        dwarf_buf_error(buf, "DW_FORM_indirect to DW_FORM_implicit_const", 0);
        return 0;
      }
      return read_attribute((enum dwarf_form)form, 0, buf, is_dwarf64, version,
                            addrsize, dwarf_sections, altlink, val);
    }
    case DW_FORM_sec_offset:
      val->encoding = ATTR_VAL_REF_SECTION;
      val->u.uint = read_offset(buf, is_dwarf64);
      return 1;
    case DW_FORM_exprloc:
      val->encoding = ATTR_VAL_EXPR;
      return advance(buf, read_uleb128(buf));
    case DW_FORM_flag_present:
      val->encoding = ATTR_VAL_UINT;
      val->u.uint = 1;
      return 1;
    case DW_FORM_ref_sig8:
      val->encoding = ATTR_VAL_REF_TYPE;
      val->u.uint = read_uint64(buf);
      return 1;
    case DW_FORM_strx:
    case DW_FORM_strx1:
    case DW_FORM_strx2:
    case DW_FORM_strx3:
    case DW_FORM_strx4: {
      uint64_t offset;

      switch (form) {
        case DW_FORM_strx:
          offset = read_uleb128(buf);
          break;
        case DW_FORM_strx1:
          offset = read_byte(buf);
          break;
        case DW_FORM_strx2:
          offset = read_uint16(buf);
          break;
        case DW_FORM_strx3:
          offset = read_uint24(buf);
          break;
        case DW_FORM_strx4:
          offset = read_uint32(buf);
          break;
        default:

          return 0;
      }
      val->encoding = ATTR_VAL_STRING_INDEX;
      val->u.uint = offset;
      return 1;
    }
    case DW_FORM_addrx:
    case DW_FORM_addrx1:
    case DW_FORM_addrx2:
    case DW_FORM_addrx3:
    case DW_FORM_addrx4: {
      uint64_t offset;

      switch (form) {
        case DW_FORM_addrx:
          offset = read_uleb128(buf);
          break;
        case DW_FORM_addrx1:
          offset = read_byte(buf);
          break;
        case DW_FORM_addrx2:
          offset = read_uint16(buf);
          break;
        case DW_FORM_addrx3:
          offset = read_uint24(buf);
          break;
        case DW_FORM_addrx4:
          offset = read_uint32(buf);
          break;
        default:

          return 0;
      }
      val->encoding = ATTR_VAL_ADDRESS_INDEX;
      val->u.uint = offset;
      return 1;
    }
    case DW_FORM_ref_sup4:
      val->encoding = ATTR_VAL_REF_SECTION;
      val->u.uint = read_uint32(buf);
      return 1;
    case DW_FORM_ref_sup8:
      val->encoding = ATTR_VAL_REF_SECTION;
      val->u.uint = read_uint64(buf);
      return 1;
    case DW_FORM_implicit_const:
      val->encoding = ATTR_VAL_UINT;
      val->u.uint = implicit_val;
      return 1;
    case DW_FORM_loclistx:

      val->encoding = ATTR_VAL_REF_SECTION;
      val->u.uint = read_uleb128(buf);
      return 1;
    case DW_FORM_rnglistx:
      val->encoding = ATTR_VAL_RNGLISTS_INDEX;
      val->u.uint = read_uleb128(buf);
      return 1;
    case DW_FORM_GNU_addr_index:
      val->encoding = ATTR_VAL_REF_SECTION;
      val->u.uint = read_uleb128(buf);
      return 1;
    case DW_FORM_GNU_str_index:
      val->encoding = ATTR_VAL_REF_SECTION;
      val->u.uint = read_uleb128(buf);
      return 1;
    case DW_FORM_GNU_ref_alt:
      val->u.uint = read_offset(buf, is_dwarf64);
      if (altlink == NULL) {
        val->encoding = ATTR_VAL_NONE;
        return 1;
      }
      val->encoding = ATTR_VAL_REF_ALT_INFO;
      return 1;
    case DW_FORM_strp_sup:
    case DW_FORM_GNU_strp_alt: {
      uint64_t offset;

      offset = read_offset(buf, is_dwarf64);
      if (altlink == NULL) {
        val->encoding = ATTR_VAL_NONE;
        return 1;
      }
      if (offset >= altlink->dwarf_sections.size[DEBUG_STR]) {
        dwarf_buf_error(buf, "DW_FORM_strp_sup out of range", 0);
        return 0;
      }
      val->encoding = ATTR_VAL_STRING;
      val->u.string =
          (const char *)altlink->dwarf_sections.data[DEBUG_STR] + offset;
      return 1;
    }
    default:
      dwarf_buf_error(buf, "unrecognized DWARF form", -1);
      return 0;
  }
}

static int resolve_string(const struct dwarf_sections *dwarf_sections,
                          int is_dwarf64, int is_bigendian,
                          uint64_t str_offsets_base, const struct attr_val *val,
                          backtrace_error_callback error_callback, void *data,
                          const char **string) {
  switch (val->encoding) {
    case ATTR_VAL_STRING:
      *string = val->u.string;
      return 1;

    case ATTR_VAL_STRING_INDEX: {
      uint64_t offset;
      struct dwarf_buf offset_buf;

      offset = val->u.uint * (is_dwarf64 ? 8 : 4) + str_offsets_base;
      if (offset + (is_dwarf64 ? 8 : 4) >
          dwarf_sections->size[DEBUG_STR_OFFSETS]) {
        error_callback(data, "DW_FORM_strx value out of range", 0);
        return 0;
      }

      offset_buf.name = ".debug_str_offsets";
      offset_buf.start = dwarf_sections->data[DEBUG_STR_OFFSETS];
      offset_buf.buf = dwarf_sections->data[DEBUG_STR_OFFSETS] + offset;
      offset_buf.left = dwarf_sections->size[DEBUG_STR_OFFSETS] - offset;
      offset_buf.is_bigendian = is_bigendian;
      offset_buf.error_callback = error_callback;
      offset_buf.data = data;
      offset_buf.reported_underflow = 0;

      offset = read_offset(&offset_buf, is_dwarf64);
      if (offset >= dwarf_sections->size[DEBUG_STR]) {
        dwarf_buf_error(&offset_buf, "DW_FORM_strx offset out of range", 0);
        return 0;
      }
      *string = (const char *)dwarf_sections->data[DEBUG_STR] + offset;
      return 1;
    }

    default:
      return 1;
  }
}

static int resolve_addr_index(const struct dwarf_sections *dwarf_sections,
                              uint64_t addr_base, int addrsize,
                              int is_bigendian, uint64_t addr_index,
                              backtrace_error_callback error_callback,
                              void *data, uint64_t *address) {
  uint64_t offset;
  struct dwarf_buf addr_buf;

  offset = addr_index * addrsize + addr_base;
  if (offset + addrsize > dwarf_sections->size[DEBUG_ADDR]) {
    error_callback(data, "DW_FORM_addrx value out of range", 0);
    return 0;
  }

  addr_buf.name = ".debug_addr";
  addr_buf.start = dwarf_sections->data[DEBUG_ADDR];
  addr_buf.buf = dwarf_sections->data[DEBUG_ADDR] + offset;
  addr_buf.left = dwarf_sections->size[DEBUG_ADDR] - offset;
  addr_buf.is_bigendian = is_bigendian;
  addr_buf.error_callback = error_callback;
  addr_buf.data = data;
  addr_buf.reported_underflow = 0;

  *address = read_address(&addr_buf, addrsize);
  return 1;
}

static int units_search(const void *vkey, const void *ventry) {
  const size_t *key = (const size_t *)vkey;
  const struct unit *entry = *((const struct unit *const *)ventry);
  size_t offset;

  offset = *key;
  if (offset < entry->low_offset)
    return -1;
  else if (offset >= entry->high_offset)
    return 1;
  else
    return 0;
}

static struct unit *find_unit(struct unit **pu, size_t units_count,
                              size_t offset) {
  struct unit **u;
  u = bsearch(&offset, pu, units_count, sizeof(struct unit *), units_search);
  return u == NULL ? NULL : *u;
}

static int function_addrs_compare(const void *v1, const void *v2) {
  const struct function_addrs *a1 = (const struct function_addrs *)v1;
  const struct function_addrs *a2 = (const struct function_addrs *)v2;

  if (a1->low < a2->low) return -1;
  if (a1->low > a2->low) return 1;
  if (a1->high < a2->high) return 1;
  if (a1->high > a2->high) return -1;
  return strcmp(a1->function->name, a2->function->name);
}

static int function_addrs_search(const void *vkey, const void *ventry) {
  const uintptr_t *key = (const uintptr_t *)vkey;
  const struct function_addrs *entry = (const struct function_addrs *)ventry;
  uintptr_t pc;

  pc = *key;
  if (pc < entry->low)
    return -1;
  else if (pc > (entry + 1)->low)
    return 1;
  else
    return 0;
}

static int add_unit_addr(struct backtrace_state *state, void *rdata,
                         uint64_t lowpc, uint64_t highpc,
                         backtrace_error_callback error_callback, void *data,
                         void *pvec) {
  struct unit *u = (struct unit *)rdata;
  struct unit_addrs_vector *vec = (struct unit_addrs_vector *)pvec;
  struct unit_addrs *p;

  if (vec->count > 0) {
    p = (struct unit_addrs *)vec->vec.base + (vec->count - 1);
    if ((lowpc == p->high || lowpc == p->high + 1) && u == p->u) {
      if (highpc > p->high) p->high = highpc;
      return 1;
    }
  }

  p = ((struct unit_addrs *)backtrace_vector_grow(
      state, sizeof(struct unit_addrs), error_callback, data, &vec->vec));
  if (p == NULL) return 0;

  p->low = lowpc;
  p->high = highpc;
  p->u = u;

  ++vec->count;

  return 1;
}

static int unit_addrs_compare(const void *v1, const void *v2) {
  const struct unit_addrs *a1 = (const struct unit_addrs *)v1;
  const struct unit_addrs *a2 = (const struct unit_addrs *)v2;

  if (a1->low < a2->low) return -1;
  if (a1->low > a2->low) return 1;
  if (a1->high < a2->high) return 1;
  if (a1->high > a2->high) return -1;
  if (a1->u->lineoff < a2->u->lineoff) return -1;
  if (a1->u->lineoff > a2->u->lineoff) return 1;
  return 0;
}

static int unit_addrs_search(const void *vkey, const void *ventry) {
  const uintptr_t *key = (const uintptr_t *)vkey;
  const struct unit_addrs *entry = (const struct unit_addrs *)ventry;
  uintptr_t pc;

  pc = *key;
  if (pc < entry->low)
    return -1;
  else if (pc > (entry + 1)->low)
    return 1;
  else
    return 0;
}

static int line_compare(const void *v1, const void *v2) {
  const struct line *ln1 = (const struct line *)v1;
  const struct line *ln2 = (const struct line *)v2;

  if (ln1->pc < ln2->pc)
    return -1;
  else if (ln1->pc > ln2->pc)
    return 1;
  else if (ln1->idx < ln2->idx)
    return -1;
  else if (ln1->idx > ln2->idx)
    return 1;
  else
    return 0;
}

static int line_search(const void *vkey, const void *ventry) {
  const uintptr_t *key = (const uintptr_t *)vkey;
  const struct line *entry = (const struct line *)ventry;
  uintptr_t pc;

  pc = *key;
  if (pc < entry->pc)
    return -1;
  else if (pc >= (entry + 1)->pc)
    return 1;
  else
    return 0;
}

static int abbrev_compare(const void *v1, const void *v2) {
  const struct abbrev *a1 = (const struct abbrev *)v1;
  const struct abbrev *a2 = (const struct abbrev *)v2;

  if (a1->code < a2->code)
    return -1;
  else if (a1->code > a2->code)
    return 1;
  else {
    return 0;
  }
}

static int read_abbrevs(struct backtrace_state *state, uint64_t abbrev_offset,
                        const unsigned char *dwarf_abbrev,
                        size_t dwarf_abbrev_size, int is_bigendian,
                        backtrace_error_callback error_callback, void *data,
                        struct abbrevs *abbrevs) {
  struct dwarf_buf abbrev_buf;
  struct dwarf_buf count_buf;
  size_t num_abbrevs;

  abbrevs->num_abbrevs = 0;
  abbrevs->abbrevs = NULL;

  if (abbrev_offset >= dwarf_abbrev_size) {
    error_callback(data, "abbrev offset out of range", 0);
    return 0;
  }

  abbrev_buf.name = ".debug_abbrev";
  abbrev_buf.start = dwarf_abbrev;
  abbrev_buf.buf = dwarf_abbrev + abbrev_offset;
  abbrev_buf.left = dwarf_abbrev_size - abbrev_offset;
  abbrev_buf.is_bigendian = is_bigendian;
  abbrev_buf.error_callback = error_callback;
  abbrev_buf.data = data;
  abbrev_buf.reported_underflow = 0;

  count_buf = abbrev_buf;
  num_abbrevs = 0;
  while (read_uleb128(&count_buf) != 0) {
    if (count_buf.reported_underflow) return 0;
    ++num_abbrevs;

    read_uleb128(&count_buf);

    read_byte(&count_buf);

    while (read_uleb128(&count_buf) != 0) {
      uint64_t form;

      form = read_uleb128(&count_buf);
      if ((enum dwarf_form)form == DW_FORM_implicit_const)
        read_sleb128(&count_buf);
    }

    read_uleb128(&count_buf);
  }

  if (count_buf.reported_underflow) return 0;

  if (num_abbrevs == 0) return 1;

  abbrevs->abbrevs = ((struct abbrev *)backtrace_alloc(
      state, num_abbrevs * sizeof(struct abbrev), error_callback, data));
  if (abbrevs->abbrevs == NULL) return 0;
  abbrevs->num_abbrevs = num_abbrevs;
  memset(abbrevs->abbrevs, 0, num_abbrevs * sizeof(struct abbrev));

  num_abbrevs = 0;
  while (1) {
    uint64_t code;
    struct abbrev a;
    size_t num_attrs;
    struct attr *attrs;

    if (abbrev_buf.reported_underflow) goto fail;

    code = read_uleb128(&abbrev_buf);
    if (code == 0) break;

    a.code = code;
    a.tag = (enum dwarf_tag)read_uleb128(&abbrev_buf);
    a.has_children = read_byte(&abbrev_buf);

    count_buf = abbrev_buf;
    num_attrs = 0;
    while (read_uleb128(&count_buf) != 0) {
      uint64_t form;

      ++num_attrs;
      form = read_uleb128(&count_buf);
      if ((enum dwarf_form)form == DW_FORM_implicit_const)
        read_sleb128(&count_buf);
    }

    if (num_attrs == 0) {
      attrs = NULL;
      read_uleb128(&abbrev_buf);
      read_uleb128(&abbrev_buf);
    } else {
      attrs = ((struct attr *)backtrace_alloc(state, num_attrs * sizeof *attrs,
                                              error_callback, data));
      if (attrs == NULL) goto fail;
      num_attrs = 0;
      while (1) {
        uint64_t name;
        uint64_t form;

        name = read_uleb128(&abbrev_buf);
        form = read_uleb128(&abbrev_buf);
        if (name == 0) break;
        attrs[num_attrs].name = (enum dwarf_attribute)name;
        attrs[num_attrs].form = (enum dwarf_form)form;
        if ((enum dwarf_form)form == DW_FORM_implicit_const)
          attrs[num_attrs].val = read_sleb128(&abbrev_buf);
        else
          attrs[num_attrs].val = 0;
        ++num_attrs;
      }
    }

    a.num_attrs = num_attrs;
    a.attrs = attrs;

    abbrevs->abbrevs[num_abbrevs] = a;
    ++num_abbrevs;
  }

  backtrace_qsort(abbrevs->abbrevs, abbrevs->num_abbrevs, sizeof(struct abbrev),
                  abbrev_compare);

  return 1;

fail:
  free_abbrevs(state, abbrevs, error_callback, data);
  return 0;
}

static const struct abbrev *lookup_abbrev(
    struct abbrevs *abbrevs, uint64_t code,
    backtrace_error_callback error_callback, void *data) {
  struct abbrev key;
  void *p;

  if (code - 1 < abbrevs->num_abbrevs &&
      abbrevs->abbrevs[code - 1].code == code)
    return &abbrevs->abbrevs[code - 1];

  memset(&key, 0, sizeof key);
  key.code = code;
  p = bsearch(&key, abbrevs->abbrevs, abbrevs->num_abbrevs,
              sizeof(struct abbrev), abbrev_compare);
  if (p == NULL) {
    error_callback(data, "invalid abbreviation code", 0);
    return NULL;
  }
  return (const struct abbrev *)p;
}

struct pcrange {
  uint64_t lowpc;
  int have_lowpc;
  int lowpc_is_addr_index;
  uint64_t highpc;
  int have_highpc;
  int highpc_is_relative;
  int highpc_is_addr_index;
  uint64_t ranges;
  int have_ranges;
  int ranges_is_index;
};

static void update_pcrange(const struct attr *attr, const struct attr_val *val,
                           struct pcrange *pcrange) {
  switch (attr->name) {
    case DW_AT_low_pc:
      if (val->encoding == ATTR_VAL_ADDRESS) {
        pcrange->lowpc = val->u.uint;
        pcrange->have_lowpc = 1;
      } else if (val->encoding == ATTR_VAL_ADDRESS_INDEX) {
        pcrange->lowpc = val->u.uint;
        pcrange->have_lowpc = 1;
        pcrange->lowpc_is_addr_index = 1;
      }
      break;

    case DW_AT_high_pc:
      if (val->encoding == ATTR_VAL_ADDRESS) {
        pcrange->highpc = val->u.uint;
        pcrange->have_highpc = 1;
      } else if (val->encoding == ATTR_VAL_UINT) {
        pcrange->highpc = val->u.uint;
        pcrange->have_highpc = 1;
        pcrange->highpc_is_relative = 1;
      } else if (val->encoding == ATTR_VAL_ADDRESS_INDEX) {
        pcrange->highpc = val->u.uint;
        pcrange->have_highpc = 1;
        pcrange->highpc_is_addr_index = 1;
      }
      break;

    case DW_AT_ranges:
      if (val->encoding == ATTR_VAL_UINT ||
          val->encoding == ATTR_VAL_REF_SECTION) {
        pcrange->ranges = val->u.uint;
        pcrange->have_ranges = 1;
      } else if (val->encoding == ATTR_VAL_RNGLISTS_INDEX) {
        pcrange->ranges = val->u.uint;
        pcrange->have_ranges = 1;
        pcrange->ranges_is_index = 1;
      }
      break;

    default:
      break;
  }
}

static int add_low_high_range(
    struct backtrace_state *state, const struct dwarf_sections *dwarf_sections,
    uintptr_t base_address, int is_bigendian, struct unit *u,
    const struct pcrange *pcrange,
    int (*add_range)(struct backtrace_state *state, void *rdata, uint64_t lowpc,
                     uint64_t highpc, backtrace_error_callback error_callback,
                     void *data, void *vec),
    void *rdata, backtrace_error_callback error_callback, void *data,
    void *vec) {
  uint64_t lowpc;
  uint64_t highpc;

  lowpc = pcrange->lowpc;
  if (pcrange->lowpc_is_addr_index) {
    if (!resolve_addr_index(dwarf_sections, u->addr_base, u->addrsize,
                            is_bigendian, lowpc, error_callback, data, &lowpc))
      return 0;
  }

  highpc = pcrange->highpc;
  if (pcrange->highpc_is_addr_index) {
    if (!resolve_addr_index(dwarf_sections, u->addr_base, u->addrsize,
                            is_bigendian, highpc, error_callback, data,
                            &highpc))
      return 0;
  }
  if (pcrange->highpc_is_relative) highpc += lowpc;

  lowpc += base_address;
  highpc += base_address;

  return add_range(state, rdata, lowpc, highpc, error_callback, data, vec);
}

static int add_ranges_from_ranges(
    struct backtrace_state *state, const struct dwarf_sections *dwarf_sections,
    uintptr_t base_address, int is_bigendian, struct unit *u, uint64_t base,
    const struct pcrange *pcrange,
    int (*add_range)(struct backtrace_state *state, void *rdata, uint64_t lowpc,
                     uint64_t highpc, backtrace_error_callback error_callback,
                     void *data, void *vec),
    void *rdata, backtrace_error_callback error_callback, void *data,
    void *vec) {
  struct dwarf_buf ranges_buf;

  if (pcrange->ranges >= dwarf_sections->size[DEBUG_RANGES]) {
    error_callback(data, "ranges offset out of range", 0);
    return 0;
  }

  ranges_buf.name = ".debug_ranges";
  ranges_buf.start = dwarf_sections->data[DEBUG_RANGES];
  ranges_buf.buf = dwarf_sections->data[DEBUG_RANGES] + pcrange->ranges;
  ranges_buf.left = dwarf_sections->size[DEBUG_RANGES] - pcrange->ranges;
  ranges_buf.is_bigendian = is_bigendian;
  ranges_buf.error_callback = error_callback;
  ranges_buf.data = data;
  ranges_buf.reported_underflow = 0;

  while (1) {
    uint64_t low;
    uint64_t high;

    if (ranges_buf.reported_underflow) return 0;

    low = read_address(&ranges_buf, u->addrsize);
    high = read_address(&ranges_buf, u->addrsize);

    if (low == 0 && high == 0) break;

    if (is_highest_address(low, u->addrsize))
      base = high;
    else {
      if (!add_range(state, rdata, low + base + base_address,
                     high + base + base_address, error_callback, data, vec))
        return 0;
    }
  }

  if (ranges_buf.reported_underflow) return 0;

  return 1;
}

static int add_ranges_from_rnglists(
    struct backtrace_state *state, const struct dwarf_sections *dwarf_sections,
    uintptr_t base_address, int is_bigendian, struct unit *u, uint64_t base,
    const struct pcrange *pcrange,
    int (*add_range)(struct backtrace_state *state, void *rdata, uint64_t lowpc,
                     uint64_t highpc, backtrace_error_callback error_callback,
                     void *data, void *vec),
    void *rdata, backtrace_error_callback error_callback, void *data,
    void *vec) {
  uint64_t offset;
  struct dwarf_buf rnglists_buf;

  if (!pcrange->ranges_is_index)
    offset = pcrange->ranges;
  else
    offset = u->rnglists_base + pcrange->ranges * (u->is_dwarf64 ? 8 : 4);
  if (offset >= dwarf_sections->size[DEBUG_RNGLISTS]) {
    error_callback(data, "rnglists offset out of range", 0);
    return 0;
  }

  rnglists_buf.name = ".debug_rnglists";
  rnglists_buf.start = dwarf_sections->data[DEBUG_RNGLISTS];
  rnglists_buf.buf = dwarf_sections->data[DEBUG_RNGLISTS] + offset;
  rnglists_buf.left = dwarf_sections->size[DEBUG_RNGLISTS] - offset;
  rnglists_buf.is_bigendian = is_bigendian;
  rnglists_buf.error_callback = error_callback;
  rnglists_buf.data = data;
  rnglists_buf.reported_underflow = 0;

  if (pcrange->ranges_is_index) {
    offset = read_offset(&rnglists_buf, u->is_dwarf64);
    offset += u->rnglists_base;
    if (offset >= dwarf_sections->size[DEBUG_RNGLISTS]) {
      error_callback(data, "rnglists index offset out of range", 0);
      return 0;
    }
    rnglists_buf.buf = dwarf_sections->data[DEBUG_RNGLISTS] + offset;
    rnglists_buf.left = dwarf_sections->size[DEBUG_RNGLISTS] - offset;
  }

  while (1) {
    unsigned char rle;

    rle = read_byte(&rnglists_buf);
    if (rle == DW_RLE_end_of_list) break;
    switch (rle) {
      case DW_RLE_base_addressx: {
        uint64_t index;

        index = read_uleb128(&rnglists_buf);
        if (!resolve_addr_index(dwarf_sections, u->addr_base, u->addrsize,
                                is_bigendian, index, error_callback, data,
                                &base))
          return 0;
      } break;

      case DW_RLE_startx_endx: {
        uint64_t index;
        uint64_t low;
        uint64_t high;

        index = read_uleb128(&rnglists_buf);
        if (!resolve_addr_index(dwarf_sections, u->addr_base, u->addrsize,
                                is_bigendian, index, error_callback, data,
                                &low))
          return 0;
        index = read_uleb128(&rnglists_buf);
        if (!resolve_addr_index(dwarf_sections, u->addr_base, u->addrsize,
                                is_bigendian, index, error_callback, data,
                                &high))
          return 0;
        if (!add_range(state, rdata, low + base_address, high + base_address,
                       error_callback, data, vec))
          return 0;
      } break;

      case DW_RLE_startx_length: {
        uint64_t index;
        uint64_t low;
        uint64_t length;

        index = read_uleb128(&rnglists_buf);
        if (!resolve_addr_index(dwarf_sections, u->addr_base, u->addrsize,
                                is_bigendian, index, error_callback, data,
                                &low))
          return 0;
        length = read_uleb128(&rnglists_buf);
        low += base_address;
        if (!add_range(state, rdata, low, low + length, error_callback, data,
                       vec))
          return 0;
      } break;

      case DW_RLE_offset_pair: {
        uint64_t low;
        uint64_t high;

        low = read_uleb128(&rnglists_buf);
        high = read_uleb128(&rnglists_buf);
        if (!add_range(state, rdata, low + base + base_address,
                       high + base + base_address, error_callback, data, vec))
          return 0;
      } break;

      case DW_RLE_base_address:
        base = read_address(&rnglists_buf, u->addrsize);
        break;

      case DW_RLE_start_end: {
        uint64_t low;
        uint64_t high;

        low = read_address(&rnglists_buf, u->addrsize);
        high = read_address(&rnglists_buf, u->addrsize);
        if (!add_range(state, rdata, low + base_address, high + base_address,
                       error_callback, data, vec))
          return 0;
      } break;

      case DW_RLE_start_length: {
        uint64_t low;
        uint64_t length;

        low = read_address(&rnglists_buf, u->addrsize);
        length = read_uleb128(&rnglists_buf);
        low += base_address;
        if (!add_range(state, rdata, low, low + length, error_callback, data,
                       vec))
          return 0;
      } break;

      default:
        dwarf_buf_error(&rnglists_buf, "unrecognized DW_RLE value", -1);
        return 0;
    }
  }

  if (rnglists_buf.reported_underflow) return 0;

  return 1;
}

static int add_ranges(
    struct backtrace_state *state, const struct dwarf_sections *dwarf_sections,
    uintptr_t base_address, int is_bigendian, struct unit *u, uint64_t base,
    const struct pcrange *pcrange,
    int (*add_range)(struct backtrace_state *state, void *rdata, uint64_t lowpc,
                     uint64_t highpc, backtrace_error_callback error_callback,
                     void *data, void *vec),
    void *rdata, backtrace_error_callback error_callback, void *data,
    void *vec) {
  if (pcrange->have_lowpc && pcrange->have_highpc)
    return add_low_high_range(state, dwarf_sections, base_address, is_bigendian,
                              u, pcrange, add_range, rdata, error_callback,
                              data, vec);

  if (!pcrange->have_ranges) {
    return 1;
  }

  if (u->version < 5)
    return add_ranges_from_ranges(state, dwarf_sections, base_address,
                                  is_bigendian, u, base, pcrange, add_range,
                                  rdata, error_callback, data, vec);
  else
    return add_ranges_from_rnglists(state, dwarf_sections, base_address,
                                    is_bigendian, u, base, pcrange, add_range,
                                    rdata, error_callback, data, vec);
}

static int find_address_ranges(
    struct backtrace_state *state, uintptr_t base_address,
    struct dwarf_buf *unit_buf, const struct dwarf_sections *dwarf_sections,
    int is_bigendian, struct dwarf_data *altlink,
    backtrace_error_callback error_callback, void *data, struct unit *u,
    struct unit_addrs_vector *addrs, enum dwarf_tag *unit_tag) {
  while (unit_buf->left > 0) {
    uint64_t code;
    const struct abbrev *abbrev;
    struct pcrange pcrange;
    struct attr_val name_val;
    int have_name_val;
    struct attr_val comp_dir_val;
    int have_comp_dir_val;
    size_t i;

    code = read_uleb128(unit_buf);
    if (code == 0) return 1;

    abbrev = lookup_abbrev(&u->abbrevs, code, error_callback, data);
    if (abbrev == NULL) return 0;

    if (unit_tag != NULL) *unit_tag = abbrev->tag;

    memset(&pcrange, 0, sizeof pcrange);
    memset(&name_val, 0, sizeof name_val);
    have_name_val = 0;
    memset(&comp_dir_val, 0, sizeof comp_dir_val);
    have_comp_dir_val = 0;
    for (i = 0; i < abbrev->num_attrs; ++i) {
      struct attr_val val;

      if (!read_attribute(abbrev->attrs[i].form, abbrev->attrs[i].val, unit_buf,
                          u->is_dwarf64, u->version, u->addrsize,
                          dwarf_sections, altlink, &val))
        return 0;

      switch (abbrev->attrs[i].name) {
        case DW_AT_low_pc:
        case DW_AT_high_pc:
        case DW_AT_ranges:
          update_pcrange(&abbrev->attrs[i], &val, &pcrange);
          break;

        case DW_AT_stmt_list:
          if ((abbrev->tag == DW_TAG_compile_unit ||
               abbrev->tag == DW_TAG_skeleton_unit) &&
              (val.encoding == ATTR_VAL_UINT ||
               val.encoding == ATTR_VAL_REF_SECTION))
            u->lineoff = val.u.uint;
          break;

        case DW_AT_name:
          if (abbrev->tag == DW_TAG_compile_unit ||
              abbrev->tag == DW_TAG_skeleton_unit) {
            name_val = val;
            have_name_val = 1;
          }
          break;

        case DW_AT_comp_dir:
          if (abbrev->tag == DW_TAG_compile_unit ||
              abbrev->tag == DW_TAG_skeleton_unit) {
            comp_dir_val = val;
            have_comp_dir_val = 1;
          }
          break;

        case DW_AT_str_offsets_base:
          if ((abbrev->tag == DW_TAG_compile_unit ||
               abbrev->tag == DW_TAG_skeleton_unit) &&
              val.encoding == ATTR_VAL_REF_SECTION)
            u->str_offsets_base = val.u.uint;
          break;

        case DW_AT_addr_base:
          if ((abbrev->tag == DW_TAG_compile_unit ||
               abbrev->tag == DW_TAG_skeleton_unit) &&
              val.encoding == ATTR_VAL_REF_SECTION)
            u->addr_base = val.u.uint;
          break;

        case DW_AT_rnglists_base:
          if ((abbrev->tag == DW_TAG_compile_unit ||
               abbrev->tag == DW_TAG_skeleton_unit) &&
              val.encoding == ATTR_VAL_REF_SECTION)
            u->rnglists_base = val.u.uint;
          break;

        default:
          break;
      }
    }

    if (have_name_val) {
      if (!resolve_string(dwarf_sections, u->is_dwarf64, is_bigendian,
                          u->str_offsets_base, &name_val, error_callback, data,
                          &u->filename))
        return 0;
    }
    if (have_comp_dir_val) {
      if (!resolve_string(dwarf_sections, u->is_dwarf64, is_bigendian,
                          u->str_offsets_base, &comp_dir_val, error_callback,
                          data, &u->comp_dir))
        return 0;
    }

    if (abbrev->tag == DW_TAG_compile_unit ||
        abbrev->tag == DW_TAG_subprogram ||
        abbrev->tag == DW_TAG_skeleton_unit) {
      if (!add_ranges(state, dwarf_sections, base_address, is_bigendian, u,
                      pcrange.lowpc, &pcrange, add_unit_addr, (void *)u,
                      error_callback, data, (void *)addrs))
        return 0;

      if ((abbrev->tag == DW_TAG_compile_unit ||
           abbrev->tag == DW_TAG_skeleton_unit) &&
          (pcrange.have_ranges || (pcrange.have_lowpc && pcrange.have_highpc)))
        return 1;
    }

    if (abbrev->has_children) {
      if (!find_address_ranges(state, base_address, unit_buf, dwarf_sections,
                               is_bigendian, altlink, error_callback, data, u,
                               addrs, NULL))
        return 0;
    }
  }

  return 1;
}

static int build_address_map(struct backtrace_state *state,
                             uintptr_t base_address,
                             const struct dwarf_sections *dwarf_sections,
                             int is_bigendian, struct dwarf_data *altlink,
                             backtrace_error_callback error_callback,
                             void *data, struct unit_addrs_vector *addrs,
                             struct unit_vector *unit_vec) {
  struct dwarf_buf info;
  struct backtrace_vector units;
  size_t units_count;
  size_t i;
  struct unit **pu;
  size_t unit_offset = 0;
  struct unit_addrs *pa;

  memset(&addrs->vec, 0, sizeof addrs->vec);
  memset(&unit_vec->vec, 0, sizeof unit_vec->vec);
  addrs->count = 0;
  unit_vec->count = 0;

  info.name = ".debug_info";
  info.start = dwarf_sections->data[DEBUG_INFO];
  info.buf = info.start;
  info.left = dwarf_sections->size[DEBUG_INFO];
  info.is_bigendian = is_bigendian;
  info.error_callback = error_callback;
  info.data = data;
  info.reported_underflow = 0;

  memset(&units, 0, sizeof units);
  units_count = 0;

  while (info.left > 0) {
    const unsigned char *unit_data_start;
    uint64_t len;
    int is_dwarf64;
    struct dwarf_buf unit_buf;
    int version;
    int unit_type;
    uint64_t abbrev_offset;
    int addrsize;
    struct unit *u;
    enum dwarf_tag unit_tag;

    if (info.reported_underflow) goto fail;

    unit_data_start = info.buf;

    len = read_initial_length(&info, &is_dwarf64);
    unit_buf = info;
    unit_buf.left = len;

    if (!advance(&info, len)) goto fail;

    version = read_uint16(&unit_buf);
    if (version < 2 || version > 5) {
      dwarf_buf_error(&unit_buf, "unrecognized DWARF version", -1);
      goto fail;
    }

    if (version < 5)
      unit_type = 0;
    else {
      unit_type = read_byte(&unit_buf);
      if (unit_type == DW_UT_type || unit_type == DW_UT_split_type) {
        continue;
      }
    }

    pu = ((struct unit **)backtrace_vector_grow(state, sizeof(struct unit *),
                                                error_callback, data, &units));
    if (pu == NULL) goto fail;

    u = ((struct unit *)backtrace_alloc(state, sizeof *u, error_callback,
                                        data));
    if (u == NULL) goto fail;

    *pu = u;
    ++units_count;

    if (version < 5)
      addrsize = 0;
    else
      addrsize = read_byte(&unit_buf);

    memset(&u->abbrevs, 0, sizeof u->abbrevs);
    abbrev_offset = read_offset(&unit_buf, is_dwarf64);
    if (!read_abbrevs(state, abbrev_offset, dwarf_sections->data[DEBUG_ABBREV],
                      dwarf_sections->size[DEBUG_ABBREV], is_bigendian,
                      error_callback, data, &u->abbrevs))
      goto fail;

    if (version < 5) addrsize = read_byte(&unit_buf);

    switch (unit_type) {
      case 0:
        break;
      case DW_UT_compile:
      case DW_UT_partial:
        break;
      case DW_UT_skeleton:
      case DW_UT_split_compile:
        read_uint64(&unit_buf);
        break;
      default:
        break;
    }

    u->low_offset = unit_offset;
    unit_offset += len + (is_dwarf64 ? 12 : 4);
    u->high_offset = unit_offset;
    u->unit_data = unit_buf.buf;
    u->unit_data_len = unit_buf.left;
    u->unit_data_offset = unit_buf.buf - unit_data_start;
    u->version = version;
    u->is_dwarf64 = is_dwarf64;
    u->addrsize = addrsize;
    u->filename = NULL;
    u->comp_dir = NULL;
    u->abs_filename = NULL;
    u->lineoff = 0;
    u->str_offsets_base = 0;
    u->addr_base = 0;
    u->rnglists_base = 0;

    u->lines = NULL;
    u->lines_count = 0;
    u->function_addrs = NULL;
    u->function_addrs_count = 0;

    if (!find_address_ranges(state, base_address, &unit_buf, dwarf_sections,
                             is_bigendian, altlink, error_callback, data, u,
                             addrs, &unit_tag))
      goto fail;

    if (unit_buf.reported_underflow) goto fail;
  }
  if (info.reported_underflow) goto fail;

  pa = ((struct unit_addrs *)backtrace_vector_grow(
      state, sizeof(struct unit_addrs), error_callback, data, &addrs->vec));
  if (pa == NULL) goto fail;
  pa->low = 0;
  --pa->low;
  pa->high = pa->low;
  pa->u = NULL;

  unit_vec->vec = units;
  unit_vec->count = units_count;
  return 1;

fail:
  if (units_count > 0) {
    pu = (struct unit **)units.base;
    for (i = 0; i < units_count; i++) {
      free_abbrevs(state, &pu[i]->abbrevs, error_callback, data);
      backtrace_free(state, pu[i], sizeof **pu, error_callback, data);
    }
    backtrace_vector_free(state, &units, error_callback, data);
  }
  if (addrs->count > 0) {
    backtrace_vector_free(state, &addrs->vec, error_callback, data);
    addrs->count = 0;
  }
  return 0;
}

static int add_line(struct backtrace_state *state, struct dwarf_data *ddata,
                    uintptr_t pc, const char *filename, int lineno,
                    backtrace_error_callback error_callback, void *data,
                    struct line_vector *vec) {
  struct line *ln;

  if (vec->count > 0) {
    ln = (struct line *)vec->vec.base + (vec->count - 1);
    if (pc == ln->pc && filename == ln->filename && lineno == ln->lineno)
      return 1;
  }

  ln = ((struct line *)backtrace_vector_grow(state, sizeof(struct line),
                                             error_callback, data, &vec->vec));
  if (ln == NULL) return 0;

  ln->pc = pc + ddata->base_address;

  ln->filename = filename;
  ln->lineno = lineno;
  ln->idx = vec->count;

  ++vec->count;

  return 1;
}

static void free_line_header(struct backtrace_state *state,
                             struct line_header *hdr,
                             backtrace_error_callback error_callback,
                             void *data) {
  if (hdr->dirs_count != 0)
    backtrace_free(state, hdr->dirs, hdr->dirs_count * sizeof(const char *),
                   error_callback, data);
  backtrace_free(state, hdr->filenames, hdr->filenames_count * sizeof(char *),
                 error_callback, data);
}

static int read_v2_paths(struct backtrace_state *state, struct unit *u,
                         struct dwarf_buf *hdr_buf, struct line_header *hdr) {
  const unsigned char *p;
  const unsigned char *pend;
  size_t i;

  hdr->dirs_count = 0;
  p = hdr_buf->buf;
  pend = p + hdr_buf->left;
  while (p < pend && *p != '\0') {
    p += strnlen((const char *)p, pend - p) + 1;
    ++hdr->dirs_count;
  }

  ++hdr->dirs_count;
  hdr->dirs = ((const char **)backtrace_alloc(
      state, hdr->dirs_count * sizeof(const char *), hdr_buf->error_callback,
      hdr_buf->data));
  if (hdr->dirs == NULL) return 0;

  hdr->dirs[0] = u->comp_dir;
  i = 1;
  while (*hdr_buf->buf != '\0') {
    if (hdr_buf->reported_underflow) return 0;

    hdr->dirs[i] = read_string(hdr_buf);
    if (hdr->dirs[i] == NULL) return 0;
    ++i;
  }
  if (!advance(hdr_buf, 1)) return 0;

  hdr->filenames_count = 0;
  p = hdr_buf->buf;
  pend = p + hdr_buf->left;
  while (p < pend && *p != '\0') {
    p += strnlen((const char *)p, pend - p) + 1;
    p += leb128_len(p);
    p += leb128_len(p);
    p += leb128_len(p);
    ++hdr->filenames_count;
  }

  ++hdr->filenames_count;
  hdr->filenames = ((const char **)backtrace_alloc(
      state, hdr->filenames_count * sizeof(char *), hdr_buf->error_callback,
      hdr_buf->data));
  if (hdr->filenames == NULL) return 0;
  hdr->filenames[0] = u->filename;
  i = 1;
  while (*hdr_buf->buf != '\0') {
    const char *filename;
    uint64_t dir_index;

    if (hdr_buf->reported_underflow) return 0;

    filename = read_string(hdr_buf);
    if (filename == NULL) return 0;
    dir_index = read_uleb128(hdr_buf);
    if (IS_ABSOLUTE_PATH(filename) ||
        (dir_index < hdr->dirs_count && hdr->dirs[dir_index] == NULL))
      hdr->filenames[i] = filename;
    else {
      const char *dir;
      size_t dir_len;
      size_t filename_len;
      char *s;

      if (dir_index < hdr->dirs_count)
        dir = hdr->dirs[dir_index];
      else {
        dwarf_buf_error(hdr_buf,
                        ("invalid directory index in "
                         "line number program header"),
                        0);
        return 0;
      }
      dir_len = strlen(dir);
      filename_len = strlen(filename);
      s = ((char *)backtrace_alloc(state, dir_len + filename_len + 2,
                                   hdr_buf->error_callback, hdr_buf->data));
      if (s == NULL) return 0;
      memcpy(s, dir, dir_len);

      s[dir_len] = '/';
      memcpy(s + dir_len + 1, filename, filename_len + 1);
      hdr->filenames[i] = s;
    }

    read_uleb128(hdr_buf);
    read_uleb128(hdr_buf);

    ++i;
  }

  return 1;
}

static int read_lnct(struct backtrace_state *state, struct dwarf_data *ddata,
                     struct unit *u, struct dwarf_buf *hdr_buf,
                     const struct line_header *hdr, size_t formats_count,
                     const struct line_header_format *formats,
                     const char **string) {
  size_t i;
  const char *dir;
  const char *path;

  dir = NULL;
  path = NULL;
  for (i = 0; i < formats_count; i++) {
    struct attr_val val;

    if (!read_attribute(formats[i].form, 0, hdr_buf, u->is_dwarf64, u->version,
                        hdr->addrsize, &ddata->dwarf_sections, ddata->altlink,
                        &val))
      return 0;
    switch (formats[i].lnct) {
      case DW_LNCT_path:
        if (!resolve_string(&ddata->dwarf_sections, u->is_dwarf64,
                            ddata->is_bigendian, u->str_offsets_base, &val,
                            hdr_buf->error_callback, hdr_buf->data, &path))
          return 0;
        break;
      case DW_LNCT_directory_index:
        if (val.encoding == ATTR_VAL_UINT) {
          if (val.u.uint >= hdr->dirs_count) {
            dwarf_buf_error(hdr_buf,
                            ("invalid directory index in "
                             "line number program header"),
                            0);
            return 0;
          }
          dir = hdr->dirs[val.u.uint];
        }
        break;
      default:

        break;
    }
  }

  if (path == NULL) {
    dwarf_buf_error(hdr_buf, "missing file name in line number program header",
                    0);
    return 0;
  }

  if (dir == NULL)
    *string = path;
  else {
    size_t dir_len;
    size_t path_len;
    char *s;

    dir_len = strlen(dir);
    path_len = strlen(path);
    s = (char *)backtrace_alloc(state, dir_len + path_len + 2,
                                hdr_buf->error_callback, hdr_buf->data);
    if (s == NULL) return 0;
    memcpy(s, dir, dir_len);

    s[dir_len] = '/';
    memcpy(s + dir_len + 1, path, path_len + 1);
    *string = s;
  }

  return 1;
}

static int read_line_header_format_entries(
    struct backtrace_state *state, struct dwarf_data *ddata, struct unit *u,
    struct dwarf_buf *hdr_buf, struct line_header *hdr, size_t *pcount,
    const char ***ppaths) {
  size_t formats_count;
  struct line_header_format *formats;
  size_t paths_count;
  const char **paths;
  size_t i;
  int ret;

  formats_count = read_byte(hdr_buf);
  if (formats_count == 0)
    formats = NULL;
  else {
    formats = ((struct line_header_format *)backtrace_alloc(
        state, (formats_count * sizeof(struct line_header_format)),
        hdr_buf->error_callback, hdr_buf->data));
    if (formats == NULL) return 0;

    for (i = 0; i < formats_count; i++) {
      formats[i].lnct = (int)read_uleb128(hdr_buf);
      formats[i].form = (enum dwarf_form)read_uleb128(hdr_buf);
    }
  }

  paths_count = read_uleb128(hdr_buf);
  if (paths_count == 0) {
    *pcount = 0;
    *ppaths = NULL;
    ret = 1;
    goto exit;
  }

  paths =
      ((const char **)backtrace_alloc(state, paths_count * sizeof(const char *),
                                      hdr_buf->error_callback, hdr_buf->data));
  if (paths == NULL) {
    ret = 0;
    goto exit;
  }
  for (i = 0; i < paths_count; i++) {
    if (!read_lnct(state, ddata, u, hdr_buf, hdr, formats_count, formats,
                   &paths[i])) {
      backtrace_free(state, paths, paths_count * sizeof(const char *),
                     hdr_buf->error_callback, hdr_buf->data);
      ret = 0;
      goto exit;
    }
  }

  *pcount = paths_count;
  *ppaths = paths;

  ret = 1;

exit:
  if (formats != NULL)
    backtrace_free(state, formats,
                   formats_count * sizeof(struct line_header_format),
                   hdr_buf->error_callback, hdr_buf->data);

  return ret;
}

static int read_line_header(struct backtrace_state *state,
                            struct dwarf_data *ddata, struct unit *u,
                            int is_dwarf64, struct dwarf_buf *line_buf,
                            struct line_header *hdr) {
  uint64_t hdrlen;
  struct dwarf_buf hdr_buf;

  hdr->version = read_uint16(line_buf);
  if (hdr->version < 2 || hdr->version > 5) {
    dwarf_buf_error(line_buf, "unsupported line number version", -1);
    return 0;
  }

  if (hdr->version < 5)
    hdr->addrsize = u->addrsize;
  else {
    hdr->addrsize = read_byte(line_buf);

    if (read_byte(line_buf) != 0) {
      dwarf_buf_error(line_buf, "non-zero segment_selector_size not supported",
                      -1);
      return 0;
    }
  }

  hdrlen = read_offset(line_buf, is_dwarf64);

  hdr_buf = *line_buf;
  hdr_buf.left = hdrlen;

  if (!advance(line_buf, hdrlen)) return 0;

  hdr->min_insn_len = read_byte(&hdr_buf);
  if (hdr->version < 4)
    hdr->max_ops_per_insn = 1;
  else
    hdr->max_ops_per_insn = read_byte(&hdr_buf);

  read_byte(&hdr_buf);

  hdr->line_base = read_sbyte(&hdr_buf);
  hdr->line_range = read_byte(&hdr_buf);

  hdr->opcode_base = read_byte(&hdr_buf);
  hdr->opcode_lengths = hdr_buf.buf;
  if (!advance(&hdr_buf, hdr->opcode_base - 1)) return 0;

  if (hdr->version < 5) {
    if (!read_v2_paths(state, u, &hdr_buf, hdr)) return 0;
  } else {
    if (!read_line_header_format_entries(state, ddata, u, &hdr_buf, hdr,
                                         &hdr->dirs_count, &hdr->dirs))
      return 0;
    if (!read_line_header_format_entries(state, ddata, u, &hdr_buf, hdr,
                                         &hdr->filenames_count,
                                         &hdr->filenames))
      return 0;
  }

  if (hdr_buf.reported_underflow) return 0;

  return 1;
}

static int read_line_program(struct backtrace_state *state,
                             struct dwarf_data *ddata,
                             const struct line_header *hdr,
                             struct dwarf_buf *line_buf,
                             struct line_vector *vec) {
  uint64_t address;
  unsigned int op_index;
  const char *reset_filename;
  const char *filename;
  int lineno;

  address = 0;
  op_index = 0;
  if (hdr->filenames_count > 1)
    reset_filename = hdr->filenames[1];
  else
    reset_filename = "";
  filename = reset_filename;
  lineno = 1;
  while (line_buf->left > 0) {
    unsigned int op;

    op = read_byte(line_buf);
    if (op >= hdr->opcode_base) {
      unsigned int advance;

      op -= hdr->opcode_base;
      advance = op / hdr->line_range;
      address +=
          (hdr->min_insn_len * (op_index + advance) / hdr->max_ops_per_insn);
      op_index = (op_index + advance) % hdr->max_ops_per_insn;
      lineno += hdr->line_base + (int)(op % hdr->line_range);
      add_line(state, ddata, address, filename, lineno,
               line_buf->error_callback, line_buf->data, vec);
    } else if (op == DW_LNS_extended_op) {
      uint64_t len;

      len = read_uleb128(line_buf);
      op = read_byte(line_buf);
      switch (op) {
        case DW_LNE_end_sequence:

          address = 0;
          op_index = 0;
          filename = reset_filename;
          lineno = 1;
          break;
        case DW_LNE_set_address:
          address = read_address(line_buf, hdr->addrsize);
          break;
        case DW_LNE_define_file: {
          const char *f;
          unsigned int dir_index;

          f = read_string(line_buf);
          if (f == NULL) return 0;
          dir_index = read_uleb128(line_buf);

          read_uleb128(line_buf);
          read_uleb128(line_buf);
          if (IS_ABSOLUTE_PATH(f))
            filename = f;
          else {
            const char *dir;
            size_t dir_len;
            size_t f_len;
            char *p;

            if (dir_index < hdr->dirs_count)
              dir = hdr->dirs[dir_index];
            else {
              dwarf_buf_error(line_buf,
                              ("invalid directory index "
                               "in line number program"),
                              0);
              return 0;
            }
            dir_len = strlen(dir);
            f_len = strlen(f);
            p = ((char *)backtrace_alloc(state, dir_len + f_len + 2,
                                         line_buf->error_callback,
                                         line_buf->data));
            if (p == NULL) return 0;
            memcpy(p, dir, dir_len);

            p[dir_len] = '/';
            memcpy(p + dir_len + 1, f, f_len + 1);
            filename = p;
          }
        } break;
        case DW_LNE_set_discriminator:

          read_uleb128(line_buf);
          break;
        default:
          if (!advance(line_buf, len - 1)) return 0;
          break;
      }
    } else {
      switch (op) {
        case DW_LNS_copy:
          add_line(state, ddata, address, filename, lineno,
                   line_buf->error_callback, line_buf->data, vec);
          break;
        case DW_LNS_advance_pc: {
          uint64_t advance;

          advance = read_uleb128(line_buf);
          address += (hdr->min_insn_len * (op_index + advance) /
                      hdr->max_ops_per_insn);
          op_index = (op_index + advance) % hdr->max_ops_per_insn;
        } break;
        case DW_LNS_advance_line:
          lineno += (int)read_sleb128(line_buf);
          break;
        case DW_LNS_set_file: {
          uint64_t fileno;

          fileno = read_uleb128(line_buf);
          if (fileno >= hdr->filenames_count) {
            dwarf_buf_error(line_buf,
                            ("invalid file number in "
                             "line number program"),
                            0);
            return 0;
          }
          filename = hdr->filenames[fileno];
        } break;
        case DW_LNS_set_column:
          read_uleb128(line_buf);
          break;
        case DW_LNS_negate_stmt:
          break;
        case DW_LNS_set_basic_block:
          break;
        case DW_LNS_const_add_pc: {
          unsigned int advance;

          op = 255 - hdr->opcode_base;
          advance = op / hdr->line_range;
          address += (hdr->min_insn_len * (op_index + advance) /
                      hdr->max_ops_per_insn);
          op_index = (op_index + advance) % hdr->max_ops_per_insn;
        } break;
        case DW_LNS_fixed_advance_pc:
          address += read_uint16(line_buf);
          op_index = 0;
          break;
        case DW_LNS_set_prologue_end:
          break;
        case DW_LNS_set_epilogue_begin:
          break;
        case DW_LNS_set_isa:
          read_uleb128(line_buf);
          break;
        default: {
          unsigned int i;

          for (i = hdr->opcode_lengths[op - 1]; i > 0; --i)
            read_uleb128(line_buf);
        } break;
      }
    }
  }

  return 1;
}

static int read_line_info(struct backtrace_state *state,
                          struct dwarf_data *ddata,
                          backtrace_error_callback error_callback, void *data,
                          struct unit *u, struct line_header *hdr,
                          struct line **lines, size_t *lines_count) {
  struct line_vector vec;
  struct dwarf_buf line_buf;
  uint64_t len;
  int is_dwarf64;
  struct line *ln;

  memset(&vec.vec, 0, sizeof vec.vec);
  vec.count = 0;

  memset(hdr, 0, sizeof *hdr);

  if (u->lineoff != (off_t)(size_t)u->lineoff ||
      (size_t)u->lineoff >= ddata->dwarf_sections.size[DEBUG_LINE]) {
    error_callback(data, "unit line offset out of range", 0);
    goto fail;
  }

  line_buf.name = ".debug_line";
  line_buf.start = ddata->dwarf_sections.data[DEBUG_LINE];
  line_buf.buf = ddata->dwarf_sections.data[DEBUG_LINE] + u->lineoff;
  line_buf.left = ddata->dwarf_sections.size[DEBUG_LINE] - u->lineoff;
  line_buf.is_bigendian = ddata->is_bigendian;
  line_buf.error_callback = error_callback;
  line_buf.data = data;
  line_buf.reported_underflow = 0;

  len = read_initial_length(&line_buf, &is_dwarf64);
  line_buf.left = len;

  if (!read_line_header(state, ddata, u, is_dwarf64, &line_buf, hdr)) goto fail;

  if (!read_line_program(state, ddata, hdr, &line_buf, &vec)) goto fail;

  if (line_buf.reported_underflow) goto fail;

  if (vec.count == 0) {
    goto fail;
  }

  ln = ((struct line *)backtrace_vector_grow(state, sizeof(struct line),
                                             error_callback, data, &vec.vec));
  if (ln == NULL) goto fail;
  ln->pc = (uintptr_t)-1;
  ln->filename = NULL;
  ln->lineno = 0;
  ln->idx = 0;

  if (!backtrace_vector_release(state, &vec.vec, error_callback, data))
    goto fail;

  ln = (struct line *)vec.vec.base;
  backtrace_qsort(ln, vec.count, sizeof(struct line), line_compare);

  *lines = ln;
  *lines_count = vec.count;

  return 1;

fail:
  backtrace_vector_free(state, &vec.vec, error_callback, data);
  free_line_header(state, hdr, error_callback, data);
  *lines = (struct line *)(uintptr_t)-1;
  *lines_count = 0;
  return 0;
}

static const char *read_referenced_name(struct dwarf_data *, struct unit *,
                                        uint64_t, backtrace_error_callback,
                                        void *);

static const char *read_referenced_name_from_attr(
    struct dwarf_data *ddata, struct unit *u, struct attr *attr,
    struct attr_val *val, backtrace_error_callback error_callback, void *data) {
  switch (attr->name) {
    case DW_AT_abstract_origin:
    case DW_AT_specification:
      break;
    default:
      return NULL;
  }

  if (attr->form == DW_FORM_ref_sig8) return NULL;

  if (val->encoding == ATTR_VAL_REF_INFO) {
    struct unit *unit =
        find_unit(ddata->units, ddata->units_count, val->u.uint);
    if (unit == NULL) return NULL;

    uint64_t offset = val->u.uint - unit->low_offset;
    return read_referenced_name(ddata, unit, offset, error_callback, data);
  }

  if (val->encoding == ATTR_VAL_UINT || val->encoding == ATTR_VAL_REF_UNIT)
    return read_referenced_name(ddata, u, val->u.uint, error_callback, data);

  if (val->encoding == ATTR_VAL_REF_ALT_INFO) {
    struct unit *alt_unit = find_unit(ddata->altlink->units,
                                      ddata->altlink->units_count, val->u.uint);
    if (alt_unit == NULL) return NULL;

    uint64_t offset = val->u.uint - alt_unit->low_offset;
    return read_referenced_name(ddata->altlink, alt_unit, offset,
                                error_callback, data);
  }

  return NULL;
}

static const char *read_referenced_name(struct dwarf_data *ddata,
                                        struct unit *u, uint64_t offset,
                                        backtrace_error_callback error_callback,
                                        void *data) {
  struct dwarf_buf unit_buf;
  uint64_t code;
  const struct abbrev *abbrev;
  const char *ret;
  size_t i;

  if (offset < u->unit_data_offset ||
      offset - u->unit_data_offset >= u->unit_data_len) {
    error_callback(data, "abstract origin or specification out of range", 0);
    return NULL;
  }

  offset -= u->unit_data_offset;

  unit_buf.name = ".debug_info";
  unit_buf.start = ddata->dwarf_sections.data[DEBUG_INFO];
  unit_buf.buf = u->unit_data + offset;
  unit_buf.left = u->unit_data_len - offset;
  unit_buf.is_bigendian = ddata->is_bigendian;
  unit_buf.error_callback = error_callback;
  unit_buf.data = data;
  unit_buf.reported_underflow = 0;

  code = read_uleb128(&unit_buf);
  if (code == 0) {
    dwarf_buf_error(&unit_buf, "invalid abstract origin or specification", 0);
    return NULL;
  }

  abbrev = lookup_abbrev(&u->abbrevs, code, error_callback, data);
  if (abbrev == NULL) return NULL;

  ret = NULL;
  for (i = 0; i < abbrev->num_attrs; ++i) {
    struct attr_val val;

    if (!read_attribute(abbrev->attrs[i].form, abbrev->attrs[i].val, &unit_buf,
                        u->is_dwarf64, u->version, u->addrsize,
                        &ddata->dwarf_sections, ddata->altlink, &val))
      return NULL;

    switch (abbrev->attrs[i].name) {
      case DW_AT_name:

        if (ret != NULL) break;
        if (!resolve_string(&ddata->dwarf_sections, u->is_dwarf64,
                            ddata->is_bigendian, u->str_offsets_base, &val,
                            error_callback, data, &ret))
          return NULL;
        break;

      case DW_AT_linkage_name:
      case DW_AT_MIPS_linkage_name:

      {
        const char *s;

        s = NULL;
        if (!resolve_string(&ddata->dwarf_sections, u->is_dwarf64,
                            ddata->is_bigendian, u->str_offsets_base, &val,
                            error_callback, data, &s))
          return NULL;
        if (s != NULL) return s;
      } break;

      case DW_AT_specification:

      {
        const char *name;

        name = read_referenced_name_from_attr(ddata, u, &abbrev->attrs[i], &val,
                                              error_callback, data);
        if (name != NULL) ret = name;
      } break;

      default:
        break;
    }
  }

  return ret;
}

static int add_function_range(struct backtrace_state *state, void *rdata,
                              uint64_t lowpc, uint64_t highpc,
                              backtrace_error_callback error_callback,
                              void *data, void *pvec) {
  struct function *function = (struct function *)rdata;
  struct function_vector *vec = (struct function_vector *)pvec;
  struct function_addrs *p;

  if (vec->count > 0) {
    p = (struct function_addrs *)vec->vec.base + (vec->count - 1);
    if ((lowpc == p->high || lowpc == p->high + 1) && function == p->function) {
      if (highpc > p->high) p->high = highpc;
      return 1;
    }
  }

  p = ((struct function_addrs *)backtrace_vector_grow(
      state, sizeof(struct function_addrs), error_callback, data, &vec->vec));
  if (p == NULL) return 0;

  p->low = lowpc;
  p->high = highpc;
  p->function = function;

  ++vec->count;

  return 1;
}

static int read_function_entry(struct backtrace_state *state,
                               struct dwarf_data *ddata, struct unit *u,
                               uint64_t base, struct dwarf_buf *unit_buf,
                               const struct line_header *lhdr,
                               backtrace_error_callback error_callback,
                               void *data, struct function_vector *vec_function,
                               struct function_vector *vec_inlined) {
  while (unit_buf->left > 0) {
    uint64_t code;
    const struct abbrev *abbrev;
    int is_function;
    struct function *function;
    struct function_vector *vec;
    size_t i;
    struct pcrange pcrange;
    int have_linkage_name;

    code = read_uleb128(unit_buf);
    if (code == 0) return 1;

    abbrev = lookup_abbrev(&u->abbrevs, code, error_callback, data);
    if (abbrev == NULL) return 0;

    is_function = (abbrev->tag == DW_TAG_subprogram ||
                   abbrev->tag == DW_TAG_entry_point ||
                   abbrev->tag == DW_TAG_inlined_subroutine);

    if (abbrev->tag == DW_TAG_inlined_subroutine)
      vec = vec_inlined;
    else
      vec = vec_function;

    function = NULL;
    if (is_function) {
      function = ((struct function *)backtrace_alloc(state, sizeof *function,
                                                     error_callback, data));
      if (function == NULL) return 0;
      memset(function, 0, sizeof *function);
    }

    memset(&pcrange, 0, sizeof pcrange);
    have_linkage_name = 0;
    for (i = 0; i < abbrev->num_attrs; ++i) {
      struct attr_val val;

      if (!read_attribute(abbrev->attrs[i].form, abbrev->attrs[i].val, unit_buf,
                          u->is_dwarf64, u->version, u->addrsize,
                          &ddata->dwarf_sections, ddata->altlink, &val))
        return 0;

      if ((abbrev->tag == DW_TAG_compile_unit ||
           abbrev->tag == DW_TAG_skeleton_unit) &&
          abbrev->attrs[i].name == DW_AT_low_pc) {
        if (val.encoding == ATTR_VAL_ADDRESS)
          base = val.u.uint;
        else if (val.encoding == ATTR_VAL_ADDRESS_INDEX) {
          if (!resolve_addr_index(&ddata->dwarf_sections, u->addr_base,
                                  u->addrsize, ddata->is_bigendian, val.u.uint,
                                  error_callback, data, &base))
            return 0;
        }
      }

      if (is_function) {
        switch (abbrev->attrs[i].name) {
          case DW_AT_call_file:
            if (val.encoding == ATTR_VAL_UINT) {
              if (val.u.uint >= lhdr->filenames_count) {
                dwarf_buf_error(unit_buf,
                                ("invalid file number in "
                                 "DW_AT_call_file attribute"),
                                0);
                return 0;
              }
              function->caller_filename = lhdr->filenames[val.u.uint];
            }
            break;

          case DW_AT_call_line:
            if (val.encoding == ATTR_VAL_UINT)
              function->caller_lineno = val.u.uint;
            break;

          case DW_AT_abstract_origin:
          case DW_AT_specification:

            if (have_linkage_name) break;
            {
              const char *name;

              name = read_referenced_name_from_attr(ddata, u, &abbrev->attrs[i],
                                                    &val, error_callback, data);
              if (name != NULL) function->name = name;
            }
            break;

          case DW_AT_name:

            if (function->name != NULL) break;
            if (!resolve_string(&ddata->dwarf_sections, u->is_dwarf64,
                                ddata->is_bigendian, u->str_offsets_base, &val,
                                error_callback, data, &function->name))
              return 0;
            break;

          case DW_AT_linkage_name:
          case DW_AT_MIPS_linkage_name:

          {
            const char *s;

            s = NULL;
            if (!resolve_string(&ddata->dwarf_sections, u->is_dwarf64,
                                ddata->is_bigendian, u->str_offsets_base, &val,
                                error_callback, data, &s))
              return 0;
            if (s != NULL) {
              function->name = s;
              have_linkage_name = 1;
            }
          } break;

          case DW_AT_low_pc:
          case DW_AT_high_pc:
          case DW_AT_ranges:
            update_pcrange(&abbrev->attrs[i], &val, &pcrange);
            break;

          default:
            break;
        }
      }
    }

    if (is_function && function->name == NULL) {
      backtrace_free(state, function, sizeof *function, error_callback, data);
      is_function = 0;
    }

    if (is_function) {
      if (pcrange.have_ranges || (pcrange.have_lowpc && pcrange.have_highpc)) {
        if (!add_ranges(state, &ddata->dwarf_sections, ddata->base_address,
                        ddata->is_bigendian, u, base, &pcrange,
                        add_function_range, (void *)function, error_callback,
                        data, (void *)vec))
          return 0;
      } else {
        backtrace_free(state, function, sizeof *function, error_callback, data);
        is_function = 0;
      }
    }

    if (abbrev->has_children) {
      if (!is_function) {
        if (!read_function_entry(state, ddata, u, base, unit_buf, lhdr,
                                 error_callback, data, vec_function,
                                 vec_inlined))
          return 0;
      } else {
        struct function_vector fvec;

        memset(&fvec, 0, sizeof fvec);

        if (!read_function_entry(state, ddata, u, base, unit_buf, lhdr,
                                 error_callback, data, vec_function, &fvec))
          return 0;

        if (fvec.count > 0) {
          struct function_addrs *p;
          struct function_addrs *faddrs;

          p = ((struct function_addrs *)backtrace_vector_grow(
              state, sizeof(struct function_addrs), error_callback, data,
              &fvec.vec));
          if (p == NULL) return 0;
          p->low = 0;
          --p->low;
          p->high = p->low;
          p->function = NULL;

          if (!backtrace_vector_release(state, &fvec.vec, error_callback, data))
            return 0;

          faddrs = (struct function_addrs *)fvec.vec.base;
          backtrace_qsort(faddrs, fvec.count, sizeof(struct function_addrs),
                          function_addrs_compare);

          function->function_addrs = faddrs;
          function->function_addrs_count = fvec.count;
        }
      }
    }
  }

  return 1;
}

static void read_function_info(
    struct backtrace_state *state, struct dwarf_data *ddata,
    const struct line_header *lhdr, backtrace_error_callback error_callback,
    void *data, struct unit *u, struct function_vector *fvec,
    struct function_addrs **ret_addrs, size_t *ret_addrs_count) {
  struct function_vector lvec;
  struct function_vector *pfvec;
  struct dwarf_buf unit_buf;
  struct function_addrs *p;
  struct function_addrs *addrs;
  size_t addrs_count;

  if (fvec != NULL)
    pfvec = fvec;
  else {
    memset(&lvec, 0, sizeof lvec);
    pfvec = &lvec;
  }

  unit_buf.name = ".debug_info";
  unit_buf.start = ddata->dwarf_sections.data[DEBUG_INFO];
  unit_buf.buf = u->unit_data;
  unit_buf.left = u->unit_data_len;
  unit_buf.is_bigendian = ddata->is_bigendian;
  unit_buf.error_callback = error_callback;
  unit_buf.data = data;
  unit_buf.reported_underflow = 0;

  while (unit_buf.left > 0) {
    if (!read_function_entry(state, ddata, u, 0, &unit_buf, lhdr,
                             error_callback, data, pfvec, pfvec))
      return;
  }

  if (pfvec->count == 0) return;

  p = ((struct function_addrs *)backtrace_vector_grow(
      state, sizeof(struct function_addrs), error_callback, data, &pfvec->vec));
  if (p == NULL) return;
  p->low = 0;
  --p->low;
  p->high = p->low;
  p->function = NULL;

  addrs_count = pfvec->count;

  if (fvec == NULL) {
    if (!backtrace_vector_release(state, &lvec.vec, error_callback, data))
      return;
    addrs = (struct function_addrs *)pfvec->vec.base;
  } else {
    addrs = ((struct function_addrs *)backtrace_vector_finish(
        state, &fvec->vec, error_callback, data));
    if (addrs == NULL) return;
    fvec->count = 0;
  }

  backtrace_qsort(addrs, addrs_count, sizeof(struct function_addrs),
                  function_addrs_compare);

  *ret_addrs = addrs;
  *ret_addrs_count = addrs_count;
}

static int report_inlined_functions(uintptr_t pc, struct function *function,
                                    backtrace_full_callback callback,
                                    void *data, const char **filename,
                                    int *lineno) {
  struct function_addrs *p;
  struct function_addrs *match;
  struct function *inlined;
  int ret;

  if (function->function_addrs_count == 0) return 0;

  if (pc + 1 == 0) return 0;

  p = ((struct function_addrs *)bsearch(
      &pc, function->function_addrs, function->function_addrs_count,
      sizeof(struct function_addrs), function_addrs_search));
  if (p == NULL) return 0;

  while (pc == (p + 1)->low) ++p;
  match = NULL;
  while (1) {
    if (pc < p->high) {
      match = p;
      break;
    }
    if (p == function->function_addrs) break;
    if ((p - 1)->low < p->low) break;
    --p;
  }
  if (match == NULL) return 0;

  inlined = match->function;

  ret = report_inlined_functions(pc, inlined, callback, data, filename, lineno);
  if (ret != 0) return ret;

  ret = callback(data, pc, *filename, *lineno, inlined->name);
  if (ret != 0) return ret;

  *filename = inlined->caller_filename;
  *lineno = inlined->caller_lineno;

  return 0;
}

static int dwarf_lookup_pc(struct backtrace_state *state,
                           struct dwarf_data *ddata, uintptr_t pc,
                           backtrace_full_callback callback,
                           backtrace_error_callback error_callback, void *data,
                           int *found) {
  struct unit_addrs *entry;
  int found_entry;
  struct unit *u;
  int new_data;
  struct line *lines;
  struct line *ln;
  struct function_addrs *p;
  struct function_addrs *fmatch;
  struct function *function;
  const char *filename;
  int lineno;
  int ret;

  *found = 1;

  entry = (ddata->addrs_count == 0 || pc + 1 == 0
               ? NULL
               : bsearch(&pc, ddata->addrs, ddata->addrs_count,
                         sizeof(struct unit_addrs), unit_addrs_search));

  if (entry == NULL) {
    *found = 0;
    return 0;
  }

  while (pc == (entry + 1)->low) ++entry;
  found_entry = 0;
  while (1) {
    if (pc < entry->high) {
      found_entry = 1;
      break;
    }
    if (entry == ddata->addrs) break;
    if ((entry - 1)->low < entry->low) break;
    --entry;
  }
  if (!found_entry) {
    *found = 0;
    return 0;
  }

  u = entry->u;
  lines = u->lines;

  while (entry > ddata->addrs && pc >= (entry - 1)->low &&
         pc < (entry - 1)->high) {
    if (state->threaded)
      lines = (struct line *)backtrace_atomic_load_pointer(&u->lines);

    if (lines != (struct line *)(uintptr_t)-1) break;

    --entry;

    u = entry->u;
    lines = u->lines;
  }

  if (state->threaded) lines = backtrace_atomic_load_pointer(&u->lines);

  new_data = 0;
  if (lines == NULL) {
    struct function_addrs *function_addrs;
    size_t function_addrs_count;
    struct line_header lhdr;
    size_t count;

    function_addrs = NULL;
    function_addrs_count = 0;
    if (read_line_info(state, ddata, error_callback, data, entry->u, &lhdr,
                       &lines, &count)) {
      struct function_vector *pfvec;

      if (state->threaded)
        pfvec = NULL;
      else
        pfvec = &ddata->fvec;
      read_function_info(state, ddata, &lhdr, error_callback, data, entry->u,
                         pfvec, &function_addrs, &function_addrs_count);
      free_line_header(state, &lhdr, error_callback, data);
      new_data = 1;
    }

    if (!state->threaded) {
      u->lines_count = count;
      u->function_addrs = function_addrs;
      u->function_addrs_count = function_addrs_count;
      u->lines = lines;
    } else {
      backtrace_atomic_store_size_t(&u->lines_count, count);
      backtrace_atomic_store_pointer(&u->function_addrs, function_addrs);
      backtrace_atomic_store_size_t(&u->function_addrs_count,
                                    function_addrs_count);
      backtrace_atomic_store_pointer(&u->lines, lines);
    }
  }

  if (lines == (struct line *)(uintptr_t)-1) {
    if (new_data)
      return dwarf_lookup_pc(state, ddata, pc, callback, error_callback, data,
                             found);
    return callback(data, pc, NULL, 0, NULL);
  }

  ln = (struct line *)bsearch(&pc, lines, entry->u->lines_count,
                              sizeof(struct line), line_search);
  if (ln == NULL) {
    if (entry->u->abs_filename == NULL) {
      const char *filename;

      filename = entry->u->filename;
      if (filename != NULL && !IS_ABSOLUTE_PATH(filename) &&
          entry->u->comp_dir != NULL) {
        size_t filename_len;
        const char *dir;
        size_t dir_len;
        char *s;

        filename_len = strlen(filename);
        dir = entry->u->comp_dir;
        dir_len = strlen(dir);
        s = (char *)backtrace_alloc(state, dir_len + filename_len + 2,
                                    error_callback, data);
        if (s == NULL) {
          *found = 0;
          return 0;
        }
        memcpy(s, dir, dir_len);

        s[dir_len] = '/';
        memcpy(s + dir_len + 1, filename, filename_len + 1);
        filename = s;
      }
      entry->u->abs_filename = filename;
    }

    return callback(data, pc, entry->u->abs_filename, 0, NULL);
  }

  if (entry->u->function_addrs_count == 0)
    return callback(data, pc, ln->filename, ln->lineno, NULL);

  p = ((struct function_addrs *)bsearch(
      &pc, entry->u->function_addrs, entry->u->function_addrs_count,
      sizeof(struct function_addrs), function_addrs_search));
  if (p == NULL) return callback(data, pc, ln->filename, ln->lineno, NULL);

  while (pc == (p + 1)->low) ++p;
  fmatch = NULL;
  while (1) {
    if (pc < p->high) {
      fmatch = p;
      break;
    }
    if (p == entry->u->function_addrs) break;
    if ((p - 1)->low < p->low) break;
    --p;
  }
  if (fmatch == NULL) return callback(data, pc, ln->filename, ln->lineno, NULL);

  function = fmatch->function;

  filename = ln->filename;
  lineno = ln->lineno;

  ret = report_inlined_functions(pc, function, callback, data, &filename,
                                 &lineno);
  if (ret != 0) return ret;

  return callback(data, pc, filename, lineno, function->name);
}

static int dwarf_fileline(struct backtrace_state *state, uintptr_t pc,
                          backtrace_full_callback callback,
                          backtrace_error_callback error_callback, void *data) {
  struct dwarf_data *ddata;
  int found;
  int ret;

  if (!state->threaded) {
    for (ddata = (struct dwarf_data *)state->fileline_data; ddata != NULL;
         ddata = ddata->next) {
      ret = dwarf_lookup_pc(state, ddata, pc, callback, error_callback, data,
                            &found);
      if (ret != 0 || found) return ret;
    }
  } else {
    struct dwarf_data **pp;

    pp = (struct dwarf_data **)(void *)&state->fileline_data;
    while (1) {
      ddata = backtrace_atomic_load_pointer(pp);
      if (ddata == NULL) break;

      ret = dwarf_lookup_pc(state, ddata, pc, callback, error_callback, data,
                            &found);
      if (ret != 0 || found) return ret;

      pp = &ddata->next;
    }
  }

  return callback(data, pc, NULL, 0, NULL);
}

static struct dwarf_data *build_dwarf_data(
    struct backtrace_state *state, uintptr_t base_address,
    const struct dwarf_sections *dwarf_sections, int is_bigendian,
    struct dwarf_data *altlink, backtrace_error_callback error_callback,
    void *data) {
  struct unit_addrs_vector addrs_vec;
  struct unit_addrs *addrs;
  size_t addrs_count;
  struct unit_vector units_vec;
  struct unit **units;
  size_t units_count;
  struct dwarf_data *fdata;

  if (!build_address_map(state, base_address, dwarf_sections, is_bigendian,
                         altlink, error_callback, data, &addrs_vec, &units_vec))
    return NULL;

  if (!backtrace_vector_release(state, &addrs_vec.vec, error_callback, data))
    return NULL;
  if (!backtrace_vector_release(state, &units_vec.vec, error_callback, data))
    return NULL;
  addrs = (struct unit_addrs *)addrs_vec.vec.base;
  units = (struct unit **)units_vec.vec.base;
  addrs_count = addrs_vec.count;
  units_count = units_vec.count;
  backtrace_qsort(addrs, addrs_count, sizeof(struct unit_addrs),
                  unit_addrs_compare);

  fdata = ((struct dwarf_data *)backtrace_alloc(
      state, sizeof(struct dwarf_data), error_callback, data));
  if (fdata == NULL) return NULL;

  fdata->next = NULL;
  fdata->altlink = altlink;
  fdata->base_address = base_address;
  fdata->addrs = addrs;
  fdata->addrs_count = addrs_count;
  fdata->units = units;
  fdata->units_count = units_count;
  fdata->dwarf_sections = *dwarf_sections;
  fdata->is_bigendian = is_bigendian;
  memset(&fdata->fvec, 0, sizeof fdata->fvec);

  return fdata;
}

int backtrace_dwarf_add(struct backtrace_state *state, uintptr_t base_address,
                        const struct dwarf_sections *dwarf_sections,
                        int is_bigendian, struct dwarf_data *fileline_altlink,
                        backtrace_error_callback error_callback, void *data,
                        fileline *fileline_fn,
                        struct dwarf_data **fileline_entry) {
  struct dwarf_data *fdata;

  fdata = build_dwarf_data(state, base_address, dwarf_sections, is_bigendian,
                           fileline_altlink, error_callback, data);
  if (fdata == NULL) return 0;

  if (fileline_entry != NULL) *fileline_entry = fdata;

  if (!state->threaded) {
    struct dwarf_data **pp;

    for (pp = (struct dwarf_data **)(void *)&state->fileline_data; *pp != NULL;
         pp = &(*pp)->next)
      ;
    *pp = fdata;
  } else {
    while (1) {
      struct dwarf_data **pp;

      pp = (struct dwarf_data **)(void *)&state->fileline_data;

      while (1) {
        struct dwarf_data *p;

        p = backtrace_atomic_load_pointer(pp);

        if (p == NULL) break;

        pp = &p->next;
      }

      if (__sync_bool_compare_and_swap(pp, NULL, fdata)) break;
    }
  }

  *fileline_fn = dwarf_fileline;

  return 1;
}

// fileline.c:
#include <errno.h>
#include <fcntl.h>
#include <stdlib.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>

#if defined(HAVE_KERN_PROC_ARGS) || defined(HAVE_KERN_PROC)
#include <sys/sysctl.h>
#endif

#ifdef HAVE_MACH_O_DYLD_H
#include <mach-o/dyld.h>
#endif

#ifndef HAVE_GETEXECNAME
#define getexecname() NULL
#endif

#if !defined(HAVE_KERN_PROC_ARGS) && !defined(HAVE_KERN_PROC)

#define sysctl_exec_name1(state, error_callback, data) NULL
#define sysctl_exec_name2(state, error_callback, data) NULL

#else
static char *sysctl_exec_name(struct backtrace_state *state, int mib0, int mib1,
                              int mib2, int mib3,
                              backtrace_error_callback error_callback,
                              void *data) {
  int mib[4];
  size_t len;
  char *name;
  size_t rlen;

  mib[0] = mib0;
  mib[1] = mib1;
  mib[2] = mib2;
  mib[3] = mib3;

  if (sysctl(mib, 4, NULL, &len, NULL, 0) < 0) return NULL;
  name = (char *)backtrace_alloc(state, len, error_callback, data);
  if (name == NULL) return NULL;
  rlen = len;
  if (sysctl(mib, 4, name, &rlen, NULL, 0) < 0) {
    backtrace_free(state, name, len, error_callback, data);
    return NULL;
  }
  return name;
}

#ifdef HAVE_KERN_PROC_ARGS

static char *sysctl_exec_name1(struct backtrace_state *state,
                               backtrace_error_callback error_callback,
                               void *data) {
  return sysctl_exec_name(state, CTL_KERN, KERN_PROC_ARGS, -1,
                          KERN_PROC_PATHNAME, error_callback, data);
}

#else

#define sysctl_exec_name1(state, error_callback, data) NULL

#endif

#ifdef HAVE_KERN_PROC

static char *sysctl_exec_name2(struct backtrace_state *state,
                               backtrace_error_callback error_callback,
                               void *data) {
  return sysctl_exec_name(state, CTL_KERN, KERN_PROC, KERN_PROC_PATHNAME, -1,
                          error_callback, data);
}

#else

#define sysctl_exec_name2(state, error_callback, data) NULL

#endif

#endif
#ifdef HAVE_MACH_O_DYLD_H

static char *macho_get_executable_path(struct backtrace_state *state,
                                       backtrace_error_callback error_callback,
                                       void *data) {
  uint32_t len;
  char *name;

  len = 0;
  if (_NSGetExecutablePath(NULL, &len) == 0) return NULL;
  name = (char *)backtrace_alloc(state, len, error_callback, data);
  if (name == NULL) return NULL;
  if (_NSGetExecutablePath(name, &len) != 0) {
    backtrace_free(state, name, len, error_callback, data);
    return NULL;
  }
  return name;
}

#else
#define macho_get_executable_path(state, error_callback, data) NULL

#endif

static int fileline_initialize(struct backtrace_state *state,
                               backtrace_error_callback error_callback,
                               void *data) {
  int failed;
  fileline fileline_fn;
  int pass;
  int called_error_callback;
  int descriptor;
  const char *filename;
  char buf[64];

  if (!state->threaded)
    failed = state->fileline_initialization_failed;
  else
    failed = backtrace_atomic_load_int(&state->fileline_initialization_failed);

  if (failed) {
    error_callback(data, "failed to read executable information", -1);
    return 0;
  }

  if (!state->threaded)
    fileline_fn = state->fileline_fn;
  else
    fileline_fn = backtrace_atomic_load_pointer(&state->fileline_fn);
  if (fileline_fn != NULL) return 1;

  descriptor = -1;
  called_error_callback = 0;
  for (pass = 0; pass < 8; ++pass) {
    int does_not_exist;

    switch (pass) {
      case 0:
        filename = state->filename;
        break;
      case 1:
        filename = getexecname();
        break;
      case 2:
        filename = "/proc/self/exe";
        break;
      case 3:
        filename = "/proc/curproc/file";
        break;
      case 4:
        snprintf(buf, sizeof(buf), "/proc/%ld/object/a.out", (long)getpid());
        filename = buf;
        break;
      case 5:
        filename = sysctl_exec_name1(state, error_callback, data);
        break;
      case 6:
        filename = sysctl_exec_name2(state, error_callback, data);
        break;
      case 7:
        filename = macho_get_executable_path(state, error_callback, data);
        break;
      default:
        abort();
    }

    if (filename == NULL) continue;

    descriptor =
        backtrace_open(filename, error_callback, data, &does_not_exist);
    if (descriptor < 0 && !does_not_exist) {
      called_error_callback = 1;
      break;
    }
    if (descriptor >= 0) break;
  }

  if (descriptor < 0) {
    if (!called_error_callback) {
      if (state->filename != NULL)
        error_callback(data, state->filename, ENOENT);
      else
        error_callback(data, "libbacktrace could not find executable to open",
                       0);
    }
    failed = 1;
  }

  if (!failed) {
    if (!backtrace_initialize(state, filename, descriptor, error_callback, data,
                              &fileline_fn))
      failed = 1;
  }

  if (failed) {
    if (!state->threaded)
      state->fileline_initialization_failed = 1;
    else
      backtrace_atomic_store_int(&state->fileline_initialization_failed, 1);
    return 0;
  }

  if (!state->threaded)
    state->fileline_fn = fileline_fn;
  else {
    backtrace_atomic_store_pointer(&state->fileline_fn, fileline_fn);
  }

  return 1;
}

int backtrace_pcinfo(struct backtrace_state *state, uintptr_t pc,
                     backtrace_full_callback callback,
                     backtrace_error_callback error_callback, void *data) {
  if (!fileline_initialize(state, error_callback, data)) return 0;

  if (state->fileline_initialization_failed) return 0;

  return state->fileline_fn(state, pc, callback, error_callback, data);
}

int backtrace_syminfo(struct backtrace_state *state, uintptr_t pc,
                      backtrace_syminfo_callback callback,
                      backtrace_error_callback error_callback, void *data) {
  if (!fileline_initialize(state, error_callback, data)) return 0;

  if (state->fileline_initialization_failed) return 0;

  state->syminfo_fn(state, pc, callback, error_callback, data);
  return 1;
}

void backtrace_syminfo_to_full_callback(void *data, uintptr_t pc,
                                        const char *symname,
                                        uintptr_t symval ATTRIBUTE_UNUSED,
                                        uintptr_t symsize ATTRIBUTE_UNUSED) {
  struct backtrace_call_full *bdata = (struct backtrace_call_full *)data;

  bdata->ret = bdata->full_callback(bdata->full_data, pc, NULL, 0, symname);
}

void backtrace_syminfo_to_full_error_callback(void *data, const char *msg,
                                              int errnum) {
  struct backtrace_call_full *bdata = (struct backtrace_call_full *)data;

  bdata->full_error_callback(bdata->full_data, msg, errnum);
}

// posix.c:
#include <errno.h>
#include <fcntl.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>

#ifndef O_BINARY
#define O_BINARY 0
#endif

#ifndef O_CLOEXEC
#define O_CLOEXEC 0
#endif

#ifndef FD_CLOEXEC
#define FD_CLOEXEC 1
#endif

int backtrace_open(const char *filename,
                   backtrace_error_callback error_callback, void *data,
                   int *does_not_exist) {
  int descriptor;

  if (does_not_exist != NULL) *does_not_exist = 0;

  descriptor = open(filename, (int)(O_RDONLY | O_BINARY | O_CLOEXEC));
  if (descriptor < 0) {
    if (does_not_exist != NULL && (errno == ENOENT || errno == EACCES))
      *does_not_exist = 1;
    else
      error_callback(data, filename, errno);
    return -1;
  }

#ifdef HAVE_FCNTL

  fcntl(descriptor, F_SETFD, FD_CLOEXEC);
#endif

  return descriptor;
}

int backtrace_close(int descriptor, backtrace_error_callback error_callback,
                    void *data) {
  if (close(descriptor) < 0) {
    error_callback(data, "close", errno);
    return 0;
  }
  return 1;
}

// print.c:
#include <stdio.h>
#include <string.h>
#include <sys/types.h>

struct print_data {
  struct backtrace_state *state;
  FILE *f;
};

static int print_callback(void *data, uintptr_t pc, const char *filename,
                          int lineno, const char *function) {
  struct print_data *pdata = (struct print_data *)data;

  fprintf(pdata->f, "0x%lx %s\n\t%s:%d\n", (unsigned long)pc,
          function == NULL ? "???" : function,
          filename == NULL ? "???" : filename, lineno);
  return 0;
}

static void error_callback(void *data, const char *msg, int errnum) {
  struct print_data *pdata = (struct print_data *)data;

  if (pdata->state->filename != NULL)
    fprintf(stderr, "%s: ", pdata->state->filename);
  fprintf(stderr, "libbacktrace: %s", msg);
  if (errnum > 0) fprintf(stderr, ": %s", strerror(errnum));
  fputc('\n', stderr);
}

void __attribute__((noinline))
backtrace_print(struct backtrace_state *state, int skip, FILE *f) {
  struct print_data data;

  data.state = state;
  data.f = f;
  backtrace_full(state, skip + 1, print_callback, error_callback,
                 (void *)&data);
}

// sort.c:
#include <stddef.h>
#include <sys/types.h>

static void swap(char *a, char *b, size_t size) {
  size_t i;

  for (i = 0; i < size; i++, a++, b++) {
    char t;

    t = *a;
    *a = *b;
    *b = t;
  }
}

void backtrace_qsort(void *basearg, size_t count, size_t size,
                     int (*compar)(const void *, const void *)) {
  char *base = (char *)basearg;
  size_t i;
  size_t mid;

tail_recurse:
  if (count < 2) return;

  swap(base, base + (count / 2) * size, size);

  mid = 0;
  for (i = 1; i < count; i++) {
    if ((*compar)(base, base + i * size) > 0) {
      ++mid;
      if (i != mid) swap(base + mid * size, base + i * size, size);
    }
  }

  if (mid > 0) swap(base, base + mid * size, size);

  if (2 * mid < count) {
    backtrace_qsort(base, mid, size, compar);
    base += (mid + 1) * size;
    count -= mid + 1;
    goto tail_recurse;
  } else {
    backtrace_qsort(base + (mid + 1) * size, count - (mid + 1), size, compar);
    count = mid;
    goto tail_recurse;
  }
}

// state.c:
#include <string.h>
#include <sys/types.h>

struct backtrace_state *backtrace_create_state(
    const char *filename, int threaded, backtrace_error_callback error_callback,
    void *data) {
  struct backtrace_state init_state;
  struct backtrace_state *state;

#ifndef HAVE_SYNC_FUNCTIONS
  if (threaded) {
    error_callback(data, "backtrace library does not support threads", 0);
    return NULL;
  }
#endif

  memset(&init_state, 0, sizeof init_state);
  init_state.filename = filename;
  init_state.threaded = threaded;

  state = ((struct backtrace_state *)backtrace_alloc(&init_state, sizeof *state,
                                                     error_callback, data));
  if (state == NULL) return NULL;
  *state = init_state;

  return state;
}

// backtrace.c:
#include <sys/types.h>

#ifdef BACKTRACE_SUPPORTED
#include <unwind.h>

struct backtrace_data {
  int skip;

  struct backtrace_state *state;

  backtrace_full_callback callback;

  backtrace_error_callback error_callback;

  void *data;

  int ret;

  int can_alloc;
};

static _Unwind_Reason_Code unwind(struct _Unwind_Context *context,
                                  void *vdata) {
  struct backtrace_data *bdata = (struct backtrace_data *)vdata;
  uintptr_t pc;
  int ip_before_insn = 0;

#ifdef HAVE_GETIPINFO
  pc = _Unwind_GetIPInfo(context, &ip_before_insn);
#else
  pc = _Unwind_GetIP(context);
#endif

  if (bdata->skip > 0) {
    --bdata->skip;
    return _URC_NO_REASON;
  }

  if (!ip_before_insn) --pc;

  if (!bdata->can_alloc)
    bdata->ret = bdata->callback(bdata->data, pc, NULL, 0, NULL);
  else
    bdata->ret = backtrace_pcinfo(bdata->state, pc, bdata->callback,
                                  bdata->error_callback, bdata->data);
  if (bdata->ret != 0) return _URC_END_OF_STACK;

  return _URC_NO_REASON;
}

int __attribute__((noinline))
backtrace_full(struct backtrace_state *state, int skip,
               backtrace_full_callback callback,
               backtrace_error_callback error_callback, void *data) {
  struct backtrace_data bdata;
  void *p;

  bdata.skip = skip + 1;
  bdata.state = state;
  bdata.callback = callback;
  bdata.error_callback = error_callback;
  bdata.data = data;
  bdata.ret = 0;

  p = backtrace_alloc(state, 4096, NULL, NULL);
  if (p == NULL)
    bdata.can_alloc = 0;
  else {
    backtrace_free(state, p, 4096, NULL, NULL);
    bdata.can_alloc = 1;
  }

  _Unwind_Backtrace(unwind, &bdata);
  return bdata.ret;
}
#else
// Copied from nounwind.c
int
backtrace_full (struct backtrace_state *state ATTRIBUTE_UNUSED,
		int skip ATTRIBUTE_UNUSED,
		backtrace_full_callback callback ATTRIBUTE_UNUSED,
		backtrace_error_callback error_callback, void *data)
{
  error_callback (data,
		  "no stack trace because unwind library not available",
		  0);
  return 0;
}
#endif

// simple.c:
#ifdef BACKTRACE_SUPPORTED
#include <unwind.h>

struct backtrace_simple_data {
  int skip;

  struct backtrace_state *state;

  backtrace_simple_callback callback;

  backtrace_error_callback error_callback;

  void *data;

  int ret;
};

static _Unwind_Reason_Code simple_unwind(struct _Unwind_Context *context,
                                         void *vdata) {
  struct backtrace_simple_data *bdata = (struct backtrace_simple_data *)vdata;
  uintptr_t pc;
  int ip_before_insn = 0;

#ifdef HAVE_GETIPINFO
  pc = _Unwind_GetIPInfo(context, &ip_before_insn);
#else
  pc = _Unwind_GetIP(context);
#endif

  if (bdata->skip > 0) {
    --bdata->skip;
    return _URC_NO_REASON;
  }

  if (!ip_before_insn) --pc;

  bdata->ret = bdata->callback(bdata->data, pc);

  if (bdata->ret != 0) return _URC_END_OF_STACK;

  return _URC_NO_REASON;
}

int __attribute__((noinline))
backtrace_simple(struct backtrace_state *state, int skip,
                 backtrace_simple_callback callback,
                 backtrace_error_callback error_callback, void *data) {
  struct backtrace_simple_data bdata;

  bdata.skip = skip + 1;
  bdata.state = state;
  bdata.callback = callback;
  bdata.error_callback = error_callback;
  bdata.data = data;
  bdata.ret = 0;
  _Unwind_Backtrace(simple_unwind, &bdata);
  return bdata.ret;
}
#else
int
backtrace_simple (struct backtrace_state *state ATTRIBUTE_UNUSED,
		  int skip ATTRIBUTE_UNUSED,
		  backtrace_simple_callback callback ATTRIBUTE_UNUSED,
		  backtrace_error_callback error_callback, void *data)
{
  error_callback (data,
		  "no stack trace because unwind library not available",
		  0);
  return 0;
}
#endif