mirror of
https://github.com/PashaBibko/LX.git
synced 2026-04-03 17:39:02 +00:00
159 lines
5.2 KiB
C++
159 lines
5.2 KiB
C++
#include <LX-Common.h>
|
|
|
|
#include <Parser.h>
|
|
#include <Lexer.h>
|
|
|
|
namespace LX
|
|
{
|
|
// Different errors thrown by main //
|
|
struct InvalidFilePath : public RuntimeError
|
|
{
|
|
GENERATE_LX_ERROR_REQUIRED_FUNCTION_DECLARATIONS;
|
|
|
|
InvalidFilePath(const std::string& _name, const std::filesystem::path& _path)
|
|
: name(_name), path(_path)
|
|
{}
|
|
|
|
std::string name;
|
|
std::filesystem::path path;
|
|
};
|
|
|
|
void InvalidFilePath::PrintToConsole() const
|
|
{
|
|
// Tells the user the input file could not be found and how to fix the issue //
|
|
LX::PrintStringAsColor("Error: ", LX::Color::LIGHT_RED);
|
|
std::cout << "Invalid " << name << ": ";
|
|
LX::PrintStringAsColor(path.string().c_str(), LX::Color::WHITE);
|
|
std::cout << "\n\nMake sure the file exists and the process has the correct path to the file\n";
|
|
}
|
|
|
|
const char* InvalidFilePath::ErrorType() const
|
|
{
|
|
return "Invalid File Path";
|
|
}
|
|
}
|
|
|
|
extern "C" int __declspec(dllexport) GenIR(const char* a_inpPath, const char* a_outPath, const char* a_logPath)
|
|
{
|
|
// Creates the file paths outside of the try-catch so they can be used in errors //
|
|
std::filesystem::path inpPath;
|
|
std::filesystem::path outPath;
|
|
std::filesystem::path logPath;
|
|
|
|
// Creates the contents string outside of the try-catch so they can be used in errors //
|
|
std::string contents;
|
|
LX::Token::source = &contents;
|
|
|
|
// Creates the log-file out of the try-catch so it can be closed propely if an error is thrown //
|
|
std::unique_ptr<std::ofstream> log = nullptr;
|
|
|
|
try
|
|
{
|
|
// Turns the file paths into the C++ type for handling them //
|
|
inpPath = a_inpPath;
|
|
outPath = a_outPath;
|
|
|
|
// Checks the input file exists and opens it //
|
|
LX::ThrowIf<LX::InvalidFilePath>(std::filesystem::exists(inpPath) == false, "input file path", inpPath);
|
|
std::ifstream inpFile(inpPath, std::ios::binary | std::ios::ate); // Opens in binary at the end for microptimisation //
|
|
LX::ThrowIf<LX::InvalidFilePath>(inpFile.is_open() == false, "input file path", inpPath);
|
|
|
|
// Copies the file into the string //
|
|
const std::streamsize len = inpFile.tellg(); // Gets length of file because it was opened at the end
|
|
inpFile.seekg(0, std::ios::beg); // Goes back to the beginning
|
|
contents = std::string(len, '\0'); // Allocates all the space for the string
|
|
inpFile.read(&contents[0], len); // Transfers file contents to string
|
|
|
|
// Opens / Creates the output file //
|
|
std::ofstream outFile(outPath);
|
|
LX::ThrowIf<LX::InvalidFilePath>(outFile.is_open() == false, "output file path", outPath);
|
|
outFile.close(); // Opened just to check we can
|
|
|
|
// Opens the log file (if there is one specified //
|
|
if (a_logPath != nullptr)
|
|
{
|
|
logPath = a_logPath;
|
|
log = std::make_unique<std::ofstream>(logPath);
|
|
LX::ThrowIf<LX::InvalidFilePath>(log->is_open() == false, "log file path", logPath);
|
|
}
|
|
|
|
// Prints the full paths to the console to let the user know compiling is being done //
|
|
std::cout << std::filesystem::absolute(inpPath) << " -> " << std::filesystem::absolute(outPath) << std::endl;
|
|
|
|
// Create tokens out of the input file //
|
|
LX::InvalidCharInSource::s_Source = &contents;
|
|
LX::InvalidCharInSource::s_SourceFile = &inpPath;
|
|
std::vector<LX::Token>tokens = LX::LexicalAnalyze(contents, len, log.get());
|
|
LX::SafeFlush(log.get());
|
|
|
|
// Turns the tokens into an AST //
|
|
LX::UnexpectedToken::s_Source = &contents;
|
|
LX::UnexpectedToken::s_SourceFile = &inpPath;
|
|
|
|
LX::FileAST AST = LX::TurnTokensIntoAbstractSyntaxTree(tokens, log.get());
|
|
LX::SafeFlush(log.get());
|
|
|
|
// Turns the AST into LLVM IR //
|
|
LX::GenerateIR(AST, inpPath.filename().string(), outPath);
|
|
LX::SafeFlush(log.get());
|
|
|
|
// Returns success
|
|
return 0;
|
|
}
|
|
|
|
catch(LX::RuntimeError& e)
|
|
{
|
|
// Closes the log to save everything outputted to it after logging the error //
|
|
LX::SafeLog(log.get(), LX::LOG_BREAK, "Error thrown of type: ", e.ErrorType(), LX::LOG_BREAK);
|
|
if (log != nullptr) { log->close(); }
|
|
|
|
// Logs the errors type to the console if built as Debug //
|
|
#ifdef _DEBUG
|
|
std::cout << "LX::RuntimeError thrown of type: ";
|
|
LX::PrintStringAsColor(e.ErrorType(), LX::Color::WHITE);
|
|
std::cout << "\n";
|
|
#endif // _DEBUG
|
|
|
|
// Prints the error to the console and returns //
|
|
e.PrintToConsole();
|
|
return -1;
|
|
}
|
|
|
|
// Catches any std errors, there should be none //
|
|
catch (std::exception& e)
|
|
{
|
|
// Closes the log if it is open to get as much info as possible //
|
|
if (log != nullptr) { log->close(); }
|
|
|
|
// Prints the std exception to the console //
|
|
// Any errors here are problems with the code //
|
|
std::cout << "An error occured. Please report this on the github page.\n" << std::endl;
|
|
std::cout << e.what() << std::endl;
|
|
|
|
// Exit code -1 means an undefined error //
|
|
return -1;
|
|
}
|
|
|
|
// Catches errors that i was too lazy to code //
|
|
catch (int)
|
|
{
|
|
// Closes the log if it is open to get as much info as possible //
|
|
if (log != nullptr) { log->close(); }
|
|
|
|
std::cout << "An placeholder error occured. Maybe use a language that wasn't coded by a lazy person.\n" << std::endl;
|
|
|
|
// Exit code -1 means an undefined error //
|
|
return -1;
|
|
}
|
|
|
|
// Default catches any non-specified errors //
|
|
catch (...)
|
|
{
|
|
// Closes the log if it is open to get as much info as possible //
|
|
if (log != nullptr) { log->close(); }
|
|
|
|
// Exit code -1 means an undefined error //
|
|
return -1;
|
|
}
|
|
}
|