|  | /* BFD back-end for binary objects. | 
|  | Copyright (C) 1994-2024 Free Software Foundation, Inc. | 
|  | Written by Ian Lance Taylor, Cygnus Support, <ian@cygnus.com> | 
|  |  | 
|  | This file is part of BFD, the Binary File Descriptor library. | 
|  |  | 
|  | This program is free software; you can redistribute it and/or modify | 
|  | it under the terms of the GNU General Public License as published by | 
|  | the Free Software Foundation; either version 3 of the License, or | 
|  | (at your option) any later version. | 
|  |  | 
|  | This program is distributed in the hope that it will be useful, | 
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | GNU General Public License for more details. | 
|  |  | 
|  | You should have received a copy of the GNU General Public License | 
|  | along with this program; if not, write to the Free Software | 
|  | Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, | 
|  | MA 02110-1301, USA.  */ | 
|  |  | 
|  | /* This is a BFD backend which may be used to read or write binary | 
|  | objects.  Historically, it was used as an output format for objcopy | 
|  | in order to generate raw binary data, but is now used for other | 
|  | purposes as well. | 
|  |  | 
|  | This is very simple.  The only complication is that the real data | 
|  | will start at some address X, and in some cases we will not want to | 
|  | include X zeroes just to get to that point.  Since the start | 
|  | address is not meaningful for this object file format, we use it | 
|  | instead to indicate the number of zeroes to skip at the start of | 
|  | the file.  objcopy cooperates by specially setting the start | 
|  | address to zero by default.  */ | 
|  |  | 
|  | #include "sysdep.h" | 
|  | #include "bfd.h" | 
|  | #include "safe-ctype.h" | 
|  | #include "libbfd.h" | 
|  |  | 
|  | /* Any bfd we create by reading a binary file has three symbols: | 
|  | a start symbol, an end symbol, and an absolute length symbol.  */ | 
|  | #define BIN_SYMS 3 | 
|  |  | 
|  | /* Create a binary object.  Invoked via bfd_set_format.  */ | 
|  |  | 
|  | static bool | 
|  | binary_mkobject (bfd *abfd ATTRIBUTE_UNUSED) | 
|  | { | 
|  | return true; | 
|  | } | 
|  |  | 
|  | /* Any file may be considered to be a binary file, provided the target | 
|  | was not defaulted.  That is, it must be explicitly specified as | 
|  | being binary.  */ | 
|  |  | 
|  | static bfd_cleanup | 
|  | binary_object_p (bfd *abfd) | 
|  | { | 
|  | struct stat statbuf; | 
|  | asection *sec; | 
|  | flagword flags; | 
|  |  | 
|  | if (abfd->target_defaulted) | 
|  | { | 
|  | bfd_set_error (bfd_error_wrong_format); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | abfd->symcount = BIN_SYMS; | 
|  |  | 
|  | /* Find the file size.  */ | 
|  | if (bfd_stat (abfd, &statbuf) < 0) | 
|  | { | 
|  | bfd_set_error (bfd_error_system_call); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | /* One data section.  */ | 
|  | flags = SEC_ALLOC | SEC_LOAD | SEC_DATA | SEC_HAS_CONTENTS; | 
|  | sec = bfd_make_section_with_flags (abfd, ".data", flags); | 
|  | if (sec == NULL) | 
|  | return NULL; | 
|  | sec->vma = 0; | 
|  | sec->size = statbuf.st_size; | 
|  | sec->filepos = 0; | 
|  |  | 
|  | abfd->tdata.any = (void *) sec; | 
|  |  | 
|  | return _bfd_no_cleanup; | 
|  | } | 
|  |  | 
|  | #define binary_close_and_cleanup     _bfd_generic_close_and_cleanup | 
|  | #define binary_bfd_free_cached_info  _bfd_generic_bfd_free_cached_info | 
|  | #define binary_new_section_hook      _bfd_generic_new_section_hook | 
|  |  | 
|  | /* Get contents of the only section.  */ | 
|  |  | 
|  | static bool | 
|  | binary_get_section_contents (bfd *abfd, | 
|  | asection *section, | 
|  | void * location, | 
|  | file_ptr offset, | 
|  | bfd_size_type count) | 
|  | { | 
|  | if (bfd_seek (abfd, section->filepos + offset, SEEK_SET) != 0 | 
|  | || bfd_read (location, count, abfd) != count) | 
|  | return false; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | /* Return the amount of memory needed to read the symbol table.  */ | 
|  |  | 
|  | static long | 
|  | binary_get_symtab_upper_bound (bfd *abfd ATTRIBUTE_UNUSED) | 
|  | { | 
|  | return (BIN_SYMS + 1) * sizeof (asymbol *); | 
|  | } | 
|  |  | 
|  | /* Create a symbol name based on the bfd's filename.  */ | 
|  |  | 
|  | static char * | 
|  | mangle_name (bfd *abfd, char *suffix) | 
|  | { | 
|  | bfd_size_type size; | 
|  | char *buf; | 
|  | char *p; | 
|  |  | 
|  | size = (strlen (bfd_get_filename (abfd)) | 
|  | + strlen (suffix) | 
|  | + sizeof "_binary__"); | 
|  |  | 
|  | buf = (char *) bfd_alloc (abfd, size); | 
|  | if (buf == NULL) | 
|  | return ""; | 
|  |  | 
|  | sprintf (buf, "_binary_%s_%s", bfd_get_filename (abfd), suffix); | 
|  |  | 
|  | /* Change any non-alphanumeric characters to underscores.  */ | 
|  | for (p = buf; *p; p++) | 
|  | if (! ISALNUM (*p)) | 
|  | *p = '_'; | 
|  |  | 
|  | return buf; | 
|  | } | 
|  |  | 
|  | /* Return the symbol table.  */ | 
|  |  | 
|  | static long | 
|  | binary_canonicalize_symtab (bfd *abfd, asymbol **alocation) | 
|  | { | 
|  | asection *sec = (asection *) abfd->tdata.any; | 
|  | asymbol *syms; | 
|  | unsigned int i; | 
|  | size_t amt = BIN_SYMS * sizeof (asymbol); | 
|  |  | 
|  | syms = (asymbol *) bfd_alloc (abfd, amt); | 
|  | if (syms == NULL) | 
|  | return -1; | 
|  |  | 
|  | /* Start symbol.  */ | 
|  | syms[0].the_bfd = abfd; | 
|  | syms[0].name = mangle_name (abfd, "start"); | 
|  | syms[0].value = 0; | 
|  | syms[0].flags = BSF_GLOBAL; | 
|  | syms[0].section = sec; | 
|  | syms[0].udata.p = NULL; | 
|  |  | 
|  | /* End symbol.  */ | 
|  | syms[1].the_bfd = abfd; | 
|  | syms[1].name = mangle_name (abfd, "end"); | 
|  | syms[1].value = sec->size; | 
|  | syms[1].flags = BSF_GLOBAL; | 
|  | syms[1].section = sec; | 
|  | syms[1].udata.p = NULL; | 
|  |  | 
|  | /* Size symbol.  */ | 
|  | syms[2].the_bfd = abfd; | 
|  | syms[2].name = mangle_name (abfd, "size"); | 
|  | syms[2].value = sec->size; | 
|  | syms[2].flags = BSF_GLOBAL; | 
|  | syms[2].section = bfd_abs_section_ptr; | 
|  | syms[2].udata.p = NULL; | 
|  |  | 
|  | for (i = 0; i < BIN_SYMS; i++) | 
|  | *alocation++ = syms++; | 
|  | *alocation = NULL; | 
|  |  | 
|  | return BIN_SYMS; | 
|  | } | 
|  |  | 
|  | #define binary_make_empty_symbol  _bfd_generic_make_empty_symbol | 
|  | #define binary_print_symbol       _bfd_nosymbols_print_symbol | 
|  | #define binary_get_symbol_version_string \ | 
|  | _bfd_nosymbols_get_symbol_version_string | 
|  |  | 
|  | /* Get information about a symbol.  */ | 
|  |  | 
|  | static void | 
|  | binary_get_symbol_info (bfd *ignore_abfd ATTRIBUTE_UNUSED, | 
|  | asymbol *symbol, | 
|  | symbol_info *ret) | 
|  | { | 
|  | bfd_symbol_info (symbol, ret); | 
|  | } | 
|  |  | 
|  | #define binary_bfd_is_local_label_name	    bfd_generic_is_local_label_name | 
|  | #define binary_get_lineno		   _bfd_nosymbols_get_lineno | 
|  | #define binary_find_nearest_line	   _bfd_nosymbols_find_nearest_line | 
|  | #define binary_find_nearest_line_with_alt  _bfd_nosymbols_find_nearest_line_with_alt | 
|  | #define binary_find_line		   _bfd_nosymbols_find_line | 
|  | #define binary_find_inliner_info	   _bfd_nosymbols_find_inliner_info | 
|  | #define binary_bfd_make_debug_symbol	   _bfd_nosymbols_bfd_make_debug_symbol | 
|  | #define binary_read_minisymbols		   _bfd_generic_read_minisymbols | 
|  | #define binary_minisymbol_to_symbol	   _bfd_generic_minisymbol_to_symbol | 
|  | #define binary_bfd_is_target_special_symbol _bfd_bool_bfd_asymbol_false | 
|  |  | 
|  | /* Set the architecture of a binary file.  */ | 
|  | #define binary_set_arch_mach _bfd_generic_set_arch_mach | 
|  |  | 
|  | /* Write section contents of a binary file.  */ | 
|  |  | 
|  | static bool | 
|  | binary_set_section_contents (bfd *abfd, | 
|  | asection *sec, | 
|  | const void * data, | 
|  | file_ptr offset, | 
|  | bfd_size_type size) | 
|  | { | 
|  | if (size == 0) | 
|  | return true; | 
|  |  | 
|  | if (! abfd->output_has_begun) | 
|  | { | 
|  | bool found_low; | 
|  | bfd_vma low; | 
|  | asection *s; | 
|  |  | 
|  | /* The lowest section LMA sets the virtual address of the start | 
|  | of the file.  We use this to set the file position of all the | 
|  | sections.  */ | 
|  | found_low = false; | 
|  | low = 0; | 
|  | for (s = abfd->sections; s != NULL; s = s->next) | 
|  | if (((s->flags | 
|  | & (SEC_HAS_CONTENTS | SEC_LOAD | SEC_ALLOC | SEC_NEVER_LOAD)) | 
|  | == (SEC_HAS_CONTENTS | SEC_LOAD | SEC_ALLOC)) | 
|  | && (s->size > 0) | 
|  | && (! found_low || s->lma < low)) | 
|  | { | 
|  | low = s->lma; | 
|  | found_low = true; | 
|  | } | 
|  |  | 
|  | for (s = abfd->sections; s != NULL; s = s->next) | 
|  | { | 
|  | unsigned int opb = bfd_octets_per_byte (abfd, s); | 
|  |  | 
|  | s->filepos = (s->lma - low) * opb; | 
|  |  | 
|  | /* Skip following warning check for sections that will not | 
|  | occupy file space.  */ | 
|  | if ((s->flags | 
|  | & (SEC_HAS_CONTENTS | SEC_ALLOC | SEC_NEVER_LOAD)) | 
|  | != (SEC_HAS_CONTENTS | SEC_ALLOC) | 
|  | || (s->size == 0)) | 
|  | continue; | 
|  |  | 
|  | /* If attempting to generate a binary file from a bfd with | 
|  | LMA's all over the place, huge (sparse?) binary files may | 
|  | result.  This condition attempts to detect this situation | 
|  | and print a warning.  Better heuristics would be nice to | 
|  | have.  */ | 
|  |  | 
|  | if (s->filepos < 0) | 
|  | _bfd_error_handler | 
|  | /* xgettext:c-format */ | 
|  | (_("warning: writing section `%pA' at huge (ie negative) " | 
|  | "file offset"), | 
|  | s); | 
|  | } | 
|  |  | 
|  | abfd->output_has_begun = true; | 
|  | } | 
|  |  | 
|  | /* We don't want to output anything for a section that is neither | 
|  | loaded nor allocated.  The contents of such a section are not | 
|  | meaningful in the binary format.  */ | 
|  | if ((sec->flags & (SEC_LOAD | SEC_ALLOC)) == 0) | 
|  | return true; | 
|  | if ((sec->flags & SEC_NEVER_LOAD) != 0) | 
|  | return true; | 
|  |  | 
|  | return _bfd_generic_set_section_contents (abfd, sec, data, offset, size); | 
|  | } | 
|  |  | 
|  | /* No space is required for header information.  */ | 
|  |  | 
|  | static int | 
|  | binary_sizeof_headers (bfd *abfd ATTRIBUTE_UNUSED, | 
|  | struct bfd_link_info *info ATTRIBUTE_UNUSED) | 
|  | { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | #define binary_bfd_get_relocated_section_contents  bfd_generic_get_relocated_section_contents | 
|  | #define binary_bfd_relax_section		   bfd_generic_relax_section | 
|  | #define binary_bfd_gc_sections			   bfd_generic_gc_sections | 
|  | #define binary_bfd_lookup_section_flags		   bfd_generic_lookup_section_flags | 
|  | #define binary_bfd_merge_sections		   bfd_generic_merge_sections | 
|  | #define binary_bfd_is_group_section		   bfd_generic_is_group_section | 
|  | #define binary_bfd_group_name			   bfd_generic_group_name | 
|  | #define binary_bfd_discard_group		   bfd_generic_discard_group | 
|  | #define binary_section_already_linked		  _bfd_generic_section_already_linked | 
|  | #define binary_bfd_define_common_symbol		   bfd_generic_define_common_symbol | 
|  | #define binary_bfd_link_hide_symbol		   _bfd_generic_link_hide_symbol | 
|  | #define binary_bfd_define_start_stop		   bfd_generic_define_start_stop | 
|  | #define binary_bfd_link_hash_table_create	  _bfd_generic_link_hash_table_create | 
|  | #define binary_bfd_link_just_syms		  _bfd_generic_link_just_syms | 
|  | #define binary_bfd_copy_link_hash_symbol_type	  _bfd_generic_copy_link_hash_symbol_type | 
|  | #define binary_bfd_link_add_symbols		  _bfd_generic_link_add_symbols | 
|  | #define binary_bfd_final_link			  _bfd_generic_final_link | 
|  | #define binary_bfd_link_split_section		  _bfd_generic_link_split_section | 
|  | #define binary_get_section_contents_in_window	  _bfd_generic_get_section_contents_in_window | 
|  | #define binary_bfd_link_check_relocs		  _bfd_generic_link_check_relocs | 
|  |  | 
|  | const bfd_target binary_vec = | 
|  | { | 
|  | "binary",			/* name */ | 
|  | bfd_target_unknown_flavour,	/* flavour */ | 
|  | BFD_ENDIAN_UNKNOWN,		/* byteorder */ | 
|  | BFD_ENDIAN_UNKNOWN,		/* header_byteorder */ | 
|  | EXEC_P,			/* object_flags */ | 
|  | (SEC_ALLOC | SEC_LOAD | SEC_READONLY | SEC_CODE | SEC_DATA | 
|  | | SEC_ROM | SEC_HAS_CONTENTS), /* section_flags */ | 
|  | 0,				/* symbol_leading_char */ | 
|  | ' ',				/* ar_pad_char */ | 
|  | 16,				/* ar_max_namelen */ | 
|  | 255,				/* match priority.  */ | 
|  | TARGET_KEEP_UNUSED_SECTION_SYMBOLS, /* keep unused section symbols.  */ | 
|  | bfd_getb64, bfd_getb_signed_64, bfd_putb64, | 
|  | bfd_getb32, bfd_getb_signed_32, bfd_putb32, | 
|  | bfd_getb16, bfd_getb_signed_16, bfd_putb16,	/* data */ | 
|  | bfd_getb64, bfd_getb_signed_64, bfd_putb64, | 
|  | bfd_getb32, bfd_getb_signed_32, bfd_putb32, | 
|  | bfd_getb16, bfd_getb_signed_16, bfd_putb16,	/* hdrs */ | 
|  | {				/* bfd_check_format */ | 
|  | _bfd_dummy_target, | 
|  | binary_object_p, | 
|  | _bfd_dummy_target, | 
|  | _bfd_dummy_target, | 
|  | }, | 
|  | {				/* bfd_set_format */ | 
|  | _bfd_bool_bfd_false_error, | 
|  | binary_mkobject, | 
|  | _bfd_bool_bfd_false_error, | 
|  | _bfd_bool_bfd_false_error, | 
|  | }, | 
|  | {				/* bfd_write_contents */ | 
|  | _bfd_bool_bfd_false_error, | 
|  | _bfd_bool_bfd_true, | 
|  | _bfd_bool_bfd_false_error, | 
|  | _bfd_bool_bfd_false_error, | 
|  | }, | 
|  |  | 
|  | BFD_JUMP_TABLE_GENERIC (binary), | 
|  | BFD_JUMP_TABLE_COPY (_bfd_generic), | 
|  | BFD_JUMP_TABLE_CORE (_bfd_nocore), | 
|  | BFD_JUMP_TABLE_ARCHIVE (_bfd_noarchive), | 
|  | BFD_JUMP_TABLE_SYMBOLS (binary), | 
|  | BFD_JUMP_TABLE_RELOCS (_bfd_norelocs), | 
|  | BFD_JUMP_TABLE_WRITE (binary), | 
|  | BFD_JUMP_TABLE_LINK (binary), | 
|  | BFD_JUMP_TABLE_DYNAMIC (_bfd_nodynamic), | 
|  |  | 
|  | NULL, | 
|  |  | 
|  | NULL | 
|  | }; |