1 1.5 christos /* $NetBSD: libdwarf_str.c,v 1.5 2024/03/03 17:37:33 christos Exp $ */ 2 1.2 christos 3 1.1 christos /*- 4 1.5 christos * Copyright (c) 2009,2010,2023 Kai Wang 5 1.1 christos * All rights reserved. 6 1.1 christos * 7 1.1 christos * Redistribution and use in source and binary forms, with or without 8 1.1 christos * modification, are permitted provided that the following conditions 9 1.1 christos * are met: 10 1.1 christos * 1. Redistributions of source code must retain the above copyright 11 1.1 christos * notice, this list of conditions and the following disclaimer. 12 1.1 christos * 2. Redistributions in binary form must reproduce the above copyright 13 1.1 christos * notice, this list of conditions and the following disclaimer in the 14 1.1 christos * documentation and/or other materials provided with the distribution. 15 1.1 christos * 16 1.1 christos * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 17 1.1 christos * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18 1.1 christos * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19 1.1 christos * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 20 1.1 christos * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 21 1.1 christos * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 22 1.1 christos * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 23 1.1 christos * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 24 1.1 christos * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 25 1.1 christos * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 26 1.1 christos * SUCH DAMAGE. 27 1.1 christos */ 28 1.1 christos 29 1.1 christos #include "_libdwarf.h" 30 1.1 christos 31 1.5 christos __RCSID("$NetBSD: libdwarf_str.c,v 1.5 2024/03/03 17:37:33 christos Exp $"); 32 1.5 christos ELFTC_VCSID("Id: libdwarf_str.c 4016 2023-10-15 05:39:46Z kaiwang27"); 33 1.1 christos 34 1.1 christos #define _INIT_DWARF_STRTAB_SIZE 1024 35 1.1 christos 36 1.1 christos int 37 1.1 christos _dwarf_strtab_add(Dwarf_Debug dbg, char *string, uint64_t *off, 38 1.1 christos Dwarf_Error *error) 39 1.1 christos { 40 1.1 christos size_t len; 41 1.1 christos 42 1.1 christos assert(dbg != NULL && string != NULL); 43 1.1 christos 44 1.1 christos len = strlen(string) + 1; 45 1.1 christos while (dbg->dbg_strtab_size + len > dbg->dbg_strtab_cap) { 46 1.1 christos dbg->dbg_strtab_cap *= 2; 47 1.1 christos dbg->dbg_strtab = realloc(dbg->dbg_strtab, 48 1.1 christos (size_t) dbg->dbg_strtab_cap); 49 1.1 christos if (dbg->dbg_strtab == NULL) { 50 1.1 christos DWARF_SET_ERROR(dbg, error, DW_DLE_MEMORY); 51 1.1 christos return (DW_DLE_MEMORY); 52 1.1 christos } 53 1.1 christos } 54 1.1 christos 55 1.1 christos if (off != NULL) 56 1.1 christos *off = dbg->dbg_strtab_size; 57 1.1 christos 58 1.5 christos memcpy(&dbg->dbg_strtab[dbg->dbg_strtab_size], string, len - 1); 59 1.1 christos dbg->dbg_strtab_size += len; 60 1.1 christos dbg->dbg_strtab[dbg->dbg_strtab_size - 1] = '\0'; 61 1.1 christos 62 1.1 christos return (DW_DLE_NONE); 63 1.1 christos } 64 1.1 christos 65 1.1 christos char * 66 1.1 christos _dwarf_strtab_get_table(Dwarf_Debug dbg) 67 1.1 christos { 68 1.1 christos 69 1.1 christos assert(dbg != NULL); 70 1.1 christos 71 1.1 christos return (dbg->dbg_strtab); 72 1.1 christos } 73 1.1 christos 74 1.5 christos char * 75 1.5 christos _dwarf_strtab_get_line_table(Dwarf_Debug dbg) 76 1.5 christos { 77 1.5 christos 78 1.5 christos assert(dbg != NULL); 79 1.5 christos 80 1.5 christos return (dbg->dbg_line_strtab); 81 1.5 christos } 82 1.5 christos 83 1.1 christos int 84 1.1 christos _dwarf_strtab_init(Dwarf_Debug dbg, Dwarf_Error *error) 85 1.1 christos { 86 1.1 christos Dwarf_Section *ds; 87 1.1 christos 88 1.1 christos assert(dbg != NULL); 89 1.1 christos 90 1.1 christos if (dbg->dbg_mode == DW_DLC_READ || dbg->dbg_mode == DW_DLC_RDWR) { 91 1.1 christos ds = _dwarf_find_section(dbg, ".debug_str"); 92 1.1 christos if (ds == NULL) { 93 1.1 christos dbg->dbg_strtab = NULL; 94 1.1 christos dbg->dbg_strtab_cap = dbg->dbg_strtab_size = 0; 95 1.1 christos return (DW_DLE_NONE); 96 1.1 christos } 97 1.1 christos 98 1.1 christos dbg->dbg_strtab_cap = dbg->dbg_strtab_size = ds->ds_size; 99 1.1 christos 100 1.1 christos if (dbg->dbg_mode == DW_DLC_RDWR) { 101 1.1 christos if ((dbg->dbg_strtab = malloc((size_t) ds->ds_size)) == 102 1.1 christos NULL) { 103 1.1 christos DWARF_SET_ERROR(dbg, error, DW_DLE_MEMORY); 104 1.1 christos return (DW_DLE_MEMORY); 105 1.1 christos } 106 1.1 christos memcpy(dbg->dbg_strtab, ds->ds_data, ds->ds_size); 107 1.1 christos } else 108 1.1 christos dbg->dbg_strtab = (char *) ds->ds_data; 109 1.5 christos 110 1.5 christos ds = _dwarf_find_section(dbg, ".debug_line_str"); 111 1.5 christos if (ds != NULL) { 112 1.5 christos dbg->dbg_line_strtab = (char *) ds->ds_data; 113 1.5 christos } 114 1.1 christos } else { 115 1.1 christos /* DW_DLC_WRITE */ 116 1.1 christos 117 1.1 christos dbg->dbg_strtab_cap = _INIT_DWARF_STRTAB_SIZE; 118 1.1 christos dbg->dbg_strtab_size = 0; 119 1.1 christos 120 1.1 christos if ((dbg->dbg_strtab = malloc((size_t) dbg->dbg_strtab_cap)) == 121 1.1 christos NULL) { 122 1.1 christos DWARF_SET_ERROR(dbg, error, DW_DLE_MEMORY); 123 1.1 christos return (DW_DLE_MEMORY); 124 1.1 christos } 125 1.1 christos 126 1.1 christos dbg->dbg_strtab[0] = '\0'; 127 1.1 christos } 128 1.1 christos 129 1.1 christos return (DW_DLE_NONE); 130 1.1 christos } 131 1.1 christos 132 1.1 christos void 133 1.1 christos _dwarf_strtab_cleanup(Dwarf_Debug dbg) 134 1.1 christos { 135 1.1 christos 136 1.1 christos assert(dbg != NULL); 137 1.1 christos 138 1.1 christos if (dbg->dbg_mode == DW_DLC_RDWR || dbg->dbg_mode == DW_DLC_WRITE) 139 1.1 christos free(dbg->dbg_strtab); 140 1.1 christos } 141 1.1 christos 142 1.1 christos int 143 1.1 christos _dwarf_strtab_gen(Dwarf_P_Debug dbg, Dwarf_Error *error) 144 1.1 christos { 145 1.1 christos Dwarf_P_Section ds; 146 1.1 christos int ret; 147 1.1 christos 148 1.1 christos assert(dbg != NULL); 149 1.1 christos 150 1.1 christos if ((ret = _dwarf_section_init(dbg, &ds, ".debug_str", 0, error)) != 151 1.1 christos DW_DLE_NONE) 152 1.1 christos return (ret); 153 1.1 christos 154 1.1 christos if (dbg->dbg_strtab_size > ds->ds_cap) { 155 1.1 christos ds->ds_data = realloc(ds->ds_data, 156 1.1 christos (size_t) dbg->dbg_strtab_size); 157 1.1 christos if (ds->ds_data == NULL) { 158 1.1 christos _dwarf_section_free(dbg, &ds); 159 1.1 christos DWARF_SET_ERROR(dbg, error, DW_DLE_MEMORY); 160 1.1 christos return (DW_DLE_MEMORY); 161 1.1 christos } 162 1.1 christos ds->ds_cap = dbg->dbg_strtab_size; 163 1.1 christos } 164 1.1 christos 165 1.1 christos memcpy(ds->ds_data, dbg->dbg_strtab, dbg->dbg_strtab_size); 166 1.1 christos ds->ds_size = dbg->dbg_strtab_size; 167 1.1 christos 168 1.1 christos /* 169 1.1 christos * Inform application the creation of .debug_str ELF section. 170 1.1 christos * Note that .debug_str use a different format than usual ELF 171 1.1 christos * string table, so it should not have SHT_STRTAB as its type. 172 1.1 christos */ 173 1.1 christos ret = _dwarf_section_callback(dbg, ds, SHT_PROGBITS, 0, 0, 0, error); 174 1.1 christos 175 1.1 christos return (ret); 176 1.1 christos } 177