Fixed changelog entry. Indented m68k-tdep.c
[deliverable/binutils-gdb.git] / binutils / objcopy.c
CommitLineData
252b5132 1/* objcopy.c -- copy object file from input to output, optionally massaging it.
8c2bc687 2 Copyright 1991, 1992, 1993, 1994, 1995, 1996, 1997, 1998, 1999, 2000,
12b87c89 3 2001, 2002
252b5132
RH
4 Free Software Foundation, Inc.
5
6 This file is part of GNU Binutils.
7
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 2 of the License, or
11 (at your option) any later version.
12
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.
17
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., 59 Temple Place - Suite 330, Boston, MA
21 02111-1307, USA. */
22\f
23#include "bfd.h"
24#include "progress.h"
25#include "bucomm.h"
26#include "getopt.h"
27#include "libiberty.h"
28#include "budbg.h"
5af11cab 29#include "filenames.h"
252b5132
RH
30#include <sys/stat.h>
31
32/* A list of symbols to explicitly strip out, or to keep. A linked
33 list is good enough for a small number from the command line, but
34 this will slow things down a lot if many symbols are being
0af11b59 35 deleted. */
252b5132
RH
36
37struct symlist
38{
39 const char *name;
40 struct symlist *next;
41};
42
57938635
AM
43/* A list to support redefine_sym. */
44struct redefine_node
45{
46 char *source;
47 char *target;
48 struct redefine_node *next;
49};
50
594ef5db
NC
51typedef struct section_rename
52{
53 const char * old_name;
54 const char * new_name;
55 flagword flags;
56 struct section_rename * next;
57}
58section_rename;
59
60/* List of sections to be renamed. */
61static section_rename * section_rename_list;
62
252b5132
RH
63static void copy_usage PARAMS ((FILE *, int));
64static void strip_usage PARAMS ((FILE *, int));
65static flagword parse_flags PARAMS ((const char *));
66static struct section_list *find_section_list PARAMS ((const char *, boolean));
67static void setup_section PARAMS ((bfd *, asection *, PTR));
68static void copy_section PARAMS ((bfd *, asection *, PTR));
69static void get_sections PARAMS ((bfd *, asection *, PTR));
70static int compare_section_lma PARAMS ((const PTR, const PTR));
71static void add_specific_symbol PARAMS ((const char *, struct symlist **));
16b2b71c 72static void add_specific_symbols PARAMS ((const char *, struct symlist **));
252b5132
RH
73static boolean is_specified_symbol PARAMS ((const char *, struct symlist *));
74static boolean is_strip_section PARAMS ((bfd *, asection *));
75static unsigned int filter_symbols
76 PARAMS ((bfd *, bfd *, asymbol **, asymbol **, long));
77static void mark_symbols_used_in_relocations PARAMS ((bfd *, asection *, PTR));
78static void filter_bytes PARAMS ((char *, bfd_size_type *));
79static boolean write_debugging_info PARAMS ((bfd *, PTR, long *, asymbol ***));
80static void copy_object PARAMS ((bfd *, bfd *));
81static void copy_archive PARAMS ((bfd *, bfd *, const char *));
82static void copy_file
83 PARAMS ((const char *, const char *, const char *, const char *));
84static int strip_main PARAMS ((int, char **));
85static int copy_main PARAMS ((int, char **));
57938635
AM
86static const char *lookup_sym_redefinition PARAMS((const char *));
87static void redefine_list_append PARAMS ((const char *, const char *));
594ef5db
NC
88static const char * find_section_rename PARAMS ((bfd *, sec_ptr, flagword *));
89static void add_section_rename PARAMS ((const char *, const char *, flagword));
252b5132
RH
90
91#define RETURN_NONFATAL(s) {bfd_nonfatal (s); status = 1; return;}
92
93static asymbol **isympp = NULL; /* Input symbols */
94static asymbol **osympp = NULL; /* Output symbols that survive stripping */
95
96/* If `copy_byte' >= 0, copy only that byte of every `interleave' bytes. */
97static int copy_byte = -1;
98static int interleave = 4;
99
0af11b59 100static boolean verbose; /* Print file and target names. */
252b5132
RH
101static boolean preserve_dates; /* Preserve input file timestamp. */
102static int status = 0; /* Exit status. */
103
104enum strip_action
105 {
106 STRIP_UNDEF,
107 STRIP_NONE, /* don't strip */
108 STRIP_DEBUG, /* strip all debugger symbols */
109 STRIP_UNNEEDED, /* strip unnecessary symbols */
110 STRIP_ALL /* strip all symbols */
111 };
112
0af11b59 113/* Which symbols to remove. */
252b5132
RH
114static enum strip_action strip_symbols;
115
116enum locals_action
117 {
118 LOCALS_UNDEF,
119 LOCALS_START_L, /* discard locals starting with L */
120 LOCALS_ALL /* discard all locals */
121 };
122
123/* Which local symbols to remove. Overrides STRIP_ALL. */
124static enum locals_action discard_locals;
125
126/* What kind of change to perform. */
127enum change_action
128{
129 CHANGE_IGNORE,
130 CHANGE_MODIFY,
131 CHANGE_SET
132};
133
134/* Structure used to hold lists of sections and actions to take. */
135struct section_list
136{
137 struct section_list * next; /* Next section to change. */
138 const char * name; /* Section name. */
139 boolean used; /* Whether this entry was used. */
140 boolean remove; /* Whether to remove this section. */
f91ea849 141 boolean copy; /* Whether to copy this section. */
252b5132
RH
142 enum change_action change_vma;/* Whether to change or set VMA. */
143 bfd_vma vma_val; /* Amount to change by or set to. */
144 enum change_action change_lma;/* Whether to change or set LMA. */
145 bfd_vma lma_val; /* Amount to change by or set to. */
146 boolean set_flags; /* Whether to set the section flags. */
147 flagword flags; /* What to set the section flags to. */
148};
149
150static struct section_list *change_sections;
594ef5db
NC
151
152/* True if some sections are to be removed. */
252b5132 153static boolean sections_removed;
594ef5db
NC
154
155/* True if only some sections are to be copied. */
f91ea849 156static boolean sections_copied;
252b5132
RH
157
158/* Changes to the start address. */
159static bfd_vma change_start = 0;
160static boolean set_start_set = false;
161static bfd_vma set_start;
162
163/* Changes to section addresses. */
164static bfd_vma change_section_address = 0;
165
166/* Filling gaps between sections. */
167static boolean gap_fill_set = false;
168static bfd_byte gap_fill = 0;
169
170/* Pad to a given address. */
171static boolean pad_to_set = false;
172static bfd_vma pad_to;
173
1ae8b3d2
AO
174/* Use alternate machine code? */
175static int use_alt_mach_code = 0;
176
252b5132 177/* List of sections to add. */
252b5132
RH
178struct section_add
179{
180 /* Next section to add. */
181 struct section_add *next;
182 /* Name of section to add. */
183 const char *name;
184 /* Name of file holding section contents. */
185 const char *filename;
186 /* Size of file. */
187 size_t size;
188 /* Contents of file. */
189 bfd_byte *contents;
190 /* BFD section, after it has been added. */
191 asection *section;
192};
193
594ef5db 194/* List of sections to add to the output BFD. */
252b5132
RH
195static struct section_add *add_sections;
196
197/* Whether to convert debugging information. */
252b5132
RH
198static boolean convert_debugging = false;
199
200/* Whether to change the leading character in symbol names. */
252b5132
RH
201static boolean change_leading_char = false;
202
203/* Whether to remove the leading character from global symbol names. */
252b5132
RH
204static boolean remove_leading_char = false;
205
16b2b71c
NC
206/* List of symbols to strip, keep, localize, keep-global, weaken,
207 or redefine. */
252b5132
RH
208static struct symlist *strip_specific_list = NULL;
209static struct symlist *keep_specific_list = NULL;
210static struct symlist *localize_specific_list = NULL;
16b2b71c 211static struct symlist *keepglobal_specific_list = NULL;
252b5132 212static struct symlist *weaken_specific_list = NULL;
57938635 213static struct redefine_node *redefine_sym_list = NULL;
252b5132
RH
214
215/* If this is true, we weaken global symbols (set BSF_WEAK). */
252b5132
RH
216static boolean weaken = false;
217
218/* 150 isn't special; it's just an arbitrary non-ASCII char value. */
219
220#define OPTION_ADD_SECTION 150
221#define OPTION_CHANGE_ADDRESSES (OPTION_ADD_SECTION + 1)
222#define OPTION_CHANGE_LEADING_CHAR (OPTION_CHANGE_ADDRESSES + 1)
223#define OPTION_CHANGE_START (OPTION_CHANGE_LEADING_CHAR + 1)
224#define OPTION_CHANGE_SECTION_ADDRESS (OPTION_CHANGE_START + 1)
225#define OPTION_CHANGE_SECTION_LMA (OPTION_CHANGE_SECTION_ADDRESS + 1)
226#define OPTION_CHANGE_SECTION_VMA (OPTION_CHANGE_SECTION_LMA + 1)
227#define OPTION_CHANGE_WARNINGS (OPTION_CHANGE_SECTION_VMA + 1)
228#define OPTION_DEBUGGING (OPTION_CHANGE_WARNINGS + 1)
229#define OPTION_GAP_FILL (OPTION_DEBUGGING + 1)
230#define OPTION_NO_CHANGE_WARNINGS (OPTION_GAP_FILL + 1)
231#define OPTION_PAD_TO (OPTION_NO_CHANGE_WARNINGS + 1)
232#define OPTION_REMOVE_LEADING_CHAR (OPTION_PAD_TO + 1)
233#define OPTION_SET_SECTION_FLAGS (OPTION_REMOVE_LEADING_CHAR + 1)
234#define OPTION_SET_START (OPTION_SET_SECTION_FLAGS + 1)
235#define OPTION_STRIP_UNNEEDED (OPTION_SET_START + 1)
236#define OPTION_WEAKEN (OPTION_STRIP_UNNEEDED + 1)
57938635 237#define OPTION_REDEFINE_SYM (OPTION_WEAKEN + 1)
420496c1
NC
238#define OPTION_SREC_LEN (OPTION_REDEFINE_SYM + 1)
239#define OPTION_SREC_FORCES3 (OPTION_SREC_LEN + 1)
16b2b71c
NC
240#define OPTION_STRIP_SYMBOLS (OPTION_SREC_FORCES3 + 1)
241#define OPTION_KEEP_SYMBOLS (OPTION_STRIP_SYMBOLS + 1)
242#define OPTION_LOCALIZE_SYMBOLS (OPTION_KEEP_SYMBOLS + 1)
243#define OPTION_KEEPGLOBAL_SYMBOLS (OPTION_LOCALIZE_SYMBOLS + 1)
244#define OPTION_WEAKEN_SYMBOLS (OPTION_KEEPGLOBAL_SYMBOLS + 1)
594ef5db 245#define OPTION_RENAME_SECTION (OPTION_WEAKEN_SYMBOLS + 1)
1ae8b3d2 246#define OPTION_ALT_MACH_CODE (OPTION_RENAME_SECTION + 1)
252b5132
RH
247
248/* Options to handle if running as "strip". */
249
250static struct option strip_options[] =
251{
252 {"discard-all", no_argument, 0, 'x'},
253 {"discard-locals", no_argument, 0, 'X'},
254 {"format", required_argument, 0, 'F'}, /* Obsolete */
255 {"help", no_argument, 0, 'h'},
256 {"input-format", required_argument, 0, 'I'}, /* Obsolete */
257 {"input-target", required_argument, 0, 'I'},
258 {"keep-symbol", required_argument, 0, 'K'},
259 {"output-format", required_argument, 0, 'O'}, /* Obsolete */
260 {"output-target", required_argument, 0, 'O'},
af3bdff7 261 {"output-file", required_argument, 0, 'o'},
252b5132
RH
262 {"preserve-dates", no_argument, 0, 'p'},
263 {"remove-section", required_argument, 0, 'R'},
264 {"strip-all", no_argument, 0, 's'},
265 {"strip-debug", no_argument, 0, 'S'},
266 {"strip-unneeded", no_argument, 0, OPTION_STRIP_UNNEEDED},
267 {"strip-symbol", required_argument, 0, 'N'},
268 {"target", required_argument, 0, 'F'},
269 {"verbose", no_argument, 0, 'v'},
270 {"version", no_argument, 0, 'V'},
271 {0, no_argument, 0, 0}
272};
273
274/* Options to handle if running as "objcopy". */
275
276static struct option copy_options[] =
277{
278 {"add-section", required_argument, 0, OPTION_ADD_SECTION},
279 {"adjust-start", required_argument, 0, OPTION_CHANGE_START},
280 {"adjust-vma", required_argument, 0, OPTION_CHANGE_ADDRESSES},
281 {"adjust-section-vma", required_argument, 0, OPTION_CHANGE_SECTION_ADDRESS},
282 {"adjust-warnings", no_argument, 0, OPTION_CHANGE_WARNINGS},
43a0748c 283 {"binary-architecture", required_argument, 0, 'B'},
252b5132
RH
284 {"byte", required_argument, 0, 'b'},
285 {"change-addresses", required_argument, 0, OPTION_CHANGE_ADDRESSES},
286 {"change-leading-char", no_argument, 0, OPTION_CHANGE_LEADING_CHAR},
287 {"change-section-address", required_argument, 0, OPTION_CHANGE_SECTION_ADDRESS},
288 {"change-section-lma", required_argument, 0, OPTION_CHANGE_SECTION_LMA},
289 {"change-section-vma", required_argument, 0, OPTION_CHANGE_SECTION_VMA},
290 {"change-start", required_argument, 0, OPTION_CHANGE_START},
291 {"change-warnings", no_argument, 0, OPTION_CHANGE_WARNINGS},
292 {"debugging", no_argument, 0, OPTION_DEBUGGING},
293 {"discard-all", no_argument, 0, 'x'},
294 {"discard-locals", no_argument, 0, 'X'},
f91ea849 295 {"only-section", required_argument, 0, 'j'},
252b5132
RH
296 {"format", required_argument, 0, 'F'}, /* Obsolete */
297 {"gap-fill", required_argument, 0, OPTION_GAP_FILL},
298 {"help", no_argument, 0, 'h'},
299 {"input-format", required_argument, 0, 'I'}, /* Obsolete */
300 {"input-target", required_argument, 0, 'I'},
301 {"interleave", required_argument, 0, 'i'},
302 {"keep-symbol", required_argument, 0, 'K'},
303 {"no-adjust-warnings", no_argument, 0, OPTION_NO_CHANGE_WARNINGS},
304 {"no-change-warnings", no_argument, 0, OPTION_NO_CHANGE_WARNINGS},
305 {"output-format", required_argument, 0, 'O'}, /* Obsolete */
306 {"output-target", required_argument, 0, 'O'},
307 {"pad-to", required_argument, 0, OPTION_PAD_TO},
308 {"preserve-dates", no_argument, 0, 'p'},
309 {"localize-symbol", required_argument, 0, 'L'},
16b2b71c 310 {"keep-global-symbol", required_argument, 0, 'G'},
252b5132
RH
311 {"remove-leading-char", no_argument, 0, OPTION_REMOVE_LEADING_CHAR},
312 {"remove-section", required_argument, 0, 'R'},
594ef5db 313 {"rename-section", required_argument, 0, OPTION_RENAME_SECTION},
252b5132
RH
314 {"set-section-flags", required_argument, 0, OPTION_SET_SECTION_FLAGS},
315 {"set-start", required_argument, 0, OPTION_SET_START},
316 {"strip-all", no_argument, 0, 'S'},
317 {"strip-debug", no_argument, 0, 'g'},
318 {"strip-unneeded", no_argument, 0, OPTION_STRIP_UNNEEDED},
319 {"strip-symbol", required_argument, 0, 'N'},
320 {"target", required_argument, 0, 'F'},
321 {"verbose", no_argument, 0, 'v'},
322 {"version", no_argument, 0, 'V'},
323 {"weaken", no_argument, 0, OPTION_WEAKEN},
324 {"weaken-symbol", required_argument, 0, 'W'},
57938635 325 {"redefine-sym", required_argument, 0, OPTION_REDEFINE_SYM},
420496c1
NC
326 {"srec-len", required_argument, 0, OPTION_SREC_LEN},
327 {"srec-forceS3", no_argument, 0, OPTION_SREC_FORCES3},
16b2b71c
NC
328 {"keep-symbols", required_argument, 0, OPTION_KEEP_SYMBOLS},
329 {"strip-symbols", required_argument, 0, OPTION_STRIP_SYMBOLS},
330 {"keep-global-symbols", required_argument, 0, OPTION_KEEPGLOBAL_SYMBOLS},
331 {"localize-symbols", required_argument, 0, OPTION_LOCALIZE_SYMBOLS},
332 {"weaken-symbols", required_argument, 0, OPTION_WEAKEN_SYMBOLS},
1ae8b3d2 333 {"alt-machine-code", required_argument, 0, OPTION_ALT_MACH_CODE},
252b5132
RH
334 {0, no_argument, 0, 0}
335};
336
337/* IMPORTS */
338extern char *program_name;
339
340/* This flag distinguishes between strip and objcopy:
341 1 means this is 'strip'; 0 means this is 'objcopy'.
0af11b59 342 -1 means if we should use argv[0] to decide. */
252b5132
RH
343extern int is_strip;
344
420496c1
NC
345/* The maximum length of an S record. This variable is declared in srec.c
346 and can be modified by the --srec-len parameter. */
347extern unsigned int Chunk;
348
349/* Restrict the generation of Srecords to type S3 only.
350 This variable is declare in bfd/srec.c and can be toggled
351 on by the --srec-forceS3 command line switch. */
352extern boolean S3Forced;
252b5132 353
43a0748c
NC
354/* Defined in bfd/binary.c. Used to set architecture of input binary files. */
355extern enum bfd_architecture bfd_external_binary_architecture;
356
594ef5db 357\f
252b5132
RH
358static void
359copy_usage (stream, exit_status)
360 FILE *stream;
361 int exit_status;
362{
8b53311e
NC
363 fprintf (stream, _("Usage: %s [option(s)] in-file [out-file]\n"), program_name);
364 fprintf (stream, _(" Copies a binary file, possibly transforming it in the process\n"));
6364e0b4 365 fprintf (stream, _(" The options are:\n"));
252b5132 366 fprintf (stream, _("\
d5bcb29d
NC
367 -I --input-target <bfdname> Assume input file is in format <bfdname>\n\
368 -O --output-target <bfdname> Create an output file in format <bfdname>\n\
43a0748c 369 -B --binary-architecture <arch> Set arch of output file, when input is binary\n\
d5bcb29d
NC
370 -F --target <bfdname> Set both input and output format to <bfdname>\n\
371 --debugging Convert debugging information, if possible\n\
372 -p --preserve-dates Copy modified/access timestamps to the output\n\
373 -j --only-section <name> Only copy section <name> into the output\n\
374 -R --remove-section <name> Remove section <name> from the output\n\
375 -S --strip-all Remove all symbol and relocation information\n\
376 -g --strip-debug Remove all debugging symbols\n\
377 --strip-unneeded Remove all symbols not needed by relocations\n\
378 -N --strip-symbol <name> Do not copy symbol <name>\n\
379 -K --keep-symbol <name> Only copy symbol <name>\n\
380 -L --localize-symbol <name> Force symbol <name> to be marked as a local\n\
16b2b71c 381 -G --keep-global-symbol <name> Localize all symbols except <name>\n\
d5bcb29d
NC
382 -W --weaken-symbol <name> Force symbol <name> to be marked as a weak\n\
383 --weaken Force all global symbols to be marked as weak\n\
384 -x --discard-all Remove all non-global symbols\n\
385 -X --discard-locals Remove any compiler-generated symbols\n\
386 -i --interleave <number> Only copy one out of every <number> bytes\n\
387 -b --byte <num> Select byte <num> in every interleaved block\n\
388 --gap-fill <val> Fill gaps between sections with <val>\n\
389 --pad-to <addr> Pad the last section up to address <addr>\n\
390 --set-start <addr> Set the start address to <addr>\n\
391 {--change-start|--adjust-start} <incr>\n\
392 Add <incr> to the start address\n\
393 {--change-addresses|--adjust-vma} <incr>\n\
394 Add <incr> to LMA, VMA and start addresses\n\
395 {--change-section-address|--adjust-section-vma} <name>{=|+|-}<val>\n\
396 Change LMA and VMA of section <name> by <val>\n\
397 --change-section-lma <name>{=|+|-}<val>\n\
398 Change the LMA of section <name> by <val>\n\
399 --change-section-vma <name>{=|+|-}<val>\n\
400 Change the VMA of section <name> by <val>\n\
401 {--[no-]change-warnings|--[no-]adjust-warnings}\n\
402 Warn if a named section does not exist\n\
403 --set-section-flags <name>=<flags>\n\
404 Set section <name>'s properties to <flags>\n\
405 --add-section <name>=<file> Add section <name> found in <file> to output\n\
594ef5db 406 --rename-section <old>=<new>[,<flags>] Rename section <old> to <new>\n\
d5bcb29d
NC
407 --change-leading-char Force output format's leading character style\n\
408 --remove-leading-char Remove leading character from global symbols\n\
57938635 409 --redefine-sym <old>=<new> Redefine symbol name <old> to <new>\n\
420496c1
NC
410 --srec-len <number> Restrict the length of generated Srecords\n\
411 --srec-forceS3 Restrict the type of generated Srecords to S3\n\
16b2b71c
NC
412 --strip-symbols <file> -N for all symbols listed in <file>\n\
413 --keep-symbols <file> -K for all symbols listed in <file>\n\
414 --localize-symbols <file> -L for all symbols listed in <file>\n\
415 --keep-global-symbols <file> -G for all symbols listed in <file>\n\
416 --weaken-symbols <file> -W for all symbols listed in <file>\n\
1ae8b3d2 417 --alt-machine-code <index> Use alternate machine code for output\n\
d5bcb29d
NC
418 -v --verbose List all object files modified\n\
419 -V --version Display this program's version number\n\
420 -h --help Display this output\n\
421"));
252b5132
RH
422 list_supported_targets (program_name, stream);
423 if (exit_status == 0)
8ad3436c 424 fprintf (stream, _("Report bugs to %s\n"), REPORT_BUGS_TO);
252b5132
RH
425 exit (exit_status);
426}
427
428static void
429strip_usage (stream, exit_status)
430 FILE *stream;
431 int exit_status;
432{
8b53311e
NC
433 fprintf (stream, _("Usage: %s <option(s)> in-file(s)\n"), program_name);
434 fprintf (stream, _(" Removes symbols and sections from files\n"));
6364e0b4 435 fprintf (stream, _(" The options are:\n"));
252b5132 436 fprintf (stream, _("\
8b53311e
NC
437 -I --input-target=<bfdname> Assume input file is in format <bfdname>\n\
438 -O --output-target=<bfdname> Create an output file in format <bfdname>\n\
439 -F --target=<bfdname> Set both input and output format to <bfdname>\n\
d5bcb29d 440 -p --preserve-dates Copy modified/access timestamps to the output\n\
8b53311e 441 -R --remove-section=<name> Remove section <name> from the output\n\
d5bcb29d 442 -s --strip-all Remove all symbol and relocation information\n\
15c82623 443 -g -S -d --strip-debug Remove all debugging symbols\n\
d5bcb29d 444 --strip-unneeded Remove all symbols not needed by relocations\n\
8b53311e
NC
445 -N --strip-symbol=<name> Do not copy symbol <name>\n\
446 -K --keep-symbol=<name> Only copy symbol <name>\n\
d5bcb29d
NC
447 -x --discard-all Remove all non-global symbols\n\
448 -X --discard-locals Remove any compiler-generated symbols\n\
449 -v --verbose List all object files modified\n\
450 -V --version Display this program's version number\n\
451 -h --help Display this output\n\
452 -o <file> Place stripped output into <file>\n\
453"));
454
252b5132
RH
455 list_supported_targets (program_name, stream);
456 if (exit_status == 0)
8ad3436c 457 fprintf (stream, _("Report bugs to %s\n"), REPORT_BUGS_TO);
252b5132
RH
458 exit (exit_status);
459}
460
461/* Parse section flags into a flagword, with a fatal error if the
462 string can't be parsed. */
463
464static flagword
465parse_flags (s)
466 const char *s;
467{
468 flagword ret;
469 const char *snext;
470 int len;
471
472 ret = SEC_NO_FLAGS;
473
474 do
475 {
476 snext = strchr (s, ',');
477 if (snext == NULL)
478 len = strlen (s);
479 else
480 {
481 len = snext - s;
482 ++snext;
483 }
484
485 if (0) ;
486#define PARSE_FLAG(fname,fval) \
487 else if (strncasecmp (fname, s, len) == 0) ret |= fval
488 PARSE_FLAG ("alloc", SEC_ALLOC);
489 PARSE_FLAG ("load", SEC_LOAD);
3994e2c6 490 PARSE_FLAG ("noload", SEC_NEVER_LOAD);
252b5132 491 PARSE_FLAG ("readonly", SEC_READONLY);
3994e2c6 492 PARSE_FLAG ("debug", SEC_DEBUGGING);
252b5132
RH
493 PARSE_FLAG ("code", SEC_CODE);
494 PARSE_FLAG ("data", SEC_DATA);
495 PARSE_FLAG ("rom", SEC_ROM);
3994e2c6 496 PARSE_FLAG ("share", SEC_SHARED);
252b5132
RH
497 PARSE_FLAG ("contents", SEC_HAS_CONTENTS);
498#undef PARSE_FLAG
499 else
500 {
501 char *copy;
502
503 copy = xmalloc (len + 1);
504 strncpy (copy, s, len);
505 copy[len] = '\0';
506 non_fatal (_("unrecognized section flag `%s'"), copy);
57938635
AM
507 fatal (_("supported flags: %s"),
508 "alloc, load, noload, readonly, debug, code, data, rom, share, contents");
252b5132
RH
509 }
510
511 s = snext;
512 }
513 while (s != NULL);
514
515 return ret;
516}
517
518/* Find and optionally add an entry in the change_sections list. */
519
520static struct section_list *
521find_section_list (name, add)
522 const char *name;
523 boolean add;
524{
525 register struct section_list *p;
526
527 for (p = change_sections; p != NULL; p = p->next)
528 if (strcmp (p->name, name) == 0)
529 return p;
530
531 if (! add)
532 return NULL;
533
534 p = (struct section_list *) xmalloc (sizeof (struct section_list));
535 p->name = name;
536 p->used = false;
537 p->remove = false;
f91ea849 538 p->copy = false;
252b5132
RH
539 p->change_vma = CHANGE_IGNORE;
540 p->change_lma = CHANGE_IGNORE;
541 p->vma_val = 0;
542 p->lma_val = 0;
543 p->set_flags = false;
544 p->flags = 0;
545
546 p->next = change_sections;
547 change_sections = p;
548
549 return p;
550}
551
552/* Add a symbol to strip_specific_list. */
553
57938635 554static void
252b5132
RH
555add_specific_symbol (name, list)
556 const char *name;
557 struct symlist **list;
558{
559 struct symlist *tmp_list;
560
561 tmp_list = (struct symlist *) xmalloc (sizeof (struct symlist));
562 tmp_list->name = name;
563 tmp_list->next = *list;
564 *list = tmp_list;
565}
566
0af11b59 567/* Add symbols listed in `filename' to strip_specific_list. */
16b2b71c
NC
568
569#define IS_WHITESPACE(c) ((c) == ' ' || (c) == '\t')
570#define IS_LINE_TERMINATOR(c) ((c) == '\n' || (c) == '\r' || (c) == '\0')
571
572static void
573add_specific_symbols (filename, list)
574 const char *filename;
575 struct symlist **list;
576{
577 struct stat st;
578 FILE * f;
579 char * line;
580 char * buffer;
581 unsigned int line_count;
0af11b59 582
16b2b71c
NC
583 if (stat (filename, & st) < 0)
584 fatal (_("cannot stat: %s: %s"), filename, strerror (errno));
585 if (st.st_size == 0)
586 return;
587
588 buffer = (char *) xmalloc (st.st_size + 2);
589 f = fopen (filename, FOPEN_RT);
590 if (f == NULL)
591 fatal (_("cannot open: %s: %s"), filename, strerror (errno));
592
593 if (fread (buffer, 1, st.st_size, f) == 0 || ferror (f))
594 fatal (_("%s: fread failed"), filename);
595
596 fclose (f);
597 buffer [st.st_size] = '\n';
598 buffer [st.st_size + 1] = '\0';
599
600 line_count = 1;
0af11b59 601
16b2b71c
NC
602 for (line = buffer; * line != '\0'; line ++)
603 {
604 char * eol;
605 char * name;
606 char * name_end;
607 int finished = false;
608
609 for (eol = line;; eol ++)
610 {
611 switch (* eol)
612 {
613 case '\n':
614 * eol = '\0';
615 /* Cope with \n\r. */
616 if (eol[1] == '\r')
617 ++ eol;
618 finished = true;
619 break;
0af11b59 620
16b2b71c
NC
621 case '\r':
622 * eol = '\0';
623 /* Cope with \r\n. */
624 if (eol[1] == '\n')
625 ++ eol;
626 finished = true;
627 break;
0af11b59 628
16b2b71c
NC
629 case 0:
630 finished = true;
631 break;
0af11b59 632
16b2b71c
NC
633 case '#':
634 /* Line comment, Terminate the line here, in case a
635 name is present and then allow the rest of the
636 loop to find the real end of the line. */
637 * eol = '\0';
638 break;
0af11b59 639
16b2b71c
NC
640 default:
641 break;
642 }
643
644 if (finished)
645 break;
646 }
647
648 /* A name may now exist somewhere between 'line' and 'eol'.
649 Strip off leading whitespace and trailing whitespace,
650 then add it to the list. */
651 for (name = line; IS_WHITESPACE (* name); name ++)
652 ;
653 for (name_end = name;
654 (! IS_WHITESPACE (* name_end))
655 && (! IS_LINE_TERMINATOR (* name_end));
0af11b59
KH
656 name_end ++)
657 ;
16b2b71c
NC
658
659 if (! IS_LINE_TERMINATOR (* name_end))
660 {
661 char * extra;
662
663 for (extra = name_end + 1; IS_WHITESPACE (* extra); extra ++)
664 ;
665
666 if (! IS_LINE_TERMINATOR (* extra))
667 non_fatal (_("Ignoring rubbish found on line %d of %s"),
668 line_count, filename);
669 }
0af11b59 670
16b2b71c
NC
671 * name_end = '\0';
672
673 if (name_end > name)
674 add_specific_symbol (name, list);
675
676 /* Advance line pointer to end of line. The 'eol ++' in the for
677 loop above will then advance us to the start of the next line. */
678 line = eol;
679 line_count ++;
680 }
681}
682
252b5132
RH
683/* See whether a symbol should be stripped or kept based on
684 strip_specific_list and keep_symbols. */
685
686static boolean
687is_specified_symbol (name, list)
688 const char *name;
689 struct symlist *list;
690{
691 struct symlist *tmp_list;
692
693 for (tmp_list = list; tmp_list; tmp_list = tmp_list->next)
594ef5db
NC
694 if (strcmp (name, tmp_list->name) == 0)
695 return true;
696
252b5132
RH
697 return false;
698}
699
700/* See if a section is being removed. */
701
702static boolean
703is_strip_section (abfd, sec)
b4c96d0d 704 bfd *abfd ATTRIBUTE_UNUSED;
252b5132
RH
705 asection *sec;
706{
707 struct section_list *p;
708
709 if ((bfd_get_section_flags (abfd, sec) & SEC_DEBUGGING) != 0
710 && (strip_symbols == STRIP_DEBUG
711 || strip_symbols == STRIP_UNNEEDED
712 || strip_symbols == STRIP_ALL
713 || discard_locals == LOCALS_ALL
714 || convert_debugging))
715 return true;
716
f91ea849 717 if (! sections_removed && ! sections_copied)
252b5132 718 return false;
f91ea849 719
252b5132 720 p = find_section_list (bfd_get_section_name (abfd, sec), false);
f91ea849
ILT
721 if (sections_removed && p != NULL && p->remove)
722 return true;
723 if (sections_copied && (p == NULL || ! p->copy))
724 return true;
725 return false;
252b5132
RH
726}
727
728/* Choose which symbol entries to copy; put the result in OSYMS.
729 We don't copy in place, because that confuses the relocs.
730 Return the number of symbols to print. */
731
732static unsigned int
733filter_symbols (abfd, obfd, osyms, isyms, symcount)
734 bfd *abfd;
735 bfd *obfd;
736 asymbol **osyms, **isyms;
737 long symcount;
738{
739 register asymbol **from = isyms, **to = osyms;
740 long src_count = 0, dst_count = 0;
d8121479
L
741 int relocatable = (abfd->flags & (HAS_RELOC | EXEC_P | DYNAMIC))
742 == HAS_RELOC;
252b5132
RH
743
744 for (; src_count < symcount; src_count++)
745 {
746 asymbol *sym = from[src_count];
747 flagword flags = sym->flags;
748 const char *name = bfd_asymbol_name (sym);
749 int keep;
24e01a36 750 boolean undefined;
252b5132 751
57938635
AM
752 if (redefine_sym_list)
753 {
754 const char *old_name, *new_name;
755
756 old_name = bfd_asymbol_name (sym);
757 new_name = lookup_sym_redefinition (old_name);
758 name = bfd_asymbol_name (sym) = new_name;
759 }
760
252b5132
RH
761 if (change_leading_char
762 && (bfd_get_symbol_leading_char (abfd)
763 != bfd_get_symbol_leading_char (obfd))
764 && (bfd_get_symbol_leading_char (abfd) == '\0'
765 || (name[0] == bfd_get_symbol_leading_char (abfd))))
766 {
767 if (bfd_get_symbol_leading_char (obfd) == '\0')
768 name = bfd_asymbol_name (sym) = name + 1;
769 else
770 {
771 char *n;
772
773 n = xmalloc (strlen (name) + 2);
774 n[0] = bfd_get_symbol_leading_char (obfd);
775 if (bfd_get_symbol_leading_char (abfd) == '\0')
776 strcpy (n + 1, name);
777 else
778 strcpy (n + 1, name + 1);
779 name = bfd_asymbol_name (sym) = n;
780 }
781 }
782
24e01a36
L
783 undefined = bfd_is_und_section (bfd_get_section (sym));
784
252b5132
RH
785 if (remove_leading_char
786 && ((flags & BSF_GLOBAL) != 0
787 || (flags & BSF_WEAK) != 0
24e01a36 788 || undefined
252b5132
RH
789 || bfd_is_com_section (bfd_get_section (sym)))
790 && name[0] == bfd_get_symbol_leading_char (abfd))
791 name = bfd_asymbol_name (sym) = name + 1;
792
793 if (strip_symbols == STRIP_ALL)
794 keep = 0;
795 else if ((flags & BSF_KEEP) != 0 /* Used in relocation. */
796 || ((flags & BSF_SECTION_SYM) != 0
797 && ((*bfd_get_section (sym)->symbol_ptr_ptr)->flags
798 & BSF_KEEP) != 0))
799 keep = 1;
0af11b59 800 else if (relocatable /* Relocatable file. */
d8121479
L
801 && (flags & (BSF_GLOBAL | BSF_WEAK)) != 0)
802 keep = 1;
16b2b71c
NC
803 else if (bfd_decode_symclass (sym) == 'I')
804 /* Global symbols in $idata sections need to be retained
805 even if relocatable is false. External users of the
806 library containing the $idata section may reference these
807 symbols. */
af3bdff7 808 keep = 1;
252b5132
RH
809 else if ((flags & BSF_GLOBAL) != 0 /* Global symbol. */
810 || (flags & BSF_WEAK) != 0
24e01a36 811 || undefined
252b5132
RH
812 || bfd_is_com_section (bfd_get_section (sym)))
813 keep = strip_symbols != STRIP_UNNEEDED;
814 else if ((flags & BSF_DEBUGGING) != 0) /* Debugging symbol. */
815 keep = (strip_symbols != STRIP_DEBUG
816 && strip_symbols != STRIP_UNNEEDED
817 && ! convert_debugging);
af3bdff7
NC
818 else if (bfd_get_section (sym)->comdat)
819 /* COMDAT sections store special information in local
820 symbols, so we cannot risk stripping any of them. */
821 keep = 1;
252b5132
RH
822 else /* Local symbol. */
823 keep = (strip_symbols != STRIP_UNNEEDED
824 && (discard_locals != LOCALS_ALL
825 && (discard_locals != LOCALS_START_L
826 || ! bfd_is_local_label (abfd, sym))));
827
828 if (keep && is_specified_symbol (name, strip_specific_list))
829 keep = 0;
830 if (!keep && is_specified_symbol (name, keep_specific_list))
831 keep = 1;
832 if (keep && is_strip_section (abfd, bfd_get_section (sym)))
833 keep = 0;
e0c60db2 834
252b5132
RH
835 if (keep && (flags & BSF_GLOBAL) != 0
836 && (weaken || is_specified_symbol (name, weaken_specific_list)))
837 {
838 sym->flags &=~ BSF_GLOBAL;
839 sym->flags |= BSF_WEAK;
840 }
24e01a36 841 if (keep && !undefined && (flags & (BSF_GLOBAL | BSF_WEAK))
16b2b71c
NC
842 && (is_specified_symbol (name, localize_specific_list)
843 || (keepglobal_specific_list != NULL
844 && ! is_specified_symbol (name, keepglobal_specific_list))))
252b5132
RH
845 {
846 sym->flags &= ~(BSF_GLOBAL | BSF_WEAK);
847 sym->flags |= BSF_LOCAL;
848 }
849
850 if (keep)
851 to[dst_count++] = sym;
852 }
853
854 to[dst_count] = NULL;
855
856 return dst_count;
857}
858
594ef5db
NC
859/* Find the redefined name of symbol SOURCE. */
860
57938635
AM
861static const char *
862lookup_sym_redefinition (source)
863 const char *source;
864{
57938635
AM
865 struct redefine_node *list;
866
57938635 867 for (list = redefine_sym_list; list != NULL; list = list->next)
594ef5db
NC
868 if (strcmp (source, list->source) == 0)
869 return list->target;
870
871 return source;
57938635
AM
872}
873
594ef5db 874/* Add a node to a symbol redefine list. */
57938635
AM
875
876static void
877redefine_list_append (source, target)
878 const char *source;
879 const char *target;
880{
881 struct redefine_node **p;
882 struct redefine_node *list;
883 struct redefine_node *new_node;
884
885 for (p = &redefine_sym_list; (list = *p) != NULL; p = &list->next)
886 {
887 if (strcmp (source, list->source) == 0)
594ef5db
NC
888 fatal (_("%s: Multiple redefinition of symbol \"%s\""),
889 "--redefine-sym",
890 source);
57938635
AM
891
892 if (strcmp (target, list->target) == 0)
594ef5db
NC
893 fatal (_("%s: Symbol \"%s\" is target of more than one redefinition"),
894 "--redefine-sym",
895 target);
57938635
AM
896 }
897
898 new_node = (struct redefine_node *) xmalloc (sizeof (struct redefine_node));
899
900 new_node->source = strdup (source);
901 new_node->target = strdup (target);
902 new_node->next = NULL;
903
904 *p = new_node;
905}
906
252b5132
RH
907/* Keep only every `copy_byte'th byte in MEMHUNK, which is *SIZE bytes long.
908 Adjust *SIZE. */
909
910static void
911filter_bytes (memhunk, size)
912 char *memhunk;
913 bfd_size_type *size;
914{
915 char *from = memhunk + copy_byte, *to = memhunk, *end = memhunk + *size;
916
917 for (; from < end; from += interleave)
918 *to++ = *from;
594ef5db 919
b4c96d0d 920 if (*size % interleave > (bfd_size_type) copy_byte)
252b5132
RH
921 *size = (*size / interleave) + 1;
922 else
923 *size /= interleave;
924}
925
926/* Copy object file IBFD onto OBFD. */
927
928static void
929copy_object (ibfd, obfd)
930 bfd *ibfd;
931 bfd *obfd;
932{
933 bfd_vma start;
934 long symcount;
935 asection **osections = NULL;
936 bfd_size_type *gaps = NULL;
937 bfd_size_type max_gap = 0;
938 long symsize;
939 PTR dhandle;
940
23719f39
NC
941 if (ibfd->xvec->byteorder != obfd->xvec->byteorder
942 && ibfd->xvec->byteorder != BFD_ENDIAN_UNKNOWN
943 && obfd->xvec->byteorder != BFD_ENDIAN_UNKNOWN)
944 {
945 fatal (_("Unable to change endianness of input file(s)"));
946 return;
947 }
252b5132
RH
948
949 if (!bfd_set_format (obfd, bfd_get_format (ibfd)))
950 RETURN_NONFATAL (bfd_get_filename (obfd));
951
952 if (verbose)
953 printf (_("copy from %s(%s) to %s(%s)\n"),
954 bfd_get_filename (ibfd), bfd_get_target (ibfd),
955 bfd_get_filename (obfd), bfd_get_target (obfd));
956
957 if (set_start_set)
958 start = set_start;
959 else
960 start = bfd_get_start_address (ibfd);
961 start += change_start;
962
0af11b59
KH
963 /* Neither the start address nor the flags
964 need to be set for a core file. */
4dd67f29
MS
965 if (bfd_get_format (obfd) != bfd_core)
966 {
967 if (!bfd_set_start_address (obfd, start)
968 || !bfd_set_file_flags (obfd,
969 (bfd_get_file_flags (ibfd)
970 & bfd_applicable_file_flags (obfd))))
971 RETURN_NONFATAL (bfd_get_filename (ibfd));
972 }
252b5132 973
594ef5db 974 /* Copy architecture of input file to output file. */
252b5132
RH
975 if (!bfd_set_arch_mach (obfd, bfd_get_arch (ibfd),
976 bfd_get_mach (ibfd)))
977 non_fatal (_("Warning: Output file cannot represent architecture %s"),
978 bfd_printable_arch_mach (bfd_get_arch (ibfd),
979 bfd_get_mach (ibfd)));
57938635 980
252b5132
RH
981 if (!bfd_set_format (obfd, bfd_get_format (ibfd)))
982 RETURN_NONFATAL (bfd_get_filename (ibfd));
983
984 if (isympp)
985 free (isympp);
57938635 986
252b5132
RH
987 if (osympp != isympp)
988 free (osympp);
989
990 /* BFD mandates that all output sections be created and sizes set before
991 any output is done. Thus, we traverse all sections multiple times. */
992 bfd_map_over_sections (ibfd, setup_section, (void *) obfd);
993
994 if (add_sections != NULL)
995 {
996 struct section_add *padd;
997 struct section_list *pset;
998
999 for (padd = add_sections; padd != NULL; padd = padd->next)
1000 {
1001 padd->section = bfd_make_section (obfd, padd->name);
1002 if (padd->section == NULL)
1003 {
1004 non_fatal (_("can't create section `%s': %s"),
1005 padd->name, bfd_errmsg (bfd_get_error ()));
1006 status = 1;
1007 return;
1008 }
1009 else
1010 {
1011 flagword flags;
57938635 1012
252b5132
RH
1013 if (! bfd_set_section_size (obfd, padd->section, padd->size))
1014 RETURN_NONFATAL (bfd_get_filename (obfd));
1015
1016 pset = find_section_list (padd->name, false);
1017 if (pset != NULL)
1018 pset->used = true;
1019
1020 if (pset != NULL && pset->set_flags)
1021 flags = pset->flags | SEC_HAS_CONTENTS;
1022 else
1023 flags = SEC_HAS_CONTENTS | SEC_READONLY | SEC_DATA;
57938635 1024
252b5132
RH
1025 if (! bfd_set_section_flags (obfd, padd->section, flags))
1026 RETURN_NONFATAL (bfd_get_filename (obfd));
1027
1028 if (pset != NULL)
1029 {
1030 if (pset->change_vma != CHANGE_IGNORE)
1031 if (! bfd_set_section_vma (obfd, padd->section, pset->vma_val))
1032 RETURN_NONFATAL (bfd_get_filename (obfd));
57938635 1033
252b5132
RH
1034 if (pset->change_lma != CHANGE_IGNORE)
1035 {
1036 padd->section->lma = pset->lma_val;
57938635 1037
252b5132
RH
1038 if (! bfd_set_section_alignment
1039 (obfd, padd->section,
1040 bfd_section_alignment (obfd, padd->section)))
1041 RETURN_NONFATAL (bfd_get_filename (obfd));
1042 }
1043 }
1044 }
1045 }
1046 }
1047
1048 if (gap_fill_set || pad_to_set)
1049 {
1050 asection **set;
1051 unsigned int c, i;
1052
1053 /* We must fill in gaps between the sections and/or we must pad
1054 the last section to a specified address. We do this by
1055 grabbing a list of the sections, sorting them by VMA, and
1056 increasing the section sizes as required to fill the gaps.
1057 We write out the gap contents below. */
1058
1059 c = bfd_count_sections (obfd);
1060 osections = (asection **) xmalloc (c * sizeof (asection *));
1061 set = osections;
1062 bfd_map_over_sections (obfd, get_sections, (void *) &set);
1063
1064 qsort (osections, c, sizeof (asection *), compare_section_lma);
1065
1066 gaps = (bfd_size_type *) xmalloc (c * sizeof (bfd_size_type));
1067 memset (gaps, 0, c * sizeof (bfd_size_type));
1068
1069 if (gap_fill_set)
1070 {
1071 for (i = 0; i < c - 1; i++)
1072 {
1073 flagword flags;
1074 bfd_size_type size;
1075 bfd_vma gap_start, gap_stop;
1076
1077 flags = bfd_get_section_flags (obfd, osections[i]);
1078 if ((flags & SEC_HAS_CONTENTS) == 0
1079 || (flags & SEC_LOAD) == 0)
1080 continue;
1081
1082 size = bfd_section_size (obfd, osections[i]);
1083 gap_start = bfd_section_lma (obfd, osections[i]) + size;
1084 gap_stop = bfd_section_lma (obfd, osections[i + 1]);
1085 if (gap_start < gap_stop)
1086 {
1087 if (! bfd_set_section_size (obfd, osections[i],
1088 size + (gap_stop - gap_start)))
1089 {
1090 non_fatal (_("Can't fill gap after %s: %s"),
0af11b59
KH
1091 bfd_get_section_name (obfd, osections[i]),
1092 bfd_errmsg (bfd_get_error ()));
252b5132
RH
1093 status = 1;
1094 break;
1095 }
1096 gaps[i] = gap_stop - gap_start;
1097 if (max_gap < gap_stop - gap_start)
1098 max_gap = gap_stop - gap_start;
1099 }
1100 }
1101 }
1102
1103 if (pad_to_set)
1104 {
1105 bfd_vma lma;
1106 bfd_size_type size;
1107
1108 lma = bfd_section_lma (obfd, osections[c - 1]);
1109 size = bfd_section_size (obfd, osections[c - 1]);
1110 if (lma + size < pad_to)
1111 {
1112 if (! bfd_set_section_size (obfd, osections[c - 1],
1113 pad_to - lma))
1114 {
1115 non_fatal (_("Can't add padding to %s: %s"),
0af11b59
KH
1116 bfd_get_section_name (obfd, osections[c - 1]),
1117 bfd_errmsg (bfd_get_error ()));
252b5132
RH
1118 status = 1;
1119 }
1120 else
1121 {
1122 gaps[c - 1] = pad_to - (lma + size);
1123 if (max_gap < pad_to - (lma + size))
1124 max_gap = pad_to - (lma + size);
1125 }
1126 }
1127 }
1128 }
1129
594ef5db
NC
1130 /* Symbol filtering must happen after the output sections
1131 have been created, but before their contents are set. */
252b5132
RH
1132 dhandle = NULL;
1133 symsize = bfd_get_symtab_upper_bound (ibfd);
1134 if (symsize < 0)
1135 RETURN_NONFATAL (bfd_get_filename (ibfd));
57938635 1136
252b5132
RH
1137 osympp = isympp = (asymbol **) xmalloc (symsize);
1138 symcount = bfd_canonicalize_symtab (ibfd, isympp);
1139 if (symcount < 0)
1140 RETURN_NONFATAL (bfd_get_filename (ibfd));
57938635 1141
252b5132
RH
1142 if (convert_debugging)
1143 dhandle = read_debugging_info (ibfd, isympp, symcount);
57938635
AM
1144
1145 if (strip_symbols == STRIP_DEBUG
252b5132
RH
1146 || strip_symbols == STRIP_ALL
1147 || strip_symbols == STRIP_UNNEEDED
1148 || discard_locals != LOCALS_UNDEF
1149 || strip_specific_list != NULL
1150 || keep_specific_list != NULL
1151 || localize_specific_list != NULL
16b2b71c 1152 || keepglobal_specific_list != NULL
252b5132
RH
1153 || weaken_specific_list != NULL
1154 || sections_removed
f91ea849 1155 || sections_copied
252b5132
RH
1156 || convert_debugging
1157 || change_leading_char
1158 || remove_leading_char
57938635 1159 || redefine_sym_list
252b5132
RH
1160 || weaken)
1161 {
1162 /* Mark symbols used in output relocations so that they
1163 are kept, even if they are local labels or static symbols.
57938635 1164
252b5132
RH
1165 Note we iterate over the input sections examining their
1166 relocations since the relocations for the output sections
1167 haven't been set yet. mark_symbols_used_in_relocations will
1168 ignore input sections which have no corresponding output
1169 section. */
1170 if (strip_symbols != STRIP_ALL)
1171 bfd_map_over_sections (ibfd,
1172 mark_symbols_used_in_relocations,
1173 (PTR)isympp);
1174 osympp = (asymbol **) xmalloc ((symcount + 1) * sizeof (asymbol *));
1175 symcount = filter_symbols (ibfd, obfd, osympp, isympp, symcount);
1176 }
1177
1178 if (convert_debugging && dhandle != NULL)
1179 {
1180 if (! write_debugging_info (obfd, dhandle, &symcount, &osympp))
1181 {
1182 status = 1;
1183 return;
1184 }
1185 }
1186
1187 bfd_set_symtab (obfd, osympp, symcount);
1188
1189 /* This has to happen after the symbol table has been set. */
1190 bfd_map_over_sections (ibfd, copy_section, (void *) obfd);
1191
1192 if (add_sections != NULL)
1193 {
1194 struct section_add *padd;
1195
1196 for (padd = add_sections; padd != NULL; padd = padd->next)
1197 {
1198 if (! bfd_set_section_contents (obfd, padd->section,
1199 (PTR) padd->contents,
1200 (file_ptr) 0,
1201 (bfd_size_type) padd->size))
1202 RETURN_NONFATAL (bfd_get_filename (obfd));
1203 }
1204 }
1205
1206 if (gap_fill_set || pad_to_set)
1207 {
1208 bfd_byte *buf;
1209 int c, i;
1210
1211 /* Fill in the gaps. */
252b5132
RH
1212 if (max_gap > 8192)
1213 max_gap = 8192;
1214 buf = (bfd_byte *) xmalloc (max_gap);
1215 memset (buf, gap_fill, (size_t) max_gap);
1216
1217 c = bfd_count_sections (obfd);
1218 for (i = 0; i < c; i++)
1219 {
1220 if (gaps[i] != 0)
1221 {
1222 bfd_size_type left;
1223 file_ptr off;
1224
1225 left = gaps[i];
1226 off = bfd_section_size (obfd, osections[i]) - left;
594ef5db 1227
252b5132
RH
1228 while (left > 0)
1229 {
1230 bfd_size_type now;
1231
1232 if (left > 8192)
1233 now = 8192;
1234 else
1235 now = left;
1236
1237 if (! bfd_set_section_contents (obfd, osections[i], buf,
1238 off, now))
1239 RETURN_NONFATAL (bfd_get_filename (obfd));
1240
1241 left -= now;
1242 off += now;
1243 }
1244 }
1245 }
1246 }
1247
1248 /* Allow the BFD backend to copy any private data it understands
1249 from the input BFD to the output BFD. This is done last to
1250 permit the routine to look at the filtered symbol table, which is
1251 important for the ECOFF code at least. */
594ef5db 1252 if (! bfd_copy_private_bfd_data (ibfd, obfd))
252b5132
RH
1253 {
1254 non_fatal (_("%s: error copying private BFD data: %s"),
1255 bfd_get_filename (obfd),
1256 bfd_errmsg (bfd_get_error ()));
1257 status = 1;
1258 return;
1259 }
1ae8b3d2
AO
1260
1261 /* Switch to the alternate machine code. We have to do this at the
1262 very end, because we only initialize the header when we create
1263 the first section. */
1264 if (use_alt_mach_code != 0)
1265 {
1266 if (!bfd_alt_mach_code (obfd, use_alt_mach_code))
1267 non_fatal (_("unknown alternate machine code, ignored"));
1268 }
252b5132
RH
1269}
1270
4c168fa3
AM
1271#undef MKDIR
1272#if defined (_WIN32) && !defined (__CYGWIN32__)
1273#define MKDIR(DIR, MODE) mkdir (DIR)
1274#else
1275#define MKDIR(DIR, MODE) mkdir (DIR, MODE)
1276#endif
1277
252b5132
RH
1278/* Read each archive element in turn from IBFD, copy the
1279 contents to temp file, and keep the temp file handle. */
1280
1281static void
1282copy_archive (ibfd, obfd, output_target)
1283 bfd *ibfd;
1284 bfd *obfd;
1285 const char *output_target;
1286{
1287 struct name_list
1288 {
1289 struct name_list *next;
4c168fa3 1290 const char *name;
252b5132
RH
1291 bfd *obfd;
1292 } *list, *l;
1293 bfd **ptr = &obfd->archive_head;
1294 bfd *this_element;
1295 char *dir = make_tempname (bfd_get_filename (obfd));
1296
1297 /* Make a temp directory to hold the contents. */
4c168fa3 1298 if (MKDIR (dir, 0700) != 0)
252b5132
RH
1299 {
1300 fatal (_("cannot mkdir %s for archive copying (error: %s)"),
1301 dir, strerror (errno));
1302 }
1303 obfd->has_armap = ibfd->has_armap;
1304
1305 list = NULL;
1306
1307 this_element = bfd_openr_next_archived_file (ibfd, NULL);
594ef5db 1308
b667df2e
AM
1309 if (!bfd_set_format (obfd, bfd_get_format (ibfd)))
1310 RETURN_NONFATAL (bfd_get_filename (obfd));
1311
252b5132
RH
1312 while (!status && this_element != (bfd *) NULL)
1313 {
4c168fa3
AM
1314 char *output_name;
1315 bfd *output_bfd;
252b5132 1316 bfd *last_element;
8066d1a2
AS
1317 struct stat buf;
1318 int stat_status = 0;
1319
4c168fa3
AM
1320 /* Create an output file for this member. */
1321 output_name = concat (dir, "/",
1322 bfd_get_filename (this_element), (char *) 0);
1323
1324 /* If the file already exists, make another temp dir. */
1325 if (stat (output_name, &buf) >= 0)
1326 {
1327 output_name = make_tempname (output_name);
1328 if (MKDIR (output_name, 0700) != 0)
1329 {
1330 fatal (_("cannot mkdir %s for archive copying (error: %s)"),
1331 output_name, strerror (errno));
1332 }
1333 l = (struct name_list *) xmalloc (sizeof (struct name_list));
1334 l->name = output_name;
1335 l->next = list;
1336 l->obfd = NULL;
1337 list = l;
1338 output_name = concat (output_name, "/",
1339 bfd_get_filename (this_element), (char *) 0);
1340 }
1341
1342 output_bfd = bfd_openw (output_name, output_target);
8066d1a2
AS
1343 if (preserve_dates)
1344 {
1345 stat_status = bfd_stat_arch_elt (this_element, &buf);
594ef5db 1346
8066d1a2
AS
1347 if (stat_status != 0)
1348 non_fatal (_("internal stat error on %s"),
1349 bfd_get_filename (this_element));
1350 }
252b5132
RH
1351
1352 l = (struct name_list *) xmalloc (sizeof (struct name_list));
1353 l->name = output_name;
1354 l->next = list;
1355 list = l;
1356
1357 if (output_bfd == (bfd *) NULL)
1358 RETURN_NONFATAL (output_name);
1359
252b5132
RH
1360 if (bfd_check_format (this_element, bfd_object) == true)
1361 copy_object (this_element, output_bfd);
1362
1363 if (!bfd_close (output_bfd))
1364 {
1365 bfd_nonfatal (bfd_get_filename (output_bfd));
0af11b59 1366 /* Error in new object file. Don't change archive. */
252b5132
RH
1367 status = 1;
1368 }
1369
8066d1a2
AS
1370 if (preserve_dates && stat_status == 0)
1371 set_times (output_name, &buf);
1372
252b5132
RH
1373 /* Open the newly output file and attach to our list. */
1374 output_bfd = bfd_openr (output_name, output_target);
1375
1376 l->obfd = output_bfd;
1377
1378 *ptr = output_bfd;
1379 ptr = &output_bfd->next;
1380
1381 last_element = this_element;
1382
1383 this_element = bfd_openr_next_archived_file (ibfd, last_element);
1384
1385 bfd_close (last_element);
1386 }
1387 *ptr = (bfd *) NULL;
1388
1389 if (!bfd_close (obfd))
1390 RETURN_NONFATAL (bfd_get_filename (obfd));
1391
1392 if (!bfd_close (ibfd))
1393 RETURN_NONFATAL (bfd_get_filename (ibfd));
1394
1395 /* Delete all the files that we opened. */
1396 for (l = list; l != NULL; l = l->next)
1397 {
4c168fa3
AM
1398 if (l->obfd == NULL)
1399 rmdir (l->name);
1400 else
1401 {
1402 bfd_close (l->obfd);
1403 unlink (l->name);
1404 }
252b5132
RH
1405 }
1406 rmdir (dir);
1407}
1408
1409/* The top-level control. */
1410
1411static void
1412copy_file (input_filename, output_filename, input_target, output_target)
1413 const char *input_filename;
1414 const char *output_filename;
1415 const char *input_target;
1416 const char *output_target;
1417{
1418 bfd *ibfd;
1419 char **matching;
1420
1421 /* To allow us to do "strip *" without dying on the first
1422 non-object file, failures are nonfatal. */
252b5132
RH
1423 ibfd = bfd_openr (input_filename, input_target);
1424 if (ibfd == NULL)
1425 RETURN_NONFATAL (input_filename);
1426
1427 if (bfd_check_format (ibfd, bfd_archive))
1428 {
1429 bfd *obfd;
1430
1431 /* bfd_get_target does not return the correct value until
1432 bfd_check_format succeeds. */
1433 if (output_target == NULL)
1434 output_target = bfd_get_target (ibfd);
1435
1436 obfd = bfd_openw (output_filename, output_target);
1437 if (obfd == NULL)
1438 RETURN_NONFATAL (output_filename);
1439
1440 copy_archive (ibfd, obfd, output_target);
1441 }
4dd67f29
MS
1442 else if (bfd_check_format_matches (ibfd, bfd_object, &matching)
1443 || bfd_check_format_matches (ibfd, bfd_core, &matching))
252b5132
RH
1444 {
1445 bfd *obfd;
1446
1447 /* bfd_get_target does not return the correct value until
1448 bfd_check_format succeeds. */
1449 if (output_target == NULL)
1450 output_target = bfd_get_target (ibfd);
1451
1452 obfd = bfd_openw (output_filename, output_target);
1453 if (obfd == NULL)
1454 RETURN_NONFATAL (output_filename);
1455
1456 copy_object (ibfd, obfd);
1457
1458 if (!bfd_close (obfd))
1459 RETURN_NONFATAL (output_filename);
1460
1461 if (!bfd_close (ibfd))
1462 RETURN_NONFATAL (input_filename);
1463 }
1464 else
1465 {
1466 bfd_nonfatal (input_filename);
57938635 1467
252b5132
RH
1468 if (bfd_get_error () == bfd_error_file_ambiguously_recognized)
1469 {
1470 list_matching_formats (matching);
1471 free (matching);
1472 }
57938635 1473
252b5132
RH
1474 status = 1;
1475 }
1476}
1477
594ef5db
NC
1478/* Add a name to the section renaming list. */
1479
1480static void
1481add_section_rename (old_name, new_name, flags)
1482 const char * old_name;
1483 const char * new_name;
1484 flagword flags;
1485{
1486 section_rename * rename;
1487
1488 /* Check for conflicts first. */
1489 for (rename = section_rename_list; rename != NULL; rename = rename->next)
1490 if (strcmp (rename->old_name, old_name) == 0)
1491 {
1492 /* Silently ignore duplicate definitions. */
1493 if (strcmp (rename->new_name, new_name) == 0
1494 && rename->flags == flags)
1495 return;
0af11b59 1496
594ef5db
NC
1497 fatal (_("Multiple renames of section %s"), old_name);
1498 }
1499
1500 rename = (section_rename *) xmalloc (sizeof (* rename));
1501
1502 rename->old_name = old_name;
1503 rename->new_name = new_name;
1504 rename->flags = flags;
1505 rename->next = section_rename_list;
0af11b59 1506
594ef5db
NC
1507 section_rename_list = rename;
1508}
1509
1510/* Check the section rename list for a new name of the input section
1511 ISECTION. Return the new name if one is found.
1512 Also set RETURNED_FLAGS to the flags to be used for this section. */
1513
1514static const char *
1515find_section_rename (ibfd, isection, returned_flags)
1516 bfd * ibfd ATTRIBUTE_UNUSED;
1517 sec_ptr isection;
1518 flagword * returned_flags;
1519{
1520 const char * old_name = bfd_section_name (ibfd, isection);
1521 section_rename * rename;
1522
1523 /* Default to using the flags of the input section. */
1524 * returned_flags = bfd_get_section_flags (ibfd, isection);
1525
1526 for (rename = section_rename_list; rename != NULL; rename = rename->next)
1527 if (strcmp (rename->old_name, old_name) == 0)
1528 {
1529 if (rename->flags != (flagword) -1)
1530 * returned_flags = rename->flags;
1531
1532 return rename->new_name;
1533 }
1534
1535 return old_name;
1536}
1537
1538/* Create a section in OBFD with the same
1539 name and attributes as ISECTION in IBFD. */
252b5132
RH
1540
1541static void
1542setup_section (ibfd, isection, obfdarg)
1543 bfd *ibfd;
1544 sec_ptr isection;
1545 PTR obfdarg;
1546{
1547 bfd *obfd = (bfd *) obfdarg;
1548 struct section_list *p;
1549 sec_ptr osection;
1550 bfd_size_type size;
1551 bfd_vma vma;
1552 bfd_vma lma;
1553 flagword flags;
1a89cc7d 1554 const char *err;
594ef5db 1555 const char * name;
0af11b59 1556
252b5132
RH
1557 if ((bfd_get_section_flags (ibfd, isection) & SEC_DEBUGGING) != 0
1558 && (strip_symbols == STRIP_DEBUG
1559 || strip_symbols == STRIP_UNNEEDED
1560 || strip_symbols == STRIP_ALL
1561 || discard_locals == LOCALS_ALL
1562 || convert_debugging))
1563 return;
1564
1565 p = find_section_list (bfd_section_name (ibfd, isection), false);
1566 if (p != NULL)
1567 p->used = true;
1568
f91ea849
ILT
1569 if (sections_removed && p != NULL && p->remove)
1570 return;
1571 if (sections_copied && (p == NULL || ! p->copy))
252b5132
RH
1572 return;
1573
594ef5db
NC
1574 /* Get the, possibly new, name of the output section. */
1575 name = find_section_rename (ibfd, isection, & flags);
0af11b59 1576
594ef5db 1577 osection = bfd_make_section_anyway (obfd, name);
57938635 1578
252b5132
RH
1579 if (osection == NULL)
1580 {
1a89cc7d 1581 err = _("making");
252b5132
RH
1582 goto loser;
1583 }
1584
1585 size = bfd_section_size (ibfd, isection);
1586 if (copy_byte >= 0)
1587 size = (size + interleave - 1) / interleave;
1588 if (! bfd_set_section_size (obfd, osection, size))
1589 {
1a89cc7d 1590 err = _("size");
252b5132
RH
1591 goto loser;
1592 }
57938635 1593
252b5132
RH
1594 vma = bfd_section_vma (ibfd, isection);
1595 if (p != NULL && p->change_vma == CHANGE_MODIFY)
1596 vma += p->vma_val;
1597 else if (p != NULL && p->change_vma == CHANGE_SET)
1598 vma = p->vma_val;
1599 else
1600 vma += change_section_address;
57938635 1601
252b5132
RH
1602 if (! bfd_set_section_vma (obfd, osection, vma))
1603 {
1a89cc7d 1604 err = _("vma");
252b5132
RH
1605 goto loser;
1606 }
1607
1608 lma = isection->lma;
1609 if ((p != NULL) && p->change_lma != CHANGE_IGNORE)
1610 {
1611 if (p->change_lma == CHANGE_MODIFY)
1612 lma += p->lma_val;
1613 else if (p->change_lma == CHANGE_SET)
1614 lma = p->lma_val;
1615 else
1616 abort ();
1617 }
1618 else
1619 lma += change_section_address;
57938635 1620
252b5132
RH
1621 osection->lma = lma;
1622
1623 /* FIXME: This is probably not enough. If we change the LMA we
1624 may have to recompute the header for the file as well. */
1625 if (bfd_set_section_alignment (obfd,
1626 osection,
1627 bfd_section_alignment (ibfd, isection))
1628 == false)
1629 {
1a89cc7d 1630 err = _("alignment");
252b5132
RH
1631 goto loser;
1632 }
1633
252b5132 1634 if (p != NULL && p->set_flags)
17978339 1635 flags = p->flags | (flags & (SEC_HAS_CONTENTS | SEC_RELOC));
252b5132
RH
1636 if (!bfd_set_section_flags (obfd, osection, flags))
1637 {
1a89cc7d 1638 err = _("flags");
252b5132
RH
1639 goto loser;
1640 }
1641
bc408b8a
JJ
1642 /* Copy merge entity size. */
1643 osection->entsize = isection->entsize;
1644
252b5132
RH
1645 /* This used to be mangle_section; we do here to avoid using
1646 bfd_get_section_by_name since some formats allow multiple
1647 sections with the same name. */
1648 isection->output_section = osection;
1649 isection->output_offset = 0;
1650
1651 /* Allow the BFD backend to copy any private data it understands
1652 from the input section to the output section. */
1653 if (!bfd_copy_private_section_data (ibfd, isection, obfd, osection))
1654 {
1a89cc7d 1655 err = _("private data");
252b5132
RH
1656 goto loser;
1657 }
1658
594ef5db 1659 /* All went well. */
252b5132
RH
1660 return;
1661
1662loser:
1663 non_fatal (_("%s: section `%s': error in %s: %s"),
1664 bfd_get_filename (ibfd),
1665 bfd_section_name (ibfd, isection),
1666 err, bfd_errmsg (bfd_get_error ()));
1667 status = 1;
1668}
1669
1670/* Copy the data of input section ISECTION of IBFD
1671 to an output section with the same name in OBFD.
1672 If stripping then don't copy any relocation info. */
1673
1674static void
1675copy_section (ibfd, isection, obfdarg)
1676 bfd *ibfd;
1677 sec_ptr isection;
1678 PTR obfdarg;
1679{
1680 bfd *obfd = (bfd *) obfdarg;
1681 struct section_list *p;
1682 arelent **relpp;
1683 long relcount;
1684 sec_ptr osection;
1685 bfd_size_type size;
1686 long relsize;
dc156bc0 1687 flagword flags;
252b5132 1688
594ef5db
NC
1689 /* If we have already failed earlier on,
1690 do not keep on generating complaints now. */
252b5132
RH
1691 if (status != 0)
1692 return;
57938635 1693
dc156bc0
AM
1694 flags = bfd_get_section_flags (ibfd, isection);
1695 if ((flags & SEC_DEBUGGING) != 0
252b5132
RH
1696 && (strip_symbols == STRIP_DEBUG
1697 || strip_symbols == STRIP_UNNEEDED
1698 || strip_symbols == STRIP_ALL
1699 || discard_locals == LOCALS_ALL
1700 || convert_debugging))
e0c60db2 1701 return;
252b5132 1702
dc156bc0
AM
1703 if ((flags & SEC_GROUP) != 0)
1704 return;
1705
252b5132
RH
1706 p = find_section_list (bfd_section_name (ibfd, isection), false);
1707
f91ea849
ILT
1708 if (sections_removed && p != NULL && p->remove)
1709 return;
1710 if (sections_copied && (p == NULL || ! p->copy))
252b5132
RH
1711 return;
1712
1713 osection = isection->output_section;
1714 size = bfd_get_section_size_before_reloc (isection);
1715
1716 if (size == 0 || osection == 0)
1717 return;
1718
0af11b59 1719 /* Core files do not need to be relocated. */
4dd67f29
MS
1720 if (bfd_get_format (obfd) == bfd_core)
1721 relsize = 0;
1722 else
1723 relsize = bfd_get_reloc_upper_bound (ibfd, isection);
1724
252b5132
RH
1725 if (relsize < 0)
1726 RETURN_NONFATAL (bfd_get_filename (ibfd));
57938635 1727
252b5132
RH
1728 if (relsize == 0)
1729 bfd_set_reloc (obfd, osection, (arelent **) NULL, 0);
1730 else
1731 {
1732 relpp = (arelent **) xmalloc (relsize);
1733 relcount = bfd_canonicalize_reloc (ibfd, isection, relpp, isympp);
1734 if (relcount < 0)
1735 RETURN_NONFATAL (bfd_get_filename (ibfd));
57938635 1736
252b5132
RH
1737 if (strip_symbols == STRIP_ALL)
1738 {
1739 /* Remove relocations which are not in
0af11b59 1740 keep_strip_specific_list. */
252b5132
RH
1741 arelent **temp_relpp;
1742 long temp_relcount = 0;
1743 long i;
57938635 1744
252b5132
RH
1745 temp_relpp = (arelent **) xmalloc (relsize);
1746 for (i = 0; i < relcount; i++)
1747 if (is_specified_symbol
1748 (bfd_asymbol_name (*relpp [i]->sym_ptr_ptr),
1749 keep_specific_list))
1750 temp_relpp [temp_relcount++] = relpp [i];
1751 relcount = temp_relcount;
1752 free (relpp);
1753 relpp = temp_relpp;
1754 }
e0c60db2 1755
252b5132
RH
1756 bfd_set_reloc (obfd, osection,
1757 (relcount == 0 ? (arelent **) NULL : relpp), relcount);
1758 }
57938635 1759
252b5132
RH
1760 isection->_cooked_size = isection->_raw_size;
1761 isection->reloc_done = true;
1762
0af11b59 1763 if (bfd_get_section_flags (ibfd, isection) & SEC_HAS_CONTENTS
4dd67f29 1764 && bfd_get_section_flags (obfd, osection) & SEC_HAS_CONTENTS)
252b5132
RH
1765 {
1766 PTR memhunk = (PTR) xmalloc ((unsigned) size);
1767
1768 if (!bfd_get_section_contents (ibfd, isection, memhunk, (file_ptr) 0,
1769 size))
1770 RETURN_NONFATAL (bfd_get_filename (ibfd));
1771
57938635 1772 if (copy_byte >= 0)
252b5132
RH
1773 filter_bytes (memhunk, &size);
1774
1775 if (!bfd_set_section_contents (obfd, osection, memhunk, (file_ptr) 0,
1776 size))
1777 RETURN_NONFATAL (bfd_get_filename (obfd));
1778
1779 free (memhunk);
1780 }
1781 else if (p != NULL && p->set_flags && (p->flags & SEC_HAS_CONTENTS) != 0)
1782 {
1783 PTR memhunk = (PTR) xmalloc ((unsigned) size);
1784
1785 /* We don't permit the user to turn off the SEC_HAS_CONTENTS
1786 flag--they can just remove the section entirely and add it
1787 back again. However, we do permit them to turn on the
1788 SEC_HAS_CONTENTS flag, and take it to mean that the section
1789 contents should be zeroed out. */
1790
1791 memset (memhunk, 0, size);
1792 if (! bfd_set_section_contents (obfd, osection, memhunk, (file_ptr) 0,
1793 size))
1794 RETURN_NONFATAL (bfd_get_filename (obfd));
1795 free (memhunk);
1796 }
1797}
1798
1799/* Get all the sections. This is used when --gap-fill or --pad-to is
1800 used. */
1801
1802static void
1803get_sections (obfd, osection, secppparg)
b4c96d0d 1804 bfd *obfd ATTRIBUTE_UNUSED;
252b5132
RH
1805 asection *osection;
1806 PTR secppparg;
1807{
1808 asection ***secppp = (asection ***) secppparg;
1809
1810 **secppp = osection;
1811 ++(*secppp);
1812}
1813
1814/* Sort sections by VMA. This is called via qsort, and is used when
1815 --gap-fill or --pad-to is used. We force non loadable or empty
1816 sections to the front, where they are easier to ignore. */
1817
1818static int
1819compare_section_lma (arg1, arg2)
1820 const PTR arg1;
1821 const PTR arg2;
1822{
1823 const asection **sec1 = (const asection **) arg1;
1824 const asection **sec2 = (const asection **) arg2;
1825 flagword flags1, flags2;
1826
1827 /* Sort non loadable sections to the front. */
1828 flags1 = (*sec1)->flags;
1829 flags2 = (*sec2)->flags;
1830 if ((flags1 & SEC_HAS_CONTENTS) == 0
1831 || (flags1 & SEC_LOAD) == 0)
1832 {
1833 if ((flags2 & SEC_HAS_CONTENTS) != 0
1834 && (flags2 & SEC_LOAD) != 0)
1835 return -1;
1836 }
1837 else
1838 {
1839 if ((flags2 & SEC_HAS_CONTENTS) == 0
1840 || (flags2 & SEC_LOAD) == 0)
1841 return 1;
1842 }
1843
1844 /* Sort sections by LMA. */
1845 if ((*sec1)->lma > (*sec2)->lma)
1846 return 1;
1847 else if ((*sec1)->lma < (*sec2)->lma)
1848 return -1;
1849
1850 /* Sort sections with the same LMA by size. */
1851 if ((*sec1)->_raw_size > (*sec2)->_raw_size)
1852 return 1;
1853 else if ((*sec1)->_raw_size < (*sec2)->_raw_size)
1854 return -1;
1855
1856 return 0;
1857}
1858
1859/* Mark all the symbols which will be used in output relocations with
1860 the BSF_KEEP flag so that those symbols will not be stripped.
1861
1862 Ignore relocations which will not appear in the output file. */
1863
1864static void
1865mark_symbols_used_in_relocations (ibfd, isection, symbolsarg)
1866 bfd *ibfd;
1867 sec_ptr isection;
1868 PTR symbolsarg;
1869{
1870 asymbol **symbols = (asymbol **) symbolsarg;
1871 long relsize;
1872 arelent **relpp;
1873 long relcount, i;
1874
1875 /* Ignore an input section with no corresponding output section. */
1876 if (isection->output_section == NULL)
1877 return;
1878
1879 relsize = bfd_get_reloc_upper_bound (ibfd, isection);
1880 if (relsize < 0)
1881 bfd_fatal (bfd_get_filename (ibfd));
1882
1883 if (relsize == 0)
1884 return;
1885
1886 relpp = (arelent **) xmalloc (relsize);
1887 relcount = bfd_canonicalize_reloc (ibfd, isection, relpp, symbols);
1888 if (relcount < 0)
1889 bfd_fatal (bfd_get_filename (ibfd));
1890
ec5d57d5
NC
1891 /* Examine each symbol used in a relocation. If it's not one of the
1892 special bfd section symbols, then mark it with BSF_KEEP. */
252b5132
RH
1893 for (i = 0; i < relcount; i++)
1894 {
ec5d57d5
NC
1895 if (*relpp[i]->sym_ptr_ptr != bfd_com_section_ptr->symbol
1896 && *relpp[i]->sym_ptr_ptr != bfd_abs_section_ptr->symbol
1897 && *relpp[i]->sym_ptr_ptr != bfd_und_section_ptr->symbol)
1898 (*relpp[i]->sym_ptr_ptr)->flags |= BSF_KEEP;
252b5132
RH
1899 }
1900
1901 if (relpp != NULL)
1902 free (relpp);
1903}
1904
1905/* Write out debugging information. */
1906
1907static boolean
1908write_debugging_info (obfd, dhandle, symcountp, symppp)
1909 bfd *obfd;
1910 PTR dhandle;
b4c96d0d
ILT
1911 long *symcountp ATTRIBUTE_UNUSED;
1912 asymbol ***symppp ATTRIBUTE_UNUSED;
252b5132
RH
1913{
1914 if (bfd_get_flavour (obfd) == bfd_target_ieee_flavour)
1915 return write_ieee_debugging_info (obfd, dhandle);
1916
1917 if (bfd_get_flavour (obfd) == bfd_target_coff_flavour
1918 || bfd_get_flavour (obfd) == bfd_target_elf_flavour)
1919 {
1920 bfd_byte *syms, *strings;
1921 bfd_size_type symsize, stringsize;
1922 asection *stabsec, *stabstrsec;
1923
1924 if (! write_stabs_in_sections_debugging_info (obfd, dhandle, &syms,
1925 &symsize, &strings,
1926 &stringsize))
1927 return false;
1928
1929 stabsec = bfd_make_section (obfd, ".stab");
1930 stabstrsec = bfd_make_section (obfd, ".stabstr");
1931 if (stabsec == NULL
1932 || stabstrsec == NULL
1933 || ! bfd_set_section_size (obfd, stabsec, symsize)
1934 || ! bfd_set_section_size (obfd, stabstrsec, stringsize)
1935 || ! bfd_set_section_alignment (obfd, stabsec, 2)
1936 || ! bfd_set_section_alignment (obfd, stabstrsec, 0)
1937 || ! bfd_set_section_flags (obfd, stabsec,
1938 (SEC_HAS_CONTENTS
1939 | SEC_READONLY
1940 | SEC_DEBUGGING))
1941 || ! bfd_set_section_flags (obfd, stabstrsec,
1942 (SEC_HAS_CONTENTS
1943 | SEC_READONLY
1944 | SEC_DEBUGGING)))
1945 {
1946 non_fatal (_("%s: can't create debugging section: %s"),
1947 bfd_get_filename (obfd),
1948 bfd_errmsg (bfd_get_error ()));
1949 return false;
1950 }
1951
1952 /* We can get away with setting the section contents now because
1953 the next thing the caller is going to do is copy over the
1954 real sections. We may someday have to split the contents
1955 setting out of this function. */
1956 if (! bfd_set_section_contents (obfd, stabsec, syms, (file_ptr) 0,
1957 symsize)
1958 || ! bfd_set_section_contents (obfd, stabstrsec, strings,
1959 (file_ptr) 0, stringsize))
1960 {
1961 non_fatal (_("%s: can't set debugging section contents: %s"),
1962 bfd_get_filename (obfd),
1963 bfd_errmsg (bfd_get_error ()));
1964 return false;
1965 }
1966
1967 return true;
1968 }
1969
1970 non_fatal (_("%s: don't know how to write debugging information for %s"),
1971 bfd_get_filename (obfd), bfd_get_target (obfd));
1972 return false;
1973}
1974
1975static int
1976strip_main (argc, argv)
1977 int argc;
1978 char *argv[];
1979{
1980 char *input_target = NULL, *output_target = NULL;
1981 boolean show_version = false;
1982 int c, i;
1983 struct section_list *p;
1984 char *output_file = NULL;
1985
8b53311e 1986 while ((c = getopt_long (argc, argv, "I:O:F:K:N:R:o:sSpdgxXHhVv",
252b5132
RH
1987 strip_options, (int *) 0)) != EOF)
1988 {
1989 switch (c)
1990 {
1991 case 'I':
1992 input_target = optarg;
1993 break;
1994 case 'O':
1995 output_target = optarg;
1996 break;
1997 case 'F':
1998 input_target = output_target = optarg;
1999 break;
2000 case 'R':
2001 p = find_section_list (optarg, true);
2002 p->remove = true;
2003 sections_removed = true;
2004 break;
2005 case 's':
2006 strip_symbols = STRIP_ALL;
2007 break;
2008 case 'S':
2009 case 'g':
db4f6831 2010 case 'd': /* Historic BSD alias for -g. Used by early NetBSD. */
252b5132
RH
2011 strip_symbols = STRIP_DEBUG;
2012 break;
2013 case OPTION_STRIP_UNNEEDED:
2014 strip_symbols = STRIP_UNNEEDED;
2015 break;
2016 case 'K':
2017 add_specific_symbol (optarg, &keep_specific_list);
2018 break;
2019 case 'N':
2020 add_specific_symbol (optarg, &strip_specific_list);
2021 break;
2022 case 'o':
2023 output_file = optarg;
2024 break;
2025 case 'p':
2026 preserve_dates = true;
2027 break;
2028 case 'x':
2029 discard_locals = LOCALS_ALL;
2030 break;
2031 case 'X':
2032 discard_locals = LOCALS_START_L;
2033 break;
2034 case 'v':
2035 verbose = true;
2036 break;
2037 case 'V':
2038 show_version = true;
2039 break;
2040 case 0:
594ef5db
NC
2041 /* We've been given a long option. */
2042 break;
8b53311e 2043 case 'H':
252b5132
RH
2044 case 'h':
2045 strip_usage (stdout, 0);
2046 default:
2047 strip_usage (stderr, 1);
2048 }
2049 }
2050
2051 if (show_version)
2052 print_version ("strip");
2053
2054 /* Default is to strip all symbols. */
2055 if (strip_symbols == STRIP_UNDEF
2056 && discard_locals == LOCALS_UNDEF
2057 && strip_specific_list == NULL)
2058 strip_symbols = STRIP_ALL;
2059
2060 if (output_target == (char *) NULL)
2061 output_target = input_target;
2062
2063 i = optind;
2064 if (i == argc
2065 || (output_file != NULL && (i + 1) < argc))
2066 strip_usage (stderr, 1);
2067
2068 for (; i < argc; i++)
2069 {
2070 int hold_status = status;
2071 struct stat statbuf;
2072 char *tmpname;
2073
2074 if (preserve_dates)
2075 {
2076 if (stat (argv[i], &statbuf) < 0)
2077 {
2078 non_fatal (_("%s: cannot stat: %s"), argv[i], strerror (errno));
2079 continue;
2080 }
2081 }
2082
2083 if (output_file != NULL)
2084 tmpname = output_file;
2085 else
2086 tmpname = make_tempname (argv[i]);
2087 status = 0;
2088
2089 copy_file (argv[i], tmpname, input_target, output_target);
2090 if (status == 0)
2091 {
2092 if (preserve_dates)
2093 set_times (tmpname, &statbuf);
2094 if (output_file == NULL)
2095 smart_rename (tmpname, argv[i], preserve_dates);
2096 status = hold_status;
2097 }
2098 else
2099 unlink (tmpname);
2100 if (output_file == NULL)
2101 free (tmpname);
2102 }
2103
2104 return 0;
2105}
2106
2107static int
2108copy_main (argc, argv)
2109 int argc;
2110 char *argv[];
2111{
43a0748c 2112 char * binary_architecture = NULL;
252b5132
RH
2113 char *input_filename = NULL, *output_filename = NULL;
2114 char *input_target = NULL, *output_target = NULL;
2115 boolean show_version = false;
2116 boolean change_warn = true;
2117 int c;
2118 struct section_list *p;
2119 struct stat statbuf;
2120
8b53311e 2121 while ((c = getopt_long (argc, argv, "b:B:i:I:j:K:N:s:O:d:F:L:G:R:SpgxXHhVvW:",
252b5132
RH
2122 copy_options, (int *) 0)) != EOF)
2123 {
2124 switch (c)
2125 {
2126 case 'b':
2127 copy_byte = atoi (optarg);
2128 if (copy_byte < 0)
2129 fatal (_("byte number must be non-negative"));
2130 break;
57938635 2131
0af11b59
KH
2132 case 'B':
2133 binary_architecture = optarg;
2134 break;
43a0748c 2135
252b5132
RH
2136 case 'i':
2137 interleave = atoi (optarg);
2138 if (interleave < 1)
2139 fatal (_("interleave must be positive"));
2140 break;
57938635 2141
252b5132
RH
2142 case 'I':
2143 case 's': /* "source" - 'I' is preferred */
2144 input_target = optarg;
2145 break;
57938635 2146
252b5132
RH
2147 case 'O':
2148 case 'd': /* "destination" - 'O' is preferred */
2149 output_target = optarg;
2150 break;
57938635 2151
252b5132
RH
2152 case 'F':
2153 input_target = output_target = optarg;
2154 break;
57938635 2155
f91ea849
ILT
2156 case 'j':
2157 p = find_section_list (optarg, true);
2158 if (p->remove)
2159 fatal (_("%s both copied and removed"), optarg);
2160 p->copy = true;
2161 sections_copied = true;
2162 break;
57938635 2163
252b5132
RH
2164 case 'R':
2165 p = find_section_list (optarg, true);
f91ea849
ILT
2166 if (p->copy)
2167 fatal (_("%s both copied and removed"), optarg);
252b5132
RH
2168 p->remove = true;
2169 sections_removed = true;
2170 break;
57938635 2171
252b5132
RH
2172 case 'S':
2173 strip_symbols = STRIP_ALL;
2174 break;
57938635 2175
252b5132
RH
2176 case 'g':
2177 strip_symbols = STRIP_DEBUG;
2178 break;
57938635 2179
252b5132
RH
2180 case OPTION_STRIP_UNNEEDED:
2181 strip_symbols = STRIP_UNNEEDED;
2182 break;
57938635 2183
252b5132
RH
2184 case 'K':
2185 add_specific_symbol (optarg, &keep_specific_list);
2186 break;
57938635 2187
252b5132
RH
2188 case 'N':
2189 add_specific_symbol (optarg, &strip_specific_list);
2190 break;
57938635 2191
252b5132
RH
2192 case 'L':
2193 add_specific_symbol (optarg, &localize_specific_list);
2194 break;
57938635 2195
16b2b71c
NC
2196 case 'G':
2197 add_specific_symbol (optarg, &keepglobal_specific_list);
2198 break;
2199
252b5132
RH
2200 case 'W':
2201 add_specific_symbol (optarg, &weaken_specific_list);
2202 break;
57938635 2203
252b5132
RH
2204 case 'p':
2205 preserve_dates = true;
2206 break;
57938635 2207
252b5132
RH
2208 case 'x':
2209 discard_locals = LOCALS_ALL;
2210 break;
57938635 2211
252b5132
RH
2212 case 'X':
2213 discard_locals = LOCALS_START_L;
2214 break;
57938635 2215
252b5132
RH
2216 case 'v':
2217 verbose = true;
2218 break;
57938635 2219
252b5132
RH
2220 case 'V':
2221 show_version = true;
2222 break;
57938635 2223
252b5132
RH
2224 case OPTION_WEAKEN:
2225 weaken = true;
2226 break;
57938635 2227
252b5132
RH
2228 case OPTION_ADD_SECTION:
2229 {
2230 const char *s;
2231 struct stat st;
2232 struct section_add *pa;
2233 int len;
2234 char *name;
2235 FILE *f;
2236
2237 s = strchr (optarg, '=');
57938635 2238
252b5132 2239 if (s == NULL)
57938635 2240 fatal (_("bad format for %s"), "--add-section");
252b5132
RH
2241
2242 if (stat (s + 1, & st) < 0)
2243 fatal (_("cannot stat: %s: %s"), s + 1, strerror (errno));
2244
2245 pa = (struct section_add *) xmalloc (sizeof (struct section_add));
2246
2247 len = s - optarg;
2248 name = (char *) xmalloc (len + 1);
2249 strncpy (name, optarg, len);
2250 name[len] = '\0';
2251 pa->name = name;
2252
2253 pa->filename = s + 1;
2254
2255 pa->size = st.st_size;
2256
2257 pa->contents = (bfd_byte *) xmalloc (pa->size);
2258 f = fopen (pa->filename, FOPEN_RB);
57938635 2259
252b5132
RH
2260 if (f == NULL)
2261 fatal (_("cannot open: %s: %s"), pa->filename, strerror (errno));
57938635 2262
252b5132
RH
2263 if (fread (pa->contents, 1, pa->size, f) == 0
2264 || ferror (f))
2265 fatal (_("%s: fread failed"), pa->filename);
2266
2267 fclose (f);
2268
2269 pa->next = add_sections;
2270 add_sections = pa;
2271 }
2272 break;
57938635 2273
252b5132
RH
2274 case OPTION_CHANGE_START:
2275 change_start = parse_vma (optarg, "--change-start");
2276 break;
57938635 2277
252b5132
RH
2278 case OPTION_CHANGE_SECTION_ADDRESS:
2279 case OPTION_CHANGE_SECTION_LMA:
2280 case OPTION_CHANGE_SECTION_VMA:
2281 {
2282 const char *s;
2283 int len;
2284 char *name;
b4c96d0d 2285 char *option = NULL;
252b5132 2286 bfd_vma val;
b4c96d0d 2287 enum change_action what = CHANGE_IGNORE;
57938635 2288
252b5132
RH
2289 switch (c)
2290 {
b4c96d0d
ILT
2291 case OPTION_CHANGE_SECTION_ADDRESS:
2292 option = "--change-section-address";
2293 break;
2294 case OPTION_CHANGE_SECTION_LMA:
2295 option = "--change-section-lma";
2296 break;
2297 case OPTION_CHANGE_SECTION_VMA:
2298 option = "--change-section-vma";
2299 break;
252b5132 2300 }
57938635 2301
252b5132
RH
2302 s = strchr (optarg, '=');
2303 if (s == NULL)
2304 {
2305 s = strchr (optarg, '+');
2306 if (s == NULL)
2307 {
2308 s = strchr (optarg, '-');
2309 if (s == NULL)
2310 fatal (_("bad format for %s"), option);
2311 }
2312 }
2313
2314 len = s - optarg;
2315 name = (char *) xmalloc (len + 1);
2316 strncpy (name, optarg, len);
2317 name[len] = '\0';
2318
2319 p = find_section_list (name, true);
2320
2321 val = parse_vma (s + 1, option);
2322
2323 switch (*s)
2324 {
2325 case '=': what = CHANGE_SET; break;
2326 case '-': val = - val; /* Drop through. */
2327 case '+': what = CHANGE_MODIFY; break;
2328 }
57938635 2329
252b5132
RH
2330 switch (c)
2331 {
2332 case OPTION_CHANGE_SECTION_ADDRESS:
2333 p->change_vma = what;
2334 p->vma_val = val;
2335 /* Drop through. */
57938635 2336
252b5132
RH
2337 case OPTION_CHANGE_SECTION_LMA:
2338 p->change_lma = what;
2339 p->lma_val = val;
2340 break;
57938635 2341
252b5132
RH
2342 case OPTION_CHANGE_SECTION_VMA:
2343 p->change_vma = what;
2344 p->vma_val = val;
2345 break;
2346 }
2347 }
2348 break;
57938635 2349
252b5132
RH
2350 case OPTION_CHANGE_ADDRESSES:
2351 change_section_address = parse_vma (optarg, "--change-addresses");
2352 change_start = change_section_address;
2353 break;
57938635 2354
252b5132
RH
2355 case OPTION_CHANGE_WARNINGS:
2356 change_warn = true;
2357 break;
57938635 2358
252b5132
RH
2359 case OPTION_CHANGE_LEADING_CHAR:
2360 change_leading_char = true;
2361 break;
57938635 2362
252b5132
RH
2363 case OPTION_DEBUGGING:
2364 convert_debugging = true;
2365 break;
57938635 2366
252b5132
RH
2367 case OPTION_GAP_FILL:
2368 {
2369 bfd_vma gap_fill_vma;
2370
2371 gap_fill_vma = parse_vma (optarg, "--gap-fill");
2372 gap_fill = (bfd_byte) gap_fill_vma;
2373 if ((bfd_vma) gap_fill != gap_fill_vma)
2374 {
2375 char buff[20];
57938635 2376
252b5132 2377 sprintf_vma (buff, gap_fill_vma);
57938635 2378
252b5132
RH
2379 non_fatal (_("Warning: truncating gap-fill from 0x%s to 0x%x"),
2380 buff, gap_fill);
2381 }
2382 gap_fill_set = true;
2383 }
2384 break;
57938635 2385
252b5132
RH
2386 case OPTION_NO_CHANGE_WARNINGS:
2387 change_warn = false;
2388 break;
57938635 2389
252b5132
RH
2390 case OPTION_PAD_TO:
2391 pad_to = parse_vma (optarg, "--pad-to");
2392 pad_to_set = true;
2393 break;
57938635 2394
252b5132
RH
2395 case OPTION_REMOVE_LEADING_CHAR:
2396 remove_leading_char = true;
2397 break;
57938635
AM
2398
2399 case OPTION_REDEFINE_SYM:
2400 {
2401 /* Push this redefinition onto redefine_symbol_list. */
2402
2403 int len;
2404 const char *s;
2405 const char *nextarg;
2406 char *source, *target;
2407
2408 s = strchr (optarg, '=');
2409 if (s == NULL)
594ef5db 2410 fatal (_("bad format for %s"), "--redefine-sym");
57938635
AM
2411
2412 len = s - optarg;
2413 source = (char *) xmalloc (len + 1);
2414 strncpy (source, optarg, len);
2415 source[len] = '\0';
2416
2417 nextarg = s + 1;
2418 len = strlen (nextarg);
2419 target = (char *) xmalloc (len + 1);
2420 strcpy (target, nextarg);
2421
2422 redefine_list_append (source, target);
2423
2424 free (source);
2425 free (target);
2426 }
2427 break;
2428
252b5132
RH
2429 case OPTION_SET_SECTION_FLAGS:
2430 {
2431 const char *s;
2432 int len;
2433 char *name;
2434
2435 s = strchr (optarg, '=');
2436 if (s == NULL)
57938635 2437 fatal (_("bad format for %s"), "--set-section-flags");
252b5132
RH
2438
2439 len = s - optarg;
2440 name = (char *) xmalloc (len + 1);
2441 strncpy (name, optarg, len);
2442 name[len] = '\0';
2443
2444 p = find_section_list (name, true);
2445
2446 p->set_flags = true;
2447 p->flags = parse_flags (s + 1);
2448 }
2449 break;
57938635 2450
594ef5db
NC
2451 case OPTION_RENAME_SECTION:
2452 {
2453 flagword flags;
3bcfb3e4
AM
2454 const char *eq, *fl;
2455 char *old_name;
2456 char *new_name;
594ef5db
NC
2457 unsigned int len;
2458
3bcfb3e4
AM
2459 eq = strchr (optarg, '=');
2460 if (eq == NULL)
594ef5db
NC
2461 fatal (_("bad format for %s"), "--rename-section");
2462
3bcfb3e4 2463 len = eq - optarg;
594ef5db 2464 if (len == 0)
3bcfb3e4 2465 fatal (_("bad format for %s"), "--rename-section");
594ef5db
NC
2466
2467 old_name = (char *) xmalloc (len + 1);
2468 strncpy (old_name, optarg, len);
2469 old_name[len] = 0;
2470
3bcfb3e4
AM
2471 eq++;
2472 fl = strchr (eq, ',');
2473 if (fl)
594ef5db 2474 {
3bcfb3e4
AM
2475 flags = parse_flags (fl + 1);
2476 len = fl - eq;
594ef5db
NC
2477 }
2478 else
2479 {
594ef5db 2480 flags = -1;
3bcfb3e4 2481 len = strlen (eq);
594ef5db
NC
2482 }
2483
3bcfb3e4
AM
2484 if (len == 0)
2485 fatal (_("bad format for %s"), "--rename-section");
2486
2487 new_name = (char *) xmalloc (len + 1);
2488 strncpy (new_name, eq, len);
2489 new_name[len] = 0;
2490
594ef5db
NC
2491 add_section_rename (old_name, new_name, flags);
2492 }
2493 break;
2494
252b5132
RH
2495 case OPTION_SET_START:
2496 set_start = parse_vma (optarg, "--set-start");
2497 set_start_set = true;
2498 break;
57938635 2499
0af11b59
KH
2500 case OPTION_SREC_LEN:
2501 Chunk = parse_vma (optarg, "--srec-len");
2502 break;
420496c1 2503
0af11b59 2504 case OPTION_SREC_FORCES3:
420496c1 2505 S3Forced = true;
0af11b59 2506 break;
420496c1 2507
16b2b71c
NC
2508 case OPTION_STRIP_SYMBOLS:
2509 add_specific_symbols (optarg, &strip_specific_list);
2510 break;
2511
2512 case OPTION_KEEP_SYMBOLS:
2513 add_specific_symbols (optarg, &keep_specific_list);
2514 break;
2515
2516 case OPTION_LOCALIZE_SYMBOLS:
2517 add_specific_symbols (optarg, &localize_specific_list);
2518 break;
2519
2520 case OPTION_KEEPGLOBAL_SYMBOLS:
2521 add_specific_symbols (optarg, &keepglobal_specific_list);
2522 break;
2523
2524 case OPTION_WEAKEN_SYMBOLS:
2525 add_specific_symbols (optarg, &weaken_specific_list);
2526 break;
2527
1ae8b3d2
AO
2528 case OPTION_ALT_MACH_CODE:
2529 use_alt_mach_code = atoi (optarg);
2530 if (use_alt_mach_code <= 0)
2531 fatal (_("alternate machine code index must be positive"));
2532 break;
2533
252b5132
RH
2534 case 0:
2535 break; /* we've been given a long option */
57938635 2536
8b53311e 2537 case 'H':
252b5132
RH
2538 case 'h':
2539 copy_usage (stdout, 0);
57938635 2540
252b5132
RH
2541 default:
2542 copy_usage (stderr, 1);
2543 }
2544 }
2545
2546 if (show_version)
2547 print_version ("objcopy");
2548
2549 if (copy_byte >= interleave)
2550 fatal (_("byte number must be less than interleave"));
2551
2552 if (optind == argc || optind + 2 < argc)
2553 copy_usage (stderr, 1);
2554
2555 input_filename = argv[optind];
2556 if (optind + 1 < argc)
2557 output_filename = argv[optind + 1];
2558
2559 /* Default is to strip no symbols. */
2560 if (strip_symbols == STRIP_UNDEF && discard_locals == LOCALS_UNDEF)
2561 strip_symbols = STRIP_NONE;
2562
2563 if (output_target == (char *) NULL)
2564 output_target = input_target;
2565
43a0748c 2566 if (binary_architecture != (char *) NULL)
252b5132 2567 {
43a0748c 2568 if (input_target && strcmp (input_target, "binary") == 0)
0af11b59
KH
2569 {
2570 const bfd_arch_info_type * temp_arch_info;
43a0748c
NC
2571
2572 temp_arch_info = bfd_scan_arch (binary_architecture);
2573
0af11b59
KH
2574 if (temp_arch_info != NULL)
2575 bfd_external_binary_architecture = temp_arch_info->arch;
2576 else
2577 fatal (_("architecture %s unknown"), binary_architecture);
2578 }
43a0748c
NC
2579 else
2580 {
2581 non_fatal (_("Warning: input target 'binary' required for binary architecture parameter."));
2582 non_fatal (_(" Argument %s ignored"), binary_architecture);
2583 }
252b5132
RH
2584 }
2585
43a0748c
NC
2586 if (preserve_dates)
2587 if (stat (input_filename, & statbuf) < 0)
2588 fatal (_("Cannot stat: %s: %s"), input_filename, strerror (errno));
2589
252b5132
RH
2590 /* If there is no destination file then create a temp and rename
2591 the result into the input. */
2592
2593 if (output_filename == (char *) NULL)
2594 {
2595 char *tmpname = make_tempname (input_filename);
2596
2597 copy_file (input_filename, tmpname, input_target, output_target);
2598 if (status == 0)
57938635 2599 {
252b5132
RH
2600 if (preserve_dates)
2601 set_times (tmpname, &statbuf);
2602 smart_rename (tmpname, input_filename, preserve_dates);
2603 }
2604 else
2605 unlink (tmpname);
2606 }
2607 else
2608 {
2609 copy_file (input_filename, output_filename, input_target, output_target);
594ef5db 2610
252b5132
RH
2611 if (status == 0 && preserve_dates)
2612 set_times (output_filename, &statbuf);
2613 }
2614
2615 if (change_warn)
2616 {
2617 for (p = change_sections; p != NULL; p = p->next)
2618 {
2619 if (! p->used)
2620 {
2621 if (p->change_vma != CHANGE_IGNORE)
2622 {
2623 char buff [20];
2624
2625 sprintf_vma (buff, p->vma_val);
57938635 2626
252b5132 2627 /* xgettext:c-format */
57938635
AM
2628 non_fatal (_("%s %s%c0x%s never used"),
2629 "--change-section-vma",
252b5132
RH
2630 p->name,
2631 p->change_vma == CHANGE_SET ? '=' : '+',
2632 buff);
2633 }
57938635 2634
252b5132
RH
2635 if (p->change_lma != CHANGE_IGNORE)
2636 {
2637 char buff [20];
2638
2639 sprintf_vma (buff, p->lma_val);
57938635 2640
252b5132 2641 /* xgettext:c-format */
57938635
AM
2642 non_fatal (_("%s %s%c0x%s never used"),
2643 "--change-section-lma",
252b5132
RH
2644 p->name,
2645 p->change_lma == CHANGE_SET ? '=' : '+',
2646 buff);
2647 }
2648 }
2649 }
2650 }
2651
2652 return 0;
2653}
2654
65de42c0
TS
2655int main PARAMS ((int, char **));
2656
252b5132
RH
2657int
2658main (argc, argv)
2659 int argc;
2660 char *argv[];
2661{
2662#if defined (HAVE_SETLOCALE) && defined (HAVE_LC_MESSAGES)
2663 setlocale (LC_MESSAGES, "");
3882b010
L
2664#endif
2665#if defined (HAVE_SETLOCALE)
2666 setlocale (LC_CTYPE, "");
252b5132
RH
2667#endif
2668 bindtextdomain (PACKAGE, LOCALEDIR);
2669 textdomain (PACKAGE);
2670
2671 program_name = argv[0];
2672 xmalloc_set_program_name (program_name);
2673
2674 START_PROGRESS (program_name, 0);
2675
2676 strip_symbols = STRIP_UNDEF;
2677 discard_locals = LOCALS_UNDEF;
2678
2679 bfd_init ();
2680 set_default_bfd_target ();
2681
2682 if (is_strip < 0)
2683 {
2684 int i = strlen (program_name);
5af11cab
AM
2685#ifdef HAVE_DOS_BASED_FILE_SYSTEM
2686 /* Drop the .exe suffix, if any. */
2687 if (i > 4 && FILENAME_CMP (program_name + i - 4, ".exe") == 0)
2688 {
2689 i -= 4;
2690 program_name[i] = '\0';
2691 }
2692#endif
2693 is_strip = (i >= 5 && FILENAME_CMP (program_name + i - 5, "strip") == 0);
252b5132
RH
2694 }
2695
2696 if (is_strip)
2697 strip_main (argc, argv);
2698 else
2699 copy_main (argc, argv);
2700
2701 END_PROGRESS (program_name);
2702
2703 return status;
2704}
This page took 0.239226 seconds and 4 git commands to generate.