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 lrtables/SLR1Itemset.cpp
) )
add_library(Parsodus-backends # add_library(Parsodus-backends
backends/cpp.cpp # )
)
add_library(pds add_library(pds
backend.cpp
driver.cpp driver.cpp
template.cpp
inputparser.cpp inputparser.cpp
backendmanager.cpp
"${CMAKE_CURRENT_BINARY_DIR}/ParsodusLexer.cpp" "${CMAKE_CURRENT_BINARY_DIR}/ParsodusLexer.cpp"
) )
@ -42,7 +44,7 @@ add_executable(Parsodus
target_link_libraries(Parsodus target_link_libraries(Parsodus
Parsodus-util Parsodus-util
Parsodus-tables Parsodus-tables
Parsodus-backends # Parsodus-backends
pds pds
mstch::mstch) 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 "ParsodusLexer.h"
#include "Parsodus/inputparser.h" #include "Parsodus/inputparser.h"
#include "Parsodus/util/parserType.h"
namespace pds { namespace pds {
@ -32,7 +33,7 @@ namespace pds {
lexColon(lex, token); lexColon(lex, token);
if(token.type == ParsodusLexer::PARSERTYPE) { if(token.type == ParsodusLexer::PARSERTYPE) {
if(token.content == "lalr(1)") { if(token.content == "lalr(1)") {
config.parserType = Config::ParserType::LALR_1; config.parserType = util::ParserType::LALR_1;
} else } else
throw InputParserException("Unkown parser type"); throw InputParserException("Unkown parser type");
} else { } else {

View File

@ -2,9 +2,11 @@
#include <fstream> #include <fstream>
#include "optparse.h" #include "optparse.h"
#include "Parsodus/inputparser.h" #include "Parsodus/inputparser.h"
#include "Parsodus/backendmanager.h"
#include "Parsodus/lrtables/generator.h" #include "Parsodus/lrtables/generator.h"
#include "Parsodus/lrtables/SLR1Itemset.h" #include "Parsodus/lrtables/SLR1Itemset.h"
#include "Parsodus/backends/cppLR.h"
#include "Parsodus/util/parserType.h"
int main(int argc, char** argv) { int main(int argc, char** argv) {
@ -30,7 +32,10 @@ int main(int argc, char** argv) {
auto config = pds::InputParser::parseInput(infile); 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... // Reporting what the inputparser found, to be removed...
std::cout << "Start: " << config.grammar.start << std::endl; std::cout << "Start: " << config.grammar.start << std::endl;

View File

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