1cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky//===-- llvm-ar.cpp - LLVM archive librarian utility ----------------------===//
2cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky//
3cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky//                     The LLVM Compiler Infrastructure
4cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky//
5cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// This file is distributed under the University of Illinois Open Source
6cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// License. See LICENSE.TXT for details.
7cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky//
8cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky//===----------------------------------------------------------------------===//
9cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky//
10cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// Builds up (relatively) standard unix archive files (.a) containing LLVM
11cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// bitcode or other files.
12cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky//
13cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky//===----------------------------------------------------------------------===//
14cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
15cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/ADT/StringSwitch.h"
16cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/ADT/Triple.h"
17cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/IR/LLVMContext.h"
18cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/IR/Module.h"
19cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/LibDriver/LibDriver.h"
20cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Object/Archive.h"
21cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Object/ArchiveWriter.h"
22cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Object/MachO.h"
23cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Object/ObjectFile.h"
24cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Support/CommandLine.h"
25cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Support/Errc.h"
26cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Support/FileSystem.h"
27cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Support/Format.h"
28cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Support/LineIterator.h"
29cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Support/ManagedStatic.h"
30cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Support/MemoryBuffer.h"
31cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Support/Path.h"
32cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Support/PrettyStackTrace.h"
33cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Support/Signals.h"
34cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Support/TargetSelect.h"
35cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Support/ToolOutputFile.h"
36cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include "llvm/Support/raw_ostream.h"
37cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include <algorithm>
38cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include <cstdlib>
39cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include <memory>
40cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
41cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#if !defined(_MSC_VER) && !defined(__MINGW32__)
42cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include <unistd.h>
43cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#else
44cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#include <io.h>
45cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky#endif
46cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
47cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckyusing namespace llvm;
48cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
49cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// The name this program was invoked as.
50cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic StringRef ToolName;
51cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
52cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// Show the error message and exit.
53cd66d540cead3f8200b0c73bad9c276d67896c3dDavid SrbeckyLLVM_ATTRIBUTE_NORETURN static void fail(Twine Error) {
54cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  outs() << ToolName << ": " << Error << ".\n";
55cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  exit(1);
56cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky}
57cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
58cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic void failIfError(std::error_code EC, Twine Context = "") {
59cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  if (!EC)
60cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    return;
61cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
62cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  std::string ContextStr = Context.str();
63cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  if (ContextStr == "")
64cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    fail(EC.message());
65cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  fail(Context + ": " + EC.message());
66cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky}
67cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
68cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic void failIfError(Error E, Twine Context = "") {
69cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  if (!E)
70cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    return;
71cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
72cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  handleAllErrors(std::move(E), [&](const llvm::ErrorInfoBase &EIB) {
73cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    std::string ContextStr = Context.str();
74cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    if (ContextStr == "")
75cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky      fail(EIB.message());
76cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    fail(Context + ": " + EIB.message());
77cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  });
78cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky}
79cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
80cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// llvm-ar/llvm-ranlib remaining positional arguments.
81cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic cl::list<std::string>
82cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    RestOfArgs(cl::Positional, cl::ZeroOrMore,
83cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky               cl::desc("[relpos] [count] <archive-file> [members]..."));
84cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
85cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic cl::opt<bool> MRI("M", cl::desc(""));
86cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic cl::opt<std::string> Plugin("plugin", cl::desc("plugin (ignored for compatibility"));
87cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
88cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckynamespace {
89cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckyenum Format { Default, GNU, BSD };
90cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky}
91cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
92cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic cl::opt<Format>
93cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    FormatOpt("format", cl::desc("Archive format to create"),
94cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky              cl::values(clEnumValN(Default, "default", "default"),
95cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky                         clEnumValN(GNU, "gnu", "gnu"),
96cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky                         clEnumValN(BSD, "bsd", "bsd"), clEnumValEnd));
97cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
98cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic std::string Options;
99cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
100cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// Provide additional help output explaining the operations and modifiers of
101cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// llvm-ar. This object instructs the CommandLine library to print the text of
102cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// the constructor when the --help option is given.
103cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic cl::extrahelp MoreHelp(
104cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "\nOPERATIONS:\n"
105cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  d[NsS]       - delete file(s) from the archive\n"
106cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  m[abiSs]     - move file(s) in the archive\n"
107cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  p[kN]        - print file(s) found in the archive\n"
108cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  q[ufsS]      - quick append file(s) to the archive\n"
109cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  r[abfiuRsS]  - replace or insert file(s) into the archive\n"
110cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  t            - display contents of archive\n"
111cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  x[No]        - extract file(s) from the archive\n"
112cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "\nMODIFIERS (operation specific):\n"
113cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  [a] - put file(s) after [relpos]\n"
114cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  [b] - put file(s) before [relpos] (same as [i])\n"
115cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  [i] - put file(s) before [relpos] (same as [b])\n"
116cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  [o] - preserve original dates\n"
117cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  [s] - create an archive index (cf. ranlib)\n"
118cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  [S] - do not build a symbol table\n"
119cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  [u] - update only files newer than archive contents\n"
120cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "\nMODIFIERS (generic):\n"
121cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  [c] - do not warn if the library had to be created\n"
122cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  "  [v] - be verbose about actions taken\n"
123cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky);
124cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
125cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// This enumeration delineates the kinds of operations on an archive
126cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// that are permitted.
127cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckyenum ArchiveOperation {
128cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  Print,            ///< Print the contents of the archive
129cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  Delete,           ///< Delete the specified members
130cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  Move,             ///< Move members to end or as given by {a,b,i} modifiers
131cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  QuickAppend,      ///< Quickly append to end of archive
132cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  ReplaceOrInsert,  ///< Replace or Insert members
133cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  DisplayTable,     ///< Display the table of contents
134cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  Extract,          ///< Extract files back to file system
135cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  CreateSymTab      ///< Create a symbol table in an existing archive
136cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky};
137cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
138cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// Modifiers to follow operation to vary behavior
139cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic bool AddAfter = false;      ///< 'a' modifier
140cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic bool AddBefore = false;     ///< 'b' modifier
141cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic bool Create = false;        ///< 'c' modifier
142cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic bool OriginalDates = false; ///< 'o' modifier
143cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic bool OnlyUpdate = false;    ///< 'u' modifier
144cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic bool Verbose = false;       ///< 'v' modifier
145cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic bool Symtab = true;         ///< 's' modifier
146cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic bool Deterministic = true;  ///< 'D' and 'U' modifiers
147cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic bool Thin = false;          ///< 'T' modifier
148cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
149cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// Relative Positional Argument (for insert/move). This variable holds
150cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// the name of the archive member to which the 'a', 'b' or 'i' modifier
151cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// refers. Only one of 'a', 'b' or 'i' can be specified so we only need
152cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// one variable.
153cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic std::string RelPos;
154cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
155cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// This variable holds the name of the archive file as given on the
156cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// command line.
157cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic std::string ArchiveName;
158cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
159cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// This variable holds the list of member files to proecess, as given
160cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// on the command line.
161cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic std::vector<StringRef> Members;
162cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
163cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// Show the error message, the help message and exit.
164cd66d540cead3f8200b0c73bad9c276d67896c3dDavid SrbeckyLLVM_ATTRIBUTE_NORETURN static void
165cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckyshow_help(const std::string &msg) {
166cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  errs() << ToolName << ": " << msg << "\n\n";
167cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  cl::PrintHelpMessage();
168cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  std::exit(1);
169cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky}
170cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
171cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// Extract the member filename from the command line for the [relpos] argument
172cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// associated with a, b, and i modifiers
173cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic void getRelPos() {
174cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  if(RestOfArgs.size() == 0)
175cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    show_help("Expected [relpos] for a, b, or i modifier");
176cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  RelPos = RestOfArgs[0];
177cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  RestOfArgs.erase(RestOfArgs.begin());
178cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky}
179cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
180cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic void getOptions() {
181cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  if(RestOfArgs.size() == 0)
182cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    show_help("Expected options");
183cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  Options = RestOfArgs[0];
184cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  RestOfArgs.erase(RestOfArgs.begin());
185cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky}
186cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
187cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// Get the archive file name from the command line
188cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic void getArchive() {
189cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  if(RestOfArgs.size() == 0)
190cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    show_help("An archive name must be specified");
191cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  ArchiveName = RestOfArgs[0];
192cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  RestOfArgs.erase(RestOfArgs.begin());
193cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky}
194cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
195cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// Copy over remaining items in RestOfArgs to our Members vector
196cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic void getMembers() {
197cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  for (auto &Arg : RestOfArgs)
198cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    Members.push_back(Arg);
199cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky}
200cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
201cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic void runMRIScript();
202cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
203cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// Parse the command line options as presented and return the operation
204cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// specified. Process all modifiers and check to make sure that constraints on
205cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky// modifier/operation pairs have not been violated.
206cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbeckystatic ArchiveOperation parseCommandLine() {
207cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  if (MRI) {
208cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    if (!RestOfArgs.empty())
209cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky      fail("Cannot mix -M and other options");
210cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    runMRIScript();
211cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  }
212cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
213cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  getOptions();
214cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
215cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  // Keep track of number of operations. We can only specify one
216cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  // per execution.
217cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  unsigned NumOperations = 0;
218cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
219cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  // Keep track of the number of positional modifiers (a,b,i). Only
220cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  // one can be specified.
221cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  unsigned NumPositional = 0;
222cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
223cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  // Keep track of which operation was requested
224cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  ArchiveOperation Operation;
225cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
226cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  bool MaybeJustCreateSymTab = false;
227cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky
228cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky  for(unsigned i=0; i<Options.size(); ++i) {
229cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    switch(Options[i]) {
230cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 'd': ++NumOperations; Operation = Delete; break;
231cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 'm': ++NumOperations; Operation = Move ; break;
232cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 'p': ++NumOperations; Operation = Print; break;
233cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 'q': ++NumOperations; Operation = QuickAppend; break;
234cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 'r': ++NumOperations; Operation = ReplaceOrInsert; break;
235cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 't': ++NumOperations; Operation = DisplayTable; break;
236cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 'x': ++NumOperations; Operation = Extract; break;
237cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 'c': Create = true; break;
238cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 'l': /* accepted but unused */ break;
239cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 'o': OriginalDates = true; break;
240cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 's':
241cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky      Symtab = true;
242cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky      MaybeJustCreateSymTab = true;
243cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky      break;
244cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 'S':
245cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky      Symtab = false;
246cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky      break;
247cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 'u': OnlyUpdate = true; break;
248cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 'v': Verbose = true; break;
249cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky    case 'a':
250cd66d540cead3f8200b0c73bad9c276d67896c3dDavid Srbecky      getRelPos();
251      AddAfter = true;
252      NumPositional++;
253      break;
254    case 'b':
255      getRelPos();
256      AddBefore = true;
257      NumPositional++;
258      break;
259    case 'i':
260      getRelPos();
261      AddBefore = true;
262      NumPositional++;
263      break;
264    case 'D':
265      Deterministic = true;
266      break;
267    case 'U':
268      Deterministic = false;
269      break;
270    case 'T':
271      Thin = true;
272      break;
273    default:
274      cl::PrintHelpMessage();
275    }
276  }
277
278  // At this point, the next thing on the command line must be
279  // the archive name.
280  getArchive();
281
282  // Everything on the command line at this point is a member.
283  getMembers();
284
285 if (NumOperations == 0 && MaybeJustCreateSymTab) {
286    NumOperations = 1;
287    Operation = CreateSymTab;
288    if (!Members.empty())
289      show_help("The s operation takes only an archive as argument");
290  }
291
292  // Perform various checks on the operation/modifier specification
293  // to make sure we are dealing with a legal request.
294  if (NumOperations == 0)
295    show_help("You must specify at least one of the operations");
296  if (NumOperations > 1)
297    show_help("Only one operation may be specified");
298  if (NumPositional > 1)
299    show_help("You may only specify one of a, b, and i modifiers");
300  if (AddAfter || AddBefore) {
301    if (Operation != Move && Operation != ReplaceOrInsert)
302      show_help("The 'a', 'b' and 'i' modifiers can only be specified with "
303            "the 'm' or 'r' operations");
304  }
305  if (OriginalDates && Operation != Extract)
306    show_help("The 'o' modifier is only applicable to the 'x' operation");
307  if (OnlyUpdate && Operation != ReplaceOrInsert)
308    show_help("The 'u' modifier is only applicable to the 'r' operation");
309
310  // Return the parsed operation to the caller
311  return Operation;
312}
313
314// Implements the 'p' operation. This function traverses the archive
315// looking for members that match the path list.
316static void doPrint(StringRef Name, const object::Archive::Child &C) {
317  if (Verbose)
318    outs() << "Printing " << Name << "\n";
319
320  ErrorOr<StringRef> DataOrErr = C.getBuffer();
321  failIfError(DataOrErr.getError());
322  StringRef Data = *DataOrErr;
323  outs().write(Data.data(), Data.size());
324}
325
326// Utility function for printing out the file mode when the 't' operation is in
327// verbose mode.
328static void printMode(unsigned mode) {
329  outs() << ((mode & 004) ? "r" : "-");
330  outs() << ((mode & 002) ? "w" : "-");
331  outs() << ((mode & 001) ? "x" : "-");
332}
333
334// Implement the 't' operation. This function prints out just
335// the file names of each of the members. However, if verbose mode is requested
336// ('v' modifier) then the file type, permission mode, user, group, size, and
337// modification time are also printed.
338static void doDisplayTable(StringRef Name, const object::Archive::Child &C) {
339  if (Verbose) {
340    sys::fs::perms Mode = C.getAccessMode();
341    printMode((Mode >> 6) & 007);
342    printMode((Mode >> 3) & 007);
343    printMode(Mode & 007);
344    outs() << ' ' << C.getUID();
345    outs() << '/' << C.getGID();
346    ErrorOr<uint64_t> Size = C.getSize();
347    failIfError(Size.getError());
348    outs() << ' ' << format("%6llu", Size.get());
349    outs() << ' ' << C.getLastModified().str();
350    outs() << ' ';
351  }
352  outs() << Name << "\n";
353}
354
355// Implement the 'x' operation. This function extracts files back to the file
356// system.
357static void doExtract(StringRef Name, const object::Archive::Child &C) {
358  // Retain the original mode.
359  sys::fs::perms Mode = C.getAccessMode();
360
361  int FD;
362  failIfError(sys::fs::openFileForWrite(Name, FD, sys::fs::F_None, Mode), Name);
363
364  {
365    raw_fd_ostream file(FD, false);
366
367    // Get the data and its length
368    StringRef Data = *C.getBuffer();
369
370    // Write the data.
371    file.write(Data.data(), Data.size());
372  }
373
374  // If we're supposed to retain the original modification times, etc. do so
375  // now.
376  if (OriginalDates)
377    failIfError(
378        sys::fs::setLastModificationAndAccessTime(FD, C.getLastModified()));
379
380  if (close(FD))
381    fail("Could not close the file");
382}
383
384static bool shouldCreateArchive(ArchiveOperation Op) {
385  switch (Op) {
386  case Print:
387  case Delete:
388  case Move:
389  case DisplayTable:
390  case Extract:
391  case CreateSymTab:
392    return false;
393
394  case QuickAppend:
395  case ReplaceOrInsert:
396    return true;
397  }
398
399  llvm_unreachable("Missing entry in covered switch.");
400}
401
402static void performReadOperation(ArchiveOperation Operation,
403                                 object::Archive *OldArchive) {
404  if (Operation == Extract && OldArchive->isThin())
405    fail("extracting from a thin archive is not supported");
406
407  bool Filter = !Members.empty();
408  {
409    Error Err;
410    for (auto &C : OldArchive->children(Err)) {
411      ErrorOr<StringRef> NameOrErr = C.getName();
412      failIfError(NameOrErr.getError());
413      StringRef Name = NameOrErr.get();
414
415      if (Filter) {
416        auto I = std::find(Members.begin(), Members.end(), Name);
417        if (I == Members.end())
418          continue;
419        Members.erase(I);
420      }
421
422      switch (Operation) {
423      default:
424        llvm_unreachable("Not a read operation");
425      case Print:
426        doPrint(Name, C);
427        break;
428      case DisplayTable:
429        doDisplayTable(Name, C);
430        break;
431      case Extract:
432        doExtract(Name, C);
433        break;
434      }
435    }
436    failIfError(std::move(Err));
437  }
438
439  if (Members.empty())
440    return;
441  for (StringRef Name : Members)
442    errs() << Name << " was not found\n";
443  std::exit(1);
444}
445
446static void addMember(std::vector<NewArchiveMember> &Members,
447                      StringRef FileName, int Pos = -1) {
448  Expected<NewArchiveMember> NMOrErr =
449      NewArchiveMember::getFile(FileName, Deterministic);
450  failIfError(NMOrErr.takeError(), FileName);
451  if (Pos == -1)
452    Members.push_back(std::move(*NMOrErr));
453  else
454    Members[Pos] = std::move(*NMOrErr);
455}
456
457static void addMember(std::vector<NewArchiveMember> &Members,
458                      const object::Archive::Child &M, int Pos = -1) {
459  if (Thin && !M.getParent()->isThin())
460    fail("Cannot convert a regular archive to a thin one");
461  Expected<NewArchiveMember> NMOrErr =
462      NewArchiveMember::getOldMember(M, Deterministic);
463  failIfError(NMOrErr.takeError());
464  if (Pos == -1)
465    Members.push_back(std::move(*NMOrErr));
466  else
467    Members[Pos] = std::move(*NMOrErr);
468}
469
470enum InsertAction {
471  IA_AddOldMember,
472  IA_AddNewMeber,
473  IA_Delete,
474  IA_MoveOldMember,
475  IA_MoveNewMember
476};
477
478static InsertAction computeInsertAction(ArchiveOperation Operation,
479                                        const object::Archive::Child &Member,
480                                        StringRef Name,
481                                        std::vector<StringRef>::iterator &Pos) {
482  if (Operation == QuickAppend || Members.empty())
483    return IA_AddOldMember;
484
485  auto MI =
486      std::find_if(Members.begin(), Members.end(), [Name](StringRef Path) {
487        return Name == sys::path::filename(Path);
488      });
489
490  if (MI == Members.end())
491    return IA_AddOldMember;
492
493  Pos = MI;
494
495  if (Operation == Delete)
496    return IA_Delete;
497
498  if (Operation == Move)
499    return IA_MoveOldMember;
500
501  if (Operation == ReplaceOrInsert) {
502    StringRef PosName = sys::path::filename(RelPos);
503    if (!OnlyUpdate) {
504      if (PosName.empty())
505        return IA_AddNewMeber;
506      return IA_MoveNewMember;
507    }
508
509    // We could try to optimize this to a fstat, but it is not a common
510    // operation.
511    sys::fs::file_status Status;
512    failIfError(sys::fs::status(*MI, Status), *MI);
513    if (Status.getLastModificationTime() < Member.getLastModified()) {
514      if (PosName.empty())
515        return IA_AddOldMember;
516      return IA_MoveOldMember;
517    }
518
519    if (PosName.empty())
520      return IA_AddNewMeber;
521    return IA_MoveNewMember;
522  }
523  llvm_unreachable("No such operation");
524}
525
526// We have to walk this twice and computing it is not trivial, so creating an
527// explicit std::vector is actually fairly efficient.
528static std::vector<NewArchiveMember>
529computeNewArchiveMembers(ArchiveOperation Operation,
530                         object::Archive *OldArchive) {
531  std::vector<NewArchiveMember> Ret;
532  std::vector<NewArchiveMember> Moved;
533  int InsertPos = -1;
534  StringRef PosName = sys::path::filename(RelPos);
535  if (OldArchive) {
536    Error Err;
537    for (auto &Child : OldArchive->children(Err)) {
538      int Pos = Ret.size();
539      ErrorOr<StringRef> NameOrErr = Child.getName();
540      failIfError(NameOrErr.getError());
541      StringRef Name = NameOrErr.get();
542      if (Name == PosName) {
543        assert(AddAfter || AddBefore);
544        if (AddBefore)
545          InsertPos = Pos;
546        else
547          InsertPos = Pos + 1;
548      }
549
550      std::vector<StringRef>::iterator MemberI = Members.end();
551      InsertAction Action =
552          computeInsertAction(Operation, Child, Name, MemberI);
553      switch (Action) {
554      case IA_AddOldMember:
555        addMember(Ret, Child);
556        break;
557      case IA_AddNewMeber:
558        addMember(Ret, *MemberI);
559        break;
560      case IA_Delete:
561        break;
562      case IA_MoveOldMember:
563        addMember(Moved, Child);
564        break;
565      case IA_MoveNewMember:
566        addMember(Moved, *MemberI);
567        break;
568      }
569      if (MemberI != Members.end())
570        Members.erase(MemberI);
571    }
572    failIfError(std::move(Err));
573  }
574
575  if (Operation == Delete)
576    return Ret;
577
578  if (!RelPos.empty() && InsertPos == -1)
579    fail("Insertion point not found");
580
581  if (RelPos.empty())
582    InsertPos = Ret.size();
583
584  assert(unsigned(InsertPos) <= Ret.size());
585  int Pos = InsertPos;
586  for (auto &M : Moved) {
587    Ret.insert(Ret.begin() + Pos, std::move(M));
588    ++Pos;
589  }
590
591  for (unsigned I = 0; I != Members.size(); ++I)
592    Ret.insert(Ret.begin() + InsertPos, NewArchiveMember());
593  Pos = InsertPos;
594  for (auto &Member : Members) {
595    addMember(Ret, Member, Pos);
596    ++Pos;
597  }
598
599  return Ret;
600}
601
602static object::Archive::Kind getDefaultForHost() {
603  return Triple(sys::getProcessTriple()).isOSDarwin() ? object::Archive::K_BSD
604                                                      : object::Archive::K_GNU;
605}
606
607static object::Archive::Kind getKindFromMember(const NewArchiveMember &Member) {
608  Expected<std::unique_ptr<object::ObjectFile>> OptionalObject =
609      object::ObjectFile::createObjectFile(Member.Buf->getMemBufferRef());
610
611  if (OptionalObject)
612    return isa<object::MachOObjectFile>(**OptionalObject)
613               ? object::Archive::K_BSD
614               : object::Archive::K_GNU;
615
616  // squelch the error in case we had a non-object file
617  consumeError(OptionalObject.takeError());
618  return getDefaultForHost();
619}
620
621static void
622performWriteOperation(ArchiveOperation Operation,
623                      object::Archive *OldArchive,
624                      std::unique_ptr<MemoryBuffer> OldArchiveBuf,
625                      std::vector<NewArchiveMember> *NewMembersP) {
626  std::vector<NewArchiveMember> NewMembers;
627  if (!NewMembersP)
628    NewMembers = computeNewArchiveMembers(Operation, OldArchive);
629
630  object::Archive::Kind Kind;
631  switch (FormatOpt) {
632  case Default:
633    if (Thin)
634      Kind = object::Archive::K_GNU;
635    else if (OldArchive)
636      Kind = OldArchive->kind();
637    else if (NewMembersP)
638      Kind = NewMembersP->size() ? getKindFromMember(NewMembersP->front())
639                                 : getDefaultForHost();
640    else
641      Kind = NewMembers.size() ? getKindFromMember(NewMembers.front())
642                               : getDefaultForHost();
643    break;
644  case GNU:
645    Kind = object::Archive::K_GNU;
646    break;
647  case BSD:
648    if (Thin)
649      fail("Only the gnu format has a thin mode");
650    Kind = object::Archive::K_BSD;
651    break;
652  }
653
654  std::pair<StringRef, std::error_code> Result =
655      writeArchive(ArchiveName, NewMembersP ? *NewMembersP : NewMembers, Symtab,
656                   Kind, Deterministic, Thin, std::move(OldArchiveBuf));
657  failIfError(Result.second, Result.first);
658}
659
660static void createSymbolTable(object::Archive *OldArchive) {
661  // When an archive is created or modified, if the s option is given, the
662  // resulting archive will have a current symbol table. If the S option
663  // is given, it will have no symbol table.
664  // In summary, we only need to update the symbol table if we have none.
665  // This is actually very common because of broken build systems that think
666  // they have to run ranlib.
667  if (OldArchive->hasSymbolTable())
668    return;
669
670  performWriteOperation(CreateSymTab, OldArchive, nullptr, nullptr);
671}
672
673static void performOperation(ArchiveOperation Operation,
674                             object::Archive *OldArchive,
675                             std::unique_ptr<MemoryBuffer> OldArchiveBuf,
676                             std::vector<NewArchiveMember> *NewMembers) {
677  switch (Operation) {
678  case Print:
679  case DisplayTable:
680  case Extract:
681    performReadOperation(Operation, OldArchive);
682    return;
683
684  case Delete:
685  case Move:
686  case QuickAppend:
687  case ReplaceOrInsert:
688    performWriteOperation(Operation, OldArchive, std::move(OldArchiveBuf),
689                          NewMembers);
690    return;
691  case CreateSymTab:
692    createSymbolTable(OldArchive);
693    return;
694  }
695  llvm_unreachable("Unknown operation.");
696}
697
698static int performOperation(ArchiveOperation Operation,
699                            std::vector<NewArchiveMember> *NewMembers) {
700  // Create or open the archive object.
701  ErrorOr<std::unique_ptr<MemoryBuffer>> Buf =
702      MemoryBuffer::getFile(ArchiveName, -1, false);
703  std::error_code EC = Buf.getError();
704  if (EC && EC != errc::no_such_file_or_directory)
705    fail("error opening '" + ArchiveName + "': " + EC.message() + "!");
706
707  if (!EC) {
708    Error Err;
709    object::Archive Archive(Buf.get()->getMemBufferRef(), Err);
710    EC = errorToErrorCode(std::move(Err));
711    failIfError(EC,
712                "error loading '" + ArchiveName + "': " + EC.message() + "!");
713    performOperation(Operation, &Archive, std::move(Buf.get()), NewMembers);
714    return 0;
715  }
716
717  assert(EC == errc::no_such_file_or_directory);
718
719  if (!shouldCreateArchive(Operation)) {
720    failIfError(EC, Twine("error loading '") + ArchiveName + "'");
721  } else {
722    if (!Create) {
723      // Produce a warning if we should and we're creating the archive
724      errs() << ToolName << ": creating " << ArchiveName << "\n";
725    }
726  }
727
728  performOperation(Operation, nullptr, nullptr, NewMembers);
729  return 0;
730}
731
732static void runMRIScript() {
733  enum class MRICommand { AddLib, AddMod, Create, Save, End, Invalid };
734
735  ErrorOr<std::unique_ptr<MemoryBuffer>> Buf = MemoryBuffer::getSTDIN();
736  failIfError(Buf.getError());
737  const MemoryBuffer &Ref = *Buf.get();
738  bool Saved = false;
739  std::vector<NewArchiveMember> NewMembers;
740  std::vector<std::unique_ptr<MemoryBuffer>> ArchiveBuffers;
741  std::vector<std::unique_ptr<object::Archive>> Archives;
742
743  for (line_iterator I(Ref, /*SkipBlanks*/ true, ';'), E; I != E; ++I) {
744    StringRef Line = *I;
745    StringRef CommandStr, Rest;
746    std::tie(CommandStr, Rest) = Line.split(' ');
747    Rest = Rest.trim();
748    if (!Rest.empty() && Rest.front() == '"' && Rest.back() == '"')
749      Rest = Rest.drop_front().drop_back();
750    auto Command = StringSwitch<MRICommand>(CommandStr.lower())
751                       .Case("addlib", MRICommand::AddLib)
752                       .Case("addmod", MRICommand::AddMod)
753                       .Case("create", MRICommand::Create)
754                       .Case("save", MRICommand::Save)
755                       .Case("end", MRICommand::End)
756                       .Default(MRICommand::Invalid);
757
758    switch (Command) {
759    case MRICommand::AddLib: {
760      auto BufOrErr = MemoryBuffer::getFile(Rest, -1, false);
761      failIfError(BufOrErr.getError(), "Could not open library");
762      ArchiveBuffers.push_back(std::move(*BufOrErr));
763      auto LibOrErr =
764          object::Archive::create(ArchiveBuffers.back()->getMemBufferRef());
765      failIfError(errorToErrorCode(LibOrErr.takeError()),
766                  "Could not parse library");
767      Archives.push_back(std::move(*LibOrErr));
768      object::Archive &Lib = *Archives.back();
769      {
770        Error Err;
771        for (auto &Member : Lib.children(Err))
772          addMember(NewMembers, Member);
773        failIfError(std::move(Err));
774      }
775      break;
776    }
777    case MRICommand::AddMod:
778      addMember(NewMembers, Rest);
779      break;
780    case MRICommand::Create:
781      Create = true;
782      if (!ArchiveName.empty())
783        fail("Editing multiple archives not supported");
784      if (Saved)
785        fail("File already saved");
786      ArchiveName = Rest;
787      break;
788    case MRICommand::Save:
789      Saved = true;
790      break;
791    case MRICommand::End:
792      break;
793    case MRICommand::Invalid:
794      fail("Unknown command: " + CommandStr);
795    }
796  }
797
798  // Nothing to do if not saved.
799  if (Saved)
800    performOperation(ReplaceOrInsert, &NewMembers);
801  exit(0);
802}
803
804static int ar_main() {
805  // Do our own parsing of the command line because the CommandLine utility
806  // can't handle the grouped positional parameters without a dash.
807  ArchiveOperation Operation = parseCommandLine();
808  return performOperation(Operation, nullptr);
809}
810
811static int ranlib_main() {
812  if (RestOfArgs.size() != 1)
813    fail(ToolName + " takes just one archive as an argument");
814  ArchiveName = RestOfArgs[0];
815  return performOperation(CreateSymTab, nullptr);
816}
817
818int main(int argc, char **argv) {
819  ToolName = argv[0];
820  // Print a stack trace if we signal out.
821  sys::PrintStackTraceOnErrorSignal(argv[0]);
822  PrettyStackTraceProgram X(argc, argv);
823  llvm_shutdown_obj Y;  // Call llvm_shutdown() on exit.
824
825  llvm::InitializeAllTargetInfos();
826  llvm::InitializeAllTargetMCs();
827  llvm::InitializeAllAsmParsers();
828
829  StringRef Stem = sys::path::stem(ToolName);
830  if (Stem.find("ranlib") == StringRef::npos &&
831      Stem.find("lib") != StringRef::npos)
832    return libDriverMain(makeArrayRef(argv, argc));
833
834  // Have the command line options parsed and handle things
835  // like --help and --version.
836  cl::ParseCommandLineOptions(argc, argv,
837    "LLVM Archiver (llvm-ar)\n\n"
838    "  This program archives bitcode files into single libraries\n"
839  );
840
841  if (Stem.find("ranlib") != StringRef::npos)
842    return ranlib_main();
843  if (Stem.find("ar") != StringRef::npos)
844    return ar_main();
845  fail("Not ranlib, ar or lib!");
846}
847