Initial commit

This commit is contained in:
hans 2021-11-06 00:24:04 +01:00
commit 02afb71c84
11 changed files with 387 additions and 0 deletions

1
.gitignore vendored Normal file
View file

@ -0,0 +1 @@
build

14
CMakeLists.txt Normal file
View file

@ -0,0 +1,14 @@
cmake_minimum_required(VERSION 3.10)
project(biscuit_interpreter)
set(CMAKE_EXPORT_COMPILE_COMMANDS ON)
add_executable(biscuit_interpreter)
target_sources( biscuit_interpreter PRIVATE
src/main.cpp
src/parse.cpp
src/keywords.cpp )
include_directories(${CMAKE_SOURCE_DIR})

BIN
biscuit_doc.odt Normal file

Binary file not shown.

2
examples/helloworld.bisc Normal file
View file

@ -0,0 +1,2 @@
print #hello_world
print 1.01234

18
src/.ycm_extra_conf.py Normal file
View file

@ -0,0 +1,18 @@
import os
import ycm_core
flags = [
'-Wall',
'-Wextra',
'-Wno-long-long',
'-Wno-variadic-macros',
'-std=c++20',
]
SOURCE_EXTENSIONS = [ '.cpp', '.cxx', '.cc', '.c', ]
def FlagsForFile( filename, **kwargs ):
return {
'flags': flags,
'do_cache': True
}

18
src/keywords.cpp Normal file
View file

@ -0,0 +1,18 @@
#include <string>
#include <functional>
#include "keywords.hpp"
const std::vector<Keyword> keywords =
{
{
.name = "print",
.expected_num_args = -1,
.expected_token_types = {TokenType::ID_OR_LIT},
.expected_value_types = {ValueType::ANY}
},
{
.name = "exit",
.expected_num_args = 0
}
};

20
src/keywords.hpp Normal file
View file

@ -0,0 +1,20 @@
#ifndef BISCUIT_KEYWORDS_HPP_INCLUDED
#define BISCUIT_KEYWORDS_HPP_INCLUDED
#include <string>
#include <vector>
#include <array>
#include "parse.hpp"
struct Keyword
{
const std::string name;
const int expected_num_args; // -1 means: 1 to infinity, in which case all args are of same type
const std::vector<TokenType> expected_token_types;
const std::vector<ValueType> expected_value_types;
};
extern const std::vector<Keyword> keywords;
#endif

61
src/main.cpp Normal file
View file

@ -0,0 +1,61 @@
#include <iostream>
#include <vector>
#include <string>
#include <array>
#include <fstream>
#include <algorithm>
#include <exception>
#include "parse.hpp"
std::ifstream get_infile(int argc, char ** argv)
{
if(argc != 2)
{
std::cerr << "Usage: " << argv[0] << " <infile>" << std::endl;
exit(EXIT_FAILURE);
}
std::ifstream infile(argv[1]);
if(!infile)
{
std::cerr << "Could not open '" << argv[1] << "'" << std::endl;
exit(EXIT_FAILURE);
}
return infile;
}
int main(int argc, char * argv[])
{
std::ifstream infile = get_infile(argc, argv);
std::string word;
while(infile >> word)
{
try
{
Symbol sym(word);
sym.print();
}
catch(MalformedIdentifierExcept& exc)
{
std::cout << "Malformed identifier: '" << exc.malformed_str << "'"
<< "\nAborting..." << std::endl;
exit(EXIT_FAILURE);
}
catch(UnknownKeywordExcept& exc)
{
std::cout << "Unknown Keyword: '" << exc.unknown_keyword << "'"
<< "\nAborting..." << std::endl;
exit(EXIT_FAILURE);
}
catch(WrongSymbolExcept& exc)
{
std::cout << "Wrong symbol type\n Aborting..." << std::endl;
exit(EXIT_FAILURE);
}
}
return 0;
}

148
src/parse.cpp Normal file
View file

