aboutsummaryrefslogtreecommitdiffstats
path: root/tools/llvmc2/Core.cpp
blob: d08ee7c00fa8d978917bf69c27cc59a046f97ddd (plain)
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
//===--- Core.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.
//
//===----------------------------------------------------------------------===//
//
//  Core driver abstractions.
//
//===----------------------------------------------------------------------===//

#include "Core.h"
#include "Utility.h"

#include "llvm/ADT/STLExtras.h"
#include "llvm/ADT/StringExtras.h"
#include "llvm/Support/CommandLine.h"

#include <algorithm>
#include <iostream>
#include <stdexcept>

using namespace llvm;
using namespace llvmcc;

extern cl::list<std::string> InputFilenames;
extern cl::opt<std::string> OutputFilename;
extern cl::opt<bool> VerboseMode;

namespace {
  void print_string (const std::string& str) {
    std::cerr << str << ' ';
  }
}

int llvmcc::Action::Execute() {
  if (VerboseMode) {
    std::cerr << Command_ << " ";
    std::for_each(Args_.begin(), Args_.end(), print_string);
    std::cerr << '\n';
  }
  return ExecuteProgram(Command_, Args_);
}

int llvmcc::CompilationGraph::Build (const sys::Path& tempDir) const {
  sys::Path In(InputFilenames.at(0)), Out;

  // Find out which language corresponds to the suffix of the first input file
  LanguageMap::const_iterator Lang = ExtsToLangs.find(In.getSuffix());
  if (Lang == ExtsToLangs.end())
    throw std::runtime_error("Unknown suffix!");

  // Find the toolchain corresponding to this language
  ToolChainMap::const_iterator ToolsIt = ToolChains.find(Lang->second);
  if (ToolsIt == ToolChains.end())
    throw std::runtime_error("Unknown language!");
  ToolChain Tools = ToolsIt->second;

  PathVector JoinList;

  for (cl::list<std::string>::const_iterator B = InputFilenames.begin(),
        E = InputFilenames.end(); B != E; ++B) {
    In = sys::Path(*B);

    // Pass input file through the toolchain
    for (ToolChain::const_iterator B = Tools.begin(), E = Tools.end();
         B != E; ++B) {

      const Tool* CurTool = B->getPtr();

      // Is this the last step in the chain?
      if (llvm::next(B) == E || CurTool->IsLast()) {
        JoinList.push_back(In);
        break;
      }
      else {
        Out = tempDir;
        Out.appendComponent(In.getBasename());
        Out.appendSuffix(CurTool->OutputSuffix());
        Out.makeUnique(true, NULL);
        Out.eraseFromDisk();
      }

      if (CurTool->GenerateAction(In, Out).Execute() != 0)
        throw std::runtime_error("Tool returned error code!");

      In = Out; Out.clear();
    }
  }

  // Pass .o files to linker
  const Tool* JoinNode = (--Tools.end())->getPtr();

  // If the final output name is empty, set it to "a.out"
  if (!OutputFilename.empty()) {
    Out = sys::Path(OutputFilename);
  }
  else {
    Out = sys::Path("a");
    Out.appendSuffix(JoinNode->OutputSuffix());
  }

  if (JoinNode->GenerateAction(JoinList, Out).Execute() != 0)
    throw std::runtime_error("Tool returned error code!");

  return 0;
}

void llvmcc::Tool::UnpackValues (const std::string& from,
                                 std::vector<std::string>& to) const {
  SplitString(from, to, ",");
}