* script-sections.cc (Sort_output_sections::script_compare):
authorIan Lance Taylor <ian@airs.com>
Mon, 14 Mar 2011 15:22:16 +0000 (15:22 +0000)
committerIan Lance Taylor <ian@airs.com>
Mon, 14 Mar 2011 15:22:16 +0000 (15:22 +0000)
Rename from is_before, change return type.
(Sort_output_sections::operator()): Adjust accordingly.

gold/ChangeLog
gold/script-sections.cc

index 986f16ec9f56c4a64e2d22d8b640c0ee7d312a7a..ed3dc6c3416386bab39d41b66a394b879ba74017 100644 (file)
@@ -1,3 +1,9 @@
+2011-03-14  Ian Lance Taylor  <iant@google.com>
+
+       * script-sections.cc (Sort_output_sections::script_compare):
+       Rename from is_before, change return type.
+       (Sort_output_sections::operator()): Adjust accordingly.
+
 2011-03-11  Jeffrey Yasskin  <jyasskin@google.com>
 
        PR gold/12572
index bdbd811ebbaeb98dd40dacad742c81c4a6b8fbc4..10af8e11675c92a5a98d12a48970dd68ca0f965e 100644 (file)
@@ -3561,8 +3561,8 @@ class Sort_output_sections
   operator()(const Output_section* os1, const Output_section* os2) const;
 
  private:
-  bool
-  is_before(const Output_section* os1, const Output_section* os2) const;
+  int
+  script_compare(const Output_section* os1, const Output_section* os2) const;
 
  private:
   const Script_sections::Sections_elements* elements_;
@@ -3586,6 +3586,12 @@ Sort_output_sections::operator()(const Output_section* os1,
   if (os1->address() != os2->address())
     return os1->address() < os2->address();
 
+  // If the linker script says which of these sections is first, go
+  // with what it says.
+  int i = this->script_compare(os1, os2);
+  if (i != 0)
+    return i < 0;
+
   // Sort PROGBITS before NOBITS.
   bool nobits1 = os1->type() == elfcpp::SHT_NOBITS;
   bool nobits2 = os2->type() == elfcpp::SHT_NOBITS;
@@ -3604,38 +3610,46 @@ Sort_output_sections::operator()(const Output_section* os1,
     return true;
   if (!os1->is_noload() && os2->is_noload())
     return true;
-  
-  // The sections have the same address. Check the section positions 
-  // in accordance with the linker script.
-  return this->is_before(os1, os2);
+
+  // The sections seem practically identical.  Sort by name to get a
+  // stable sort.
+  return os1->name() < os2->name();
 }
 
-// Return true if OS1 comes before OS2 in ELEMENTS_.  This ensures
-// that we keep empty sections in the order in which they appear in a
-// linker script.
+// Return -1 if OS1 comes before OS2 in ELEMENTS_, 1 if comes after, 0
+// if either OS1 or OS2 is not mentioned.  This ensures that we keep
+// empty sections in the order in which they appear in a linker
+// script.
 
-bool
-Sort_output_sections::is_before(const Output_section* os1,
-                               const Output_section* os2) const
+int
+Sort_output_sections::script_compare(const Output_section* os1,
+                                    const Output_section* os2) const
 {
   if (this->elements_ == NULL)
-    return false;
+    return 0;
 
+  bool found_os1 = false;
+  bool found_os2 = false;
   for (Script_sections::Sections_elements::const_iterator
         p = this->elements_->begin();
        p != this->elements_->end();
        ++p)
     {
-      if (os1 == (*p)->get_output_section())
+      if (os2 == (*p)->get_output_section())
        {
-         for (++p; p != this->elements_->end(); ++p)
-           if (os2 == (*p)->get_output_section())
-             return true;
-         break;
+         if (found_os1)
+           return -1;
+         found_os2 = true;
+       }
+      else if (os1 == (*p)->get_output_section())
+       {
+         if (found_os2)
+           return 1;
+         found_os1 = true;
        }
     }
 
-  return false;
+  return 0;
 }
 
 // Return whether OS is a BSS section.  This is a SHT_NOBITS section.