1
0
mirror of https://github.com/exaloop/codon.git synced 2025-06-03 15:03:52 +08:00
codon/codon/app/main.cpp
2023-03-17 11:28:55 -04:00

451 lines
15 KiB
C++

// Copyright (C) 2022-2023 Exaloop Inc. <https://exaloop.io>
#include <algorithm>
#include <cstdio>
#include <cstdlib>
#include <fstream>
#include <functional>
#include <iostream>
#include <sstream>
#include <string>
#include <unordered_map>
#include <vector>
#include "codon/compiler/compiler.h"
#include "codon/compiler/error.h"
#include "codon/compiler/jit.h"
#include "codon/util/common.h"
#include "codon/util/jupyter.h"
#include "llvm/Support/CommandLine.h"
#include "llvm/Support/FileSystem.h"
namespace {
void versMsg(llvm::raw_ostream &out) {
out << CODON_VERSION_MAJOR << "." << CODON_VERSION_MINOR << "." << CODON_VERSION_PATCH
<< "\n";
}
bool isMacOS() {
#ifdef __APPLE__
return true;
#else
return false;
#endif
}
const std::vector<std::string> &supportedExtensions() {
static const std::vector<std::string> extensions = {".codon", ".py", ".seq"};
return extensions;
}
bool hasExtension(const std::string &filename, const std::string &extension) {
return filename.size() >= extension.size() &&
filename.compare(filename.size() - extension.size(), extension.size(),
extension) == 0;
}
std::string trimExtension(const std::string &filename, const std::string &extension) {
if (hasExtension(filename, extension)) {
return filename.substr(0, filename.size() - extension.size());
} else {
return filename;
}
}
std::string makeOutputFilename(const std::string &filename,
const std::string &extension) {
for (const auto &ext : supportedExtensions()) {
if (hasExtension(filename, ext))
return trimExtension(filename, ext) + extension;
}
return filename + extension;
}
void display(const codon::error::ParserErrorInfo &e) {
using codon::MessageGroupPos;
for (auto &group : e) {
for (auto &msg : group) {
MessageGroupPos pos = MessageGroupPos::NONE;
if (&msg == &group.front()) {
pos = MessageGroupPos::HEAD;
} else if (&msg == &group.back()) {
pos = MessageGroupPos::LAST;
} else {
pos = MessageGroupPos::MID;
}
codon::compilationError(msg.getMessage(), msg.getFile(), msg.getLine(),
msg.getColumn(), msg.getLength(), msg.getErrorCode(),
/*terminate=*/false, pos);
}
}
}
void initLogFlags(const llvm::cl::opt<std::string> &log) {
codon::getLogger().parse(log);
if (auto *d = getenv("CODON_DEBUG"))
codon::getLogger().parse(std::string(d));
}
enum BuildKind { LLVM, Bitcode, Object, Executable, Library, PyExtension, Detect };
enum OptMode { Debug, Release };
enum Numerics { C, Python };
} // namespace
int docMode(const std::vector<const char *> &args, const std::string &argv0) {
llvm::cl::ParseCommandLineOptions(args.size(), args.data());
std::vector<std::string> files;
for (std::string line; std::getline(std::cin, line);)
files.push_back(line);
auto compiler = std::make_unique<codon::Compiler>(args[0]);
bool failed = false;
auto result = compiler->docgen(files);
llvm::handleAllErrors(result.takeError(),
[&failed](const codon::error::ParserErrorInfo &e) {
display(e);
failed = true;
});
if (failed)
return EXIT_FAILURE;
fmt::print("{}\n", *result);
return EXIT_SUCCESS;
}
std::unique_ptr<codon::Compiler> processSource(
const std::vector<const char *> &args, bool standalone,
std::function<bool()> pyExtension = [] { return false; }) {
llvm::cl::opt<std::string> input(llvm::cl::Positional, llvm::cl::desc("<input file>"),
llvm::cl::init("-"));
auto regs = llvm::cl::getRegisteredOptions();
llvm::cl::opt<OptMode> optMode(
llvm::cl::desc("optimization mode"),
llvm::cl::values(
clEnumValN(Debug, regs.find("debug") != regs.end() ? "default" : "debug",
"Turn off compiler optimizations and show backtraces"),
clEnumValN(Release, "release",
"Turn on compiler optimizations and disable debug info")),
llvm::cl::init(Debug));
llvm::cl::list<std::string> defines(
"D", llvm::cl::Prefix,
llvm::cl::desc("Add static variable definitions. The syntax is <name>=<value>"));
llvm::cl::list<std::string> disabledOpts(
"disable-opt", llvm::cl::desc("Disable the specified IR optimization"));
llvm::cl::list<std::string> plugins("plugin",
llvm::cl::desc("Load specified plugin"));
llvm::cl::opt<std::string> log("log", llvm::cl::desc("Enable given log streams"));
llvm::cl::opt<Numerics> numerics(
"numerics", llvm::cl::desc("numerical semantics"),
llvm::cl::values(
clEnumValN(C, "c", "C semantics: best performance but deviates from Python"),
clEnumValN(Python, "py",
"Python semantics: mirrors Python but might disable optimizations "
"like vectorization")),
llvm::cl::init(C));
llvm::cl::ParseCommandLineOptions(args.size(), args.data());
initLogFlags(log);
std::unordered_map<std::string, std::string> defmap;
for (const auto &define : defines) {
auto eq = define.find('=');
if (eq == std::string::npos || !eq) {
codon::compilationWarning("ignoring malformed definition: " + define);
continue;
}
auto name = define.substr(0, eq);
auto value = define.substr(eq + 1);
if (defmap.find(name) != defmap.end()) {
codon::compilationWarning("ignoring duplicate definition: " + define);
continue;
}
defmap.emplace(name, value);
}
const bool isDebug = (optMode == OptMode::Debug);
std::vector<std::string> disabledOptsVec(disabledOpts);
auto compiler = std::make_unique<codon::Compiler>(
args[0], isDebug, disabledOptsVec,
/*isTest=*/false, (numerics == Numerics::Python), pyExtension());
compiler->getLLVMVisitor()->setStandalone(standalone);
// load plugins
for (const auto &plugin : plugins) {
bool failed = false;
llvm::handleAllErrors(
compiler->load(plugin), [&failed](const codon::error::PluginErrorInfo &e) {
codon::compilationError(e.getMessage(), /*file=*/"",
/*line=*/0, /*col=*/0, /*len*/ 0, /*errorCode*/ -1,
/*terminate=*/false);
failed = true;
});
if (failed)
return {};
}
bool failed = false;
int testFlags = 0;
if (auto *tf = getenv("CODON_TEST_FLAGS"))
testFlags = std::atoi(tf);
llvm::handleAllErrors(compiler->parseFile(input, /*testFlags=*/testFlags, defmap),
[&failed](const codon::error::ParserErrorInfo &e) {
display(e);
failed = true;
});
if (failed)
return {};
{
TIME("compile");
llvm::cantFail(compiler->compile());
}
return compiler;
}
int runMode(const std::vector<const char *> &args) {
llvm::cl::list<std::string> libs(
"l", llvm::cl::desc("Load and link the specified library"));
llvm::cl::list<std::string> progArgs(llvm::cl::ConsumeAfter,
llvm::cl::desc("<program arguments>..."));
auto compiler = processSource(args, /*standalone=*/false);
if (!compiler)
return EXIT_FAILURE;
std::vector<std::string> libsVec(libs);
std::vector<std::string> argsVec(progArgs);
argsVec.insert(argsVec.begin(), compiler->getInput());
compiler->getLLVMVisitor()->run(argsVec, libsVec);
return EXIT_SUCCESS;
}
namespace {
std::string jitExec(codon::jit::JIT *jit, const std::string &code) {
auto result = jit->execute(code);
if (auto err = result.takeError()) {
std::string output;
llvm::handleAllErrors(
std::move(err), [](const codon::error::ParserErrorInfo &e) { display(e); },
[&output](const codon::error::RuntimeErrorInfo &e) {
std::stringstream buf;
buf << e.getOutput();
buf << "\n\033[1mBacktrace:\033[0m\n";
for (const auto &line : e.getBacktrace()) {
buf << " " << line << "\n";
}
output = buf.str();
});
return output;
}
return *result;
}
void jitLoop(codon::jit::JIT *jit, std::istream &fp) {
std::string code;
for (std::string line; std::getline(fp, line);) {
if (line != "#%%") {
code += line + "\n";
} else {
fmt::print("{}[done]\n", jitExec(jit, code));
code = "";
fflush(stdout);
}
}
if (!code.empty())
fmt::print("{}[done]\n", jitExec(jit, code));
}
} // namespace
int jitMode(const std::vector<const char *> &args) {
llvm::cl::opt<std::string> input(llvm::cl::Positional, llvm::cl::desc("<input file>"),
llvm::cl::init("-"));
llvm::cl::list<std::string> plugins("plugin",
llvm::cl::desc("Load specified plugin"));
llvm::cl::opt<std::string> log("log", llvm::cl::desc("Enable given log streams"));
llvm::cl::ParseCommandLineOptions(args.size(), args.data());
initLogFlags(log);
codon::jit::JIT jit(args[0]);
// load plugins
for (const auto &plugin : plugins) {
bool failed = false;
llvm::handleAllErrors(jit.getCompiler()->load(plugin),
[&failed](const codon::error::PluginErrorInfo &e) {
codon::compilationError(e.getMessage(), /*file=*/"",
/*line=*/0, /*col=*/0, /*len=*/0,
/*errorCode*/ -1,
/*terminate=*/false);
failed = true;
});
if (failed)
return EXIT_FAILURE;
}
llvm::cantFail(jit.init());
fmt::print(">>> Codon JIT v{} <<<\n", CODON_VERSION);
if (input == "-") {
jitLoop(&jit, std::cin);
} else {
std::ifstream fileInput(input);
jitLoop(&jit, fileInput);
}
return EXIT_SUCCESS;
}
int buildMode(const std::vector<const char *> &args, const std::string &argv0) {
llvm::cl::list<std::string> libs(
"l", llvm::cl::desc("Link the specified library (only for executables)"));
llvm::cl::opt<std::string> lflags("linker-flags",
llvm::cl::desc("Pass given flags to linker"));
llvm::cl::opt<BuildKind> buildKind(
llvm::cl::desc("output type"),
llvm::cl::values(
clEnumValN(LLVM, "llvm", "Generate LLVM IR"),
clEnumValN(Bitcode, "bc", "Generate LLVM bitcode"),
clEnumValN(Object, "obj", "Generate native object file"),
clEnumValN(Executable, "exe", "Generate executable"),
clEnumValN(Library, "lib", "Generate shared library"),
clEnumValN(PyExtension, "pyext", "Generate Python extension module"),
clEnumValN(Detect, "detect",
"Detect output type based on output file extension")),
llvm::cl::init(Detect));
llvm::cl::opt<std::string> output(
"o",
llvm::cl::desc(
"Write compiled output to specified file. Supported extensions: "
"none (executable), .o (object file), .ll (LLVM IR), .bc (LLVM bitcode)"));
llvm::cl::opt<std::string> pyModule(
"module", llvm::cl::desc("Python extension module name (only applicable when "
"building Python extension module)"));
auto compiler = processSource(args, /*standalone=*/true,
[&] { return buildKind == BuildKind::PyExtension; });
if (!compiler)
return EXIT_FAILURE;
std::vector<std::string> libsVec(libs);
if (output.empty() && compiler->getInput() == "-")
codon::compilationError("output file must be specified when reading from stdin");
std::string extension;
switch (buildKind) {
case BuildKind::LLVM:
extension = ".ll";
break;
case BuildKind::Bitcode:
extension = ".bc";
break;
case BuildKind::Object:
case BuildKind::PyExtension:
extension = ".o";
break;
case BuildKind::Library:
extension = isMacOS() ? ".dylib" : ".so";
break;
case BuildKind::Executable:
case BuildKind::Detect:
extension = "";
break;
default:
seqassertn(0, "unknown build kind");
}
const std::string filename =
output.empty() ? makeOutputFilename(compiler->getInput(), extension) : output;
switch (buildKind) {
case BuildKind::LLVM:
compiler->getLLVMVisitor()->writeToLLFile(filename);
break;
case BuildKind::Bitcode:
compiler->getLLVMVisitor()->writeToBitcodeFile(filename);
break;
case BuildKind::Object:
compiler->getLLVMVisitor()->writeToObjectFile(filename);
break;
case BuildKind::Executable:
compiler->getLLVMVisitor()->writeToExecutable(filename, argv0, false, libsVec,
lflags);
break;
case BuildKind::Library:
compiler->getLLVMVisitor()->writeToExecutable(filename, argv0, true, libsVec,
lflags);
break;
case BuildKind::PyExtension:
compiler->getCache()->pyModule->name =
pyModule.empty() ? llvm::sys::path::stem(compiler->getInput()).str() : pyModule;
compiler->getLLVMVisitor()->writeToPythonExtension(*compiler->getCache()->pyModule,
filename);
break;
case BuildKind::Detect:
compiler->getLLVMVisitor()->compile(filename, argv0, libsVec, lflags);
break;
default:
seqassertn(0, "unknown build kind");
}
return EXIT_SUCCESS;
}
int jupyterMode(const std::vector<const char *> &args) {
llvm::cl::list<std::string> plugins("plugin",
llvm::cl::desc("Load specified plugin"));
llvm::cl::opt<std::string> input(llvm::cl::Positional,
llvm::cl::desc("<connection file>"),
llvm::cl::init("connection.json"));
llvm::cl::ParseCommandLineOptions(args.size(), args.data());
int code = codon::startJupyterKernel(args[0], plugins, input);
return code;
}
void showCommandsAndExit() {
codon::compilationError("Available commands: codon <run|build|doc>");
}
int otherMode(const std::vector<const char *> &args) {
llvm::cl::opt<std::string> input(llvm::cl::Positional, llvm::cl::desc("<mode>"));
llvm::cl::extrahelp("\nMODES:\n\n"
" run - run a program interactively\n"
" build - build a program\n"
" doc - generate program documentation\n");
llvm::cl::ParseCommandLineOptions(args.size(), args.data());
if (!input.empty())
showCommandsAndExit();
return EXIT_SUCCESS;
}
int main(int argc, const char **argv) {
if (argc < 2)
showCommandsAndExit();
llvm::cl::SetVersionPrinter(versMsg);
std::vector<const char *> args{argv[0]};
for (int i = 2; i < argc; i++)
args.push_back(argv[i]);
std::string mode(argv[1]);
std::string argv0 = std::string(args[0]) + " " + mode;
if (mode == "run") {
args[0] = argv0.data();
return runMode(args);
}
if (mode == "build") {
const char *oldArgv0 = args[0];
args[0] = argv0.data();
return buildMode(args, oldArgv0);
}
if (mode == "doc") {
const char *oldArgv0 = args[0];
args[0] = argv0.data();
return docMode(args, oldArgv0);
}
if (mode == "jit") {
args[0] = argv0.data();
return jitMode(args);
}
if (mode == "jupyter") {
args[0] = argv0.data();
return jupyterMode(args);
}
return otherMode({argv, argv + argc});
}