2017-11-02 05:16:06 +08:00
|
|
|
//===- llvm-objcopy.cpp ---------------------------------------------------===//
|
2017-08-01 08:33:58 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
2017-11-02 05:16:06 +08:00
|
|
|
|
2017-08-01 08:33:58 +08:00
|
|
|
#include "llvm-objcopy.h"
|
|
|
|
#include "Object.h"
|
2017-11-02 05:16:06 +08:00
|
|
|
#include "llvm/ADT/STLExtras.h"
|
|
|
|
#include "llvm/ADT/StringRef.h"
|
|
|
|
#include "llvm/ADT/Twine.h"
|
|
|
|
#include "llvm/BinaryFormat/ELF.h"
|
|
|
|
#include "llvm/Object/Binary.h"
|
|
|
|
#include "llvm/Object/ELFObjectFile.h"
|
|
|
|
#include "llvm/Object/ELFTypes.h"
|
|
|
|
#include "llvm/Object/Error.h"
|
|
|
|
#include "llvm/Support/Casting.h"
|
2017-08-01 08:33:58 +08:00
|
|
|
#include "llvm/Support/CommandLine.h"
|
2017-11-02 05:16:06 +08:00
|
|
|
#include "llvm/Support/Compiler.h"
|
|
|
|
#include "llvm/Support/Error.h"
|
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
|
|
|
#include "llvm/Support/ErrorOr.h"
|
2017-08-01 08:33:58 +08:00
|
|
|
#include "llvm/Support/FileOutputBuffer.h"
|
2017-11-02 05:16:06 +08:00
|
|
|
#include "llvm/Support/ManagedStatic.h"
|
2017-08-01 08:33:58 +08:00
|
|
|
#include "llvm/Support/PrettyStackTrace.h"
|
|
|
|
#include "llvm/Support/Signals.h"
|
2017-11-02 05:16:06 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
#include <algorithm>
|
|
|
|
#include <cassert>
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <functional>
|
|
|
|
#include <iterator>
|
2017-08-01 08:33:58 +08:00
|
|
|
#include <memory>
|
|
|
|
#include <string>
|
|
|
|
#include <system_error>
|
2017-11-02 05:16:06 +08:00
|
|
|
#include <utility>
|
2017-08-01 08:33:58 +08:00
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
using namespace object;
|
|
|
|
using namespace ELF;
|
|
|
|
|
|
|
|
// The name this program was invoked as.
|
|
|
|
static StringRef ToolName;
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
2017-10-12 07:54:34 +08:00
|
|
|
LLVM_ATTRIBUTE_NORETURN void error(Twine Message) {
|
2017-08-01 08:33:58 +08:00
|
|
|
errs() << ToolName << ": " << Message << ".\n";
|
|
|
|
errs().flush();
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
LLVM_ATTRIBUTE_NORETURN void reportError(StringRef File, std::error_code EC) {
|
|
|
|
assert(EC);
|
|
|
|
errs() << ToolName << ": '" << File << "': " << EC.message() << ".\n";
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2017-11-02 05:16:06 +08:00
|
|
|
LLVM_ATTRIBUTE_NORETURN void reportError(StringRef File, Error E) {
|
2017-08-01 08:33:58 +08:00
|
|
|
assert(E);
|
|
|
|
std::string Buf;
|
|
|
|
raw_string_ostream OS(Buf);
|
|
|
|
logAllUnhandledErrors(std::move(E), OS, "");
|
|
|
|
OS.flush();
|
|
|
|
errs() << ToolName << ": '" << File << "': " << Buf;
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
2017-11-02 05:16:06 +08:00
|
|
|
} // end namespace llvm
|
|
|
|
|
|
|
|
static cl::opt<std::string> InputFilename(cl::Positional, cl::desc("<input>"));
|
|
|
|
static cl::opt<std::string> OutputFilename(cl::Positional, cl::desc("<output>"),
|
2017-12-16 04:17:55 +08:00
|
|
|
cl::init("-"));
|
2017-11-02 05:16:06 +08:00
|
|
|
static cl::opt<std::string>
|
2017-11-15 04:36:04 +08:00
|
|
|
OutputFormat("O", cl::desc("Set output format to one of the following:"
|
2017-08-05 05:09:26 +08:00
|
|
|
"\n\tbinary"));
|
2017-11-02 05:16:06 +08:00
|
|
|
static cl::list<std::string> ToRemove("remove-section",
|
2017-11-15 04:36:04 +08:00
|
|
|
cl::desc("Remove <section>"),
|
|
|
|
cl::value_desc("section"));
|
2017-11-02 05:16:06 +08:00
|
|
|
static cl::alias ToRemoveA("R", cl::desc("Alias for remove-section"),
|
|
|
|
cl::aliasopt(ToRemove));
|
2017-11-28 02:56:01 +08:00
|
|
|
static cl::opt<bool> StripAll(
|
|
|
|
"strip-all",
|
|
|
|
cl::desc(
|
|
|
|
"Removes non-allocated sections other than .gnu.warning* sections"));
|
|
|
|
static cl::opt<bool>
|
|
|
|
StripAllGNU("strip-all-gnu",
|
|
|
|
cl::desc("Removes symbol, relocation, and debug information"));
|
2017-12-01 04:14:53 +08:00
|
|
|
static cl::list<std::string> Keep("keep", cl::desc("Keep <section>"),
|
|
|
|
cl::value_desc("section"));
|
|
|
|
static cl::list<std::string> OnlyKeep("only-keep",
|
|
|
|
cl::desc("Remove all but <section>"),
|
|
|
|
cl::value_desc("section"));
|
|
|
|
static cl::alias OnlyKeepA("j", cl::desc("Alias for only-keep"),
|
|
|
|
cl::aliasopt(OnlyKeep));
|
2017-11-14 06:13:08 +08:00
|
|
|
static cl::opt<bool> StripDebug("strip-debug",
|
|
|
|
cl::desc("Removes all debug information"));
|
2017-11-02 05:16:06 +08:00
|
|
|
static cl::opt<bool> StripSections("strip-sections",
|
|
|
|
cl::desc("Remove all section headers"));
|
2017-12-16 04:17:55 +08:00
|
|
|
static cl::opt<bool>
|
|
|
|
StripNonAlloc("strip-non-alloc",
|
|
|
|
cl::desc("Remove all non-allocated sections"));
|
2017-11-04 02:58:41 +08:00
|
|
|
static cl::opt<bool>
|
2017-11-15 04:36:04 +08:00
|
|
|
StripDWO("strip-dwo", cl::desc("Remove all DWARF .dwo sections from file"));
|
2017-11-04 02:58:41 +08:00
|
|
|
static cl::opt<bool> ExtractDWO(
|
|
|
|
"extract-dwo",
|
2017-11-15 04:36:04 +08:00
|
|
|
cl::desc("Remove all sections that are not DWARF .dwo sections from file"));
|
2017-11-04 02:58:41 +08:00
|
|
|
static cl::opt<std::string>
|
|
|
|
SplitDWO("split-dwo",
|
2017-11-15 04:36:04 +08:00
|
|
|
cl::desc("Equivalent to extract-dwo on the input file to "
|
2017-11-04 02:58:41 +08:00
|
|
|
"<dwo-file>, then strip-dwo on the input file"),
|
|
|
|
cl::value_desc("dwo-file"));
|
2017-12-19 08:47:30 +08:00
|
|
|
static cl::list<std::string> AddSection(
|
|
|
|
"add-section",
|
|
|
|
cl::desc("Make a section named <section> with the contents of <file>."),
|
|
|
|
cl::value_desc("section=file"));
|
2018-01-06 03:19:09 +08:00
|
|
|
static cl::opt<bool> LocalizeHidden(
|
|
|
|
"localize-hidden",
|
|
|
|
cl::desc(
|
|
|
|
"Mark all symbols that have hidden or internal visibility as local"));
|
2018-01-26 06:15:14 +08:00
|
|
|
static cl::opt<std::string>
|
|
|
|
AddGnuDebugLink("add-gnu-debuglink",
|
|
|
|
cl::desc("adds a .gnu_debuglink for <debug-file>"),
|
|
|
|
cl::value_desc("debug-file"));
|
2017-10-12 02:09:18 +08:00
|
|
|
|
2017-11-02 05:16:06 +08:00
|
|
|
using SectionPred = std::function<bool(const SectionBase &Sec)>;
|
2017-08-01 08:33:58 +08:00
|
|
|
|
2017-12-16 04:17:55 +08:00
|
|
|
bool IsDWOSection(const SectionBase &Sec) { return Sec.Name.endswith(".dwo"); }
|
2017-11-04 02:58:41 +08:00
|
|
|
|
2018-01-26 05:03:38 +08:00
|
|
|
template <class ELFT>
|
|
|
|
bool OnlyKeepDWOPred(const Object<ELFT> &Obj, const SectionBase &Sec) {
|
2017-11-04 02:58:41 +08:00
|
|
|
// We can't remove the section header string table.
|
2018-01-26 05:03:38 +08:00
|
|
|
if (&Sec == Obj.getSectionHeaderStrTab())
|
2017-11-04 02:58:41 +08:00
|
|
|
return false;
|
|
|
|
// Short of keeping the string table we want to keep everything that is a DWO
|
|
|
|
// section and remove everything else.
|
|
|
|
return !IsDWOSection(Sec);
|
|
|
|
}
|
|
|
|
|
2018-01-26 05:03:38 +08:00
|
|
|
template <class ELFT>
|
|
|
|
void WriteObjectFile(const Object<ELFT> &Obj, StringRef File) {
|
|
|
|
std::unique_ptr<FileOutputBuffer> Buffer;
|
|
|
|
Expected<std::unique_ptr<FileOutputBuffer>> BufferOrErr =
|
|
|
|
FileOutputBuffer::create(File, Obj.totalSize(),
|
|
|
|
FileOutputBuffer::F_executable);
|
|
|
|
handleAllErrors(BufferOrErr.takeError(), [](const ErrorInfoBase &) {
|
|
|
|
error("failed to open " + OutputFilename);
|
|
|
|
});
|
|
|
|
Buffer = std::move(*BufferOrErr);
|
|
|
|
|
|
|
|
Obj.write(*Buffer);
|
|
|
|
if (auto E = Buffer->commit())
|
|
|
|
reportError(File, errorToErrorCode(std::move(E)));
|
2018-01-26 04:24:17 +08:00
|
|
|
}
|
2017-11-04 02:58:41 +08:00
|
|
|
|
2018-01-26 05:03:38 +08:00
|
|
|
template <class ELFT>
|
|
|
|
void SplitDWOToFile(const ELFObjectFile<ELFT> &ObjFile, StringRef File) {
|
|
|
|
// Construct a second output file for the DWO sections.
|
|
|
|
ELFObject<ELFT> DWOFile(ObjFile);
|
|
|
|
|
|
|
|
DWOFile.removeSections([&](const SectionBase &Sec) {
|
|
|
|
return OnlyKeepDWOPred<ELFT>(DWOFile, Sec);
|
|
|
|
});
|
|
|
|
DWOFile.finalize();
|
|
|
|
WriteObjectFile(DWOFile, File);
|
2017-11-04 02:58:41 +08:00
|
|
|
}
|
|
|
|
|
2017-12-01 04:14:53 +08:00
|
|
|
// This function handles the high level operations of GNU objcopy including
|
|
|
|
// handling command line options. It's important to outline certain properties
|
|
|
|
// we expect to hold of the command line operations. Any operation that "keeps"
|
|
|
|
// should keep regardless of a remove. Additionally any removal should respect
|
|
|
|
// any previous removals. Lastly whether or not something is removed shouldn't
|
|
|
|
// depend a) on the order the options occur in or b) on some opaque priority
|
|
|
|
// system. The only priority is that keeps/copies overrule removes.
|
2018-01-26 05:03:38 +08:00
|
|
|
template <class ELFT> void CopyBinary(const ELFObjectFile<ELFT> &ObjFile) {
|
|
|
|
std::unique_ptr<Object<ELFT>> Obj;
|
2017-10-12 02:09:18 +08:00
|
|
|
|
2018-01-26 05:03:38 +08:00
|
|
|
if (!OutputFormat.empty() && OutputFormat != "binary")
|
|
|
|
error("invalid output format '" + OutputFormat + "'");
|
|
|
|
if (!OutputFormat.empty() && OutputFormat == "binary")
|
|
|
|
Obj = llvm::make_unique<BinaryObject<ELFT>>(ObjFile);
|
|
|
|
else
|
|
|
|
Obj = llvm::make_unique<ELFObject<ELFT>>(ObjFile);
|
|
|
|
|
|
|
|
if (!SplitDWO.empty())
|
|
|
|
SplitDWOToFile<ELFT>(ObjFile, SplitDWO.getValue());
|
2017-11-04 02:58:41 +08:00
|
|
|
|
2018-01-06 03:19:09 +08:00
|
|
|
// Localize:
|
|
|
|
|
|
|
|
if (LocalizeHidden) {
|
2018-01-26 05:03:38 +08:00
|
|
|
Obj->getSymTab()->localize([](const Symbol &Sym) {
|
2018-01-06 03:19:09 +08:00
|
|
|
return Sym.Visibility == STV_HIDDEN || Sym.Visibility == STV_INTERNAL;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-10-12 02:09:18 +08:00
|
|
|
SectionPred RemovePred = [](const SectionBase &) { return false; };
|
|
|
|
|
2017-12-01 04:14:53 +08:00
|
|
|
// Removes:
|
|
|
|
|
2017-10-11 02:47:09 +08:00
|
|
|
if (!ToRemove.empty()) {
|
2017-10-12 02:09:18 +08:00
|
|
|
RemovePred = [&](const SectionBase &Sec) {
|
2017-10-11 07:02:43 +08:00
|
|
|
return std::find(std::begin(ToRemove), std::end(ToRemove), Sec.Name) !=
|
|
|
|
std::end(ToRemove);
|
2017-10-12 02:09:18 +08:00
|
|
|
};
|
2017-10-11 02:47:09 +08:00
|
|
|
}
|
2017-10-12 02:09:18 +08:00
|
|
|
|
2017-11-04 02:58:41 +08:00
|
|
|
if (StripDWO || !SplitDWO.empty())
|
2017-11-04 04:57:09 +08:00
|
|
|
RemovePred = [RemovePred](const SectionBase &Sec) {
|
2017-11-04 02:58:41 +08:00
|
|
|
return IsDWOSection(Sec) || RemovePred(Sec);
|
|
|
|
};
|
|
|
|
|
|
|
|
if (ExtractDWO)
|
|
|
|
RemovePred = [RemovePred, &Obj](const SectionBase &Sec) {
|
2018-01-26 05:03:38 +08:00
|
|
|
return OnlyKeepDWOPred(*Obj, Sec) || RemovePred(Sec);
|
2017-11-04 02:58:41 +08:00
|
|
|
};
|
|
|
|
|
2017-11-28 02:56:01 +08:00
|
|
|
if (StripAllGNU)
|
2017-11-14 06:02:07 +08:00
|
|
|
RemovePred = [RemovePred, &Obj](const SectionBase &Sec) {
|
|
|
|
if (RemovePred(Sec))
|
|
|
|
return true;
|
|
|
|
if ((Sec.Flags & SHF_ALLOC) != 0)
|
|
|
|
return false;
|
2018-01-26 05:03:38 +08:00
|
|
|
if (&Sec == Obj->getSectionHeaderStrTab())
|
2017-11-14 06:02:07 +08:00
|
|
|
return false;
|
2017-12-16 04:17:55 +08:00
|
|
|
switch (Sec.Type) {
|
2017-11-14 06:02:07 +08:00
|
|
|
case SHT_SYMTAB:
|
|
|
|
case SHT_REL:
|
|
|
|
case SHT_RELA:
|
|
|
|
case SHT_STRTAB:
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return Sec.Name.startswith(".debug");
|
|
|
|
};
|
|
|
|
|
2017-10-12 02:09:18 +08:00
|
|
|
if (StripSections) {
|
|
|
|
RemovePred = [RemovePred](const SectionBase &Sec) {
|
|
|
|
return RemovePred(Sec) || (Sec.Flags & SHF_ALLOC) == 0;
|
|
|
|
};
|
2018-01-26 05:03:38 +08:00
|
|
|
Obj->WriteSectionHeaders = false;
|
2017-10-12 02:09:18 +08:00
|
|
|
}
|
|
|
|
|
2017-11-14 06:13:08 +08:00
|
|
|
if (StripDebug) {
|
|
|
|
RemovePred = [RemovePred](const SectionBase &Sec) {
|
|
|
|
return RemovePred(Sec) || Sec.Name.startswith(".debug");
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-11-15 02:50:24 +08:00
|
|
|
if (StripNonAlloc)
|
|
|
|
RemovePred = [RemovePred, &Obj](const SectionBase &Sec) {
|
|
|
|
if (RemovePred(Sec))
|
|
|
|
return true;
|
2018-01-26 05:03:38 +08:00
|
|
|
if (&Sec == Obj->getSectionHeaderStrTab())
|
2017-11-15 02:50:24 +08:00
|
|
|
return false;
|
|
|
|
return (Sec.Flags & SHF_ALLOC) == 0;
|
|
|
|
};
|
|
|
|
|
2017-11-28 02:56:01 +08:00
|
|
|
if (StripAll)
|
|
|
|
RemovePred = [RemovePred, &Obj](const SectionBase &Sec) {
|
|
|
|
if (RemovePred(Sec))
|
|
|
|
return true;
|
2018-01-26 05:03:38 +08:00
|
|
|
if (&Sec == Obj->getSectionHeaderStrTab())
|
2017-11-28 02:56:01 +08:00
|
|
|
return false;
|
|
|
|
if (Sec.Name.startswith(".gnu.warning"))
|
|
|
|
return false;
|
|
|
|
return (Sec.Flags & SHF_ALLOC) == 0;
|
|
|
|
};
|
|
|
|
|
2017-12-01 04:14:53 +08:00
|
|
|
// Explicit copies:
|
|
|
|
|
|
|
|
if (!OnlyKeep.empty()) {
|
|
|
|
RemovePred = [RemovePred, &Obj](const SectionBase &Sec) {
|
|
|
|
// Explicitly keep these sections regardless of previous removes.
|
|
|
|
if (std::find(std::begin(OnlyKeep), std::end(OnlyKeep), Sec.Name) !=
|
|
|
|
std::end(OnlyKeep))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Allow all implicit removes.
|
2018-01-26 05:03:38 +08:00
|
|
|
if (RemovePred(Sec)) {
|
2017-12-01 04:14:53 +08:00
|
|
|
return true;
|
2018-01-26 05:03:38 +08:00
|
|
|
}
|
2017-12-01 04:14:53 +08:00
|
|
|
|
|
|
|
// Keep special sections.
|
2018-01-26 05:03:38 +08:00
|
|
|
if (Obj->getSectionHeaderStrTab() == &Sec) {
|
2017-12-01 04:14:53 +08:00
|
|
|
return false;
|
2018-01-26 05:03:38 +08:00
|
|
|
}
|
|
|
|
if (Obj->getSymTab() == &Sec || Obj->getSymTab()->getStrTab() == &Sec) {
|
2017-12-01 04:14:53 +08:00
|
|
|
return false;
|
2018-01-26 05:03:38 +08:00
|
|
|
}
|
2017-12-01 04:14:53 +08:00
|
|
|
// Remove everything else.
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Keep.empty()) {
|
|
|
|
RemovePred = [RemovePred](const SectionBase &Sec) {
|
|
|
|
// Explicitly keep these sections regardless of previous removes.
|
|
|
|
if (std::find(std::begin(Keep), std::end(Keep), Sec.Name) !=
|
|
|
|
std::end(Keep))
|
|
|
|
return false;
|
|
|
|
// Otherwise defer to RemovePred.
|
|
|
|
return RemovePred(Sec);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-01-26 05:03:38 +08:00
|
|
|
Obj->removeSections(RemovePred);
|
2017-12-19 08:47:30 +08:00
|
|
|
|
|
|
|
if (!AddSection.empty()) {
|
|
|
|
for (const auto &Flag : AddSection) {
|
|
|
|
auto SecPair = StringRef(Flag).split("=");
|
|
|
|
auto SecName = SecPair.first;
|
|
|
|
auto File = SecPair.second;
|
|
|
|
auto BufOrErr = MemoryBuffer::getFile(File);
|
|
|
|
if (!BufOrErr)
|
|
|
|
reportError(File, BufOrErr.getError());
|
|
|
|
auto Buf = std::move(*BufOrErr);
|
|
|
|
auto BufPtr = reinterpret_cast<const uint8_t *>(Buf->getBufferStart());
|
|
|
|
auto BufSize = Buf->getBufferSize();
|
2018-01-26 05:03:38 +08:00
|
|
|
Obj->addSection(SecName, ArrayRef<uint8_t>(BufPtr, BufSize));
|
2017-12-19 08:47:30 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-26 06:15:14 +08:00
|
|
|
if (!AddGnuDebugLink.empty()) {
|
|
|
|
Obj->addGnuDebugLink(AddGnuDebugLink);
|
|
|
|
}
|
|
|
|
|
2018-01-26 05:03:38 +08:00
|
|
|
Obj->finalize();
|
|
|
|
WriteObjectFile(*Obj, OutputFilename.getValue());
|
2017-08-01 08:33:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char **argv) {
|
|
|
|
// Print a stack trace if we signal out.
|
|
|
|
sys::PrintStackTraceOnErrorSignal(argv[0]);
|
|
|
|
PrettyStackTraceProgram X(argc, argv);
|
|
|
|
llvm_shutdown_obj Y; // Call llvm_shutdown() on exit.
|
|
|
|
cl::ParseCommandLineOptions(argc, argv, "llvm objcopy utility\n");
|
|
|
|
ToolName = argv[0];
|
|
|
|
if (InputFilename.empty()) {
|
|
|
|
cl::PrintHelpMessage();
|
|
|
|
return 2;
|
|
|
|
}
|
2018-01-26 05:03:38 +08:00
|
|
|
Expected<OwningBinary<Binary>> BinaryOrErr = createBinary(InputFilename);
|
|
|
|
if (!BinaryOrErr)
|
|
|
|
reportError(InputFilename, BinaryOrErr.takeError());
|
|
|
|
Binary &Binary = *BinaryOrErr.get().getBinary();
|
|
|
|
if (auto *o = dyn_cast<ELFObjectFile<ELF64LE>>(&Binary)) {
|
|
|
|
CopyBinary(*o);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (auto *o = dyn_cast<ELFObjectFile<ELF32LE>>(&Binary)) {
|
|
|
|
CopyBinary(*o);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (auto *o = dyn_cast<ELFObjectFile<ELF64BE>>(&Binary)) {
|
|
|
|
CopyBinary(*o);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
if (auto *o = dyn_cast<ELFObjectFile<ELF32BE>>(&Binary)) {
|
|
|
|
CopyBinary(*o);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
reportError(InputFilename, object_error::invalid_file_type);
|
2017-08-01 08:33:58 +08:00
|
|
|
}
|