This is the mail archive of the
binutils@sourceware.org
mailing list for the binutils project.
[PATCH 2/2] objcopy: Add --update-section option.
- From: Andrew Burgess <andrew dot burgess at embecosm dot com>
- To: binutils at sourceware dot org
- Cc: Andrew Burgess <andrew dot burgess at embecosm dot com>
- Date: Mon, 23 Feb 2015 12:52:26 +0000
- Subject: [PATCH 2/2] objcopy: Add --update-section option.
- Authentication-results: sourceware.org; auth=none
- References: <cover dot 1424695434 dot git dot andrew dot burgess at embecosm dot com>
- References: <cover dot 1424695434 dot git dot andrew dot burgess at embecosm dot com>
New option for objcopy --update-section allows the contents of a section
to be updated while maintaining the section flags, and, for ELF files,
the section to segment mapping.
New test uses --dump-section and --update-section to check that a
section can be made larger and smaller with an update.
binutils/ChangeLog:
* objcopy.c (update_sections): New list.
(command_line_switch): Add OPTION_UPDATE_SECTION.
(copy_options): Add update-section.
(copy_usage): Document new option.
(copy_object): Update size and content of requested sections.
(is_update_section): New function.
(skip_section): Don't copy for updated sections.
(copy_main): Handle --update-section.
* doc/binutils.texi (objcopy): Add description of --update-section
option.
binutils/testsuite/ChangeLog:
* binutils-all/update-1.s: New file.
* binutils-all/update-2.s: New file.
* binutils-all/update-3.s: New file.
* binutils-all/update-section.exp: New file.
---
binutils/ChangeLog | 13 ++++
binutils/doc/binutils.texi | 10 +++
binutils/objcopy.c | 79 ++++++++++++++++++++
binutils/testsuite/ChangeLog | 7 ++
binutils/testsuite/binutils-all/update-1.s | 2 +
binutils/testsuite/binutils-all/update-2.s | 2 +
binutils/testsuite/binutils-all/update-3.s | 3 +
binutils/testsuite/binutils-all/update-section.exp | 84 ++++++++++++++++++++++
8 files changed, 200 insertions(+)
create mode 100644 binutils/testsuite/binutils-all/update-1.s
create mode 100644 binutils/testsuite/binutils-all/update-2.s
create mode 100644 binutils/testsuite/binutils-all/update-3.s
create mode 100644 binutils/testsuite/binutils-all/update-section.exp
diff --git a/binutils/ChangeLog b/binutils/ChangeLog
index 2e5985f..ad9113d 100644
--- a/binutils/ChangeLog
+++ b/binutils/ChangeLog
@@ -1,5 +1,18 @@
2015-02-20 Andrew Burgess <andrew.burgess@embecosm.com>
+ * objcopy.c (update_sections): New list.
+ (command_line_switch): Add OPTION_UPDATE_SECTION.
+ (copy_options): Add update-section.
+ (copy_usage): Document new option.
+ (copy_object): Update size and content of requested sections.
+ (is_update_section): New function.
+ (skip_section): Don't copy for updated sections.
+ (copy_main): Handle --update-section.
+ * doc/binutils.texi (objcopy): Add description of --update-section
+ option.
+
+2015-02-20 Andrew Burgess <andrew.burgess@embecosm.com>
+
* objcopy.c (init_section_add): New function.
(section_add_load_file): New function.
(copy_main): Make use of new functions.
diff --git a/binutils/doc/binutils.texi b/binutils/doc/binutils.texi
index a5bfd4c..9a26a12 100644
--- a/binutils/doc/binutils.texi
+++ b/binutils/doc/binutils.texi
@@ -1091,6 +1091,7 @@ objcopy [@option{-F} @var{bfdname}|@option{--target=}@var{bfdname}]
[@option{--set-section-flags} @var{sectionpattern}=@var{flags}]
[@option{--add-section} @var{sectionname}=@var{filename}]
[@option{--dump-section} @var{sectionname}=@var{filename}]
+ [@option{--update-section} @var{sectionname}=@var{filename}]
[@option{--rename-section} @var{oldname}=@var{newname}[,@var{flags}]]
[@option{--long-section-names} @{enable,disable,keep@}]
[@option{--change-leading-char}] [@option{--remove-leading-char}]
@@ -1489,6 +1490,15 @@ that it does not create a formatted file, it just dumps the contents
as raw binary data, without applying any relocations. The option can
be specified more than once.
+@item --update-section @var{sectionname}=@var{filename}
+Replace the contents of section named @var{sectionname} with the
+contents of file @var{filename}, replacing any existing section
+contents. The size of the section will be adjusted to the size of the
+file. The section flags for @var{sectionname} will be unchanged. For
+ELF format files the section to segment mapping will also remain
+unchanged, something which is not possible using
+@option{--remove-section} followed by @option{--add-section}.
+
@item --rename-section @var{oldname}=@var{newname}[,@var{flags}]
Rename a section from @var{oldname} to @var{newname}, optionally
changing the section's flags to @var{flags} in the process. This has
diff --git a/binutils/objcopy.c b/binutils/objcopy.c
index eb26320..7365bb2 100644
--- a/binutils/objcopy.c
+++ b/binutils/objcopy.c
@@ -186,6 +186,9 @@ struct section_add
/* List of sections to add to the output BFD. */
static struct section_add *add_sections;
+/* List of sections to update in the output BFD. */
+static struct section_add *update_sections;
+
/* List of sections to dump from the output BFD. */
static struct section_add *dump_sections;
@@ -262,6 +265,7 @@ static enum long_section_name_handling long_section_names = KEEP;
enum command_line_switch
{
OPTION_ADD_SECTION=150,
+ OPTION_UPDATE_SECTION,
OPTION_DUMP_SECTION,
OPTION_CHANGE_ADDRESSES,
OPTION_CHANGE_LEADING_CHAR,
@@ -361,6 +365,7 @@ static struct option copy_options[] =
{
{"add-gnu-debuglink", required_argument, 0, OPTION_ADD_GNU_DEBUGLINK},
{"add-section", required_argument, 0, OPTION_ADD_SECTION},
+ {"update-section", required_argument, 0, OPTION_UPDATE_SECTION},
{"adjust-start", required_argument, 0, OPTION_CHANGE_START},
{"adjust-vma", required_argument, 0, OPTION_CHANGE_ADDRESSES},
{"adjust-section-vma", required_argument, 0, OPTION_CHANGE_SECTION_ADDRESS},
@@ -553,6 +558,9 @@ copy_usage (FILE *stream, int exit_status)
--set-section-flags <name>=<flags>\n\
Set section <name>'s properties to <flags>\n\
--add-section <name>=<file> Add section <name> found in <file> to output\n\
+ --update-section <name>=<file>\n\
+ Update contents of section <name> with\n\
+ contents found in <file>.\n\
--dump-section <name>=<file> Dump the contents of section <name> into <file>\n\
--rename-section <old>=<new>[,<flags>] Rename section <old> to <new>\n\
--long-section-names {enable|disable|keep}\n\
@@ -1865,6 +1873,31 @@ copy_object (bfd *ibfd, bfd *obfd, const bfd_arch_info_type *input_arch)
}
}
+ if (update_sections != NULL)
+ {
+ struct section_add *pupdate;
+
+ for (pupdate = update_sections;
+ pupdate != NULL;
+ pupdate = pupdate->next)
+ {
+ pupdate->section = bfd_get_section_by_name (obfd, pupdate->name);
+ if (pupdate->section == NULL)
+ {
+ bfd_nonfatal_message (NULL, obfd, NULL,
+ _("can't update section '%s'"),
+ pupdate->name);
+ return FALSE;
+ }
+
+ if (! bfd_set_section_size (obfd, pupdate->section, pupdate->size))
+ {
+ bfd_nonfatal_message (NULL, obfd, pupdate->section, NULL);
+ return FALSE;
+ }
+ }
+ }
+
if (dump_sections != NULL)
{
struct section_add * pdump;
@@ -2150,6 +2183,24 @@ copy_object (bfd *ibfd, bfd *obfd, const bfd_arch_info_type *input_arch)
}
}
+ if (update_sections != NULL)
+ {
+ struct section_add *pupdate;
+
+ for (pupdate = update_sections;
+ pupdate != NULL;
+ pupdate = pupdate->next)
+ {
+ if (! bfd_set_section_contents (obfd, pupdate->section,
+ pupdate->contents,
+ 0, pupdate->size))
+ {
+ bfd_nonfatal_message (NULL, obfd, pupdate->section, NULL);
+ return FALSE;
+ }
+ }
+ }
+
if (gnu_debuglink_filename != NULL)
{
if (! bfd_fill_in_gnu_debuglink_section
@@ -2861,6 +2912,25 @@ loser:
bfd_nonfatal_message (NULL, obfd, osection, err);
}
+static bfd_boolean
+is_update_section (bfd *abfd ATTRIBUTE_UNUSED, asection *sec)
+{
+ if (update_sections != NULL)
+ {
+ struct section_add *pupdate;
+
+ for (pupdate = update_sections;
+ pupdate != NULL;
+ pupdate = pupdate->next)
+ {
+ if (strcmp (sec->name, pupdate->name) == 0)
+ return TRUE;
+ }
+ }
+
+ return FALSE;
+}
+
/* Return TRUE if input section ISECTION should be skipped. */
static bfd_boolean
@@ -2881,6 +2951,9 @@ skip_section (bfd *ibfd, sec_ptr isection)
if (is_strip_section (ibfd, isection))
return TRUE;
+ if (is_update_section (ibfd, isection))
+ return TRUE;
+
flags = bfd_get_section_flags (ibfd, isection);
if ((flags & SEC_GROUP) != 0)
return TRUE;
@@ -3795,6 +3868,12 @@ copy_main (int argc, char *argv[])
section_add_load_file (add_sections);
break;
+ case OPTION_UPDATE_SECTION:
+ update_sections = init_section_add (optarg, update_sections,
+ "--update-section");
+ section_add_load_file (update_sections);
+ break;
+
case OPTION_DUMP_SECTION:
dump_sections = init_section_add (optarg, dump_sections,
"--dump-section");
diff --git a/binutils/testsuite/ChangeLog b/binutils/testsuite/ChangeLog
index c73a6c3..05d71db 100644
--- a/binutils/testsuite/ChangeLog
+++ b/binutils/testsuite/ChangeLog
@@ -1,3 +1,10 @@
+2015-02-23 Andrew Burgess <andrew.burgess@embecosm.com>
+
+ * binutils-all/update-1.s: New file.
+ * binutils-all/update-2.s: New file.
+ * binutils-all/update-3.s: New file.
+ * binutils-all/update-section.exp: New file.
+
2015-01-01 Alan Modra <amodra@gmail.com>
Update year range in copyright notice of all files.
diff --git a/binutils/testsuite/binutils-all/update-1.s b/binutils/testsuite/binutils-all/update-1.s
new file mode 100644
index 0000000..84df14f
--- /dev/null
+++ b/binutils/testsuite/binutils-all/update-1.s
@@ -0,0 +1,2 @@
+ .section ".data", "aw"
+ .word 1, 1, 1, 1
diff --git a/binutils/testsuite/binutils-all/update-2.s b/binutils/testsuite/binutils-all/update-2.s
new file mode 100644
index 0000000..f0577a2
--- /dev/null
+++ b/binutils/testsuite/binutils-all/update-2.s
@@ -0,0 +1,2 @@
+ .section ".data", "aw"
+ .word 2, 2, 2, 2, 2, 2
diff --git a/binutils/testsuite/binutils-all/update-3.s b/binutils/testsuite/binutils-all/update-3.s
new file mode 100644
index 0000000..f902c26
--- /dev/null
+++ b/binutils/testsuite/binutils-all/update-3.s
@@ -0,0 +1,3 @@
+ .section ".data", "aw"
+ .word 3, 3
+
diff --git a/binutils/testsuite/binutils-all/update-section.exp b/binutils/testsuite/binutils-all/update-section.exp
new file mode 100644
index 0000000..7cc056e
--- /dev/null
+++ b/binutils/testsuite/binutils-all/update-section.exp
@@ -0,0 +1,84 @@
+# Copyright (C) 2015 Free Software Foundation, Inc.
+
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License as published by
+# the Free Software Foundation; either version 3 of the License, or
+# (at your option) any later version.
+#
+# This program is distributed in the hope that it will be useful,
+# but WITHOUT ANY WARRANTY; without even the implied warranty of
+# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+# GNU General Public License for more details.
+#
+# You should have received a copy of the GNU General Public License
+# along with this program; if not, write to the Free Software
+# Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA.
+
+if { [is_remote host] } then {
+ return
+}
+
+send_user "Version [binutil_version $OBJCOPY]"
+
+proc do_assemble {srcfile} {
+ global srcdir
+ global subdir
+ set objfile [regsub "\.s^" srcfile ".o"]
+ if {![binutils_assemble $srcdir/$subdir/${srcfile} tmpdir/${objfile}]} then {
+ return 0;
+ }
+ return 1;
+}
+
+proc do_objcopy {objfile extraflags} {
+ global OBJCOPY
+ global OBJCOPYFLAGS
+
+ set testname "objcopy $extraflags ${objfile}"
+ set got [binutils_run $OBJCOPY "$OBJCOPYFLAGS ${extraflags} tmpdir/${objfile}"]
+ if ![string match "" $got] then {
+ fail "objcopy ($testname)"
+ return 0
+ }
+ return 1
+}
+
+proc do_compare {file1 file2} {
+ set src1 "tmpdir/${file1}"
+ set src2 "tmpdir/${file2}"
+ set status [remote_exec build cmp "${src1} ${src2}"]
+ set exec_output [lindex $status 1]
+ set exec_output [prune_warnings $exec_output]
+
+ set testname "compare ${file1} ${file2}"
+ if [string match "" $exec_output] then {
+ pass "objcopy ($testname)"
+ } else {
+ send_log "$exec_output\n"
+ verbose "$exec_output" 1
+ fail "objcopy ($testname)"
+ return 0
+ }
+ return 1
+}
+
+#
+# Start Of Tests
+#
+
+if { ![do_assemble update-1.s]
+ || ![do_assemble update-2.s]
+ || ![do_assemble update-3.s] } then {
+ unsupported "update-section.exp"
+ return
+}
+
+# If any of these return false then a FAIL will already have been reported.
+if { ![do_objcopy update-1.o "--dump-section .data=tmpdir/dumped-contents"]
+ || ![do_objcopy update-2.o "--update-section .data=tmpdir/dumped-contents"]
+ || ![do_objcopy update-3.o "--update-section .data=tmpdir/dumped-contents"] } then {
+ return
+}
+
+do_compare update-1.o update-2.o
+do_compare update-1.o update-3.o
--
2.2.2