//===--- stkrc.cpp --- The Stacker Compiler -------------------------------===// // // The LLVM Compiler Infrastructure // // This file was developed by Reid Spencer and donated to the LLVM research // group and is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // // This is the "main" program for the Stacker Compiler. It is simply a utility // that invokes the StackerCompiler::compile method (see StackerCompiler.cpp) // // To get help using this utility, you can invoke it with: // stkrc --help - Output information about command line switches // // //===------------------------------------------------------------------------=== #include "../../lib/compiler/StackerCompiler.h" #include "llvm/Assembly/Parser.h" #include "llvm/Bytecode/Writer.h" #include "llvm/Analysis/Verifier.h" #include "llvm/Support/CommandLine.h" #include "llvm/System/Signals.h" #include <fstream> #include <iostream> #include <memory> using namespace llvm; static cl::opt<std::string> InputFilename(cl::Positional, cl::desc("<input .st file>"), cl::init("-")); static cl::opt<std::string> OutputFilename("o", cl::desc("Override output filename"), cl::value_desc("filename")); static cl::opt<bool> Force("f", cl::desc("Overwrite output files")); static cl::opt<uint32_t> StackSize("s", cl::desc("Specify program maximum stack size"), cl::init(1024), cl::value_desc("stack size")); static cl::opt<bool> DumpAsm("d", cl::desc("Print LLVM Assembly as parsed"), cl::Hidden); #ifdef PARSE_DEBUG static cl::opt<bool> ParseDebug("g", cl::desc("Turn on Bison Debugging"), cl::Hidden); #endif #ifdef FLEX_DEBUG static cl::opt<bool> FlexDebug("x", cl::desc("Turn on Flex Debugging"), cl::Hidden); #endif static cl::opt<bool> EchoSource("e", cl::desc("Print Stacker Source as parsed"), cl::Hidden); enum OptLev { None = 0, One = 1, Two = 2, Three = 3, Four = 4, Five = 5 }; static cl::opt<OptLev> OptLevel( cl::desc("Choose optimization level to apply:"), cl::init(One), cl::values( clEnumValN(None,"O0","An alias for the -O1 option"), clEnumValN(One,"O1","Optimize for compilation speed"), clEnumValN(Two,"O2","Perform simple optimizations to reduce code size"), clEnumValN(Three,"O3","More aggressive optimizations"), clEnumValN(Four,"O4","High level of optimization"), clEnumValN(Five,"O5","An alias for the -O4 option"), clEnumValEnd )); int main(int argc, char **argv) { cl::ParseCommandLineOptions(argc, argv, " stacker .st -> .bc compiler\n"); std::ostream *Out = 0; try { StackerCompiler compiler; try { #ifdef PARSE_DEBUG { extern int Stackerdebug; Stackerdebug = ParseDebug; } #endif #ifdef FLEX_DEBUG { extern int Stacker_flex_debug; Stacker_flex_debug = FlexDebug; } #endif // Parse the file now... std::auto_ptr<Module> M ( compiler.compile(InputFilename,EchoSource,OptLevel,StackSize)); if (M.get() == 0) { throw std::string("program didn't parse correctly."); } if (verifyModule(*M.get())) { throw std::string("program parsed, but does not verify as correct!"); } if (DumpAsm) std::cerr << "Here's the assembly:" << M.get(); if (OutputFilename != "") { // Specified an output filename? if (OutputFilename != "-") { // Not stdout? if (!Force && std::ifstream(OutputFilename.c_str())) { // If force is not specified, make sure not to overwrite a file! throw std::string("error opening '") + OutputFilename + "': file exists!\n" + "Use -f command line argument to force output"; return 1; } Out = new std::ofstream(OutputFilename.c_str()); } else { // Specified stdout Out = &std::cout; } } else { if (InputFilename == "-") { OutputFilename = "-"; Out = &std::cout; } else { std::string IFN = InputFilename; int Len = IFN.length(); if (IFN[Len-3] == '.' && IFN[Len-2] == 's' && IFN[Len-1] == 't') { // Source ends in .ll OutputFilename = std::string(IFN.begin(), IFN.end()-3); } else { OutputFilename = IFN; // Append a .bc to it } OutputFilename += ".bc"; if (!Force && std::ifstream(OutputFilename.c_str())) { // If force is not specified, make sure not to overwrite a file! throw std::string("error opening '") + OutputFilename + "': file exists!\n" + "Use -f command line argument to force output\n"; } Out = new std::ofstream(OutputFilename.c_str()); // Make sure that the Out file gets unlinked from the disk if we get a // SIGINT sys::RemoveFileOnSignal(sys::Path(OutputFilename)); } } if (!Out->good()) { throw std::string("error opening ") + OutputFilename + "!"; } WriteBytecodeToFile(M.get(), *Out); } catch (const ParseException &E) { std::cerr << argv[0] << ": " << E.getMessage() << "\n"; return 1; } } catch (const std::string& msg ) { std::cerr << argv[0] << ": " << msg << "\n"; return 1; } if (Out != &std::cout) delete Out; return 0; }