2008-05-30 14:17:29 +08:00
|
|
|
//===--- Action.cpp - The LLVM Compiler Driver ------------------*- C++ -*-===//
|
2008-05-07 00:34:12 +08:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2008-09-23 04:50:40 +08:00
|
|
|
#include "llvm/CompilerDriver/Action.h"
|
2009-06-26 02:20:10 +08:00
|
|
|
#include "llvm/CompilerDriver/BuiltinOptions.h"
|
2010-07-23 11:42:55 +08:00
|
|
|
#include "llvm/CompilerDriver/Error.h"
|
2010-10-29 03:33:08 +08:00
|
|
|
#include "llvm/CompilerDriver/Main.h"
|
2009-11-07 14:33:58 +08:00
|
|
|
|
2009-06-30 12:07:12 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2010-03-05 12:46:28 +08:00
|
|
|
#include "llvm/Support/SystemUtils.h"
|
2010-11-30 02:47:54 +08:00
|
|
|
#include "llvm/Support/Program.h"
|
|
|
|
#include "llvm/Support/TimeValue.h"
|
2009-11-07 14:33:58 +08:00
|
|
|
|
2008-05-07 00:34:12 +08:00
|
|
|
#include <stdexcept>
|
2009-11-07 14:33:58 +08:00
|
|
|
#include <string>
|
2008-05-07 00:34:12 +08:00
|
|
|
|
|
|
|
using namespace llvm;
|
2008-05-07 02:08:59 +08:00
|
|
|
using namespace llvmc;
|
2008-05-07 00:34:12 +08:00
|
|
|
|
2010-03-05 12:46:28 +08:00
|
|
|
namespace llvmc {
|
|
|
|
|
|
|
|
extern const char* ProgramName;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2008-05-07 00:34:12 +08:00
|
|
|
namespace {
|
2010-05-21 03:23:47 +08:00
|
|
|
|
|
|
|
void PrintString (const std::string& str) {
|
|
|
|
errs() << str << ' ';
|
|
|
|
}
|
|
|
|
|
|
|
|
void PrintCommand (const std::string& Cmd, const StrVector& Args) {
|
2010-05-21 05:11:37 +08:00
|
|
|
errs() << Cmd << ' ';
|
2010-05-21 03:23:47 +08:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2010-11-03 04:47:41 +08:00
|
|
|
int ExecuteProgram (const std::string& name, const StrVector& args) {
|
2010-10-29 03:33:08 +08:00
|
|
|
sys::Path prog(name);
|
2008-05-07 00:34:12 +08:00
|
|
|
|
2010-12-18 05:21:31 +08:00
|
|
|
if (sys::path::is_relative(prog.str())) {
|
2010-11-04 00:14:16 +08:00
|
|
|
prog = PrependMainExecutablePath(name, ProgramName,
|
|
|
|
(void *)(intptr_t)&Main);
|
2010-10-29 03:33:08 +08:00
|
|
|
|
2010-11-03 04:47:41 +08:00
|
|
|
if (!prog.canExecute()) {
|
|
|
|
prog = sys::Program::FindProgramByName(name);
|
|
|
|
if (prog.isEmpty()) {
|
|
|
|
PrintError("Can't find program '" + name + "'");
|
|
|
|
return -1;
|
|
|
|
}
|
2010-07-23 11:42:55 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!prog.canExecute()) {
|
|
|
|
PrintError("Program '" + name + "' is not executable.");
|
|
|
|
return -1;
|
2010-03-05 12:46:28 +08:00
|
|
|
}
|
2008-05-07 00:34:12 +08:00
|
|
|
|
2008-05-07 02:10:53 +08:00
|
|
|
// Build the command line vector and the redirects array.
|
2008-05-07 02:08:59 +08:00
|
|
|
const sys::Path* redirects[3] = {0,0,0};
|
|
|
|
sys::Path stdout_redirect;
|
2008-05-07 00:34:12 +08:00
|
|
|
|
2008-05-07 02:08:59 +08:00
|
|
|
std::vector<const char*> argv;
|
|
|
|
argv.reserve((args.size()+2));
|
|
|
|
argv.push_back(name.c_str());
|
|
|
|
|
2008-05-30 14:17:29 +08:00
|
|
|
for (StrVector::const_iterator B = args.begin(), E = args.end();
|
2008-05-07 02:08:59 +08:00
|
|
|
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.
|
|
|
|
|
2008-05-07 02:10:20 +08:00
|
|
|
// Invoke the program.
|
2010-05-20 03:24:32 +08:00
|
|
|
int ret = sys::Program::ExecuteAndWait(prog, &argv[0], 0, &redirects[0]);
|
|
|
|
|
2010-05-21 03:23:47 +08:00
|
|
|
if (IsSegmentationFault(ret)) {
|
2010-05-21 05:11:37 +08:00
|
|
|
errs() << "Segmentation fault: ";
|
2010-05-21 03:23:47 +08:00
|
|
|
PrintCommand(name, args);
|
2010-05-20 03:24:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2008-05-07 00:34:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-11-07 14:33:58 +08:00
|
|
|
namespace llvmc {
|
2010-05-21 03:23:47 +08:00
|
|
|
void AppendToGlobalTimeLog (const std::string& cmd, double time);
|
2009-11-07 14:33:58 +08:00
|
|
|
}
|
|
|
|
|
2010-05-21 03:23:47 +08:00
|
|
|
int llvmc::Action::Execute () const {
|
|
|
|
if (DryRun || VerboseMode)
|
|
|
|
PrintCommand(Command_, Args_);
|
|
|
|
|
2009-11-07 14:33:58 +08:00
|
|
|
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;
|
2008-05-07 00:34:12 +08:00
|
|
|
}
|