blob: 4a7befbfce53d1d1e05952b6b945e387023958ac [file] [log] [blame]
Misha Brukmande03bc02005-04-24 17:36:05 +00001//===- llvm-extract.cpp - LLVM function extraction utility ----------------===//
Misha Brukman3da94ae2005-04-22 00:00:37 +00002//
John Criswell7c0e0222003-10-20 17:47:21 +00003// The LLVM Compiler Infrastructure
4//
5// This file was developed by the LLVM research group and is distributed under
6// the University of Illinois Open Source License. See LICENSE.TXT for details.
Misha Brukman3da94ae2005-04-22 00:00:37 +00007//
John Criswell7c0e0222003-10-20 17:47:21 +00008//===----------------------------------------------------------------------===//
Chris Lattner579d9142002-05-22 20:27:00 +00009//
10// This utility changes the input module to only contain a single function,
11// which is primarily used for debugging transformations.
12//
13//===----------------------------------------------------------------------===//
14
15#include "llvm/Module.h"
16#include "llvm/PassManager.h"
Chris Lattnerc48e1db2007-05-06 05:13:17 +000017#include "llvm/Bitcode/ReaderWriter.h"
Chris Lattner33974ca2002-07-23 22:04:40 +000018#include "llvm/Transforms/IPO.h"
Brian Gaeke20408902003-10-28 22:22:16 +000019#include "llvm/Target/TargetData.h"
Reid Spencer551ccae2004-09-01 22:55:40 +000020#include "llvm/Support/CommandLine.h"
Chris Lattnerc30598b2006-12-06 01:18:01 +000021#include "llvm/Support/ManagedStatic.h"
Chris Lattnerc48e1db2007-05-06 05:13:17 +000022#include "llvm/Support/MemoryBuffer.h"
Chris Lattnerbed85ff2004-05-27 05:41:36 +000023#include "llvm/System/Signals.h"
Bill Wendling68fe61d2006-11-29 00:19:40 +000024#include <iostream>
Chris Lattner579d9142002-05-22 20:27:00 +000025#include <memory>
Chris Lattnerba9cc1f2004-02-18 16:53:59 +000026#include <fstream>
Brian Gaeked0fde302003-11-11 22:41:34 +000027using namespace llvm;
28
Chris Lattner5ff62e92002-07-22 02:10:13 +000029// InputFilename - The filename to read from.
Chris Lattnerc7a09852002-07-25 16:31:09 +000030static cl::opt<std::string>
Gabor Greifa99be512007-07-05 17:07:56 +000031InputFilename(cl::Positional, cl::desc("<input bitcode file>"),
Chris Lattner5ff62e92002-07-22 02:10:13 +000032 cl::init("-"), cl::value_desc("filename"));
Misha Brukman3da94ae2005-04-22 00:00:37 +000033
Chris Lattnerba9cc1f2004-02-18 16:53:59 +000034static cl::opt<std::string>
Misha Brukman3da94ae2005-04-22 00:00:37 +000035OutputFilename("o", cl::desc("Specify output filename"),
Chris Lattnerba9cc1f2004-02-18 16:53:59 +000036 cl::value_desc("filename"), cl::init("-"));
37
38static cl::opt<bool>
39Force("f", cl::desc("Overwrite output files"));
Chris Lattner5ff62e92002-07-22 02:10:13 +000040
Misha Brukmanca718e42004-04-22 23:07:39 +000041static cl::opt<bool>
42DeleteFn("delete", cl::desc("Delete specified function from Module"));
43
Anton Korobeynikovb10308e2007-01-28 13:31:35 +000044static cl::opt<bool>
45Relink("relink",
46 cl::desc("Turn external linkage for callees of function to delete"));
47
Chris Lattner5ff62e92002-07-22 02:10:13 +000048// ExtractFunc - The function to extract from the module... defaults to main.
Chris Lattnerc7a09852002-07-25 16:31:09 +000049static cl::opt<std::string>
Chris Lattner5ff62e92002-07-22 02:10:13 +000050ExtractFunc("func", cl::desc("Specify function to extract"), cl::init("main"),
51 cl::value_desc("function"));
52
Chris Lattner579d9142002-05-22 20:27:00 +000053int main(int argc, char **argv) {
Chris Lattnerc30598b2006-12-06 01:18:01 +000054 llvm_shutdown_obj X; // Call llvm_shutdown() on exit.
Chris Lattnerc48e1db2007-05-06 05:13:17 +000055 cl::ParseCommandLineOptions(argc, argv, " llvm extractor\n");
56 sys::PrintStackTraceOnErrorSignal();
Chris Lattner579d9142002-05-22 20:27:00 +000057
Chris Lattnerc48e1db2007-05-06 05:13:17 +000058 std::auto_ptr<Module> M;
59
Chris Lattner065344d2007-05-06 23:45:49 +000060 MemoryBuffer *Buffer = MemoryBuffer::getFileOrSTDIN(InputFilename);
Chris Lattner44dadff2007-05-06 09:29:57 +000061 if (Buffer == 0) {
Reid Spencerd2887e42007-08-08 21:19:01 +000062 cerr << argv[0] << ": Error reading file '" + InputFilename + "'\n";
Chris Lattner44dadff2007-05-06 09:29:57 +000063 return 1;
Chris Lattnerc48e1db2007-05-06 05:13:17 +000064 } else {
Chris Lattner44dadff2007-05-06 09:29:57 +000065 M.reset(ParseBitcodeFile(Buffer));
Chris Lattnerba9cc1f2004-02-18 16:53:59 +000066 }
Chris Lattner44dadff2007-05-06 09:29:57 +000067 delete Buffer;
Chris Lattnerc48e1db2007-05-06 05:13:17 +000068
69 if (M.get() == 0) {
Gabor Greifa99be512007-07-05 17:07:56 +000070 cerr << argv[0] << ": bitcode didn't read correctly.\n";
Chris Lattnerc48e1db2007-05-06 05:13:17 +000071 return 1;
72 }
73
74 // Figure out which function we should extract
75 Function *F = M.get()->getFunction(ExtractFunc);
76 if (F == 0) {
77 cerr << argv[0] << ": program doesn't contain function named '"
78 << ExtractFunc << "'!\n";
79 return 1;
80 }
81
82 // In addition to deleting all other functions, we also want to spiff it
83 // up a little bit. Do this now.
84 PassManager Passes;
85 Passes.add(new TargetData(M.get())); // Use correct TargetData
86 // Either isolate the function or delete it from the Module
87 Passes.add(createFunctionExtractionPass(F, DeleteFn, Relink));
88 if (!DeleteFn)
89 Passes.add(createGlobalDCEPass()); // Delete unreachable globals
90 Passes.add(createDeadTypeEliminationPass()); // Remove dead types...
91 Passes.add(createStripDeadPrototypesPass()); // Remove dead func decls
92
93 std::ostream *Out = 0;
94
95 if (OutputFilename != "-") { // Not stdout?
96 if (!Force && std::ifstream(OutputFilename.c_str())) {
97 // If force is not specified, make sure not to overwrite a file!
98 cerr << argv[0] << ": error opening '" << OutputFilename
99 << "': file exists!\n"
100 << "Use -f command line argument to force output\n";
101 return 1;
102 }
103 std::ios::openmode io_mode = std::ios::out | std::ios::trunc |
104 std::ios::binary;
105 Out = new std::ofstream(OutputFilename.c_str(), io_mode);
106 } else { // Specified stdout
107 // FIXME: cout is not binary!
108 Out = &std::cout;
109 }
110
Chris Lattner44dadff2007-05-06 09:29:57 +0000111 Passes.add(CreateBitcodeWriterPass(*Out));
Chris Lattnerc48e1db2007-05-06 05:13:17 +0000112 Passes.run(*M.get());
113
114 if (Out != &std::cout)
115 delete Out;
116 return 0;
Chris Lattner579d9142002-05-22 20:27:00 +0000117}