|  | // cref.cc -- cross reference for gold | 
|  |  | 
|  | // Copyright (C) 2008-2023 Free Software Foundation, Inc. | 
|  | // Written by Ian Lance Taylor <iant@google.com>. | 
|  |  | 
|  | // This file is part of gold. | 
|  |  | 
|  | // 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. | 
|  |  | 
|  | #include "gold.h" | 
|  |  | 
|  | #include <cerrno> | 
|  | #include <cstdio> | 
|  | #include <cstring> | 
|  | #include <map> | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #include "object.h" | 
|  | #include "archive.h" | 
|  | #include "symtab.h" | 
|  | #include "cref.h" | 
|  |  | 
|  | namespace gold | 
|  | { | 
|  |  | 
|  | // Class Cref_inputs.  This is used to hold the list of input files | 
|  | // for cross referencing. | 
|  |  | 
|  | class Cref_inputs | 
|  | { | 
|  | public: | 
|  | Cref_inputs() | 
|  | : objects_(), archives_(), current_(&this->objects_) | 
|  | { } | 
|  |  | 
|  | // Add an input object file. | 
|  | void | 
|  | add_object(Object* object); | 
|  |  | 
|  | // Start adding an archive.  We support nested archives for future | 
|  | // flexibility. | 
|  | void | 
|  | add_archive_start(Archive*); | 
|  |  | 
|  | // Finish adding an archive. | 
|  | void | 
|  | add_archive_stop(Archive*); | 
|  |  | 
|  | // Report symbol counts. | 
|  | void | 
|  | print_symbol_counts(const Symbol_table*, FILE*) const; | 
|  |  | 
|  | // Print a cross reference table. | 
|  | void | 
|  | print_cref(const Symbol_table*, FILE*) const; | 
|  |  | 
|  | private: | 
|  | // A list of input objects. | 
|  | typedef std::vector<Object*> Objects; | 
|  |  | 
|  | // Information we record for an archive. | 
|  | struct Archive_info | 
|  | { | 
|  | // Archive name. | 
|  | std::string name; | 
|  | // List of objects included from the archive. | 
|  | Objects* objects; | 
|  | // Number of archive members. | 
|  | size_t member_count; | 
|  | }; | 
|  |  | 
|  | // A mapping from the name of an archive to the list of objects in | 
|  | // that archive. | 
|  | typedef std::map<std::string, Archive_info> Archives; | 
|  |  | 
|  | // For --cref, we build a cross reference table which maps from | 
|  | // symbols to lists of objects.  The symbols are sorted | 
|  | // alphabetically. | 
|  |  | 
|  | class Cref_table_compare | 
|  | { | 
|  | public: | 
|  | bool | 
|  | operator()(const Symbol*, const Symbol*) const; | 
|  | }; | 
|  |  | 
|  | typedef std::map<const Symbol*, Objects*, Cref_table_compare> Cref_table; | 
|  |  | 
|  | // Report symbol counts for a list of Objects. | 
|  | void | 
|  | print_objects_symbol_counts(const Symbol_table*, FILE*, const Objects*) const; | 
|  |  | 
|  | // Report symbol counts for an object. | 
|  | void | 
|  | print_object_symbol_counts(const Symbol_table*, FILE*, const Object*) const; | 
|  |  | 
|  | // Gather cross reference information. | 
|  | void | 
|  | gather_cref(const Objects*, Cref_table*) const; | 
|  |  | 
|  | // List of input objects. | 
|  | Objects objects_; | 
|  | // List of input archives.  This is a mapping from the archive file | 
|  | // name to the list of objects. | 
|  | Archives archives_; | 
|  | // The list to which we are currently adding objects. | 
|  | Objects* current_; | 
|  | }; | 
|  |  | 
|  | // Add an object. | 
|  |  | 
|  | void | 
|  | Cref_inputs::add_object(Object* object) | 
|  | { | 
|  | this->current_->push_back(object); | 
|  | } | 
|  |  | 
|  | // Start adding an archive. | 
|  |  | 
|  | void | 
|  | Cref_inputs::add_archive_start(Archive* archive) | 
|  | { | 
|  | gold_assert(this->current_ == &this->objects_); | 
|  | if (this->archives_.find(archive->name()) == this->archives_.end()) | 
|  | { | 
|  | Archive_info* pai = &this->archives_[archive->name()]; | 
|  | pai->name = archive->filename(); | 
|  | pai->objects = new Objects(); | 
|  | pai->member_count = archive->count_members(); | 
|  | } | 
|  | this->current_ = this->archives_[archive->name()].objects; | 
|  | } | 
|  |  | 
|  | // Stop adding an archive. | 
|  |  | 
|  | void | 
|  | Cref_inputs::add_archive_stop(Archive*) | 
|  | { | 
|  | gold_assert(this->current_ != &this->objects_); | 
|  | this->current_ = &this->objects_; | 
|  | } | 
|  |  | 
|  | // Report symbol counts for an object. | 
|  |  | 
|  | void | 
|  | Cref_inputs::print_object_symbol_counts(const Symbol_table* symtab, | 
|  | FILE* f, | 
|  | const Object* object) const | 
|  | { | 
|  | size_t defined, used; | 
|  | object->get_global_symbol_counts(symtab, &defined, &used); | 
|  | fprintf(f, "symbols %s %zu %zu\n", object->name().c_str(), defined, used); | 
|  | } | 
|  |  | 
|  | // Report symbol counts for a list of inputs. | 
|  |  | 
|  | void | 
|  | Cref_inputs::print_objects_symbol_counts(const Symbol_table* symtab, | 
|  | FILE* f, | 
|  | const Objects* objects) const | 
|  | { | 
|  | for (Objects::const_iterator p = objects->begin(); | 
|  | p != objects->end(); | 
|  | ++p) | 
|  | this->print_object_symbol_counts(symtab, f, *p); | 
|  | } | 
|  |  | 
|  | // Print symbol counts.  This implements --print-symbol-counts.  This | 
|  | // is intended to be easily read by a program.  This outputs a series | 
|  | // of lines.  There are two different types of lines. | 
|  |  | 
|  | // The first is "symbols FILENAME DEFINED USED".  FILENAME is the name | 
|  | // of an object file included in the link; for an archive, this will | 
|  | // be ARCHIVEFILENAME(MEMBERNAME).  DEFINED is the number of symbols | 
|  | // which the object file defines.  USED is the number of symbols which | 
|  | // are used in the final output; this is the number of symbols which | 
|  | // appear in the final output table as having been defined by this | 
|  | // object.  These numbers will be different when weak symbols are | 
|  | // used, and they will be different for dynamic objects. | 
|  |  | 
|  | // The second is "archives FILENAME MEMBERS USED".  FILENAME is the | 
|  | // name of an archive file included in the link.  MEMBERS is the | 
|  | // number of members of the archive.  USED is the number of archive | 
|  | // members included in the link. | 
|  |  | 
|  | void | 
|  | Cref_inputs::print_symbol_counts(const Symbol_table* symtab, FILE* f) const | 
|  | { | 
|  | this->print_objects_symbol_counts(symtab, f, &this->objects_); | 
|  | for (Archives::const_iterator p = this->archives_.begin(); | 
|  | p != this->archives_.end(); | 
|  | ++p) | 
|  | { | 
|  | fprintf(f, "archive %s %zu %zu\n", p->second.name.c_str(), | 
|  | p->second.member_count, p->second.objects->size()); | 
|  | this->print_objects_symbol_counts(symtab, f, p->second.objects); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Sort symbols for the cross reference table. | 
|  |  | 
|  | bool | 
|  | Cref_inputs::Cref_table_compare::operator()(const Symbol* s1, | 
|  | const Symbol* s2) const | 
|  | { | 
|  | int i = strcmp(s1->name(), s2->name()); | 
|  | if (i != 0) | 
|  | return i < 0; | 
|  |  | 
|  | if (s1->version() == NULL) | 
|  | { | 
|  | if (s2->version() != NULL) | 
|  | return true; | 
|  | } | 
|  | else if (s2->version() == NULL) | 
|  | return false; | 
|  | else | 
|  | { | 
|  | i = strcmp(s1->version(), s2->version()); | 
|  | if (i != 0) | 
|  | return i < 0; | 
|  | } | 
|  |  | 
|  | // We should never have two different symbols with the same name and | 
|  | // version, where one doesn't forward to the other. | 
|  | if (s1 == s2) | 
|  | return false; | 
|  | if (s1->is_forwarder() && !s2->is_forwarder()) | 
|  | return true; | 
|  | if (!s1->is_forwarder() && s2->is_forwarder()) | 
|  | return false; | 
|  | gold_unreachable(); | 
|  | } | 
|  |  | 
|  | // Gather cross reference information from a list of inputs. | 
|  |  | 
|  | void | 
|  | Cref_inputs::gather_cref(const Objects* objects, Cref_table* table) const | 
|  | { | 
|  | for (Objects::const_iterator po = objects->begin(); | 
|  | po != objects->end(); | 
|  | ++po) | 
|  | { | 
|  | const Object::Symbols* symbols = (*po)->get_global_symbols(); | 
|  | if (symbols == NULL) | 
|  | continue; | 
|  | for (Object::Symbols::const_iterator ps = symbols->begin(); | 
|  | ps != symbols->end(); | 
|  | ++ps) | 
|  | { | 
|  | const Symbol* sym = *ps; | 
|  | if (sym == NULL) | 
|  | continue; | 
|  | Objects* const onull = NULL; | 
|  | std::pair<Cref_table::iterator, bool> ins = | 
|  | table->insert(std::make_pair(sym, onull)); | 
|  | Cref_table::iterator pc = ins.first; | 
|  | if (ins.second) | 
|  | pc->second = new Objects(); | 
|  | if (sym->source() == Symbol::FROM_OBJECT | 
|  | && sym->object() == *po | 
|  | && sym->is_defined()) | 
|  | pc->second->insert(pc->second->begin(), *po); | 
|  | else | 
|  | pc->second->push_back(*po); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // The column where the file name starts in a cross reference table. | 
|  |  | 
|  | static const size_t filecol = 50; | 
|  |  | 
|  | // Print a cross reference table. | 
|  |  | 
|  | void | 
|  | Cref_inputs::print_cref(const Symbol_table*, FILE* f) const | 
|  | { | 
|  | Cref_table table; | 
|  | this->gather_cref(&this->objects_, &table); | 
|  | for (Archives::const_iterator p = this->archives_.begin(); | 
|  | p != this->archives_.end(); | 
|  | ++p) | 
|  | this->gather_cref(p->second.objects, &table); | 
|  |  | 
|  | for (Cref_table::const_iterator pc = table.begin(); | 
|  | pc != table.end(); | 
|  | ++pc) | 
|  | { | 
|  | // If all the objects are dynamic, skip this symbol. | 
|  | const Symbol* sym = pc->first; | 
|  | const Objects* objects = pc->second; | 
|  | Objects::const_iterator po; | 
|  | for (po = objects->begin(); po != objects->end(); ++po) | 
|  | if (!(*po)->is_dynamic()) | 
|  | break; | 
|  | if (po == objects->end()) | 
|  | continue; | 
|  |  | 
|  | std::string s = sym->demangled_name(); | 
|  | if (sym->version() != NULL) | 
|  | { | 
|  | s += '@'; | 
|  | if (sym->is_default()) | 
|  | s += '@'; | 
|  | s += sym->version(); | 
|  | } | 
|  |  | 
|  | fputs(s.c_str(), f); | 
|  |  | 
|  | size_t len = s.length(); | 
|  |  | 
|  | for (po = objects->begin(); po != objects->end(); ++po) | 
|  | { | 
|  | int n = len < filecol ? filecol - len : 1; | 
|  | fprintf(f, "%*c%s\n", n, ' ', (*po)->name().c_str()); | 
|  | len = 0; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // Class Cref. | 
|  |  | 
|  | // Make sure the Cref_inputs object has been created. | 
|  |  | 
|  | void | 
|  | Cref::need_inputs() | 
|  | { | 
|  | if (this->inputs_ == NULL) | 
|  | this->inputs_ = new Cref_inputs(); | 
|  | } | 
|  |  | 
|  | // Add an input object file. | 
|  |  | 
|  | void | 
|  | Cref::add_object(Object* object) | 
|  | { | 
|  | this->need_inputs(); | 
|  | this->inputs_->add_object(object); | 
|  | } | 
|  |  | 
|  | // Start adding an archive. | 
|  |  | 
|  | void | 
|  | Cref::add_archive_start(Archive* archive) | 
|  | { | 
|  | this->need_inputs(); | 
|  | this->inputs_->add_archive_start(archive); | 
|  | } | 
|  |  | 
|  | // Stop adding an archive. | 
|  |  | 
|  | void | 
|  | Cref::add_archive_stop(Archive* archive) | 
|  | { | 
|  | this->inputs_->add_archive_stop(archive); | 
|  | } | 
|  |  | 
|  | // Print symbol counts. | 
|  |  | 
|  | void | 
|  | Cref::print_symbol_counts(const Symbol_table* symtab) const | 
|  | { | 
|  | if (parameters->options().user_set_print_symbol_counts() | 
|  | && this->inputs_ != NULL) | 
|  | { | 
|  | FILE* f; | 
|  | if (strcmp(parameters->options().print_symbol_counts(), "-") == 0) | 
|  | f = stdout; | 
|  | else | 
|  | { | 
|  | f = fopen(parameters->options().print_symbol_counts(), "w"); | 
|  | if (f == NULL) | 
|  | gold_error(_("cannot open symbol count file %s: %s"), | 
|  | parameters->options().print_symbol_counts(), | 
|  | strerror(errno)); | 
|  | } | 
|  | if (f != NULL) | 
|  | this->inputs_->print_symbol_counts(symtab, f); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Print a cross reference table. | 
|  |  | 
|  | void | 
|  | Cref::print_cref(const Symbol_table* symtab, FILE* f) const | 
|  | { | 
|  | fprintf(f, _("\nCross Reference Table\n\n")); | 
|  | const char* msg = _("Symbol"); | 
|  | int len = filecol - strlen(msg); | 
|  | fprintf(f, "%s%*c%s\n", msg, len, ' ', _("File")); | 
|  |  | 
|  | if (parameters->options().cref() && this->inputs_ != NULL) | 
|  | this->inputs_->print_cref(symtab, f); | 
|  | } | 
|  |  | 
|  | } // End namespace gold. |