First work on driver

This commit is contained in:
Thomas Avé 2016-12-30 17:38:42 +01:00
parent e62eae124e
commit 493ec163ec
6 changed files with 123 additions and 10 deletions

64
include/Parsodus/driver.h Normal file
View File

@ -0,0 +1,64 @@
#pragma once
#ifndef PARSODUS_DRIVER_H
#define PARSODUS_DRIVER_H
#include <memory>
#include <string>
#include "Parsodus/backendmanager.h"
namespace pds {
/**
* The main driver for Parsodus
*/
class Driver {
public:
/**
* Constructor
*
* @param backends The backendmanager, prepared with all needed supported backends
* @param inputfile An istream which should be read to be used as token rules specifications
* @param outputdir A string representing the directory where generated files should be places
* @param language The language to generate output for (backends is queried for this language)
* @param lexername The name to give to the generated lexer that is used by the parser, this gets cleaned to only contains alphanumeric chars or underscore and start with a non-digit (AKA a valid identifier)
* @param parsername The name to give to the generated parser, this gets cleaned to only contains alphanumeric chars or underscore and start with a non-digit (AKA a valid identifier)
*/
Driver(std::unique_ptr<BackendManager> backends, std::istream& inputfile, std::string outputdir, std::string language, std::string lexername, std::string parsername);
/**
* Destructor
*/
~Driver();
/**
* Run this driver, all the preparation should happen when calling the constructor
*
* @return The status code this would return if it were a main function
*/
int run();
private:
std::unique_ptr<BackendManager> m_backends;
std::istream& m_inputfile;
std::string m_outputdir;
std::string m_language;
std::string m_lexername;
std::string m_parsername;
};
/**
* Used to throw errors when the driver could generate a valid lexer name or find the backend for a language
*/
class DriverException: public std::exception {
public:
DriverException(std::string what);
virtual const char* what() const throw();
private:
std::string m_what;
};
}
#endif //PARSODUS_DRIVER_H

View File

@ -25,13 +25,15 @@ add_library(Parsodus-tables
lrtables/SLR1Itemset.cpp
)
add_library(Parsodus-backends
backends/cpp.cpp
)
# add_library(Parsodus-backends
# )
add_library(pds
backend.cpp
driver.cpp
template.cpp
inputparser.cpp
backendmanager.cpp
"${CMAKE_CURRENT_BINARY_DIR}/ParsodusLexer.cpp"
)
@ -42,7 +44,7 @@ add_executable(Parsodus
target_link_libraries(Parsodus
Parsodus-util
Parsodus-tables
Parsodus-backends
# Parsodus-backends
pds
mstch::mstch)

View File

@ -1,2 +1,43 @@
#include "Parsodus/grammar.h"
#include "Parsodus/driver.h"
#include "Parsodus/inputparser.h"
#include "Parsodus/config.h"
#include <fstream>
namespace {
/**
* Filter only valid identifier chars: alphanumeric, and not starting with a digit
*/
std::string clean(std::string in) {
std::string s;
for (char c : in) {
if ((s.length() && std::isalnum(c)) || std::isalpha(c) || c == '_')
s += c;
}
return s;
}
}
namespace pds {
Driver::Driver(std::unique_ptr<BackendManager> backends, std::istream& inputfile, std::string outputdir, std::string language, std::string lexername, std::string parsername):
m_backends(std::move(backends)), m_inputfile(inputfile), m_outputdir(outputdir), m_language(language), m_lexername(clean(lexername)), m_parsername(clean(parsername)) {
}
Driver::~Driver(){}
int Driver::run() {
if (!m_lexername.length()) throw DriverException("no valid lexer name possible");
if (!m_parsername.length()) throw DriverException("no valid parser name possible");
Config config = InputParser::parseInput(m_inputfile);
Backend* back = m_backends->findBackend(m_language, config.parserType);
if (!back) throw DriverException("Could not find a valid backend for language " + m_language + " and the given type of parser");
back->generateParser([this](std::string filename) -> std::unique_ptr<std::ostream> {
return std::unique_ptr<std::ostream>(new std::ofstream(m_outputdir + "/" + filename));
},
m_lexername, config);
return 0;
}
}

View File

@ -1,5 +1,6 @@
#include "ParsodusLexer.h"
#include "Parsodus/inputparser.h"
#include "Parsodus/util/parserType.h"
namespace pds {
@ -32,7 +33,7 @@ namespace pds {
lexColon(lex, token);
if(token.type == ParsodusLexer::PARSERTYPE) {
if(token.content == "lalr(1)") {
config.parserType = Config::ParserType::LALR_1;
config.parserType = util::ParserType::LALR_1;
} else
throw InputParserException("Unkown parser type");
} else {

View File

@ -2,9 +2,11 @@
#include <fstream>
#include "optparse.h"
#include "Parsodus/inputparser.h"
#include "Parsodus/backendmanager.h"
#include "Parsodus/lrtables/generator.h"
#include "Parsodus/lrtables/SLR1Itemset.h"
#include "Parsodus/backends/cppLR.h"
#include "Parsodus/util/parserType.h"
int main(int argc, char** argv) {
@ -30,8 +32,11 @@ int main(int argc, char** argv) {
auto config = pds::InputParser::parseInput(infile);
pds::BackendManager b;
b.registerLR<pds::backends::CppLRBackend>();
pds::Backend* back = b.findBackend("c++", pds::util::ParserType::LR_0);
std::cout << back->getName() << std::endl;
// Reporting what the inputparser found, to be removed...
std::cout << "Start: " << config.grammar.start << std::endl;
for(auto a: config.grammar.terminals)

View File

@ -3,7 +3,7 @@ add_executable(Parsodus-test
)
target_link_libraries(Parsodus-test
#Parsodus-tables
Parsodus-backends
# Parsodus-backends
pds
mstch::mstch
${GTEST_BOTH_LIBRARIES}