Update.
[deliverable/binutils-gdb.git] / ld / ldwrite.c
index b56119a39ac124ae410cca6183afc2371c2d3fdb..7c02ff65657eab38d7e8cb875f5894825ff59f86 100644 (file)
@@ -1,6 +1,6 @@
 /* ldwrite.c -- write out the linked file
-   Copyright (C) 1991, 92, 93, 94, 95, 96, 97, 1998
-   Free Software Foundation, Inc.
+   Copyright 1991, 1992, 1993, 1994, 1995, 1996, 1997, 1998, 2000, 2002,
+   2003, 2004, 2005, 2006, 2007 Free Software Foundation, Inc.
    Written by Steve Chamberlain sac@cygnus.com
 
 This file is part of GLD, the Gnu Linker.
@@ -17,30 +17,26 @@ 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.  */
+Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, MA 02110-1301, USA.  */
 
-#include "bfd.h"
 #include "sysdep.h"
+#include "bfd.h"
 #include "bfdlink.h"
 #include "libiberty.h"
+#include "safe-ctype.h"
 
 #include "ld.h"
 #include "ldexp.h"
 #include "ldlang.h"
 #include "ldwrite.h"
 #include "ldmisc.h"
-#include "ldgram.h"
+#include <ldgram.h>
 #include "ldmain.h"
 
-static void build_link_order PARAMS ((lang_statement_union_type *));
-static asection *clone_section PARAMS ((bfd *, asection *, int *));
-static void split_sections PARAMS ((bfd *, struct bfd_link_info *));
-
 /* Build link_order structures for the BFD linker.  */
 
 static void
