daily update
[binutils.git] / binutils / objcopy.c
blobe577dd30160b55bed773cae29f09da23324542ab
1 /* objcopy.c -- copy object file from input to output, optionally massaging it.
2 Copyright 1991, 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, 2000,
3 2001, 2002, 2003, 2004, 2005, 2006, 2007
4 Free Software Foundation, Inc.
6 This file is part of GNU Binutils.
8 This program is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 3 of the License, or
11 (at your option) any later version.
13 This program is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program; if not, write to the Free Software
20 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA
21 02110-1301, USA. */
23 #include "sysdep.h"
24 #include "bfd.h"
25 #include "progress.h"
26 #include "getopt.h"
27 #include "libiberty.h"
28 #include "bucomm.h"
29 #include "budbg.h"
30 #include "filenames.h"
31 #include "fnmatch.h"
32 #include "elf-bfd.h"
33 #include <sys/stat.h>
34 #include "libbfd.h"
36 /* A list of symbols to explicitly strip out, or to keep. A linked
37 list is good enough for a small number from the command line, but
38 this will slow things down a lot if many symbols are being
39 deleted. */
41 struct symlist
43 const char *name;
44 struct symlist *next;
47 /* A list to support redefine_sym. */
48 struct redefine_node
50 char *source;
51 char *target;
52 struct redefine_node *next;
55 typedef struct section_rename
57 const char * old_name;
58 const char * new_name;
59 flagword flags;
60 struct section_rename * next;
62 section_rename;
64 /* List of sections to be renamed. */
65 static section_rename *section_rename_list;
67 #define RETURN_NONFATAL(s) {bfd_nonfatal (s); status = 1; return;}
69 static asymbol **isympp = NULL; /* Input symbols. */
70 static asymbol **osympp = NULL; /* Output symbols that survive stripping. */
72 /* If `copy_byte' >= 0, copy only that byte of every `interleave' bytes. */
73 static int copy_byte = -1;
74 static int interleave = 4;
76 static bfd_boolean verbose; /* Print file and target names. */
77 static bfd_boolean preserve_dates; /* Preserve input file timestamp. */
78 static int status = 0; /* Exit status. */
80 enum strip_action
82 STRIP_UNDEF,
83 STRIP_NONE, /* Don't strip. */
84 STRIP_DEBUG, /* Strip all debugger symbols. */
85 STRIP_UNNEEDED, /* Strip unnecessary symbols. */
86 STRIP_NONDEBUG, /* Strip everything but debug info. */
87 STRIP_ALL /* Strip all symbols. */
90 /* Which symbols to remove. */
91 static enum strip_action strip_symbols;
93 enum locals_action
95 LOCALS_UNDEF,
96 LOCALS_START_L, /* Discard locals starting with L. */
97 LOCALS_ALL /* Discard all locals. */
100 /* Which local symbols to remove. Overrides STRIP_ALL. */
101 static enum locals_action discard_locals;
103 /* What kind of change to perform. */
104 enum change_action
106 CHANGE_IGNORE,
107 CHANGE_MODIFY,
108 CHANGE_SET
111 /* Structure used to hold lists of sections and actions to take. */
112 struct section_list
114 struct section_list * next; /* Next section to change. */
115 const char * name; /* Section name. */
116 bfd_boolean used; /* Whether this entry was used. */
117 bfd_boolean remove; /* Whether to remove this section. */
118 bfd_boolean copy; /* Whether to copy this section. */
119 enum change_action change_vma;/* Whether to change or set VMA. */
120 bfd_vma vma_val; /* Amount to change by or set to. */
121 enum change_action change_lma;/* Whether to change or set LMA. */
122 bfd_vma lma_val; /* Amount to change by or set to. */
123 bfd_boolean set_flags; /* Whether to set the section flags. */
124 flagword flags; /* What to set the section flags to. */
127 static struct section_list *change_sections;
129 /* TRUE if some sections are to be removed. */
130 static bfd_boolean sections_removed;
132 /* TRUE if only some sections are to be copied. */
133 static bfd_boolean sections_copied;
135 /* Changes to the start address. */
136 static bfd_vma change_start = 0;
137 static bfd_boolean set_start_set = FALSE;
138 static bfd_vma set_start;
140 /* Changes to section addresses. */
141 static bfd_vma change_section_address = 0;
143 /* Filling gaps between sections. */
144 static bfd_boolean gap_fill_set = FALSE;
145 static bfd_byte gap_fill = 0;
147 /* Pad to a given address. */
148 static bfd_boolean pad_to_set = FALSE;
149 static bfd_vma pad_to;
151 /* Use alternative machine code? */
152 static unsigned long use_alt_mach_code = 0;
154 /* Output BFD flags user wants to set or clear */
155 static flagword bfd_flags_to_set;
156 static flagword bfd_flags_to_clear;
158 /* List of sections to add. */
159 struct section_add
161 /* Next section to add. */
162 struct section_add *next;
163 /* Name of section to add. */
164 const char *name;
165 /* Name of file holding section contents. */
166 const char *filename;
167 /* Size of file. */
168 size_t size;
169 /* Contents of file. */
170 bfd_byte *contents;
171 /* BFD section, after it has been added. */
172 asection *section;
175 /* List of sections to add to the output BFD. */
176 static struct section_add *add_sections;
178 /* If non-NULL the argument to --add-gnu-debuglink.
179 This should be the filename to store in the .gnu_debuglink section. */
180 static const char * gnu_debuglink_filename = NULL;
182 /* Whether to convert debugging information. */
183 static bfd_boolean convert_debugging = FALSE;
185 /* Whether to change the leading character in symbol names. */
186 static bfd_boolean change_leading_char = FALSE;
188 /* Whether to remove the leading character from global symbol names. */
189 static bfd_boolean remove_leading_char = FALSE;
191 /* Whether to permit wildcard in symbol comparison. */
192 static bfd_boolean wildcard = FALSE;
194 /* True if --localize-hidden is in effect. */
195 static bfd_boolean localize_hidden = FALSE;
197 /* List of symbols to strip, keep, localize, keep-global, weaken,
198 or redefine. */
199 static struct symlist *strip_specific_list = NULL;
200 static struct symlist *strip_unneeded_list = NULL;
201 static struct symlist *keep_specific_list = NULL;
202 static struct symlist *localize_specific_list = NULL;
203 static struct symlist *globalize_specific_list = NULL;
204 static struct symlist *keepglobal_specific_list = NULL;
205 static struct symlist *weaken_specific_list = NULL;
206 static struct redefine_node *redefine_sym_list = NULL;
208 /* If this is TRUE, we weaken global symbols (set BSF_WEAK). */
209 static bfd_boolean weaken = FALSE;
211 /* If this is TRUE, we retain BSF_FILE symbols. */
212 static bfd_boolean keep_file_symbols = FALSE;
214 /* Prefix symbols/sections. */
215 static char *prefix_symbols_string = 0;
216 static char *prefix_sections_string = 0;
217 static char *prefix_alloc_sections_string = 0;
219 /* True if --extract-symbol was passed on the command line. */
220 static bfd_boolean extract_symbol = FALSE;
222 /* If `reverse_bytes' is nonzero, then reverse the order of every chunk
223 of <reverse_bytes> bytes within each output section. */
224 static int reverse_bytes = 0;
227 /* 150 isn't special; it's just an arbitrary non-ASCII char value. */
228 enum command_line_switch
230 OPTION_ADD_SECTION=150,
231 OPTION_CHANGE_ADDRESSES,
232 OPTION_CHANGE_LEADING_CHAR,
233 OPTION_CHANGE_START,
234 OPTION_CHANGE_SECTION_ADDRESS,
235 OPTION_CHANGE_SECTION_LMA,
236 OPTION_CHANGE_SECTION_VMA,
237 OPTION_CHANGE_WARNINGS,
238 OPTION_DEBUGGING,
239 OPTION_GAP_FILL,
240 OPTION_NO_CHANGE_WARNINGS,
241 OPTION_PAD_TO,
242 OPTION_REMOVE_LEADING_CHAR,
243 OPTION_SET_SECTION_FLAGS,
244 OPTION_SET_START,
245 OPTION_STRIP_UNNEEDED,
246 OPTION_WEAKEN,
247 OPTION_REDEFINE_SYM,
248 OPTION_REDEFINE_SYMS,
249 OPTION_SREC_LEN,
250 OPTION_SREC_FORCES3,
251 OPTION_STRIP_SYMBOLS,
252 OPTION_STRIP_UNNEEDED_SYMBOL,
253 OPTION_STRIP_UNNEEDED_SYMBOLS,
254 OPTION_KEEP_SYMBOLS,
255 OPTION_LOCALIZE_HIDDEN,
256 OPTION_LOCALIZE_SYMBOLS,
257 OPTION_GLOBALIZE_SYMBOL,
258 OPTION_GLOBALIZE_SYMBOLS,
259 OPTION_KEEPGLOBAL_SYMBOLS,
260 OPTION_WEAKEN_SYMBOLS,
261 OPTION_RENAME_SECTION,
262 OPTION_ALT_MACH_CODE,
263 OPTION_PREFIX_SYMBOLS,
264 OPTION_PREFIX_SECTIONS,
265 OPTION_PREFIX_ALLOC_SECTIONS,
266 OPTION_FORMATS_INFO,
267 OPTION_ADD_GNU_DEBUGLINK,
268 OPTION_ONLY_KEEP_DEBUG,
269 OPTION_KEEP_FILE_SYMBOLS,
270 OPTION_READONLY_TEXT,
271 OPTION_WRITABLE_TEXT,
272 OPTION_PURE,
273 OPTION_IMPURE,
274 OPTION_EXTRACT_SYMBOL,
275 OPTION_REVERSE_BYTES
278 /* Options to handle if running as "strip". */
280 static struct option strip_options[] =
282 {"discard-all", no_argument, 0, 'x'},
283 {"discard-locals", no_argument, 0, 'X'},
284 {"format", required_argument, 0, 'F'}, /* Obsolete */
285 {"help", no_argument, 0, 'h'},
286 {"info", no_argument, 0, OPTION_FORMATS_INFO},
287 {"input-format", required_argument, 0, 'I'}, /* Obsolete */
288 {"input-target", required_argument, 0, 'I'},
289 {"keep-file-symbols", no_argument, 0, OPTION_KEEP_FILE_SYMBOLS},
290 {"keep-symbol", required_argument, 0, 'K'},
291 {"only-keep-debug", no_argument, 0, OPTION_ONLY_KEEP_DEBUG},
292 {"output-format", required_argument, 0, 'O'}, /* Obsolete */
293 {"output-target", required_argument, 0, 'O'},
294 {"output-file", required_argument, 0, 'o'},
295 {"preserve-dates", no_argument, 0, 'p'},
296 {"remove-section", required_argument, 0, 'R'},
297 {"strip-all", no_argument, 0, 's'},
298 {"strip-debug", no_argument, 0, 'S'},
299 {"strip-unneeded", no_argument, 0, OPTION_STRIP_UNNEEDED},
300 {"strip-symbol", required_argument, 0, 'N'},
301 {"target", required_argument, 0, 'F'},
302 {"verbose", no_argument, 0, 'v'},
303 {"version", no_argument, 0, 'V'},
304 {"wildcard", no_argument, 0, 'w'},
305 {0, no_argument, 0, 0}
308 /* Options to handle if running as "objcopy". */
310 static struct option copy_options[] =
312 {"add-gnu-debuglink", required_argument, 0, OPTION_ADD_GNU_DEBUGLINK},
313 {"add-section", required_argument, 0, OPTION_ADD_SECTION},
314 {"adjust-start", required_argument, 0, OPTION_CHANGE_START},
315 {"adjust-vma", required_argument, 0, OPTION_CHANGE_ADDRESSES},
316 {"adjust-section-vma", required_argument, 0, OPTION_CHANGE_SECTION_ADDRESS},
317 {"adjust-warnings", no_argument, 0, OPTION_CHANGE_WARNINGS},
318 {"alt-machine-code", required_argument, 0, OPTION_ALT_MACH_CODE},
319 {"binary-architecture", required_argument, 0, 'B'},
320 {"byte", required_argument, 0, 'b'},
321 {"change-addresses", required_argument, 0, OPTION_CHANGE_ADDRESSES},
322 {"change-leading-char", no_argument, 0, OPTION_CHANGE_LEADING_CHAR},
323 {"change-section-address", required_argument, 0, OPTION_CHANGE_SECTION_ADDRESS},
324 {"change-section-lma", required_argument, 0, OPTION_CHANGE_SECTION_LMA},
325 {"change-section-vma", required_argument, 0, OPTION_CHANGE_SECTION_VMA},
326 {"change-start", required_argument, 0, OPTION_CHANGE_START},
327 {"change-warnings", no_argument, 0, OPTION_CHANGE_WARNINGS},
328 {"debugging", no_argument, 0, OPTION_DEBUGGING},
329 {"discard-all", no_argument, 0, 'x'},
330 {"discard-locals", no_argument, 0, 'X'},
331 {"extract-symbol", no_argument, 0, OPTION_EXTRACT_SYMBOL},
332 {"format", required_argument, 0, 'F'}, /* Obsolete */
333 {"gap-fill", required_argument, 0, OPTION_GAP_FILL},
334 {"globalize-symbol", required_argument, 0, OPTION_GLOBALIZE_SYMBOL},
335 {"globalize-symbols", required_argument, 0, OPTION_GLOBALIZE_SYMBOLS},
336 {"help", no_argument, 0, 'h'},
337 {"impure", no_argument, 0, OPTION_IMPURE},
338 {"info", no_argument, 0, OPTION_FORMATS_INFO},
339 {"input-format", required_argument, 0, 'I'}, /* Obsolete */
340 {"input-target", required_argument, 0, 'I'},
341 {"interleave", required_argument, 0, 'i'},
342 {"keep-file-symbols", no_argument, 0, OPTION_KEEP_FILE_SYMBOLS},
343 {"keep-global-symbol", required_argument, 0, 'G'},
344 {"keep-global-symbols", required_argument, 0, OPTION_KEEPGLOBAL_SYMBOLS},
345 {"keep-symbol", required_argument, 0, 'K'},
346 {"keep-symbols", required_argument, 0, OPTION_KEEP_SYMBOLS},
347 {"localize-hidden", no_argument, 0, OPTION_LOCALIZE_HIDDEN},
348 {"localize-symbol", required_argument, 0, 'L'},
349 {"localize-symbols", required_argument, 0, OPTION_LOCALIZE_SYMBOLS},
350 {"no-adjust-warnings", no_argument, 0, OPTION_NO_CHANGE_WARNINGS},
351 {"no-change-warnings", no_argument, 0, OPTION_NO_CHANGE_WARNINGS},
352 {"only-keep-debug", no_argument, 0, OPTION_ONLY_KEEP_DEBUG},
353 {"only-section", required_argument, 0, 'j'},
354 {"output-format", required_argument, 0, 'O'}, /* Obsolete */
355 {"output-target", required_argument, 0, 'O'},
356 {"pad-to", required_argument, 0, OPTION_PAD_TO},
357 {"prefix-symbols", required_argument, 0, OPTION_PREFIX_SYMBOLS},
358 {"prefix-sections", required_argument, 0, OPTION_PREFIX_SECTIONS},
359 {"prefix-alloc-sections", required_argument, 0, OPTION_PREFIX_ALLOC_SECTIONS},
360 {"preserve-dates", no_argument, 0, 'p'},
361 {"pure", no_argument, 0, OPTION_PURE},
362 {"readonly-text", no_argument, 0, OPTION_READONLY_TEXT},
363 {"redefine-sym", required_argument, 0, OPTION_REDEFINE_SYM},
364 {"redefine-syms", required_argument, 0, OPTION_REDEFINE_SYMS},
365 {"remove-leading-char", no_argument, 0, OPTION_REMOVE_LEADING_CHAR},
366 {"remove-section", required_argument, 0, 'R'},
367 {"rename-section", required_argument, 0, OPTION_RENAME_SECTION},
368 {"reverse-bytes", required_argument, 0, OPTION_REVERSE_BYTES},
369 {"set-section-flags", required_argument, 0, OPTION_SET_SECTION_FLAGS},
370 {"set-start", required_argument, 0, OPTION_SET_START},
371 {"srec-len", required_argument, 0, OPTION_SREC_LEN},
372 {"srec-forceS3", no_argument, 0, OPTION_SREC_FORCES3},
373 {"strip-all", no_argument, 0, 'S'},
374 {"strip-debug", no_argument, 0, 'g'},
375 {"strip-unneeded", no_argument, 0, OPTION_STRIP_UNNEEDED},
376 {"strip-unneeded-symbol", required_argument, 0, OPTION_STRIP_UNNEEDED_SYMBOL},
377 {"strip-unneeded-symbols", required_argument, 0, OPTION_STRIP_UNNEEDED_SYMBOLS},
378 {"strip-symbol", required_argument, 0, 'N'},
379 {"strip-symbols", required_argument, 0, OPTION_STRIP_SYMBOLS},
380 {"target", required_argument, 0, 'F'},
381 {"verbose", no_argument, 0, 'v'},
382 {"version", no_argument, 0, 'V'},
383 {"weaken", no_argument, 0, OPTION_WEAKEN},
384 {"weaken-symbol", required_argument, 0, 'W'},
385 {"weaken-symbols", required_argument, 0, OPTION_WEAKEN_SYMBOLS},
386 {"wildcard", no_argument, 0, 'w'},
387 {"writable-text", no_argument, 0, OPTION_WRITABLE_TEXT},
388 {0, no_argument, 0, 0}
391 /* IMPORTS */
392 extern char *program_name;
394 /* This flag distinguishes between strip and objcopy:
395 1 means this is 'strip'; 0 means this is 'objcopy'.
396 -1 means if we should use argv[0] to decide. */
397 extern int is_strip;
399 /* The maximum length of an S record. This variable is declared in srec.c
400 and can be modified by the --srec-len parameter. */
401 extern unsigned int Chunk;
403 /* Restrict the generation of Srecords to type S3 only.
404 This variable is declare in bfd/srec.c and can be toggled
405 on by the --srec-forceS3 command line switch. */
406 extern bfd_boolean S3Forced;
408 /* Defined in bfd/binary.c. Used to set architecture and machine of input
409 binary files. */
410 extern enum bfd_architecture bfd_external_binary_architecture;
411 extern unsigned long bfd_external_machine;
413 /* Forward declarations. */
414 static void setup_section (bfd *, asection *, void *);
415 static void setup_bfd_headers (bfd *, bfd *);
416 static void copy_section (bfd *, asection *, void *);
417 static void get_sections (bfd *, asection *, void *);
418 static int compare_section_lma (const void *, const void *);
419 static void mark_symbols_used_in_relocations (bfd *, asection *, void *);
420 static bfd_boolean write_debugging_info (bfd *, void *, long *, asymbol ***);
421 static const char *lookup_sym_redefinition (const char *);
423 static void
424 copy_usage (FILE *stream, int exit_status)
426 fprintf (stream, _("Usage: %s [option(s)] in-file [out-file]\n"), program_name);
427 fprintf (stream, _(" Copies a binary file, possibly transforming it in the process\n"));
428 fprintf (stream, _(" The options are:\n"));
429 fprintf (stream, _("\
430 -I --input-target <bfdname> Assume input file is in format <bfdname>\n\
431 -O --output-target <bfdname> Create an output file in format <bfdname>\n\
432 -B --binary-architecture <arch> Set arch of output file, when input is binary\n\
433 -F --target <bfdname> Set both input and output format to <bfdname>\n\
434 --debugging Convert debugging information, if possible\n\
435 -p --preserve-dates Copy modified/access timestamps to the output\n\
436 -j --only-section <name> Only copy section <name> into the output\n\
437 --add-gnu-debuglink=<file> Add section .gnu_debuglink linking to <file>\n\
438 -R --remove-section <name> Remove section <name> from the output\n\
439 -S --strip-all Remove all symbol and relocation information\n\
440 -g --strip-debug Remove all debugging symbols & sections\n\
441 --strip-unneeded Remove all symbols not needed by relocations\n\
442 -N --strip-symbol <name> Do not copy symbol <name>\n\
443 --strip-unneeded-symbol <name>\n\
444 Do not copy symbol <name> unless needed by\n\
445 relocations\n\
446 --only-keep-debug Strip everything but the debug information\n\
447 --extract-symbol Remove section contents but keep symbols\n\
448 -K --keep-symbol <name> Do not strip symbol <name>\n\
449 --keep-file-symbols Do not strip file symbol(s)\n\
450 --localize-hidden Turn all ELF hidden symbols into locals\n\
451 -L --localize-symbol <name> Force symbol <name> to be marked as a local\n\
452 --globalize-symbol <name> Force symbol <name> to be marked as a global\n\
453 -G --keep-global-symbol <name> Localize all symbols except <name>\n\
454 -W --weaken-symbol <name> Force symbol <name> to be marked as a weak\n\
455 --weaken Force all global symbols to be marked as weak\n\
456 -w --wildcard Permit wildcard in symbol comparison\n\
457 -x --discard-all Remove all non-global symbols\n\
458 -X --discard-locals Remove any compiler-generated symbols\n\
459 -i --interleave <number> Only copy one out of every <number> bytes\n\
460 -b --byte <num> Select byte <num> in every interleaved block\n\
461 --gap-fill <val> Fill gaps between sections with <val>\n\
462 --pad-to <addr> Pad the last section up to address <addr>\n\
463 --set-start <addr> Set the start address to <addr>\n\
464 {--change-start|--adjust-start} <incr>\n\
465 Add <incr> to the start address\n\
466 {--change-addresses|--adjust-vma} <incr>\n\
467 Add <incr> to LMA, VMA and start addresses\n\
468 {--change-section-address|--adjust-section-vma} <name>{=|+|-}<val>\n\
469 Change LMA and VMA of section <name> by <val>\n\
470 --change-section-lma <name>{=|+|-}<val>\n\
471 Change the LMA of section <name> by <val>\n\
472 --change-section-vma <name>{=|+|-}<val>\n\
473 Change the VMA of section <name> by <val>\n\
474 {--[no-]change-warnings|--[no-]adjust-warnings}\n\
475 Warn if a named section does not exist\n\
476 --set-section-flags <name>=<flags>\n\
477 Set section <name>'s properties to <flags>\n\
478 --add-section <name>=<file> Add section <name> found in <file> to output\n\
479 --rename-section <old>=<new>[,<flags>] Rename section <old> to <new>\n\
480 --change-leading-char Force output format's leading character style\n\
481 --remove-leading-char Remove leading character from global symbols\n\
482 --reverse-bytes=<num> Reverse <num> bytes at a time, in output sections with content\n\
483 --redefine-sym <old>=<new> Redefine symbol name <old> to <new>\n\
484 --redefine-syms <file> --redefine-sym for all symbol pairs \n\
485 listed in <file>\n\
486 --srec-len <number> Restrict the length of generated Srecords\n\
487 --srec-forceS3 Restrict the type of generated Srecords to S3\n\
488 --strip-symbols <file> -N for all symbols listed in <file>\n\
489 --strip-unneeded-symbols <file>\n\
490 --strip-unneeded-symbol for all symbols listed\n\
491 in <file>\n\
492 --keep-symbols <file> -K for all symbols listed in <file>\n\
493 --localize-symbols <file> -L for all symbols listed in <file>\n\
494 --globalize-symbols <file> --globalize-symbol for all in <file>\n\
495 --keep-global-symbols <file> -G for all symbols listed in <file>\n\
496 --weaken-symbols <file> -W for all symbols listed in <file>\n\
497 --alt-machine-code <index> Use the target's <index>'th alternative machine\n\
498 --writable-text Mark the output text as writable\n\
499 --readonly-text Make the output text write protected\n\
500 --pure Mark the output file as demand paged\n\
501 --impure Mark the output file as impure\n\
502 --prefix-symbols <prefix> Add <prefix> to start of every symbol name\n\
503 --prefix-sections <prefix> Add <prefix> to start of every section name\n\
504 --prefix-alloc-sections <prefix>\n\
505 Add <prefix> to start of every allocatable\n\
506 section name\n\
507 -v --verbose List all object files modified\n\
508 @<file> Read options from <file>\n\
509 -V --version Display this program's version number\n\
510 -h --help Display this output\n\
511 --info List object formats & architectures supported\n\
512 "));
513 list_supported_targets (program_name, stream);
514 if (REPORT_BUGS_TO[0] && exit_status == 0)
515 fprintf (stream, _("Report bugs to %s\n"), REPORT_BUGS_TO);
516 exit (exit_status);
519 static void
520 strip_usage (FILE *stream, int exit_status)
522 fprintf (stream, _("Usage: %s <option(s)> in-file(s)\n"), program_name);
523 fprintf (stream, _(" Removes symbols and sections from files\n"));
524 fprintf (stream, _(" The options are:\n"));
525 fprintf (stream, _("\
526 -I --input-target=<bfdname> Assume input file is in format <bfdname>\n\
527 -O --output-target=<bfdname> Create an output file in format <bfdname>\n\
528 -F --target=<bfdname> Set both input and output format to <bfdname>\n\
529 -p --preserve-dates Copy modified/access timestamps to the output\n\
530 -R --remove-section=<name> Remove section <name> from the output\n\
531 -s --strip-all Remove all symbol and relocation information\n\
532 -g -S -d --strip-debug Remove all debugging symbols & sections\n\
533 --strip-unneeded Remove all symbols not needed by relocations\n\
534 --only-keep-debug Strip everything but the debug information\n\
535 -N --strip-symbol=<name> Do not copy symbol <name>\n\
536 -K --keep-symbol=<name> Do not strip symbol <name>\n\
537 --keep-file-symbols Do not strip file symbol(s)\n\
538 -w --wildcard Permit wildcard in symbol comparison\n\
539 -x --discard-all Remove all non-global symbols\n\
540 -X --discard-locals Remove any compiler-generated symbols\n\
541 -v --verbose List all object files modified\n\
542 -V --version Display this program's version number\n\
543 -h --help Display this output\n\
544 --info List object formats & architectures supported\n\
545 -o <file> Place stripped output into <file>\n\
546 "));
548 list_supported_targets (program_name, stream);
549 if (REPORT_BUGS_TO[0] && exit_status == 0)
550 fprintf (stream, _("Report bugs to %s\n"), REPORT_BUGS_TO);
551 exit (exit_status);
554 /* Parse section flags into a flagword, with a fatal error if the
555 string can't be parsed. */
557 static flagword
558 parse_flags (const char *s)
560 flagword ret;
561 const char *snext;
562 int len;
564 ret = SEC_NO_FLAGS;
568 snext = strchr (s, ',');
569 if (snext == NULL)
570 len = strlen (s);
571 else
573 len = snext - s;
574 ++snext;
577 if (0) ;
578 #define PARSE_FLAG(fname,fval) \
579 else if (strncasecmp (fname, s, len) == 0) ret |= fval
580 PARSE_FLAG ("alloc", SEC_ALLOC);
581 PARSE_FLAG ("load", SEC_LOAD);
582 PARSE_FLAG ("noload", SEC_NEVER_LOAD);
583 PARSE_FLAG ("readonly", SEC_READONLY);
584 PARSE_FLAG ("debug", SEC_DEBUGGING);
585 PARSE_FLAG ("code", SEC_CODE);
586 PARSE_FLAG ("data", SEC_DATA);
587 PARSE_FLAG ("rom", SEC_ROM);
588 PARSE_FLAG ("share", SEC_COFF_SHARED);
589 PARSE_FLAG ("contents", SEC_HAS_CONTENTS);
590 #undef PARSE_FLAG
591 else
593 char *copy;
595 copy = xmalloc (len + 1);
596 strncpy (copy, s, len);
597 copy[len] = '\0';
598 non_fatal (_("unrecognized section flag `%s'"), copy);
599 fatal (_("supported flags: %s"),
600 "alloc, load, noload, readonly, debug, code, data, rom, share, contents");
603 s = snext;
605 while (s != NULL);
607 return ret;
610 /* Find and optionally add an entry in the change_sections list. */
612 static struct section_list *
613 find_section_list (const char *name, bfd_boolean add)
615 struct section_list *p;
617 for (p = change_sections; p != NULL; p = p->next)
618 if (strcmp (p->name, name) == 0)
619 return p;
621 if (! add)
622 return NULL;
624 p = xmalloc (sizeof (struct section_list));
625 p->name = name;
626 p->used = FALSE;
627 p->remove = FALSE;
628 p->copy = FALSE;
629 p->change_vma = CHANGE_IGNORE;
630 p->change_lma = CHANGE_IGNORE;
631 p->vma_val = 0;
632 p->lma_val = 0;
633 p->set_flags = FALSE;
634 p->flags = 0;
636 p->next = change_sections;
637 change_sections = p;
639 return p;
642 /* Add a symbol to strip_specific_list. */
644 static void
645 add_specific_symbol (const char *name, struct symlist **list)
647 struct symlist *tmp_list;
649 tmp_list = xmalloc (sizeof (struct symlist));
650 tmp_list->name = name;
651 tmp_list->next = *list;
652 *list = tmp_list;
655 /* Add symbols listed in `filename' to strip_specific_list. */
657 #define IS_WHITESPACE(c) ((c) == ' ' || (c) == '\t')
658 #define IS_LINE_TERMINATOR(c) ((c) == '\n' || (c) == '\r' || (c) == '\0')
660 static void
661 add_specific_symbols (const char *filename, struct symlist **list)
663 off_t size;
664 FILE * f;
665 char * line;
666 char * buffer;
667 unsigned int line_count;
669 size = get_file_size (filename);
670 if (size == 0)
672 status = 1;
673 return;
676 buffer = xmalloc (size + 2);
677 f = fopen (filename, FOPEN_RT);
678 if (f == NULL)
679 fatal (_("cannot open '%s': %s"), filename, strerror (errno));
681 if (fread (buffer, 1, size, f) == 0 || ferror (f))
682 fatal (_("%s: fread failed"), filename);
684 fclose (f);
685 buffer [size] = '\n';
686 buffer [size + 1] = '\0';
688 line_count = 1;
690 for (line = buffer; * line != '\0'; line ++)
692 char * eol;
693 char * name;
694 char * name_end;
695 int finished = FALSE;
697 for (eol = line;; eol ++)
699 switch (* eol)
701 case '\n':
702 * eol = '\0';
703 /* Cope with \n\r. */
704 if (eol[1] == '\r')
705 ++ eol;
706 finished = TRUE;
707 break;
709 case '\r':
710 * eol = '\0';
711 /* Cope with \r\n. */
712 if (eol[1] == '\n')
713 ++ eol;
714 finished = TRUE;
715 break;
717 case 0:
718 finished = TRUE;
719 break;
721 case '#':
722 /* Line comment, Terminate the line here, in case a
723 name is present and then allow the rest of the
724 loop to find the real end of the line. */
725 * eol = '\0';
726 break;
728 default:
729 break;
732 if (finished)
733 break;
736 /* A name may now exist somewhere between 'line' and 'eol'.
737 Strip off leading whitespace and trailing whitespace,
738 then add it to the list. */
739 for (name = line; IS_WHITESPACE (* name); name ++)
741 for (name_end = name;
742 (! IS_WHITESPACE (* name_end))
743 && (! IS_LINE_TERMINATOR (* name_end));
744 name_end ++)
747 if (! IS_LINE_TERMINATOR (* name_end))
749 char * extra;
751 for (extra = name_end + 1; IS_WHITESPACE (* extra); extra ++)
754 if (! IS_LINE_TERMINATOR (* extra))
755 non_fatal (_("%s:%d: Ignoring rubbish found on this line"),
756 filename, line_count);
759 * name_end = '\0';
761 if (name_end > name)
762 add_specific_symbol (name, list);
764 /* Advance line pointer to end of line. The 'eol ++' in the for
765 loop above will then advance us to the start of the next line. */
766 line = eol;
767 line_count ++;
771 /* See whether a symbol should be stripped or kept based on
772 strip_specific_list and keep_symbols. */
774 static bfd_boolean
775 is_specified_symbol (const char *name, struct symlist *list)
777 struct symlist *tmp_list;
779 if (wildcard)
781 for (tmp_list = list; tmp_list; tmp_list = tmp_list->next)
782 if (*(tmp_list->name) != '!')
784 if (!fnmatch (tmp_list->name, name, 0))
785 return TRUE;
787 else
789 if (fnmatch (tmp_list->name + 1, name, 0))
790 return TRUE;
793 else
795 for (tmp_list = list; tmp_list; tmp_list = tmp_list->next)
796 if (strcmp (name, tmp_list->name) == 0)
797 return TRUE;
800 return FALSE;
803 /* Return a pointer to the symbol used as a signature for GROUP. */
805 static asymbol *
806 group_signature (asection *group)
808 bfd *abfd = group->owner;
809 Elf_Internal_Shdr *ghdr;
811 if (bfd_get_flavour (abfd) != bfd_target_elf_flavour)
812 return NULL;
814 ghdr = &elf_section_data (group)->this_hdr;
815 if (ghdr->sh_link < elf_numsections (abfd))
817 const struct elf_backend_data *bed = get_elf_backend_data (abfd);
818 Elf_Internal_Shdr *symhdr = elf_elfsections (abfd) [ghdr->sh_link];
820 if (symhdr->sh_type == SHT_SYMTAB
821 && ghdr->sh_info < symhdr->sh_size / bed->s->sizeof_sym)
822 return isympp[ghdr->sh_info - 1];
824 return NULL;
827 /* See if a section is being removed. */
829 static bfd_boolean
830 is_strip_section (bfd *abfd ATTRIBUTE_UNUSED, asection *sec)
832 if (sections_removed || sections_copied)
834 struct section_list *p;
836 p = find_section_list (bfd_get_section_name (abfd, sec), FALSE);
838 if (sections_removed && p != NULL && p->remove)
839 return TRUE;
840 if (sections_copied && (p == NULL || ! p->copy))
841 return TRUE;
844 if ((bfd_get_section_flags (abfd, sec) & SEC_DEBUGGING) != 0)
846 if (strip_symbols == STRIP_DEBUG
847 || strip_symbols == STRIP_UNNEEDED
848 || strip_symbols == STRIP_ALL
849 || discard_locals == LOCALS_ALL
850 || convert_debugging)
851 return TRUE;
853 if (strip_symbols == STRIP_NONDEBUG)
854 return FALSE;
857 if ((bfd_get_section_flags (abfd, sec) & SEC_GROUP) != 0)
859 asymbol *gsym;
860 const char *gname;
862 /* PR binutils/3166
863 Group sections look like debugging sections but they are not.
864 (They have a non-zero size but they are not ALLOCated). */
865 if (strip_symbols == STRIP_NONDEBUG)
866 return TRUE;
868 /* PR binutils/3181
869 If we are going to strip the group signature symbol, then
870 strip the group section too. */
871 gsym = group_signature (sec);
872 if (gsym != NULL)
873 gname = gsym->name;
874 else
875 gname = sec->name;
876 if ((strip_symbols == STRIP_ALL
877 && !is_specified_symbol (gname, keep_specific_list))
878 || is_specified_symbol (gname, strip_specific_list))
879 return TRUE;
882 return FALSE;
885 /* Return true if SYM is a hidden symbol. */
887 static bfd_boolean
888 is_hidden_symbol (asymbol *sym)
890 elf_symbol_type *elf_sym;
892 elf_sym = elf_symbol_from (sym->the_bfd, sym);
893 if (elf_sym != NULL)
894 switch (ELF_ST_VISIBILITY (elf_sym->internal_elf_sym.st_other))
896 case STV_HIDDEN:
897 case STV_INTERNAL:
898 return TRUE;
900 return FALSE;
903 /* Choose which symbol entries to copy; put the result in OSYMS.
904 We don't copy in place, because that confuses the relocs.
905 Return the number of symbols to print. */
907 static unsigned int
908 filter_symbols (bfd *abfd, bfd *obfd, asymbol **osyms,
909 asymbol **isyms, long symcount)
911 asymbol **from = isyms, **to = osyms;
912 long src_count = 0, dst_count = 0;
913 int relocatable = (abfd->flags & (EXEC_P | DYNAMIC)) == 0;
915 for (; src_count < symcount; src_count++)
917 asymbol *sym = from[src_count];
918 flagword flags = sym->flags;
919 char *name = (char *) bfd_asymbol_name (sym);
920 bfd_boolean keep;
921 bfd_boolean used_in_reloc = FALSE;
922 bfd_boolean undefined;
923 bfd_boolean rem_leading_char;
924 bfd_boolean add_leading_char;
926 undefined = bfd_is_und_section (bfd_get_section (sym));
928 if (redefine_sym_list)
930 char *old_name, *new_name;
932 old_name = (char *) bfd_asymbol_name (sym);
933 new_name = (char *) lookup_sym_redefinition (old_name);
934 bfd_asymbol_name (sym) = new_name;
935 name = new_name;
938 /* Check if we will remove the current leading character. */
939 rem_leading_char =
940 (name[0] == bfd_get_symbol_leading_char (abfd))
941 && (change_leading_char
942 || (remove_leading_char
943 && ((flags & (BSF_GLOBAL | BSF_WEAK)) != 0
944 || undefined
945 || bfd_is_com_section (bfd_get_section (sym)))));
947 /* Check if we will add a new leading character. */
948 add_leading_char =
949 change_leading_char
950 && (bfd_get_symbol_leading_char (obfd) != '\0')
951 && (bfd_get_symbol_leading_char (abfd) == '\0'
952 || (name[0] == bfd_get_symbol_leading_char (abfd)));
954 /* Short circuit for change_leading_char if we can do it in-place. */
955 if (rem_leading_char && add_leading_char && !prefix_symbols_string)
957 name[0] = bfd_get_symbol_leading_char (obfd);
958 bfd_asymbol_name (sym) = name;
959 rem_leading_char = FALSE;
960 add_leading_char = FALSE;
963 /* Remove leading char. */
964 if (rem_leading_char)
965 bfd_asymbol_name (sym) = ++name;
967 /* Add new leading char and/or prefix. */
968 if (add_leading_char || prefix_symbols_string)
970 char *n, *ptr;
972 ptr = n = xmalloc (1 + strlen (prefix_symbols_string)
973 + strlen (name) + 1);
974 if (add_leading_char)
975 *ptr++ = bfd_get_symbol_leading_char (obfd);
977 if (prefix_symbols_string)
979 strcpy (ptr, prefix_symbols_string);
980 ptr += strlen (prefix_symbols_string);
983 strcpy (ptr, name);
984 bfd_asymbol_name (sym) = n;
985 name = n;
988 if (strip_symbols == STRIP_ALL)
989 keep = FALSE;
990 else if ((flags & BSF_KEEP) != 0 /* Used in relocation. */
991 || ((flags & BSF_SECTION_SYM) != 0
992 && ((*bfd_get_section (sym)->symbol_ptr_ptr)->flags
993 & BSF_KEEP) != 0))
995 keep = TRUE;
996 used_in_reloc = TRUE;
998 else if (relocatable /* Relocatable file. */
999 && (flags & (BSF_GLOBAL | BSF_WEAK)) != 0)
1000 keep = TRUE;
1001 else if (bfd_decode_symclass (sym) == 'I')
1002 /* Global symbols in $idata sections need to be retained
1003 even if relocatable is FALSE. External users of the
1004 library containing the $idata section may reference these
1005 symbols. */
1006 keep = TRUE;
1007 else if ((flags & BSF_GLOBAL) != 0 /* Global symbol. */
1008 || (flags & BSF_WEAK) != 0
1009 || undefined
1010 || bfd_is_com_section (bfd_get_section (sym)))
1011 keep = strip_symbols != STRIP_UNNEEDED;
1012 else if ((flags & BSF_DEBUGGING) != 0) /* Debugging symbol. */
1013 keep = (strip_symbols != STRIP_DEBUG
1014 && strip_symbols != STRIP_UNNEEDED
1015 && ! convert_debugging);
1016 else if (bfd_coff_get_comdat_section (abfd, bfd_get_section (sym)))
1017 /* COMDAT sections store special information in local
1018 symbols, so we cannot risk stripping any of them. */
1019 keep = TRUE;
1020 else /* Local symbol. */
1021 keep = (strip_symbols != STRIP_UNNEEDED
1022 && (discard_locals != LOCALS_ALL
1023 && (discard_locals != LOCALS_START_L
1024 || ! bfd_is_local_label (abfd, sym))));
1026 if (keep && is_specified_symbol (name, strip_specific_list))
1028 /* There are multiple ways to set 'keep' above, but if it
1029 was the relocatable symbol case, then that's an error. */
1030 if (used_in_reloc)
1032 non_fatal (_("not stripping symbol `%s' because it is named in a relocation"), name);
1033 status = 1;
1035 else
1036 keep = FALSE;
1039 if (keep
1040 && !(flags & BSF_KEEP)
1041 && is_specified_symbol (name, strip_unneeded_list))
1042 keep = FALSE;
1044 if (!keep
1045 && ((keep_file_symbols && (flags & BSF_FILE))
1046 || is_specified_symbol (name, keep_specific_list)))
1047 keep = TRUE;
1049 if (keep && is_strip_section (abfd, bfd_get_section (sym)))
1050 keep = FALSE;
1052 if (keep)
1054 if ((flags & BSF_GLOBAL) != 0
1055 && (weaken || is_specified_symbol (name, weaken_specific_list)))
1057 sym->flags &= ~ BSF_GLOBAL;
1058 sym->flags |= BSF_WEAK;
1061 if (!undefined
1062 && (flags & (BSF_GLOBAL | BSF_WEAK))
1063 && (is_specified_symbol (name, localize_specific_list)
1064 || (keepglobal_specific_list != NULL
1065 && ! is_specified_symbol (name, keepglobal_specific_list))
1066 || (localize_hidden && is_hidden_symbol (sym))))
1068 sym->flags &= ~ (BSF_GLOBAL | BSF_WEAK);
1069 sym->flags |= BSF_LOCAL;
1072 if (!undefined
1073 && (flags & BSF_LOCAL)
1074 && is_specified_symbol (name, globalize_specific_list))
1076 sym->flags &= ~ BSF_LOCAL;
1077 sym->flags |= BSF_GLOBAL;
1080 to[dst_count++] = sym;
1084 to[dst_count] = NULL;
1086 return dst_count;
1089 /* Find the redefined name of symbol SOURCE. */
1091 static const char *
1092 lookup_sym_redefinition (const char *source)
1094 struct redefine_node *list;
1096 for (list = redefine_sym_list; list != NULL; list = list->next)
1097 if (strcmp (source, list->source) == 0)
1098 return list->target;
1100 return source;
1103 /* Add a node to a symbol redefine list. */
1105 static void
1106 redefine_list_append (const char *cause, const char *source, const char *target)
1108 struct redefine_node **p;
1109 struct redefine_node *list;
1110 struct redefine_node *new_node;
1112 for (p = &redefine_sym_list; (list = *p) != NULL; p = &list->next)
1114 if (strcmp (source, list->source) == 0)
1115 fatal (_("%s: Multiple redefinition of symbol \"%s\""),
1116 cause, source);
1118 if (strcmp (target, list->target) == 0)
1119 fatal (_("%s: Symbol \"%s\" is target of more than one redefinition"),
1120 cause, target);
1123 new_node = xmalloc (sizeof (struct redefine_node));
1125 new_node->source = strdup (source);
1126 new_node->target = strdup (target);
1127 new_node->next = NULL;
1129 *p = new_node;
1132 /* Handle the --redefine-syms option. Read lines containing "old new"
1133 from the file, and add them to the symbol redefine list. */
1135 static void
1136 add_redefine_syms_file (const char *filename)
1138 FILE *file;
1139 char *buf;
1140 size_t bufsize;
1141 size_t len;
1142 size_t outsym_off;
1143 int c, lineno;
1145 file = fopen (filename, "r");
1146 if (file == NULL)
1147 fatal (_("couldn't open symbol redefinition file %s (error: %s)"),
1148 filename, strerror (errno));
1150 bufsize = 100;
1151 buf = xmalloc (bufsize);
1153 lineno = 1;
1154 c = getc (file);
1155 len = 0;
1156 outsym_off = 0;
1157 while (c != EOF)
1159 /* Collect the input symbol name. */
1160 while (! IS_WHITESPACE (c) && ! IS_LINE_TERMINATOR (c) && c != EOF)
1162 if (c == '#')
1163 goto comment;
1164 buf[len++] = c;
1165 if (len >= bufsize)
1167 bufsize *= 2;
1168 buf = xrealloc (buf, bufsize);
1170 c = getc (file);
1172 buf[len++] = '\0';
1173 if (c == EOF)
1174 break;
1176 /* Eat white space between the symbol names. */
1177 while (IS_WHITESPACE (c))
1178 c = getc (file);
1179 if (c == '#' || IS_LINE_TERMINATOR (c))
1180 goto comment;
1181 if (c == EOF)
1182 break;
1184 /* Collect the output symbol name. */
1185 outsym_off = len;
1186 while (! IS_WHITESPACE (c) && ! IS_LINE_TERMINATOR (c) && c != EOF)
1188 if (c == '#')
1189 goto comment;
1190 buf[len++] = c;
1191 if (len >= bufsize)
1193 bufsize *= 2;
1194 buf = xrealloc (buf, bufsize);
1196 c = getc (file);
1198 buf[len++] = '\0';
1199 if (c == EOF)
1200 break;
1202 /* Eat white space at end of line. */
1203 while (! IS_LINE_TERMINATOR(c) && c != EOF && IS_WHITESPACE (c))
1204 c = getc (file);
1205 if (c == '#')
1206 goto comment;
1207 /* Handle \r\n. */
1208 if ((c == '\r' && (c = getc (file)) == '\n')
1209 || c == '\n' || c == EOF)
1211 end_of_line:
1212 /* Append the redefinition to the list. */
1213 if (buf[0] != '\0')
1214 redefine_list_append (filename, &buf[0], &buf[outsym_off]);
1216 lineno++;
1217 len = 0;
1218 outsym_off = 0;
1219 if (c == EOF)
1220 break;
1221 c = getc (file);
1222 continue;
1224 else
1225 fatal (_("%s:%d: garbage found at end of line"), filename, lineno);
1226 comment:
1227 if (len != 0 && (outsym_off == 0 || outsym_off == len))
1228 fatal (_("%s:%d: missing new symbol name"), filename, lineno);
1229 buf[len++] = '\0';
1231 /* Eat the rest of the line and finish it. */
1232 while (c != '\n' && c != EOF)
1233 c = getc (file);
1234 goto end_of_line;
1237 if (len != 0)
1238 fatal (_("%s:%d: premature end of file"), filename, lineno);
1240 free (buf);
1243 /* Copy unkown object file IBFD onto OBFD.
1244 Returns TRUE upon success, FALSE otherwise. */
1246 static bfd_boolean
1247 copy_unknown_object (bfd *ibfd, bfd *obfd)
1249 char *cbuf;
1250 int tocopy;
1251 long ncopied;
1252 long size;
1253 struct stat buf;
1255 if (bfd_stat_arch_elt (ibfd, &buf) != 0)
1257 bfd_nonfatal (bfd_get_archive_filename (ibfd));
1258 return FALSE;
1261 size = buf.st_size;
1262 if (size < 0)
1264 non_fatal (_("stat returns negative size for `%s'"),
1265 bfd_get_archive_filename (ibfd));
1266 return FALSE;
1269 if (bfd_seek (ibfd, (file_ptr) 0, SEEK_SET) != 0)
1271 bfd_nonfatal (bfd_get_archive_filename (ibfd));
1272 return FALSE;
1275 if (verbose)
1276 printf (_("copy from `%s' [unknown] to `%s' [unknown]\n"),
1277 bfd_get_archive_filename (ibfd), bfd_get_filename (obfd));
1279 cbuf = xmalloc (BUFSIZE);
1280 ncopied = 0;
1281 while (ncopied < size)
1283 tocopy = size - ncopied;
1284 if (tocopy > BUFSIZE)
1285 tocopy = BUFSIZE;
1287 if (bfd_bread (cbuf, (bfd_size_type) tocopy, ibfd)
1288 != (bfd_size_type) tocopy)
1290 bfd_nonfatal (bfd_get_archive_filename (ibfd));
1291 free (cbuf);
1292 return FALSE;
1295 if (bfd_bwrite (cbuf, (bfd_size_type) tocopy, obfd)
1296 != (bfd_size_type) tocopy)
1298 bfd_nonfatal (bfd_get_filename (obfd));
1299 free (cbuf);
1300 return FALSE;
1303 ncopied += tocopy;
1306 chmod (bfd_get_filename (obfd), buf.st_mode);
1307 free (cbuf);
1308 return TRUE;
1311 /* Copy object file IBFD onto OBFD.
1312 Returns TRUE upon success, FALSE otherwise. */
1314 static bfd_boolean
1315 copy_object (bfd *ibfd, bfd *obfd)
1317 bfd_vma start;
1318 long symcount;
1319 asection **osections = NULL;
1320 asection *gnu_debuglink_section = NULL;
1321 bfd_size_type *gaps = NULL;
1322 bfd_size_type max_gap = 0;
1323 long symsize;
1324 void *dhandle;
1325 enum bfd_architecture iarch;
1326 unsigned int imach;
1328 if (ibfd->xvec->byteorder != obfd->xvec->byteorder
1329 && ibfd->xvec->byteorder != BFD_ENDIAN_UNKNOWN
1330 && obfd->xvec->byteorder != BFD_ENDIAN_UNKNOWN)
1331 fatal (_("Unable to change endianness of input file(s)"));
1333 if (!bfd_set_format (obfd, bfd_get_format (ibfd)))
1335 bfd_nonfatal (bfd_get_filename (obfd));
1336 return FALSE;
1339 if (verbose)
1340 printf (_("copy from `%s' [%s] to `%s' [%s]\n"),
1341 bfd_get_archive_filename (ibfd), bfd_get_target (ibfd),
1342 bfd_get_filename (obfd), bfd_get_target (obfd));
1344 if (extract_symbol)
1345 start = 0;
1346 else
1348 if (set_start_set)
1349 start = set_start;
1350 else
1351 start = bfd_get_start_address (ibfd);
1352 start += change_start;
1355 /* Neither the start address nor the flags
1356 need to be set for a core file. */
1357 if (bfd_get_format (obfd) != bfd_core)
1359 flagword flags;
1361 flags = bfd_get_file_flags (ibfd);
1362 flags |= bfd_flags_to_set;
1363 flags &= ~bfd_flags_to_clear;
1364 flags &= bfd_applicable_file_flags (obfd);
1366 if (!bfd_set_start_address (obfd, start)
1367 || !bfd_set_file_flags (obfd, flags))
1369 bfd_nonfatal (bfd_get_archive_filename (ibfd));
1370 return FALSE;
1374 /* Copy architecture of input file to output file. */
1375 iarch = bfd_get_arch (ibfd);
1376 imach = bfd_get_mach (ibfd);
1377 if (!bfd_set_arch_mach (obfd, iarch, imach)
1378 && (ibfd->target_defaulted
1379 || bfd_get_arch (ibfd) != bfd_get_arch (obfd)))
1381 if (bfd_get_arch (ibfd) == bfd_arch_unknown)
1382 non_fatal (_("Unable to recognise the format of the input file `%s'"),
1383 bfd_get_archive_filename (ibfd));
1384 else
1385 non_fatal (_("Warning: Output file cannot represent architecture `%s'"),
1386 bfd_printable_arch_mach (bfd_get_arch (ibfd),
1387 bfd_get_mach (ibfd)));
1388 return FALSE;
1391 if (!bfd_set_format (obfd, bfd_get_format (ibfd)))
1393 bfd_nonfatal (bfd_get_archive_filename (ibfd));
1394 return FALSE;
1397 if (isympp)
1398 free (isympp);
1400 if (osympp != isympp)
1401 free (osympp);
1403 isympp = NULL;
1404 osympp = NULL;
1406 symsize = bfd_get_symtab_upper_bound (ibfd);
1407 if (symsize < 0)
1409 bfd_nonfatal (bfd_get_archive_filename (ibfd));
1410 return FALSE;
1413 osympp = isympp = xmalloc (symsize);
1414 symcount = bfd_canonicalize_symtab (ibfd, isympp);
1415 if (symcount < 0)
1417 bfd_nonfatal (bfd_get_filename (ibfd));
1418 return FALSE;
1421 /* BFD mandates that all output sections be created and sizes set before
1422 any output is done. Thus, we traverse all sections multiple times. */
1423 bfd_map_over_sections (ibfd, setup_section, obfd);
1425 setup_bfd_headers (ibfd, obfd);
1427 if (add_sections != NULL)
1429 struct section_add *padd;
1430 struct section_list *pset;
1432 for (padd = add_sections; padd != NULL; padd = padd->next)
1434 flagword flags;
1436 pset = find_section_list (padd->name, FALSE);
1437 if (pset != NULL)
1438 pset->used = TRUE;
1440 flags = SEC_HAS_CONTENTS | SEC_READONLY | SEC_DATA;
1441 if (pset != NULL && pset->set_flags)
1442 flags = pset->flags | SEC_HAS_CONTENTS;
1444 /* bfd_make_section_with_flags() does not return very helpful
1445 error codes, so check for the most likely user error first. */
1446 if (bfd_get_section_by_name (obfd, padd->name))
1448 non_fatal (_("can't add section '%s' - it already exists!"), padd->name);
1449 return FALSE;
1451 else
1453 padd->section = bfd_make_section_with_flags (obfd, padd->name, flags);
1454 if (padd->section == NULL)
1456 non_fatal (_("can't create section `%s': %s"),
1457 padd->name, bfd_errmsg (bfd_get_error ()));
1458 return FALSE;
1462 if (! bfd_set_section_size (obfd, padd->section, padd->size))
1464 bfd_nonfatal (bfd_get_filename (obfd));
1465 return FALSE;
1468 if (pset != NULL)
1470 if (pset->change_vma != CHANGE_IGNORE)
1471 if (! bfd_set_section_vma (obfd, padd->section,
1472 pset->vma_val))
1474 bfd_nonfatal (bfd_get_filename (obfd));
1475 return FALSE;
1478 if (pset->change_lma != CHANGE_IGNORE)
1480 padd->section->lma = pset->lma_val;
1482 if (! bfd_set_section_alignment
1483 (obfd, padd->section,
1484 bfd_section_alignment (obfd, padd->section)))
1486 bfd_nonfatal (bfd_get_filename (obfd));
1487 return FALSE;
1494 if (gnu_debuglink_filename != NULL)
1496 gnu_debuglink_section = bfd_create_gnu_debuglink_section
1497 (obfd, gnu_debuglink_filename);
1499 if (gnu_debuglink_section == NULL)
1501 bfd_nonfatal (gnu_debuglink_filename);
1502 return FALSE;
1505 /* Special processing for PE format files. We
1506 have no way to distinguish PE from COFF here. */
1507 if (bfd_get_flavour (obfd) == bfd_target_coff_flavour)
1509 bfd_vma debuglink_vma;
1510 asection * highest_section;
1511 asection * sec;
1513 /* The PE spec requires that all sections be adjacent and sorted
1514 in ascending order of VMA. It also specifies that debug
1515 sections should be last. This is despite the fact that debug
1516 sections are not loaded into memory and so in theory have no
1517 use for a VMA.
1519 This means that the debuglink section must be given a non-zero
1520 VMA which makes it contiguous with other debug sections. So
1521 walk the current section list, find the section with the
1522 highest VMA and start the debuglink section after that one. */
1523 for (sec = obfd->sections, highest_section = NULL;
1524 sec != NULL;
1525 sec = sec->next)
1526 if (sec->vma > 0
1527 && (highest_section == NULL
1528 || sec->vma > highest_section->vma))
1529 highest_section = sec;
1531 if (highest_section)
1532 debuglink_vma = BFD_ALIGN (highest_section->vma
1533 + highest_section->size,
1534 /* FIXME: We ought to be using
1535 COFF_PAGE_SIZE here or maybe
1536 bfd_get_section_alignment() (if it
1537 was set) but since this is for PE
1538 and we know the required alignment
1539 it is easier just to hard code it. */
1540 0x1000);
1541 else
1542 /* Umm, not sure what to do in this case. */
1543 debuglink_vma = 0x1000;
1545 bfd_set_section_vma (obfd, gnu_debuglink_section, debuglink_vma);
1549 if (bfd_count_sections (obfd) != 0
1550 && (gap_fill_set || pad_to_set))
1552 asection **set;
1553 unsigned int c, i;
1555 /* We must fill in gaps between the sections and/or we must pad
1556 the last section to a specified address. We do this by
1557 grabbing a list of the sections, sorting them by VMA, and
1558 increasing the section sizes as required to fill the gaps.
1559 We write out the gap contents below. */
1561 c = bfd_count_sections (obfd);
1562 osections = xmalloc (c * sizeof (asection *));
1563 set = osections;
1564 bfd_map_over_sections (obfd, get_sections, &set);
1566 qsort (osections, c, sizeof (asection *), compare_section_lma);
1568 gaps = xmalloc (c * sizeof (bfd_size_type));
1569 memset (gaps, 0, c * sizeof (bfd_size_type));
1571 if (gap_fill_set)
1573 for (i = 0; i < c - 1; i++)
1575 flagword flags;
1576 bfd_size_type size;
1577 bfd_vma gap_start, gap_stop;
1579 flags = bfd_get_section_flags (obfd, osections[i]);
1580 if ((flags & SEC_HAS_CONTENTS) == 0
1581 || (flags & SEC_LOAD) == 0)
1582 continue;
1584 size = bfd_section_size (obfd, osections[i]);
1585 gap_start = bfd_section_lma (obfd, osections[i]) + size;
1586 gap_stop = bfd_section_lma (obfd, osections[i + 1]);
1587 if (gap_start < gap_stop)
1589 if (! bfd_set_section_size (obfd, osections[i],
1590 size + (gap_stop - gap_start)))
1592 non_fatal (_("Can't fill gap after %s: %s"),
1593 bfd_get_section_name (obfd, osections[i]),
1594 bfd_errmsg (bfd_get_error ()));
1595 status = 1;
1596 break;
1598 gaps[i] = gap_stop - gap_start;
1599 if (max_gap < gap_stop - gap_start)
1600 max_gap = gap_stop - gap_start;
1605 if (pad_to_set)
1607 bfd_vma lma;
1608 bfd_size_type size;
1610 lma = bfd_section_lma (obfd, osections[c - 1]);
1611 size = bfd_section_size (obfd, osections[c - 1]);
1612 if (lma + size < pad_to)
1614 if (! bfd_set_section_size (obfd, osections[c - 1],
1615 pad_to - lma))
1617 non_fatal (_("Can't add padding to %s: %s"),
1618 bfd_get_section_name (obfd, osections[c - 1]),
1619 bfd_errmsg (bfd_get_error ()));
1620 status = 1;
1622 else
1624 gaps[c - 1] = pad_to - (lma + size);
1625 if (max_gap < pad_to - (lma + size))
1626 max_gap = pad_to - (lma + size);
1632 /* Symbol filtering must happen after the output sections
1633 have been created, but before their contents are set. */
1634 dhandle = NULL;
1635 if (convert_debugging)
1636 dhandle = read_debugging_info (ibfd, isympp, symcount);
1638 if (strip_symbols == STRIP_DEBUG
1639 || strip_symbols == STRIP_ALL
1640 || strip_symbols == STRIP_UNNEEDED
1641 || strip_symbols == STRIP_NONDEBUG
1642 || discard_locals != LOCALS_UNDEF
1643 || localize_hidden
1644 || strip_specific_list != NULL
1645 || keep_specific_list != NULL
1646 || localize_specific_list != NULL
1647 || globalize_specific_list != NULL
1648 || keepglobal_specific_list != NULL
1649 || weaken_specific_list != NULL
1650 || prefix_symbols_string
1651 || sections_removed
1652 || sections_copied
1653 || convert_debugging
1654 || change_leading_char
1655 || remove_leading_char
1656 || redefine_sym_list
1657 || weaken)
1659 /* Mark symbols used in output relocations so that they
1660 are kept, even if they are local labels or static symbols.
1662 Note we iterate over the input sections examining their
1663 relocations since the relocations for the output sections
1664 haven't been set yet. mark_symbols_used_in_relocations will
1665 ignore input sections which have no corresponding output
1666 section. */
1667 if (strip_symbols != STRIP_ALL)
1668 bfd_map_over_sections (ibfd,
1669 mark_symbols_used_in_relocations,
1670 isympp);
1671 osympp = xmalloc ((symcount + 1) * sizeof (asymbol *));
1672 symcount = filter_symbols (ibfd, obfd, osympp, isympp, symcount);
1675 if (convert_debugging && dhandle != NULL)
1677 if (! write_debugging_info (obfd, dhandle, &symcount, &osympp))
1679 status = 1;
1680 return FALSE;
1684 bfd_set_symtab (obfd, osympp, symcount);
1686 /* This has to happen after the symbol table has been set. */
1687 bfd_map_over_sections (ibfd, copy_section, obfd);
1689 if (add_sections != NULL)
1691 struct section_add *padd;
1693 for (padd = add_sections; padd != NULL; padd = padd->next)
1695 if (! bfd_set_section_contents (obfd, padd->section, padd->contents,
1696 0, padd->size))
1698 bfd_nonfatal (bfd_get_filename (obfd));
1699 return FALSE;
1704 if (gnu_debuglink_filename != NULL)
1706 if (! bfd_fill_in_gnu_debuglink_section
1707 (obfd, gnu_debuglink_section, gnu_debuglink_filename))
1709 bfd_nonfatal (gnu_debuglink_filename);
1710 return FALSE;
1714 if (gap_fill_set || pad_to_set)
1716 bfd_byte *buf;
1717 int c, i;
1719 /* Fill in the gaps. */
1720 if (max_gap > 8192)
1721 max_gap = 8192;
1722 buf = xmalloc (max_gap);
1723 memset (buf, gap_fill, max_gap);
1725 c = bfd_count_sections (obfd);
1726 for (i = 0; i < c; i++)
1728 if (gaps[i] != 0)
1730 bfd_size_type left;
1731 file_ptr off;
1733 left = gaps[i];
1734 off = bfd_section_size (obfd, osections[i]) - left;
1736 while (left > 0)
1738 bfd_size_type now;
1740 if (left > 8192)
1741 now = 8192;
1742 else
1743 now = left;
1745 if (! bfd_set_section_contents (obfd, osections[i], buf,
1746 off, now))
1748 bfd_nonfatal (bfd_get_filename (obfd));
1749 return FALSE;
1752 left -= now;
1753 off += now;
1759 /* Do not copy backend data if --extract-symbol is passed; anything
1760 that needs to look at the section contents will fail. */
1761 if (extract_symbol)
1762 return TRUE;
1764 /* Allow the BFD backend to copy any private data it understands
1765 from the input BFD to the output BFD. This is done last to
1766 permit the routine to look at the filtered symbol table, which is
1767 important for the ECOFF code at least. */
1768 if (! bfd_copy_private_bfd_data (ibfd, obfd))
1770 non_fatal (_("%s: error copying private BFD data: %s"),
1771 bfd_get_filename (obfd),
1772 bfd_errmsg (bfd_get_error ()));
1773 return FALSE;
1776 /* Switch to the alternate machine code. We have to do this at the
1777 very end, because we only initialize the header when we create
1778 the first section. */
1779 if (use_alt_mach_code != 0)
1781 if (! bfd_alt_mach_code (obfd, use_alt_mach_code))
1783 non_fatal (_("this target does not support %lu alternative machine codes"),
1784 use_alt_mach_code);
1785 if (bfd_get_flavour (obfd) == bfd_target_elf_flavour)
1787 non_fatal (_("treating that number as an absolute e_machine value instead"));
1788 elf_elfheader (obfd)->e_machine = use_alt_mach_code;
1790 else
1791 non_fatal (_("ignoring the alternative value"));
1795 return TRUE;
1798 /* Read each archive element in turn from IBFD, copy the
1799 contents to temp file, and keep the temp file handle.
1800 If 'force_output_target' is TRUE then make sure that
1801 all elements in the new archive are of the type
1802 'output_target'. */
1804 static void
1805 copy_archive (bfd *ibfd, bfd *obfd, const char *output_target,
1806 bfd_boolean force_output_target)
1808 struct name_list
1810 struct name_list *next;
1811 const char *name;
1812 bfd *obfd;
1813 } *list, *l;
1814 bfd **ptr = &obfd->archive_head;
1815 bfd *this_element;
1816 char * dir;
1818 /* Make a temp directory to hold the contents. */
1819 dir = make_tempdir (bfd_get_filename (obfd));
1820 if (dir == NULL)
1821 fatal (_("cannot create tempdir for archive copying (error: %s)"),
1822 strerror (errno));
1824 obfd->has_armap = ibfd->has_armap;
1826 list = NULL;
1828 this_element = bfd_openr_next_archived_file (ibfd, NULL);
1830 if (!bfd_set_format (obfd, bfd_get_format (ibfd)))
1831 RETURN_NONFATAL (bfd_get_filename (obfd));
1833 while (!status && this_element != NULL)
1835 char *output_name;
1836 bfd *output_bfd;
1837 bfd *last_element;
1838 struct stat buf;
1839 int stat_status = 0;
1840 bfd_boolean delete = TRUE;
1842 /* Create an output file for this member. */
1843 output_name = concat (dir, "/",
1844 bfd_get_filename (this_element), (char *) 0);
1846 /* If the file already exists, make another temp dir. */
1847 if (stat (output_name, &buf) >= 0)
1849 output_name = make_tempdir (output_name);
1850 if (output_name == NULL)
1851 fatal (_("cannot create tempdir for archive copying (error: %s)"),
1852 strerror (errno));
1854 l = xmalloc (sizeof (struct name_list));
1855 l->name = output_name;
1856 l->next = list;
1857 l->obfd = NULL;
1858 list = l;
1859 output_name = concat (output_name, "/",
1860 bfd_get_filename (this_element), (char *) 0);
1863 if (preserve_dates)
1865 stat_status = bfd_stat_arch_elt (this_element, &buf);
1867 if (stat_status != 0)
1868 non_fatal (_("internal stat error on %s"),
1869 bfd_get_filename (this_element));
1872 l = xmalloc (sizeof (struct name_list));
1873 l->name = output_name;
1874 l->next = list;
1875 l->obfd = NULL;
1876 list = l;
1878 if (bfd_check_format (this_element, bfd_object))
1880 /* PR binutils/3110: Cope with archives
1881 containing multiple target types. */
1882 if (force_output_target)
1883 output_bfd = bfd_openw (output_name, output_target);
1884 else
1885 output_bfd = bfd_openw (output_name, bfd_get_target (this_element));
1887 if (output_bfd == NULL)
1888 RETURN_NONFATAL (output_name);
1890 delete = ! copy_object (this_element, output_bfd);
1892 if (! delete
1893 || bfd_get_arch (this_element) != bfd_arch_unknown)
1895 if (!bfd_close (output_bfd))
1897 bfd_nonfatal (bfd_get_filename (output_bfd));
1898 /* Error in new object file. Don't change archive. */
1899 status = 1;
1902 else
1903 goto copy_unknown_element;
1905 else
1907 non_fatal (_("Unable to recognise the format of the input file `%s'"),
1908 bfd_get_archive_filename (this_element));
1910 output_bfd = bfd_openw (output_name, output_target);
1911 copy_unknown_element:
1912 delete = !copy_unknown_object (this_element, output_bfd);
1913 if (!bfd_close_all_done (output_bfd))
1915 bfd_nonfatal (bfd_get_filename (output_bfd));
1916 /* Error in new object file. Don't change archive. */
1917 status = 1;
1921 if (delete)
1923 unlink (output_name);
1924 status = 1;
1926 else
1928 if (preserve_dates && stat_status == 0)
1929 set_times (output_name, &buf);
1931 /* Open the newly output file and attach to our list. */
1932 output_bfd = bfd_openr (output_name, output_target);
1934 l->obfd = output_bfd;
1936 *ptr = output_bfd;
1937 ptr = &output_bfd->archive_next;
1939 last_element = this_element;
1941 this_element = bfd_openr_next_archived_file (ibfd, last_element);
1943 bfd_close (last_element);
1946 *ptr = NULL;
1948 if (!bfd_close (obfd))
1949 RETURN_NONFATAL (bfd_get_filename (obfd));
1951 if (!bfd_close (ibfd))
1952 RETURN_NONFATAL (bfd_get_filename (ibfd));
1954 /* Delete all the files that we opened. */
1955 for (l = list; l != NULL; l = l->next)
1957 if (l->obfd == NULL)
1958 rmdir (l->name);
1959 else
1961 bfd_close (l->obfd);
1962 unlink (l->name);
1965 rmdir (dir);
1968 /* The top-level control. */
1970 static void
1971 copy_file (const char *input_filename, const char *output_filename,
1972 const char *input_target, const char *output_target)
1974 bfd *ibfd;
1975 char **obj_matching;
1976 char **core_matching;
1978 if (get_file_size (input_filename) < 1)
1980 status = 1;
1981 return;
1984 /* To allow us to do "strip *" without dying on the first
1985 non-object file, failures are nonfatal. */
1986 ibfd = bfd_openr (input_filename, input_target);
1987 if (ibfd == NULL)
1988 RETURN_NONFATAL (input_filename);
1990 if (bfd_check_format (ibfd, bfd_archive))
1992 bfd_boolean force_output_target;
1993 bfd *obfd;
1995 /* bfd_get_target does not return the correct value until
1996 bfd_check_format succeeds. */
1997 if (output_target == NULL)
1999 output_target = bfd_get_target (ibfd);
2000 force_output_target = FALSE;
2002 else
2003 force_output_target = TRUE;
2005 obfd = bfd_openw (output_filename, output_target);
2006 if (obfd == NULL)
2007 RETURN_NONFATAL (output_filename);
2009 copy_archive (ibfd, obfd, output_target, force_output_target);
2011 else if (bfd_check_format_matches (ibfd, bfd_object, &obj_matching))
2013 bfd *obfd;
2014 do_copy:
2016 /* bfd_get_target does not return the correct value until
2017 bfd_check_format succeeds. */
2018 if (output_target == NULL)
2019 output_target = bfd_get_target (ibfd);
2021 obfd = bfd_openw (output_filename, output_target);
2022 if (obfd == NULL)
2023 RETURN_NONFATAL (output_filename);
2025 if (! copy_object (ibfd, obfd))
2026 status = 1;
2028 if (!bfd_close (obfd))
2029 RETURN_NONFATAL (output_filename);
2031 if (!bfd_close (ibfd))
2032 RETURN_NONFATAL (input_filename);
2035 else
2037 bfd_error_type obj_error = bfd_get_error ();
2038 bfd_error_type core_error;
2040 if (bfd_check_format_matches (ibfd, bfd_core, &core_matching))
2042 /* This probably can't happen.. */
2043 if (obj_error == bfd_error_file_ambiguously_recognized)
2044 free (obj_matching);
2045 goto do_copy;
2048 core_error = bfd_get_error ();
2049 /* Report the object error in preference to the core error. */
2050 if (obj_error != core_error)
2051 bfd_set_error (obj_error);
2053 bfd_nonfatal (input_filename);
2055 if (obj_error == bfd_error_file_ambiguously_recognized)
2057 list_matching_formats (obj_matching);
2058 free (obj_matching);
2060 if (core_error == bfd_error_file_ambiguously_recognized)
2062 list_matching_formats (core_matching);
2063 free (core_matching);
2066 status = 1;
2070 /* Add a name to the section renaming list. */
2072 static void
2073 add_section_rename (const char * old_name, const char * new_name,
2074 flagword flags)
2076 section_rename * rename;
2078 /* Check for conflicts first. */
2079 for (rename = section_rename_list; rename != NULL; rename = rename->next)
2080 if (strcmp (rename->old_name, old_name) == 0)
2082 /* Silently ignore duplicate definitions. */
2083 if (strcmp (rename->new_name, new_name) == 0
2084 && rename->flags == flags)
2085 return;
2087 fatal (_("Multiple renames of section %s"), old_name);
2090 rename = xmalloc (sizeof (* rename));
2092 rename->old_name = old_name;
2093 rename->new_name = new_name;
2094 rename->flags = flags;
2095 rename->next = section_rename_list;
2097 section_rename_list = rename;
2100 /* Check the section rename list for a new name of the input section
2101 ISECTION. Return the new name if one is found.
2102 Also set RETURNED_FLAGS to the flags to be used for this section. */
2104 static const char *
2105 find_section_rename (bfd * ibfd ATTRIBUTE_UNUSED, sec_ptr isection,
2106 flagword * returned_flags)
2108 const char * old_name = bfd_section_name (ibfd, isection);
2109 section_rename * rename;
2111 /* Default to using the flags of the input section. */
2112 * returned_flags = bfd_get_section_flags (ibfd, isection);
2114 for (rename = section_rename_list; rename != NULL; rename = rename->next)
2115 if (strcmp (rename->old_name, old_name) == 0)
2117 if (rename->flags != (flagword) -1)
2118 * returned_flags = rename->flags;
2120 return rename->new_name;
2123 return old_name;
2126 /* Once each of the sections is copied, we may still need to do some
2127 finalization work for private section headers. Do that here. */
2129 static void
2130 setup_bfd_headers (bfd *ibfd, bfd *obfd)
2132 const char *err;
2134 /* Allow the BFD backend to copy any private data it understands
2135 from the input section to the output section. */
2136 if (! bfd_copy_private_header_data (ibfd, obfd))
2138 err = _("private header data");
2139 goto loser;
2142 /* All went well. */
2143 return;
2145 loser:
2146 non_fatal (_("%s: error in %s: %s"),
2147 bfd_get_filename (ibfd),
2148 err, bfd_errmsg (bfd_get_error ()));
2149 status = 1;
2152 /* Create a section in OBFD with the same
2153 name and attributes as ISECTION in IBFD. */
2155 static void
2156 setup_section (bfd *ibfd, sec_ptr isection, void *obfdarg)
2158 bfd *obfd = obfdarg;
2159 struct section_list *p;
2160 sec_ptr osection;
2161 bfd_size_type size;
2162 bfd_vma vma;
2163 bfd_vma lma;
2164 flagword flags;
2165 const char *err;
2166 const char * name;
2167 char *prefix = NULL;
2168 bfd_boolean make_nobits;
2170 if (is_strip_section (ibfd, isection))
2171 return;
2173 p = find_section_list (bfd_section_name (ibfd, isection), FALSE);
2174 if (p != NULL)
2175 p->used = TRUE;
2177 /* Get the, possibly new, name of the output section. */
2178 name = find_section_rename (ibfd, isection, & flags);
2180 /* Prefix sections. */
2181 if ((prefix_alloc_sections_string)
2182 && (bfd_get_section_flags (ibfd, isection) & SEC_ALLOC))
2183 prefix = prefix_alloc_sections_string;
2184 else if (prefix_sections_string)
2185 prefix = prefix_sections_string;
2187 if (prefix)
2189 char *n;
2191 n = xmalloc (strlen (prefix) + strlen (name) + 1);
2192 strcpy (n, prefix);
2193 strcat (n, name);
2194 name = n;
2197 make_nobits = FALSE;
2198 if (p != NULL && p->set_flags)
2199 flags = p->flags | (flags & (SEC_HAS_CONTENTS | SEC_RELOC));
2200 else if (strip_symbols == STRIP_NONDEBUG
2201 && (flags & SEC_ALLOC) != 0
2202 && (ibfd->xvec->flavour != bfd_target_elf_flavour
2203 || elf_section_type (isection) != SHT_NOTE))
2205 flags &= ~(SEC_HAS_CONTENTS | SEC_LOAD);
2206 if (obfd->xvec->flavour == bfd_target_elf_flavour)
2208 make_nobits = TRUE;
2210 /* Twiddle the input section flags so that it seems to
2211 elf.c:copy_private_bfd_data that section flags have not
2212 changed between input and output sections. This hack
2213 prevents wholesale rewriting of the program headers. */
2214 isection->flags &= ~(SEC_HAS_CONTENTS | SEC_LOAD);
2218 osection = bfd_make_section_anyway_with_flags (obfd, name, flags);
2220 if (osection == NULL)
2222 err = _("making");
2223 goto loser;
2226 if (make_nobits)
2227 elf_section_type (osection) = SHT_NOBITS;
2229 size = bfd_section_size (ibfd, isection);
2230 if (copy_byte >= 0)
2231 size = (size + interleave - 1) / interleave;
2232 else if (extract_symbol)
2233 size = 0;
2234 if (! bfd_set_section_size (obfd, osection, size))
2236 err = _("size");
2237 goto loser;
2240 vma = bfd_section_vma (ibfd, isection);
2241 if (p != NULL && p->change_vma == CHANGE_MODIFY)
2242 vma += p->vma_val;
2243 else if (p != NULL && p->change_vma == CHANGE_SET)
2244 vma = p->vma_val;
2245 else
2246 vma += change_section_address;
2248 if (! bfd_set_section_vma (obfd, osection, extract_symbol ? 0 : vma))
2250 err = _("vma");
2251 goto loser;
2254 lma = isection->lma;
2255 if ((p != NULL) && p->change_lma != CHANGE_IGNORE)
2257 if (p->change_lma == CHANGE_MODIFY)
2258 lma += p->lma_val;
2259 else if (p->change_lma == CHANGE_SET)
2260 lma = p->lma_val;
2261 else
2262 abort ();
2264 else
2265 lma += change_section_address;
2267 osection->lma = extract_symbol ? 0 : lma;
2269 /* FIXME: This is probably not enough. If we change the LMA we
2270 may have to recompute the header for the file as well. */
2271 if (!bfd_set_section_alignment (obfd,
2272 osection,
2273 bfd_section_alignment (ibfd, isection)))
2275 err = _("alignment");
2276 goto loser;
2279 /* Copy merge entity size. */
2280 osection->entsize = isection->entsize;
2282 /* This used to be mangle_section; we do here to avoid using
2283 bfd_get_section_by_name since some formats allow multiple
2284 sections with the same name. */
2285 isection->output_section = osection;
2286 isection->output_offset = extract_symbol ? vma : 0;
2288 /* Do not copy backend data if --extract-symbol is passed; anything
2289 that needs to look at the section contents will fail. */
2290 if (extract_symbol)
2291 return;
2293 /* Allow the BFD backend to copy any private data it understands
2294 from the input section to the output section. */
2295 if (!bfd_copy_private_section_data (ibfd, isection, obfd, osection))
2297 err = _("private data");
2298 goto loser;
2300 else if ((isection->flags & SEC_GROUP) != 0)
2302 asymbol *gsym = group_signature (isection);
2304 if (gsym != NULL)
2305 gsym->flags |= BSF_KEEP;
2308 /* All went well. */
2309 return;
2311 loser:
2312 non_fatal (_("%s: section `%s': error in %s: %s"),
2313 bfd_get_filename (ibfd),
2314 bfd_section_name (ibfd, isection),
2315 err, bfd_errmsg (bfd_get_error ()));
2316 status = 1;
2319 /* Copy the data of input section ISECTION of IBFD
2320 to an output section with the same name in OBFD.
2321 If stripping then don't copy any relocation info. */
2323 static void
2324 copy_section (bfd *ibfd, sec_ptr isection, void *obfdarg)
2326 bfd *obfd = obfdarg;
2327 struct section_list *p;
2328 arelent **relpp;
2329 long relcount;
2330 sec_ptr osection;
2331 bfd_size_type size;
2332 long relsize;
2333 flagword flags;
2335 /* If we have already failed earlier on,
2336 do not keep on generating complaints now. */
2337 if (status != 0)
2338 return;
2340 if (is_strip_section (ibfd, isection))
2341 return;
2343 flags = bfd_get_section_flags (ibfd, isection);
2344 if ((flags & SEC_GROUP) != 0)
2345 return;
2347 osection = isection->output_section;
2348 size = bfd_get_section_size (isection);
2350 if (size == 0 || osection == 0)
2351 return;
2353 p = find_section_list (bfd_get_section_name (ibfd, isection), FALSE);
2355 /* Core files do not need to be relocated. */
2356 if (bfd_get_format (obfd) == bfd_core)
2357 relsize = 0;
2358 else
2360 relsize = bfd_get_reloc_upper_bound (ibfd, isection);
2362 if (relsize < 0)
2364 /* Do not complain if the target does not support relocations. */
2365 if (relsize == -1 && bfd_get_error () == bfd_error_invalid_operation)
2366 relsize = 0;
2367 else
2368 RETURN_NONFATAL (bfd_get_filename (ibfd));
2372 if (relsize == 0)
2373 bfd_set_reloc (obfd, osection, NULL, 0);
2374 else
2376 relpp = xmalloc (relsize);
2377 relcount = bfd_canonicalize_reloc (ibfd, isection, relpp, isympp);
2378 if (relcount < 0)
2379 RETURN_NONFATAL (bfd_get_filename (ibfd));
2381 if (strip_symbols == STRIP_ALL)
2383 /* Remove relocations which are not in
2384 keep_strip_specific_list. */
2385 arelent **temp_relpp;
2386 long temp_relcount = 0;
2387 long i;
2389 temp_relpp = xmalloc (relsize);
2390 for (i = 0; i < relcount; i++)
2391 if (is_specified_symbol (bfd_asymbol_name (*relpp[i]->sym_ptr_ptr),
2392 keep_specific_list))
2393 temp_relpp [temp_relcount++] = relpp [i];
2394 relcount = temp_relcount;
2395 free (relpp);
2396 relpp = temp_relpp;
2399 bfd_set_reloc (obfd, osection, relcount == 0 ? NULL : relpp, relcount);
2400 if (relcount == 0)
2401 free (relpp);
2404 if (extract_symbol)
2405 return;
2407 if (bfd_get_section_flags (ibfd, isection) & SEC_HAS_CONTENTS
2408 && bfd_get_section_flags (obfd, osection) & SEC_HAS_CONTENTS)
2410 void *memhunk = xmalloc (size);
2412 if (!bfd_get_section_contents (ibfd, isection, memhunk, 0, size))
2413 RETURN_NONFATAL (bfd_get_filename (ibfd));
2415 if (reverse_bytes)
2417 /* We don't handle leftover bytes (too many possible behaviors,
2418 and we don't know what the user wants). The section length
2419 must be a multiple of the number of bytes to swap. */
2420 if ((size % reverse_bytes) == 0)
2422 unsigned long i, j;
2423 bfd_byte b;
2425 for (i = 0; i < size; i += reverse_bytes)
2426 for (j = 0; j < (unsigned long)(reverse_bytes / 2); j++)
2428 bfd_byte *m = (bfd_byte *) memhunk;
2430 b = m[i + j];
2431 m[i + j] = m[(i + reverse_bytes) - (j + 1)];
2432 m[(i + reverse_bytes) - (j + 1)] = b;
2435 else
2436 /* User must pad the section up in order to do this. */
2437 fatal (_("cannot reverse bytes: length of section %s must be evenly divisible by %d"),
2438 bfd_section_name (ibfd, isection), reverse_bytes);
2441 if (copy_byte >= 0)
2443 /* Keep only every `copy_byte'th byte in MEMHUNK. */
2444 char *from = (char *) memhunk + copy_byte;
2445 char *to = memhunk;
2446 char *end = (char *) memhunk + size;
2448 for (; from < end; from += interleave)
2449 *to++ = *from;
2451 size = (size + interleave - 1 - copy_byte) / interleave;
2452 osection->lma /= interleave;
2455 if (!bfd_set_section_contents (obfd, osection, memhunk, 0, size))
2456 RETURN_NONFATAL (bfd_get_filename (obfd));
2458 free (memhunk);
2460 else if (p != NULL && p->set_flags && (p->flags & SEC_HAS_CONTENTS) != 0)
2462 void *memhunk = xmalloc (size);
2464 /* We don't permit the user to turn off the SEC_HAS_CONTENTS
2465 flag--they can just remove the section entirely and add it
2466 back again. However, we do permit them to turn on the
2467 SEC_HAS_CONTENTS flag, and take it to mean that the section
2468 contents should be zeroed out. */
2470 memset (memhunk, 0, size);
2471 if (! bfd_set_section_contents (obfd, osection, memhunk, 0, size))
2472 RETURN_NONFATAL (bfd_get_filename (obfd));
2473 free (memhunk);
2477 /* Get all the sections. This is used when --gap-fill or --pad-to is
2478 used. */
2480 static void
2481 get_sections (bfd *obfd ATTRIBUTE_UNUSED, asection *osection, void *secppparg)
2483 asection ***secppp = secppparg;
2485 **secppp = osection;
2486 ++(*secppp);
2489 /* Sort sections by VMA. This is called via qsort, and is used when
2490 --gap-fill or --pad-to is used. We force non loadable or empty
2491 sections to the front, where they are easier to ignore. */
2493 static int
2494 compare_section_lma (const void *arg1, const void *arg2)
2496 const asection *const *sec1 = arg1;
2497 const asection *const *sec2 = arg2;
2498 flagword flags1, flags2;
2500 /* Sort non loadable sections to the front. */
2501 flags1 = (*sec1)->flags;
2502 flags2 = (*sec2)->flags;
2503 if ((flags1 & SEC_HAS_CONTENTS) == 0
2504 || (flags1 & SEC_LOAD) == 0)
2506 if ((flags2 & SEC_HAS_CONTENTS) != 0
2507 && (flags2 & SEC_LOAD) != 0)
2508 return -1;
2510 else
2512 if ((flags2 & SEC_HAS_CONTENTS) == 0
2513 || (flags2 & SEC_LOAD) == 0)
2514 return 1;
2517 /* Sort sections by LMA. */
2518 if ((*sec1)->lma > (*sec2)->lma)
2519 return 1;
2520 else if ((*sec1)->lma < (*sec2)->lma)
2521 return -1;
2523 /* Sort sections with the same LMA by size. */
2524 if (bfd_get_section_size (*sec1) > bfd_get_section_size (*sec2))
2525 return 1;
2526 else if (bfd_get_section_size (*sec1) < bfd_get_section_size (*sec2))
2527 return -1;
2529 return 0;
2532 /* Mark all the symbols which will be used in output relocations with
2533 the BSF_KEEP flag so that those symbols will not be stripped.
2535 Ignore relocations which will not appear in the output file. */
2537 static void
2538 mark_symbols_used_in_relocations (bfd *ibfd, sec_ptr isection, void *symbolsarg)
2540 asymbol **symbols = symbolsarg;
2541 long relsize;
2542 arelent **relpp;
2543 long relcount, i;
2545 /* Ignore an input section with no corresponding output section. */
2546 if (isection->output_section == NULL)
2547 return;
2549 relsize = bfd_get_reloc_upper_bound (ibfd, isection);
2550 if (relsize < 0)
2552 /* Do not complain if the target does not support relocations. */
2553 if (relsize == -1 && bfd_get_error () == bfd_error_invalid_operation)
2554 return;
2555 bfd_fatal (bfd_get_filename (ibfd));
2558 if (relsize == 0)
2559 return;
2561 relpp = xmalloc (relsize);
2562 relcount = bfd_canonicalize_reloc (ibfd, isection, relpp, symbols);
2563 if (relcount < 0)
2564 bfd_fatal (bfd_get_filename (ibfd));
2566 /* Examine each symbol used in a relocation. If it's not one of the
2567 special bfd section symbols, then mark it with BSF_KEEP. */
2568 for (i = 0; i < relcount; i++)
2570 if (*relpp[i]->sym_ptr_ptr != bfd_com_section_ptr->symbol
2571 && *relpp[i]->sym_ptr_ptr != bfd_abs_section_ptr->symbol
2572 && *relpp[i]->sym_ptr_ptr != bfd_und_section_ptr->symbol)
2573 (*relpp[i]->sym_ptr_ptr)->flags |= BSF_KEEP;
2576 if (relpp != NULL)
2577 free (relpp);
2580 /* Write out debugging information. */
2582 static bfd_boolean
2583 write_debugging_info (bfd *obfd, void *dhandle,
2584 long *symcountp ATTRIBUTE_UNUSED,
2585 asymbol ***symppp ATTRIBUTE_UNUSED)
2587 if (bfd_get_flavour (obfd) == bfd_target_ieee_flavour)
2588 return write_ieee_debugging_info (obfd, dhandle);
2590 if (bfd_get_flavour (obfd) == bfd_target_coff_flavour
2591 || bfd_get_flavour (obfd) == bfd_target_elf_flavour)
2593 bfd_byte *syms, *strings;
2594 bfd_size_type symsize, stringsize;
2595 asection *stabsec, *stabstrsec;
2596 flagword flags;
2598 if (! write_stabs_in_sections_debugging_info (obfd, dhandle, &syms,
2599 &symsize, &strings,
2600 &stringsize))
2601 return FALSE;
2603 flags = SEC_HAS_CONTENTS | SEC_READONLY | SEC_DEBUGGING;
2604 stabsec = bfd_make_section_with_flags (obfd, ".stab", flags);
2605 stabstrsec = bfd_make_section_with_flags (obfd, ".stabstr", flags);
2606 if (stabsec == NULL
2607 || stabstrsec == NULL
2608 || ! bfd_set_section_size (obfd, stabsec, symsize)
2609 || ! bfd_set_section_size (obfd, stabstrsec, stringsize)
2610 || ! bfd_set_section_alignment (obfd, stabsec, 2)
2611 || ! bfd_set_section_alignment (obfd, stabstrsec, 0))
2613 non_fatal (_("%s: can't create debugging section: %s"),
2614 bfd_get_filename (obfd),
2615 bfd_errmsg (bfd_get_error ()));
2616 return FALSE;
2619 /* We can get away with setting the section contents now because
2620 the next thing the caller is going to do is copy over the
2621 real sections. We may someday have to split the contents
2622 setting out of this function. */
2623 if (! bfd_set_section_contents (obfd, stabsec, syms, 0, symsize)
2624 || ! bfd_set_section_contents (obfd, stabstrsec, strings, 0,
2625 stringsize))
2627 non_fatal (_("%s: can't set debugging section contents: %s"),
2628 bfd_get_filename (obfd),
2629 bfd_errmsg (bfd_get_error ()));
2630 return FALSE;
2633 return TRUE;
2636 non_fatal (_("%s: don't know how to write debugging information for %s"),
2637 bfd_get_filename (obfd), bfd_get_target (obfd));
2638 return FALSE;
2641 static int
2642 strip_main (int argc, char *argv[])
2644 char *input_target = NULL;
2645 char *output_target = NULL;
2646 bfd_boolean show_version = FALSE;
2647 bfd_boolean formats_info = FALSE;
2648 int c;
2649 int i;
2650 struct section_list *p;
2651 char *output_file = NULL;
2653 while ((c = getopt_long (argc, argv, "I:O:F:K:N:R:o:sSpdgxXHhVvw",
2654 strip_options, (int *) 0)) != EOF)
2656 switch (c)
2658 case 'I':
2659 input_target = optarg;
2660 break;
2661 case 'O':
2662 output_target = optarg;
2663 break;
2664 case 'F':
2665 input_target = output_target = optarg;
2666 break;
2667 case 'R':
2668 p = find_section_list (optarg, TRUE);
2669 p->remove = TRUE;
2670 sections_removed = TRUE;
2671 break;
2672 case 's':
2673 strip_symbols = STRIP_ALL;
2674 break;
2675 case 'S':
2676 case 'g':
2677 case 'd': /* Historic BSD alias for -g. Used by early NetBSD. */
2678 strip_symbols = STRIP_DEBUG;
2679 break;
2680 case OPTION_STRIP_UNNEEDED:
2681 strip_symbols = STRIP_UNNEEDED;
2682 break;
2683 case 'K':
2684 add_specific_symbol (optarg, &keep_specific_list);
2685 break;
2686 case 'N':
2687 add_specific_symbol (optarg, &strip_specific_list);
2688 break;
2689 case 'o':
2690 output_file = optarg;
2691 break;
2692 case 'p':
2693 preserve_dates = TRUE;
2694 break;
2695 case 'x':
2696 discard_locals = LOCALS_ALL;
2697 break;
2698 case 'X':
2699 discard_locals = LOCALS_START_L;
2700 break;
2701 case 'v':
2702 verbose = TRUE;
2703 break;
2704 case 'V':
2705 show_version = TRUE;
2706 break;
2707 case OPTION_FORMATS_INFO:
2708 formats_info = TRUE;
2709 break;
2710 case OPTION_ONLY_KEEP_DEBUG:
2711 strip_symbols = STRIP_NONDEBUG;
2712 break;
2713 case OPTION_KEEP_FILE_SYMBOLS:
2714 keep_file_symbols = 1;
2715 break;
2716 case 0:
2717 /* We've been given a long option. */
2718 break;
2719 case 'w':
2720 wildcard = TRUE;
2721 break;
2722 case 'H':
2723 case 'h':
2724 strip_usage (stdout, 0);
2725 default:
2726 strip_usage (stderr, 1);
2730 if (formats_info)
2732 display_info ();
2733 return 0;
2736 if (show_version)
2737 print_version ("strip");
2739 /* Default is to strip all symbols. */
2740 if (strip_symbols == STRIP_UNDEF
2741 && discard_locals == LOCALS_UNDEF
2742 && strip_specific_list == NULL)
2743 strip_symbols = STRIP_ALL;
2745 if (output_target == NULL)
2746 output_target = input_target;
2748 i = optind;
2749 if (i == argc
2750 || (output_file != NULL && (i + 1) < argc))
2751 strip_usage (stderr, 1);
2753 for (; i < argc; i++)
2755 int hold_status = status;
2756 struct stat statbuf;
2757 char *tmpname;
2759 if (get_file_size (argv[i]) < 1)
2761 status = 1;
2762 continue;
2765 if (preserve_dates)
2766 /* No need to check the return value of stat().
2767 It has already been checked in get_file_size(). */
2768 stat (argv[i], &statbuf);
2770 if (output_file == NULL || strcmp (argv[i], output_file) == 0)
2771 tmpname = make_tempname (argv[i]);
2772 else
2773 tmpname = output_file;
2775 if (tmpname == NULL)
2777 non_fatal (_("could not create temporary file to hold stripped copy of '%s'"),
2778 argv[i]);
2779 status = 1;
2780 continue;
2783 status = 0;
2784 copy_file (argv[i], tmpname, input_target, output_target);
2785 if (status == 0)
2787 if (preserve_dates)
2788 set_times (tmpname, &statbuf);
2789 if (output_file != tmpname)
2790 smart_rename (tmpname, output_file ? output_file : argv[i],
2791 preserve_dates);
2792 status = hold_status;
2794 else
2795 unlink_if_ordinary (tmpname);
2796 if (output_file != tmpname)
2797 free (tmpname);
2800 return status;
2803 static int
2804 copy_main (int argc, char *argv[])
2806 char * binary_architecture = NULL;
2807 char *input_filename = NULL;
2808 char *output_filename = NULL;
2809 char *tmpname;
2810 char *input_target = NULL;
2811 char *output_target = NULL;
2812 bfd_boolean show_version = FALSE;
2813 bfd_boolean change_warn = TRUE;
2814 bfd_boolean formats_info = FALSE;
2815 int c;
2816 struct section_list *p;
2817 struct stat statbuf;
2819 while ((c = getopt_long (argc, argv, "b:B:i:I:j:K:N:s:O:d:F:L:G:R:SpgxXHhVvW:w",
2820 copy_options, (int *) 0)) != EOF)
2822 switch (c)
2824 case 'b':
2825 copy_byte = atoi (optarg);
2826 if (copy_byte < 0)
2827 fatal (_("byte number must be non-negative"));
2828 break;
2830 case 'B':
2831 binary_architecture = optarg;
2832 break;
2834 case 'i':
2835 interleave = atoi (optarg);
2836 if (interleave < 1)
2837 fatal (_("interleave must be positive"));
2838 break;
2840 case 'I':
2841 case 's': /* "source" - 'I' is preferred */
2842 input_target = optarg;
2843 break;
2845 case 'O':
2846 case 'd': /* "destination" - 'O' is preferred */
2847 output_target = optarg;
2848 break;
2850 case 'F':
2851 input_target = output_target = optarg;
2852 break;
2854 case 'j':
2855 p = find_section_list (optarg, TRUE);
2856 if (p->remove)
2857 fatal (_("%s both copied and removed"), optarg);
2858 p->copy = TRUE;
2859 sections_copied = TRUE;
2860 break;
2862 case 'R':
2863 p = find_section_list (optarg, TRUE);
2864 if (p->copy)
2865 fatal (_("%s both copied and removed"), optarg);
2866 p->remove = TRUE;
2867 sections_removed = TRUE;
2868 break;
2870 case 'S':
2871 strip_symbols = STRIP_ALL;
2872 break;
2874 case 'g':
2875 strip_symbols = STRIP_DEBUG;
2876 break;
2878 case OPTION_STRIP_UNNEEDED:
2879 strip_symbols = STRIP_UNNEEDED;
2880 break;
2882 case OPTION_ONLY_KEEP_DEBUG:
2883 strip_symbols = STRIP_NONDEBUG;
2884 break;
2886 case OPTION_KEEP_FILE_SYMBOLS:
2887 keep_file_symbols = 1;
2888 break;
2890 case OPTION_ADD_GNU_DEBUGLINK:
2891 gnu_debuglink_filename = optarg;
2892 break;
2894 case 'K':
2895 add_specific_symbol (optarg, &keep_specific_list);
2896 break;
2898 case 'N':
2899 add_specific_symbol (optarg, &strip_specific_list);
2900 break;
2902 case OPTION_STRIP_UNNEEDED_SYMBOL:
2903 add_specific_symbol (optarg, &strip_unneeded_list);
2904 break;
2906 case 'L':
2907 add_specific_symbol (optarg, &localize_specific_list);
2908 break;
2910 case OPTION_GLOBALIZE_SYMBOL:
2911 add_specific_symbol (optarg, &globalize_specific_list);
2912 break;
2914 case 'G':
2915 add_specific_symbol (optarg, &keepglobal_specific_list);
2916 break;
2918 case 'W':
2919 add_specific_symbol (optarg, &weaken_specific_list);
2920 break;
2922 case 'p':
2923 preserve_dates = TRUE;
2924 break;
2926 case 'w':
2927 wildcard = TRUE;
2928 break;
2930 case 'x':
2931 discard_locals = LOCALS_ALL;
2932 break;
2934 case 'X':
2935 discard_locals = LOCALS_START_L;
2936 break;
2938 case 'v':
2939 verbose = TRUE;
2940 break;
2942 case 'V':
2943 show_version = TRUE;
2944 break;
2946 case OPTION_FORMATS_INFO:
2947 formats_info = TRUE;
2948 break;
2950 case OPTION_WEAKEN:
2951 weaken = TRUE;
2952 break;
2954 case OPTION_ADD_SECTION:
2956 const char *s;
2957 off_t size;
2958 struct section_add *pa;
2959 int len;
2960 char *name;
2961 FILE *f;
2963 s = strchr (optarg, '=');
2965 if (s == NULL)
2966 fatal (_("bad format for %s"), "--add-section");
2968 size = get_file_size (s + 1);
2969 if (size < 1)
2971 status = 1;
2972 break;
2975 pa = xmalloc (sizeof (struct section_add));
2977 len = s - optarg;
2978 name = xmalloc (len + 1);
2979 strncpy (name, optarg, len);
2980 name[len] = '\0';
2981 pa->name = name;
2983 pa->filename = s + 1;
2984 pa->size = size;
2985 pa->contents = xmalloc (size);
2987 f = fopen (pa->filename, FOPEN_RB);
2989 if (f == NULL)
2990 fatal (_("cannot open: %s: %s"),
2991 pa->filename, strerror (errno));
2993 if (fread (pa->contents, 1, pa->size, f) == 0
2994 || ferror (f))
2995 fatal (_("%s: fread failed"), pa->filename);
2997 fclose (f);
2999 pa->next = add_sections;
3000 add_sections = pa;
3002 break;
3004 case OPTION_CHANGE_START:
3005 change_start = parse_vma (optarg, "--change-start");
3006 break;
3008 case OPTION_CHANGE_SECTION_ADDRESS:
3009 case OPTION_CHANGE_SECTION_LMA:
3010 case OPTION_CHANGE_SECTION_VMA:
3012 const char *s;
3013 int len;
3014 char *name;
3015 char *option = NULL;
3016 bfd_vma val;
3017 enum change_action what = CHANGE_IGNORE;
3019 switch (c)
3021 case OPTION_CHANGE_SECTION_ADDRESS:
3022 option = "--change-section-address";
3023 break;
3024 case OPTION_CHANGE_SECTION_LMA:
3025 option = "--change-section-lma";
3026 break;
3027 case OPTION_CHANGE_SECTION_VMA:
3028 option = "--change-section-vma";
3029 break;
3032 s = strchr (optarg, '=');
3033 if (s == NULL)
3035 s = strchr (optarg, '+');
3036 if (s == NULL)
3038 s = strchr (optarg, '-');
3039 if (s == NULL)
3040 fatal (_("bad format for %s"), option);
3044 len = s - optarg;
3045 name = xmalloc (len + 1);
3046 strncpy (name, optarg, len);
3047 name[len] = '\0';
3049 p = find_section_list (name, TRUE);
3051 val = parse_vma (s + 1, option);
3053 switch (*s)
3055 case '=': what = CHANGE_SET; break;
3056 case '-': val = - val; /* Drop through. */
3057 case '+': what = CHANGE_MODIFY; break;
3060 switch (c)
3062 case OPTION_CHANGE_SECTION_ADDRESS:
3063 p->change_vma = what;
3064 p->vma_val = val;
3065 /* Drop through. */
3067 case OPTION_CHANGE_SECTION_LMA:
3068 p->change_lma = what;
3069 p->lma_val = val;
3070 break;
3072 case OPTION_CHANGE_SECTION_VMA:
3073 p->change_vma = what;
3074 p->vma_val = val;
3075 break;
3078 break;
3080 case OPTION_CHANGE_ADDRESSES:
3081 change_section_address = parse_vma (optarg, "--change-addresses");
3082 change_start = change_section_address;
3083 break;
3085 case OPTION_CHANGE_WARNINGS:
3086 change_warn = TRUE;
3087 break;
3089 case OPTION_CHANGE_LEADING_CHAR:
3090 change_leading_char = TRUE;
3091 break;
3093 case OPTION_DEBUGGING:
3094 convert_debugging = TRUE;
3095 break;
3097 case OPTION_GAP_FILL:
3099 bfd_vma gap_fill_vma;
3101 gap_fill_vma = parse_vma (optarg, "--gap-fill");
3102 gap_fill = (bfd_byte) gap_fill_vma;
3103 if ((bfd_vma) gap_fill != gap_fill_vma)
3105 char buff[20];
3107 sprintf_vma (buff, gap_fill_vma);
3109 non_fatal (_("Warning: truncating gap-fill from 0x%s to 0x%x"),
3110 buff, gap_fill);
3112 gap_fill_set = TRUE;
3114 break;
3116 case OPTION_NO_CHANGE_WARNINGS:
3117 change_warn = FALSE;
3118 break;
3120 case OPTION_PAD_TO:
3121 pad_to = parse_vma (optarg, "--pad-to");
3122 pad_to_set = TRUE;
3123 break;
3125 case OPTION_REMOVE_LEADING_CHAR:
3126 remove_leading_char = TRUE;
3127 break;
3129 case OPTION_REDEFINE_SYM:
3131 /* Push this redefinition onto redefine_symbol_list. */
3133 int len;
3134 const char *s;
3135 const char *nextarg;
3136 char *source, *target;
3138 s = strchr (optarg, '=');
3139 if (s == NULL)
3140 fatal (_("bad format for %s"), "--redefine-sym");
3142 len = s - optarg;
3143 source = xmalloc (len + 1);
3144 strncpy (source, optarg, len);
3145 source[len] = '\0';
3147 nextarg = s + 1;
3148 len = strlen (nextarg);
3149 target = xmalloc (len + 1);
3150 strcpy (target, nextarg);
3152 redefine_list_append ("--redefine-sym", source, target);
3154 free (source);
3155 free (target);
3157 break;
3159 case OPTION_REDEFINE_SYMS:
3160 add_redefine_syms_file (optarg);
3161 break;
3163 case OPTION_SET_SECTION_FLAGS:
3165 const char *s;
3166 int len;
3167 char *name;
3169 s = strchr (optarg, '=');
3170 if (s == NULL)
3171 fatal (_("bad format for %s"), "--set-section-flags");
3173 len = s - optarg;
3174 name = xmalloc (len + 1);
3175 strncpy (name, optarg, len);
3176 name[len] = '\0';
3178 p = find_section_list (name, TRUE);
3180 p->set_flags = TRUE;
3181 p->flags = parse_flags (s + 1);
3183 break;
3185 case OPTION_RENAME_SECTION:
3187 flagword flags;
3188 const char *eq, *fl;
3189 char *old_name;
3190 char *new_name;
3191 unsigned int len;
3193 eq = strchr (optarg, '=');
3194 if (eq == NULL)
3195 fatal (_("bad format for %s"), "--rename-section");
3197 len = eq - optarg;
3198 if (len == 0)
3199 fatal (_("bad format for %s"), "--rename-section");
3201 old_name = xmalloc (len + 1);
3202 strncpy (old_name, optarg, len);
3203 old_name[len] = 0;
3205 eq++;
3206 fl = strchr (eq, ',');
3207 if (fl)
3209 flags = parse_flags (fl + 1);
3210 len = fl - eq;
3212 else
3214 flags = -1;
3215 len = strlen (eq);
3218 if (len == 0)
3219 fatal (_("bad format for %s"), "--rename-section");
3221 new_name = xmalloc (len + 1);
3222 strncpy (new_name, eq, len);
3223 new_name[len] = 0;
3225 add_section_rename (old_name, new_name, flags);
3227 break;
3229 case OPTION_SET_START:
3230 set_start = parse_vma (optarg, "--set-start");
3231 set_start_set = TRUE;
3232 break;
3234 case OPTION_SREC_LEN:
3235 Chunk = parse_vma (optarg, "--srec-len");
3236 break;
3238 case OPTION_SREC_FORCES3:
3239 S3Forced = TRUE;
3240 break;
3242 case OPTION_STRIP_SYMBOLS:
3243 add_specific_symbols (optarg, &strip_specific_list);
3244 break;
3246 case OPTION_STRIP_UNNEEDED_SYMBOLS:
3247 add_specific_symbols (optarg, &strip_unneeded_list);
3248 break;
3250 case OPTION_KEEP_SYMBOLS:
3251 add_specific_symbols (optarg, &keep_specific_list);
3252 break;
3254 case OPTION_LOCALIZE_HIDDEN:
3255 localize_hidden = TRUE;
3256 break;
3258 case OPTION_LOCALIZE_SYMBOLS:
3259 add_specific_symbols (optarg, &localize_specific_list);
3260 break;
3262 case OPTION_GLOBALIZE_SYMBOLS:
3263 add_specific_symbols (optarg, &globalize_specific_list);
3264 break;
3266 case OPTION_KEEPGLOBAL_SYMBOLS:
3267 add_specific_symbols (optarg, &keepglobal_specific_list);
3268 break;
3270 case OPTION_WEAKEN_SYMBOLS:
3271 add_specific_symbols (optarg, &weaken_specific_list);
3272 break;
3274 case OPTION_ALT_MACH_CODE:
3275 use_alt_mach_code = strtoul (optarg, NULL, 0);
3276 if (use_alt_mach_code == 0)
3277 fatal (_("unable to parse alternative machine code"));
3278 break;
3280 case OPTION_PREFIX_SYMBOLS:
3281 prefix_symbols_string = optarg;
3282 break;
3284 case OPTION_PREFIX_SECTIONS:
3285 prefix_sections_string = optarg;
3286 break;
3288 case OPTION_PREFIX_ALLOC_SECTIONS:
3289 prefix_alloc_sections_string = optarg;
3290 break;
3292 case OPTION_READONLY_TEXT:
3293 bfd_flags_to_set |= WP_TEXT;
3294 bfd_flags_to_clear &= ~WP_TEXT;
3295 break;
3297 case OPTION_WRITABLE_TEXT:
3298 bfd_flags_to_clear |= WP_TEXT;
3299 bfd_flags_to_set &= ~WP_TEXT;
3300 break;
3302 case OPTION_PURE:
3303 bfd_flags_to_set |= D_PAGED;
3304 bfd_flags_to_clear &= ~D_PAGED;
3305 break;
3307 case OPTION_IMPURE:
3308 bfd_flags_to_clear |= D_PAGED;
3309 bfd_flags_to_set &= ~D_PAGED;
3310 break;
3312 case OPTION_EXTRACT_SYMBOL:
3313 extract_symbol = TRUE;
3314 break;
3316 case OPTION_REVERSE_BYTES:
3318 int prev = reverse_bytes;
3320 reverse_bytes = atoi (optarg);
3321 if ((reverse_bytes <= 0) || ((reverse_bytes % 2) != 0))
3322 fatal (_("number of bytes to reverse must be positive and even"));
3324 if (prev && prev != reverse_bytes)
3325 non_fatal (_("Warning: ignoring previous --reverse-bytes value of %d"),
3326 prev);
3327 break;
3330 case 0:
3331 /* We've been given a long option. */
3332 break;
3334 case 'H':
3335 case 'h':
3336 copy_usage (stdout, 0);
3338 default:
3339 copy_usage (stderr, 1);
3343 if (formats_info)
3345 display_info ();
3346 return 0;
3349 if (show_version)
3350 print_version ("objcopy");
3352 if (copy_byte >= interleave)
3353 fatal (_("byte number must be less than interleave"));
3355 if (optind == argc || optind + 2 < argc)
3356 copy_usage (stderr, 1);
3358 input_filename = argv[optind];
3359 if (optind + 1 < argc)
3360 output_filename = argv[optind + 1];
3362 /* Default is to strip no symbols. */
3363 if (strip_symbols == STRIP_UNDEF && discard_locals == LOCALS_UNDEF)
3364 strip_symbols = STRIP_NONE;
3366 if (output_target == NULL)
3367 output_target = input_target;
3369 if (binary_architecture != NULL)
3371 if (input_target && strcmp (input_target, "binary") == 0)
3373 const bfd_arch_info_type * temp_arch_info;
3375 temp_arch_info = bfd_scan_arch (binary_architecture);
3377 if (temp_arch_info != NULL)
3379 bfd_external_binary_architecture = temp_arch_info->arch;
3380 bfd_external_machine = temp_arch_info->mach;
3382 else
3383 fatal (_("architecture %s unknown"), binary_architecture);
3385 else
3387 non_fatal (_("Warning: input target 'binary' required for binary architecture parameter."));
3388 non_fatal (_(" Argument %s ignored"), binary_architecture);
3392 if (preserve_dates)
3393 if (stat (input_filename, & statbuf) < 0)
3394 fatal (_("warning: could not locate '%s'. System error message: %s"),
3395 input_filename, strerror (errno));
3397 /* If there is no destination file, or the source and destination files
3398 are the same, then create a temp and rename the result into the input. */
3399 if (output_filename == NULL || strcmp (input_filename, output_filename) == 0)
3400 tmpname = make_tempname (input_filename);
3401 else
3402 tmpname = output_filename;
3404 if (tmpname == NULL)
3405 fatal (_("warning: could not create temporary file whilst copying '%s', (error: %s)"),
3406 input_filename, strerror (errno));
3408 copy_file (input_filename, tmpname, input_target, output_target);
3409 if (status == 0)
3411 if (preserve_dates)
3412 set_times (tmpname, &statbuf);
3413 if (tmpname != output_filename)
3414 smart_rename (tmpname, input_filename, preserve_dates);
3416 else
3417 unlink_if_ordinary (tmpname);
3419 if (change_warn)
3421 for (p = change_sections; p != NULL; p = p->next)
3423 if (! p->used)
3425 if (p->change_vma != CHANGE_IGNORE)
3427 char buff [20];
3429 sprintf_vma (buff, p->vma_val);
3431 /* xgettext:c-format */
3432 non_fatal (_("%s %s%c0x%s never used"),
3433 "--change-section-vma",
3434 p->name,
3435 p->change_vma == CHANGE_SET ? '=' : '+',
3436 buff);
3439 if (p->change_lma != CHANGE_IGNORE)
3441 char buff [20];
3443 sprintf_vma (buff, p->lma_val);
3445 /* xgettext:c-format */
3446 non_fatal (_("%s %s%c0x%s never used"),
3447 "--change-section-lma",
3448 p->name,
3449 p->change_lma == CHANGE_SET ? '=' : '+',
3450 buff);
3456 return 0;
3460 main (int argc, char *argv[])
3462 #if defined (HAVE_SETLOCALE) && defined (HAVE_LC_MESSAGES)
3463 setlocale (LC_MESSAGES, "");
3464 #endif
3465 #if defined (HAVE_SETLOCALE)
3466 setlocale (LC_CTYPE, "");
3467 #endif
3468 bindtextdomain (PACKAGE, LOCALEDIR);
3469 textdomain (PACKAGE);
3471 program_name = argv[0];
3472 xmalloc_set_program_name (program_name);
3474 START_PROGRESS (program_name, 0);
3476 expandargv (&argc, &argv);
3478 strip_symbols = STRIP_UNDEF;
3479 discard_locals = LOCALS_UNDEF;
3481 bfd_init ();
3482 set_default_bfd_target ();
3484 if (is_strip < 0)
3486 int i = strlen (program_name);
3487 #ifdef HAVE_DOS_BASED_FILE_SYSTEM
3488 /* Drop the .exe suffix, if any. */
3489 if (i > 4 && FILENAME_CMP (program_name + i - 4, ".exe") == 0)
3491 i -= 4;
3492 program_name[i] = '\0';
3494 #endif
3495 is_strip = (i >= 5 && FILENAME_CMP (program_name + i - 5, "strip") == 0);
3498 if (is_strip)
3499 strip_main (argc, argv);
3500 else
3501 copy_main (argc, argv);
3503 END_PROGRESS (program_name);
3505 return status;