1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
|
//===--- Action.cpp - The LLVM Compiler Driver ------------------*- C++ -*-===//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open
// Source License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
//
// Action class - implementation and auxiliary functions.
//
//===----------------------------------------------------------------------===//
#include "llvm/CompilerDriver/Action.h"
#include "llvm/CompilerDriver/BuiltinOptions.h"
#include "llvm/Support/raw_ostream.h"
#include "llvm/Support/SystemUtils.h"
#include "llvm/System/Program.h"
#include "llvm/System/TimeValue.h"
#include <stdexcept>
#include <string>
using namespace llvm;
using namespace llvmc;
namespace llvmc {
extern int Main(int argc, char** argv);
extern const char* ProgramName;
}
namespace {
void PrintString (const std::string& str) {
errs() << str << ' ';
}
void PrintCommand (const std::string& Cmd, const StrVector& Args) {
errs() << Cmd << ' ';
std::for_each(Args.begin(), Args.end(), &PrintString);
errs() << '\n';
}
bool IsSegmentationFault (int returnCode) {
#ifdef LLVM_ON_WIN32
return (returnCode >= 0xc0000000UL)
#else
return (returnCode < 0);
#endif
}
int ExecuteProgram (const std::string& name,
const StrVector& args) {
sys::Path prog = sys::Program::FindProgramByName(name);
if (prog.isEmpty()) {
prog = FindExecutable(name, ProgramName, (void *)(intptr_t)&Main);
if (prog.isEmpty())
throw std::runtime_error("Can't find program '" + name + "'");
}
if (!prog.canExecute())
throw std::runtime_error("Program '" + name + "' is not executable.");
// Build the command line vector and the redirects array.
const sys::Path* redirects[3] = {0,0,0};
sys::Path stdout_redirect;
std::vector<const char*> argv;
argv.reserve((args.size()+2));
argv.push_back(name.c_str());
for (StrVector::const_iterator B = args.begin(), E = args.end();
B!=E; ++B) {
if (*B == ">") {
++B;
stdout_redirect.set(*B);
redirects[1] = &stdout_redirect;
}
else {
argv.push_back((*B).c_str());
}
}
argv.push_back(0); // null terminate list.
// Invoke the program.
int ret = sys::Program::ExecuteAndWait(prog, &argv[0], 0, &redirects[0]);
if (IsSegmentationFault(ret)) {
errs() << "Segmentation fault: ";
PrintCommand(name, args);
}
return ret;
}
}
namespace llvmc {
void AppendToGlobalTimeLog (const std::string& cmd, double time);
}
int llvmc::Action::Execute () const {
if (DryRun || VerboseMode)
PrintCommand(Command_, Args_);
if (!DryRun) {
if (Time) {
sys::TimeValue now = sys::TimeValue::now();
int ret = ExecuteProgram(Command_, Args_);
sys::TimeValue now2 = sys::TimeValue::now();
now2 -= now;
double elapsed = now2.seconds() + now2.microseconds() / 1000000.0;
AppendToGlobalTimeLog(Command_, elapsed);
return ret;
}
else {
return ExecuteProgram(Command_, Args_);
}
}
return 0;
}
|