-build_link_order (statement)
-     lang_statement_union_type *statement;
+build_link_order (lang_statement_union_type *statement)
 {
   switch (statement->header.type)
     {
@@ -49,7 +45,7 @@ build_link_order (statement)
        asection *output_section;
        struct bfd_link_order *link_order;
        bfd_vma value;
-       boolean big_endian = false;
+       bfd_boolean big_endian = FALSE;
 
        output_section = statement->data_statement.output_section;
        ASSERT (output_section->owner == output_bfd);
@@ -59,8 +55,8 @@ build_link_order (statement)
          einfo (_("%P%F: bfd_new_link_order failed\n"));
 
        link_order->type = bfd_data_link_order;
-       link_order->offset = statement->data_statement.output_vma;
-       link_order->u.data.contents = (bfd_byte *) xmalloc (QUAD_SIZE);
+       link_order->offset = statement->data_statement.output_offset;
+       link_order->u.data.contents = xmalloc (QUAD_SIZE);
 
        value = statement->data_statement.value;
 
@@ -70,24 +66,24 @@ build_link_order (statement)
           endianness are big endian, so we must swap here if the
           input file is little endian.  */
        if (bfd_big_endian (output_bfd))
-         big_endian = true;
+         big_endian = TRUE;
        else if (bfd_little_endian (output_bfd))
-         big_endian = false;
+         big_endian = FALSE;
        else
          {
-           boolean swap;
+           bfd_boolean swap;
 
-           swap = false;
+           swap = FALSE;
            if (command_line.endian == ENDIAN_BIG)
-             big_endian = true;
+             big_endian = TRUE;
            else if (command_line.endian == ENDIAN_LITTLE)
              {
-               big_endian = false;
-               swap = true;
+               big_endian = FALSE;
+               swap = TRUE;
              }
            else if (command_line.endian == ENDIAN_UNSET)
              {
-               big_endian = true;
+               big_endian = TRUE;
                {
                  LANG_FOR_EACH_INPUT_STATEMENT (s)
                    {
@@ -95,8 +91,8 @@ build_link_order (statement)
                        {
                          if (bfd_little_endian (s->the_bfd))
                            {
-                             big_endian = false;
-                             swap = true;
+                             big_endian = FALSE;
+                             swap = TRUE;
                            }
                          break;
                        }
@@ -194,12 +190,10 @@ build_link_order (statement)
        if (link_order == NULL)
          einfo (_("%P%F: bfd_new_link_order failed\n"));
 
-       link_order->offset = rs->output_vma;
+       link_order->offset = rs->output_offset;
        link_order->size = bfd_get_reloc_size (rs->howto);
 
-       link_order->u.reloc.p =
-         ((struct bfd_link_order_reloc *)
-          xmalloc (sizeof (struct bfd_link_order_reloc)));
+       link_order->u.reloc.p = xmalloc (sizeof (struct bfd_link_order_reloc));
 
        link_order->u.reloc.p->reloc = rs->reloc;
        link_order->u.reloc.p->addend = rs->addend_value;
@@ -224,42 +218,46 @@ build_link_order (statement)
       break;
 
     case lang_input_section_enum:
-      /* Create a new link_order in the output section with this
-        attached */
-      if (statement->input_section.ifile->just_syms_flag == false)
-       {
-         asection *i = statement->input_section.section;
-         asection *output_section = i->output_section;
+      {
+       /* Create a new link_order in the output section with this
+          attached */
+       asection *i = statement->input_section.section;
 
-         ASSERT (output_section->owner == output_bfd);
+       if (!((lang_input_statement_type *) i->owner->usrdata)->just_syms_flag
+           && (i->flags & SEC_EXCLUDE) == 0)
+         {
+           asection *output_section = i->output_section;
 
-         if ((output_section->flags & SEC_HAS_CONTENTS) != 0)
-           {
-             struct bfd_link_order *link_order;
+           ASSERT (output_section->owner == output_bfd);
 
-             link_order = bfd_new_link_order (output_bfd, output_section);
+           if ((output_section->flags & SEC_HAS_CONTENTS) != 0
+               || ((output_section->flags & SEC_LOAD) != 0
+                   && (output_section->flags & SEC_THREAD_LOCAL)))
+             {
+               struct bfd_link_order *link_order;
 
-             if (i->flags & SEC_NEVER_LOAD)
-               {
-                 /* We've got a never load section inside one which
-                    is going to be output, we'll change it into a
-                    fill link_order */
-                 link_order->type = bfd_fill_link_order;
-                 link_order->u.fill.value = 0;
-               }
-             else
-               {
-                 link_order->type = bfd_indirect_link_order;
-                 link_order->u.indirect.section = i;
-                 ASSERT (i->output_section == output_section);
-               }
-             if (i->_cooked_size)
-               link_order->size = i->_cooked_size;
-             else
-               link_order->size = bfd_get_section_size_before_reloc (i);
-             link_order->offset = i->output_offset;
-           }
-       }
+               link_order = bfd_new_link_order (output_bfd, output_section);
+
+               if (i->flags & SEC_NEVER_LOAD)
+                 {
+                   /* We've got a never load section inside one which
+                      is going to be output, we'll change it into a
+                      fill.  */
+                   link_order->type = bfd_data_link_order;
+                   link_order->u.data.contents = (unsigned char *) "";
+                   link_order->u.data.size = 1;
+                 }
+               else
+                 {
+                   link_order->type = bfd_indirect_link_order;
+                   link_order->u.indirect.section = i;
+                   ASSERT (i->output_section == output_section);
+                 }
+               link_order->size = i->size;
+               link_order->offset = i->output_offset;
+             }
+         }
+      }
       break;
 
     case lang_padding_statement_enum:
@@ -274,10 +272,11 @@ build_link_order (statement)
        if ((output_section->flags & SEC_HAS_CONTENTS) != 0)
          {
            link_order = bfd_new_link_order (output_bfd, output_section);
-           link_order->type = bfd_fill_link_order;
+           link_order->type = bfd_data_link_order;
            link_order->size = statement->padding_statement.size;
            link_order->offset = statement->padding_statement.output_offset;
-           link_order->u.fill.value = statement->padding_statement.fill;
+           link_order->u.data.contents = statement->padding_statement.fill->data;
+           link_order->u.data.size = statement->padding_statement.fill->size;
          }
       }
       break;
@@ -288,57 +287,88 @@ build_link_order (statement)
     }
 }
 
-/* Call BFD to write out the linked file.  */
-
-
-/**********************************************************************/
+/* Return true if NAME is the name of an unsplittable section. These
+   are the stabs strings, dwarf strings.  */
 
+static bfd_boolean
+unsplittable_name (const char *name)
+{
+  if (CONST_STRNEQ (name, ".stab"))
+    {
+      /* There are several stab like string sections. We pattern match on
+        ".stab...str"  */
+      unsigned len = strlen (name);
+      if (strcmp (&name[len-3], "str") == 0)
+       return TRUE;
+    }
+  else if (strcmp (name, "$GDB_STRINGS$") == 0)
+    return TRUE;
+  return FALSE;
+}
 
 /* Wander around the input sections, make sure that
    we'll never try and create an output section with more relocs
    than will fit.. Do this by always assuming the worst case, and
-   creating new output sections with all the right bits */
+   creating new output sections with all the right bits */
 #define TESTIT 1
 static asection *
-clone_section (abfd, s, count)
-     bfd *abfd;
-     asection *s;
-     int *count;
+clone_section (bfd *abfd, asection *s, const char *name, int *count)
 {
-#define SSIZE 8
-  char sname[SSIZE];           /* ??  find the name for this size */
+  char *tname;
+  char *sname;
+  unsigned int len;    
   asection *n;
   struct bfd_link_hash_entry *h;
-  /* Invent a section name - use first five
-     chars of base section name and a digit suffix */
-  do
+
+  /* Invent a section name from the section name and a dotted numeric
+     suffix.   */
+  len = strlen (name);
+  tname = xmalloc (len + 1);
+  memcpy (tname, name, len + 1);
+  /* Remove a dotted number suffix, from a previous split link. */
+  while (len && ISDIGIT (tname[len-1]))
+    len--;
+  if (len > 1 && tname[len-1] == '.')
+    /* It was a dotted number. */
+    tname[len-1] = 0;
+
+  /* We want to use the whole of the original section name for the
+     split name, but coff can be restricted to 8 character names.  */
+  if (bfd_family_coff (abfd) && strlen (tname) > 5)
     {
-      unsigned int i;
-      char b[6];
-      for (i = 0; i < sizeof (b) - 1 && s->name[i]; i++)
-       b[i] = s->name[i];
-      b[i] = 0;
-      sprintf (sname, "%s%d", b, (*count)++);
+      /* Some section names cannot be truncated, as the name is
+        used to locate some other section.  */
+      if (CONST_STRNEQ (name, ".stab")
+         || strcmp (name, "$GDB_SYMBOLS$") == 0)
+       {
+         einfo (_ ("%F%P: cannot create split section name for %s\n"), name);
+         /* Silence gcc warnings.  einfo exits, so we never reach here.  */
+         return NULL;
+       }
+      tname[5] = 0;
     }
-  while (bfd_get_section_by_name (abfd, sname));
-
-  n = bfd_make_section_anyway (abfd, xstrdup (sname));
-
-  /* Create a symbol of the same name */
-
-  h = bfd_link_hash_lookup (link_info.hash,
-                           sname, true, true, false);
+  
+  if ((sname = bfd_get_unique_section_name (abfd, tname, count)) == NULL
+      || (n = bfd_make_section_anyway (abfd, sname)) == NULL
+      || (h = bfd_link_hash_lookup (link_info.hash,
+                                   sname, TRUE, TRUE, FALSE)) == NULL)
+    {
+      einfo (_("%F%P: clone section failed: %E\n"));
+      /* Silence gcc warnings.  einfo exits, so we never reach here.  */
+      return NULL;
+    }
+  free (tname);
+  
+  /* Set up section symbol.  */
   h->type = bfd_link_hash_defined;
   h->u.def.value = 0;
-  h->u.def.section = n   ;
-
+  h->u.def.section = n;
 
   n->flags = s->flags;
   n->vma = s->vma;
   n->user_set_vma = s->user_set_vma;
   n->lma = s->lma;
-  n->_cooked_size = 0;
-  n->_raw_size = 0;
+  n->size = 0;
   n->output_offset = s->output_offset;
   n->output_section = n;
   n->orelocation = 0;
@@ -348,12 +378,11 @@ clone_section (abfd, s, count)
 }
 
 #if TESTING
-static void 
-ds (s)
-     asection *s;
+static void
+ds (asection *s)
 {
-  struct bfd_link_order *l = s->link_order_head;
-  printf ("vma %x size %x\n", s->vma, s->_raw_size);
+  struct bfd_link_order *l = s->map_head.link_order;
+  printf ("vma %x size %x\n", s->vma, s->size);
   while (l)
     {
       if (l->type == bfd_indirect_link_order)
@@ -368,26 +397,23 @@ ds (s)
     }
   printf ("\n");
 }
-dump (s, a1, a2)
-     char *s;
-     asection *a1;
-     asection *a2;
+
+dump (char *s, asection *a1, asection *a2)
 {
   printf ("%s\n", s);
   ds (a1);
   ds (a2);
 }
 
-static void 
-sanity_check (abfd)
-     bfd *abfd;
+static void
+sanity_check (bfd *abfd)
 {
   asection *s;
   for (s = abfd->sections; s; s = s->next)
     {
       struct bfd_link_order *p;
       bfd_vma prev = 0;
-      for (p = s->link_order_head; p; p = p->next)
+      for (p = s->map_head.link_order; p; p = p->next)
        {
          if (p->offset > 100000)
            abort ();
@@ -402,35 +428,33 @@ sanity_check (abfd)
 #define dump(a, b, c)
 #endif
 
-static void 
-split_sections (abfd, info)
-     bfd *abfd;
-     struct bfd_link_info *info;
+static void
+split_sections (bfd *abfd, struct bfd_link_info *info)
 {
   asection *original_sec;
   int nsecs = abfd->section_count;
   sanity_check (abfd);
-  /* look through all the original sections */
+  /* Look through all the original sections.  */
   for (original_sec = abfd->sections;
        original_sec && nsecs;
        original_sec = original_sec->next, nsecs--)
     {
-      boolean first = true;
       int count = 0;
-      int lines = 0;
-      int relocs = 0;
-      struct bfd_link_order **pp;
+      unsigned int lines = 0;
+      unsigned int relocs = 0;
+      bfd_size_type sec_size = 0;
+      struct bfd_link_order *l;
+      struct bfd_link_order *p;
       bfd_vma vma = original_sec->vma;
-      bfd_vma shift_offset = 0;
       asection *cursor = original_sec;
 
-      /* count up the relocations and line entries to see if
-        anything would be too big to fit */
-      for (pp = &(cursor->link_order_head); *pp; pp = &((*pp)->next))
+      /* Count up the relocations and line entries to see if anything
+        would be too big to fit.  Accumulate section size too.  */
+      for (l = NULL, p = cursor->map_head.link_order; p != NULL; p = l->next)
        {
-         struct bfd_link_order *p = *pp;
-         int thislines = 0;
-         int thisrelocs = 0;
+         unsigned int thislines = 0;
+         unsigned int thisrelocs = 0;
+         bfd_size_type thissize = 0;
          if (p->type == bfd_indirect_link_order)
            {
              asection *sec;
@@ -441,80 +465,94 @@ split_sections (abfd, info)
                  || info->strip == strip_some)
                thislines = sec->lineno_count;
 
-             if (info->relocateable)
+             if (info->relocatable)
                thisrelocs = sec->reloc_count;
 
+             thissize = sec->size;
+
            }
-         else if (info->relocateable
+         else if (info->relocatable
                   && (p->type == bfd_section_reloc_link_order
                       || p->type == bfd_symbol_reloc_link_order))
            thisrelocs++;
 
-         if (! first
-             && (thisrelocs + relocs > config.split_by_reloc
-                 || thislines + lines > config.split_by_reloc
-                 || config.split_by_file))
+         if (l != NULL
+             && (thisrelocs + relocs >= config.split_by_reloc
+                 || thislines + lines >= config.split_by_reloc
+                 || (thissize + sec_size >= config.split_by_file))
+             && !unsplittable_name (cursor->name))
            {
-             /* create a new section and put this link order and the
-                following link orders into it */
-             struct bfd_link_order *l = p;
-             asection *n = clone_section (abfd, cursor, &count);
-             *pp = NULL;       /* Snip off link orders from old section */
-             n->link_order_head = l;   /* attach to new section */
-             pp = &n->link_order_head;
+             /* Create a new section and put this link order and the
+                following link orders into it.  */
+             bfd_vma shift_offset;
+             asection *n;
 
-             /* change the size of the original section and
-                update the vma of the new one */
+             n = clone_section (abfd, cursor, original_sec->name, &count);
 
-             dump ("before snip", cursor, n);
+             /* Attach the link orders to the new section and snip
+                them off from the old section.  */
+             n->map_head.link_order = p;
+             n->map_tail.link_order = cursor->map_tail.link_order;
+             cursor->map_tail.link_order = l;
+             l->next = NULL;
+             l = p;
 
-             n->_raw_size = cursor->_raw_size - l->offset;
-             cursor->_raw_size = l->offset;
+             /* Change the size of the original section and
+                update the vma of the new one.  */
 
-             vma += cursor->_raw_size;
-             n->lma = n->vma = vma;
+             dump ("before snip", cursor, n);
 
-             shift_offset = l->offset;
+             shift_offset = p->offset;
+             n->size = cursor->size - shift_offset;
+             cursor->size = shift_offset;
 
-             /* run down the chain and change the output section to
-                the right one, update the offsets too */
+             vma += shift_offset;
+             n->lma = n->vma = vma;
 
-             while (l)
+             /* Run down the chain and change the output section to
+                the right one, update the offsets too.  */
+             do
                {
-                 l->offset -= shift_offset;
-                 if (l->type == bfd_indirect_link_order)
+                 p->offset -= shift_offset;
+                 if (p->type == bfd_indirect_link_order)
                    {
-                     l->u.indirect.section->output_section = n;
-                     l->u.indirect.section->output_offset = l->offset;
+                     p->u.indirect.section->output_section = n;
+                     p->u.indirect.section->output_offset = p->offset;
                    }
-                 l = l->next;
+                 p = p->next;
                }
+             while (p);
+
              dump ("after snip", cursor, n);
              cursor = n;
              relocs = thisrelocs;
              lines = thislines;
+             sec_size = thissize;
            }
          else
            {
+             l = p;
              relocs += thisrelocs;
              lines += thislines;
+             sec_size += thissize;
            }
-
-         first = false;
        }
     }
   sanity_check (abfd);
 }
-/**********************************************************************/
+
+/* Call BFD to write out the linked file.  */
+
 void
-ldwrite ()
+ldwrite (void)
 {
   /* Reset error indicator, which can typically something like invalid
-     format from openning up the .o files */
+     format from opening up the .o files.  */
   bfd_set_error (bfd_error_no_error);
   lang_for_each_statement (build_link_order);
 
-  if (config.split_by_reloc || config.split_by_file)
+  if (config.split_by_reloc != (unsigned) -1
+      || config.split_by_file != (bfd_size_type) -1)
     split_sections (output_bfd, &link_info);
   if (!bfd_final_link (output_bfd, &link_info))
     {
@@ -523,8 +561,8 @@ ldwrite ()
         out.  */
 
       if (bfd_get_error () != bfd_error_no_error)
-       einfo (_("%F%P: final link failed: %E\n"), output_bfd);
+       einfo (_("%F%P: final link failed: %E\n"));
       else
-       xexit(1);
+       xexit (1);
     }
 }
This page took 0.030558 seconds and 4 git commands to generate.