diff options
Diffstat (limited to 'linkers/rld-outputter.cpp')
-rw-r--r-- | linkers/rld-outputter.cpp | 469 |
1 files changed, 469 insertions, 0 deletions
diff --git a/linkers/rld-outputter.cpp b/linkers/rld-outputter.cpp new file mode 100644 index 0000000..ff9032c --- /dev/null +++ b/linkers/rld-outputter.cpp @@ -0,0 +1,469 @@ +/* + * Copyright (c) 2011, Chris Johns <chrisj@rtems.org> + * + * Permission to use, copy, modify, and/or distribute this software for any + * purpose with or without fee is hereby granted, provided that the above + * copyright notice and this permission notice appear in all copies. + * + * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES + * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF + * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR + * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES + * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN + * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF + * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. + */ +/** + * @file + * + * @ingroup rtems_ld + * + * @brief RTEMS Linker. + * + */ + +#if HAVE_CONFIG_H +#include "config.h" +#endif + +#include <fstream> +#include <iostream> + +#include <errno.h> +#include <string.h> + +#include <rld.h> +#include <rld-rap.h> + +#include <sys/types.h> +#include <sys/stat.h> +#include <unistd.h> +#include "rld-process.h" + +namespace rld +{ + namespace outputter + { + int unlink (const char* path) + { +#if _WIN32 + return ::remove(path); +#else + return ::unlink (path); +#endif + } + + int link (const char* path1, const char* path2) + { +#if _WIN32 + return ::rename(path1, path2); +#else + return ::link (path1, path2); +#endif + } + + const std::string + script_text (const std::string& entry, + const std::string& exit, + const files::object_list& dependents, + const files::cache& cache, + bool not_in_archive) + { + std::ostringstream out; + files::object_list objects; + files::object_list dep_copy (dependents); + + cache.get_objects (objects); + objects.merge (dep_copy); + objects.unique (); + + if (rld::verbose () >= RLD_VERBOSE_INFO) + std::cout << " E: " << entry << std::endl; + + out << "E: " << entry << std::endl; + + if (!exit.empty ()) + { + if (rld::verbose () >= RLD_VERBOSE_INFO) + std::cout << " e: " << exit << std::endl; + out << "e: " << exit << std::endl; + } + + for (files::object_list::iterator oi = objects.begin (); + oi != objects.end (); + ++oi) + { + files::object& obj = *(*oi); + std::string name = obj.name ().basename (); + + if (not_in_archive) + { + size_t pos = name.find (':'); + if (pos != std::string::npos) + name[pos] = '_'; + pos = name.find ('@'); + if (pos != std::string::npos) + name = name.substr (0, pos); + } + + if (rld::verbose () >= RLD_VERBOSE_INFO) + std::cout << " o: " << name << std::endl; + + out << "o:" << name << std::endl; + + symbols::symtab& unresolved = obj.unresolved_symbols (); + + int count = 0; + for (symbols::symtab::iterator ursi = unresolved.begin (); + ursi != unresolved.begin (); + ++ursi) + { + symbols::symbol& urs = *((*ursi).second); + + ++count; + + if (rld::verbose () >= RLD_VERBOSE_INFO) + std::cout << " u: " << count << ':' << urs.name () << std::endl; + + out << " u:" << count << ':' << urs.name () << std::endl; + } + } + + return out.str (); + } + + void + metadata_object (files::object& metadata, + const std::string& entry, + const std::string& exit, + const files::object_list& dependents, + const files::cache& cache) + { + if (rld::verbose () >= RLD_VERBOSE_INFO) + std::cout << "metadata: " << metadata.name ().full () << std::endl; + + const std::string script = + script_text (entry, exit, dependents, cache, true); + + metadata.open (true); + metadata.begin (); + + elf::file& elf = metadata.elf (); + + elf.set_header (ET_EXEC, + elf::object_class (), + elf::object_datatype (), + elf::object_machine_type ()); + + elf::section md (elf, + elf.section_count () + 1, + ".rtemsmd", + SHT_STRTAB, + 1, + 0, + 0, + 0, + script.length ()); + + md.add_data (ELF_T_BYTE, + 1, + script.length (), + (void*) script.c_str ()); + + elf.add (md); + elf.write (); + + metadata.end (); + metadata.close (); + } + + void + archive (const std::string& name, + const std::string& entry, + const std::string& exit, + const files::object_list& dependents, + const files::cache& cache) + { + if (rld::verbose () >= RLD_VERBOSE_INFO) + std::cout << "outputter:archive: " << name + << ", dependents: " << dependents.size () << std::endl; + + std::string ext = files::extension (name); + std::string mdname = + name.substr (0, name.length () - ext.length ()) + "-metadata.o"; + + files::object metadata (mdname); + + metadata_object (metadata, entry, exit, dependents, cache); + + files::object_list dep_copy (dependents); + files::object_list objects; + + cache.get_objects (objects); + objects.merge (dep_copy); + objects.push_front (&metadata); + objects.unique (); + + files::archive arch (name); + arch.create (objects); + } + + void + archivera (const std::string& name, + const files::object_list& dependents, + files::cache& cache, + bool ra_exist, + bool ra_rap) + { + files::object_list dep_copy (dependents); + files::object_list objects; + + if (rld::verbose () >= RLD_VERBOSE_INFO) + std::cout << "outputter:archivera: " << name + << ", dependents: " << dependents.size () << std::endl; + + objects.clear (); + + files::object_list::iterator oli; + for (oli = dep_copy.begin (); oli != dep_copy.end (); ++oli) + { + files::object& object = *(*oli); + + if (ra_rap) + objects.push_back (&object); + else + { + if (object.get_archive ()) + objects.push_back (&object); + } + } + + bool exist = false; + files::object_list objects_tmp; + files::objects& objs = cache.get_objects (); + objects_tmp.clear (); + for (files::objects::iterator obi = objs.begin (); + obi != objs.end (); + ++obi) + { + files::object* obj = (*obi).second; + exist = false; + + /** + * Replace the elf object file in ra file with elf object file + * in collected object files, if exist. + */ + if (!ra_rap) + { + for (oli = objects.begin (); oli != objects.end (); ++oli) + { + files::object& object = *(*oli); + if (obj->name ().oname () == object.name ().oname ()) + { + exist = true; + break; + } + } + } + + if (!exist) + objects_tmp.push_back (obj); + } + + objects.merge (objects_tmp); + objects.unique (); + + if (objects.size ()) + { + if (ra_exist) + { + std::string new_name = "rld_XXXXXX"; + files::archive arch (new_name); + struct stat sb; + + arch.create (objects); + + if ((::stat (name.c_str (), &sb) >= 0) && S_ISREG (sb.st_mode)) + { + if (unlink (name.c_str ()) < 0) + std::cerr << "error: unlinking temp file: " << name << std::endl; + } + if (link (new_name.c_str (), name.c_str ()) < 0) + { + std::cerr << "error: linking temp file: " << name << std::endl; + } + if ((::stat (new_name.c_str (), &sb) >= 0) && S_ISREG (sb.st_mode)) + { + if (unlink (new_name.c_str ()) < 0) + std::cerr << "error: unlinking temp file: " << new_name << std::endl; + } + } + else + { + /* Create */ + files::archive arch (name); + arch.create (objects); + } + } + } + + void + script (const std::string& name, + const std::string& entry, + const std::string& exit, + const files::object_list& dependents, + const files::cache& cache) + { + if (rld::verbose () >= RLD_VERBOSE_INFO) + std::cout << "outputter:script: " << name << std::endl; + + std::fstream out (name.c_str (), + std::ios_base::out | std::ios_base::trunc); + + /* + * Tag for the shell to use. + */ + out << "!# rls" << std::endl; + + try + { + out << script_text (entry, exit, dependents, cache, false); + } + catch (...) + { + out.close (); + throw; + } + + out.close (); + } + + void + elf_application (const std::string& name, + const std::string& entry, + const std::string& exit, + const files::object_list& dependents, + const files::cache& cache) + { + if (rld::verbose () >= RLD_VERBOSE_INFO) + std::cout << "outputter:application: " << name << std::endl; + + files::object_list dep_copy (dependents); + files::object_list objects; + std::string header; + std::string script; + files::image app (name); + + header = "RELF,00000000,0001,none,00000000\n"; + header += '\0'; + + script = script_text (entry, exit, dependents, cache, true); + + cache.get_objects (objects); + objects.merge (dep_copy); + objects.unique (); + + app.open (true); + app.write (header.c_str (), header.size ()); + + #define APP_BUFFER_SIZE (128 * 1024) + + uint8_t* buffer = 0; + + try + { + buffer = new uint8_t[APP_BUFFER_SIZE]; + + for (files::object_list::iterator oi = objects.begin (); + oi != objects.end (); + ++oi) + { + files::object& obj = *(*oi); + + obj.open (); + + try + { + obj.seek (0); + + size_t in_size = obj.name ().size (); + + while (in_size) + { + size_t reading = + in_size < APP_BUFFER_SIZE ? in_size : APP_BUFFER_SIZE; + + app.write (buffer, obj.read (buffer, reading)); + + in_size -= reading; + } + } + catch (...) + { + obj.close (); + throw; + } + + obj.close (); + } + } + catch (...) + { + delete [] buffer; + app.close (); + throw; + } + + delete [] buffer; + + app.close (); + } + + bool in_archive (files::object* object) + { + if (object->get_archive ()) + return true; + return false; + } + + void + application (const std::string& name, + const std::string& entry, + const std::string& exit, + const files::object_list& dependents, + const files::cache& cache, + const symbols::table& symbols, + bool one_file) + { + if (rld::verbose () >= RLD_VERBOSE_INFO) + std::cout << "outputter:application: " << name << std::endl; + + files::object_list dep_copy (dependents); + files::object_list objects; + files::image app (name); + + if (!one_file) + dep_copy.remove_if (in_archive); + + cache.get_objects (objects); + objects.merge (dep_copy); + objects.sort (); + objects.unique (); + + app.open (true); + + try + { + rap::write (app, entry, exit, objects, symbols); + } + catch (...) + { + app.close (); + throw; + } + + app.close (); + } + + } +} |