@ -0,0 +1,148 @@
#include <iostream>
#include <exception>
#include <array>
#include "keywords.hpp"
#include "parse.hpp"
bool is_keyword(std::string str)
{
for(char &c : str)
c = std::tolower(c);
for(auto &keyword : keywords)
if(str == keyword.name)
return true;
return 0;
}
bool is_string_literal(const std::string& str)
{
if(str[0] == '#')
return true;
else
return false;
}
bool is_decimal_literal(const std::string& str)
{
for(const char &c : str)
if(!std::isdigit(c) && c != '.')
return false;
return true;
}
bool is_identifier(const std::string& str)
{
for(const char& c : str)
if(!(std::isalpha(c) || c == '_'))
return false;
return true;
}
bool Token::parse_as_keyword(const std::string &str)
{
if(!is_keyword(str))
return false;
type = TokenType::KEYWORD;
val_string = str;
return true;
}
bool Token::parse_as_literal(const std::string &str)
{
if(is_string_literal(str))
{
type = TokenType::LITERAL;
literal_type = ValueType::STRING;
auto str_copy = str;
str_copy.erase(0, 1);
val_string = str_copy;
return true;
}
else if(is_decimal_literal(str))
{
type = TokenType::LITERAL;
literal_type = ValueType::DECIMAL;
val_float = std::stof(str);
val_string = str;
return true;
}
else
{
return false;
}
}
bool Token::parse_as_identifier(const std::string &str)
{
if(is_identifier(str))
{
type = TokenType::IDENTIFIER;
val_string = str;
}
return false;
}
Token::Token(const std::string &str)
{
if (parse_as_keyword(str));
else if(parse_as_literal(str));
else if(parse_as_identifier(str));
else
{
throw MalformedIdentifierExcept(str);
}
}
void Token::print() const
{
switch(type)
{
case TokenType::IDENTIFIER:
std::cout << "Identifier: \"" << val_string << "\"";
break;
case TokenType::KEYWORD:
std::cout << "Keyword: \"" << val_string << "\"";
break;
case TokenType::LITERAL:
std::cout << "Literal of type ";
if(literal_type == ValueType::DECIMAL)
std::cout << "decimal: " << val_float;
else if(literal_type == ValueType::STRING)
std::cout << "string: \"" << val_string << "\"";
break;
case TokenType::ID_OR_LIT:
std::cerr << "Identifier or Literal?! Something went terribly wrong!!";
}
std::cout << std::endl;
}
Instruction::Instruction(std::vector<Token>& _token_list)
{
tokens = std::move(_token_list);
// resolve our keyword
auto keyword_it = std::find(keywords.begin(), keywords.end(), tokens[0].val_string);
// Throw if unknown
if(keyword_it == keywords.end())
throw UnknownKeywordExcept(tokens[0].val_string);
else
keyword_ptr = &*keyword_it;
// Make sure the keyword gets the number of arguments it needs
if(keyword_ptr->expected_num_args != -1 && (keyword_ptr->expected_num_args != tokens.size() - 1))
throw WrongArgumentCountExcept(keyword_ptr->name, keyword_ptr->expected_num_args, tokens.size() - 1);
}

94
src/parse.hpp Normal file
View file

@ -0,0 +1,94 @@
#ifndef BISCUIT_PARSE_HPP_INCLUDED
#define BISCUIT_PARSE_HPP_INCLUDED
#include <exception>
#include <string>
#include <vector>
#include "keywords.hpp"
enum TokenType
{
KEYWORD,
IDENTIFIER,
LITERAL,
ID_OR_LIT
};
enum ValueType
{
STRING,
DECIMAL,
ANY
};
struct MalformedIdentifierExcept : public std::exception
{
std::string malformed_str;
MalformedIdentifierExcept(const std::string& arg_str)
: malformed_str(arg_str) {}
};
struct UnknownKeywordExcept : public std::exception
{
std::string unknown_keyword;
UnknownKeywordExcept(const std::string& arg_str)
: unknown_keyword(arg_str) {}
};
struct WrongArgumentCountExcept : public std::exception
{
int expected, got;
std::string keyword_name;
WrongArgumentCountExcept(std::string _name, int _expected, int _got)
: expected(_expected), got(_got), keyword_name(_name) {}
};
struct WrongTokenExcept : public std::exception
{
TokenType expected, got;
std::string keyword_name;
WrongTokenExcept(std::string _name, const TokenType& _expected, const TokenType& _got)
: expected(_expected), got(_got), keyword_name(_name) {}
};
struct TypeErrorExcept : public std::exception
{
ValueType expected, got;
std::string keyword_name;
TypeErrorExcept(std::string _name, const ValueType& _expected, const ValueType& _got)
: expected(_expected), got(_got), keyword_name(_name) {}
};
struct Token
{
private:
bool parse_as_keyword(const std::string &str);
bool parse_as_literal(const std::string &str);
// if a word is not a keyword or literal, it must be identifier
bool parse_as_identifier(const std::string &str);
public:
enum TokenType type;
enum ValueType literal_type;
float val_float;
std::string val_string;
Token(const std::string &str);
void print() const;
};
class Instruction
{
std::vector<Token> tokens;
Keyword *keyword_ptr;
public:
// Try to parse a list of symbols as instructions
Instruction(std::vector<Token>& _token_list);
};
#endif

11
src/runtime.hpp Normal file
View file

@ -0,0 +1,11 @@
#ifndef BISCUIT_RUNTIME_HPP_INCLUDED
#define BISCUIT_RUNTIME_HPP_INCLUDED
#include <vector>
struct Context
{
std::vector<
};
#endif