mirror of https://github.com/RainMark/lispoo.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
331 lines
8.1 KiB
331 lines
8.1 KiB
#pragma once
|
|
|
|
#include <iostream>
|
|
#include <exception>
|
|
|
|
#include <string>
|
|
#include <vector>
|
|
#include <memory>
|
|
#include <functional>
|
|
#include <unordered_map>
|
|
|
|
namespace lispoo {
|
|
|
|
inline void oops(const std::string& err) {
|
|
std::cerr << err << std::endl;
|
|
std::exit(EXIT_FAILURE);
|
|
}
|
|
|
|
enum class Type {
|
|
Null = 0,
|
|
Float,
|
|
Integer,
|
|
Symbol,
|
|
Atom,
|
|
|
|
List,
|
|
Callable,
|
|
};
|
|
|
|
class Expr {
|
|
public:
|
|
virtual Type type() = 0;
|
|
};
|
|
|
|
template <typename T>
|
|
class Atom: public Expr {
|
|
public:
|
|
explicit Atom(const T& value) : value_(value) {}
|
|
Type type() override { return Type::Atom; }
|
|
const T& value() const { return value_; }
|
|
|
|
private:
|
|
T value_;
|
|
};
|
|
|
|
class Symbol: public Atom<std::string> {
|
|
public:
|
|
explicit Symbol(const std::string& value) : Atom(value) {}
|
|
Type type() override { return Type::Symbol; }
|
|
};
|
|
|
|
class Integer: public Atom<long> {
|
|
public:
|
|
explicit Integer(const long& value) : Atom(value) {}
|
|
Type type() override { return Type::Integer; }
|
|
};
|
|
|
|
class Float: public Atom<double> {
|
|
public:
|
|
explicit Float(const double& value) : Atom(value) {}
|
|
Type type() override { return Type::Float; }
|
|
};
|
|
|
|
class Null: public Expr {
|
|
public:
|
|
Type type() override { return Type::Null; }
|
|
};
|
|
static const std::shared_ptr<Expr> nil = std::make_shared<Null>();
|
|
|
|
class List: public Expr {
|
|
public:
|
|
Type type() override { return Type::List; }
|
|
std::vector<std::shared_ptr<Expr>>& value() { return value_; }
|
|
void append(const std::shared_ptr<Expr>& expr) { value_.emplace_back(expr); }
|
|
|
|
private:
|
|
std::vector<std::shared_ptr<Expr>> value_;
|
|
};
|
|
|
|
class Callable: public Expr {
|
|
public:
|
|
using Fn = std::function<std::shared_ptr<Expr>(const std::shared_ptr<List>&)>;
|
|
|
|
explicit Callable(Fn&& lambda) : lambda_(lambda) {}
|
|
Type type() override { return Type::Callable; }
|
|
const Fn& value() const {
|
|
return lambda_;
|
|
}
|
|
|
|
private:
|
|
Fn lambda_;
|
|
};
|
|
|
|
// environment
|
|
|
|
class Env {
|
|
public:
|
|
Env(const std::shared_ptr<Env>& env) : env_(env) {}
|
|
|
|
std::shared_ptr<Expr> get(const std::string& symbol) const {
|
|
auto it = expr_map_.find(symbol);
|
|
if (it != expr_map_.end()) {
|
|
return it->second;
|
|
}
|
|
if (env_) {
|
|
return env_->get(symbol);
|
|
}
|
|
return nil;
|
|
}
|
|
void put(const std::string& symbol, const std::shared_ptr<Expr>& expr) {
|
|
expr_map_[symbol] = expr;
|
|
}
|
|
|
|
private:
|
|
std::unordered_map<std::string, std::shared_ptr<Expr>> expr_map_;
|
|
std::shared_ptr<Env> env_;
|
|
};
|
|
static std::shared_ptr<Env> global = std::make_shared<Env>(std::shared_ptr<Env>());
|
|
inline void putenv(const std::string& symbol, Callable::Fn&& lambda) {
|
|
global->put(symbol, std::make_shared<Callable>(std::forward<Callable::Fn>(lambda)));
|
|
}
|
|
|
|
// type utils
|
|
|
|
inline bool is_par(char ch) {
|
|
return ch == '(' || ch == ')';
|
|
}
|
|
template <typename T>
|
|
inline decltype(auto) get_value(const std::shared_ptr<Expr>& expr) {
|
|
return std::static_pointer_cast<T>(expr)->value();
|
|
}
|
|
template <Type t>
|
|
inline bool is_type(const std::shared_ptr<Expr>& expr) {
|
|
return expr && expr->type() == t;
|
|
}
|
|
inline bool is_nil(const std::shared_ptr<Expr>& expr) {
|
|
return is_type<Type::Null>(expr);
|
|
}
|
|
inline bool is_number(const std::shared_ptr<Expr>& expr) {
|
|
return is_type<Type::Integer>(expr) || is_type<Type::Float>(expr);
|
|
}
|
|
inline bool is_symbol(const std::shared_ptr<Expr>& expr) {
|
|
return is_type<Type::Symbol>(expr);
|
|
}
|
|
inline bool is_true(const std::shared_ptr<Expr>& expr) {
|
|
if (!is_number(expr)) {
|
|
return false;
|
|
}
|
|
if (is_type<Type::Integer>(expr)) {
|
|
return get_value<Integer>(expr);
|
|
}
|
|
return get_value<Float>(expr);
|
|
}
|
|
|
|
// asserts
|
|
|
|
template <Type t>
|
|
inline void assert_type(const std::shared_ptr<Expr>& expr) {
|
|
if (!is_type<t>(expr)) {
|
|
oops("syntax error");
|
|
}
|
|
}
|
|
inline void assert_len(const std::shared_ptr<Expr>& expr, unsigned long expect) {
|
|
if (!is_type<Type::List>(expr)) {
|
|
oops("assert_len() failed, not List type");
|
|
}
|
|
auto value = get_value<List>(expr);
|
|
if (value.size() != expect) {
|
|
oops("assert_len() failed, expect: " + std::to_string(expect));
|
|
}
|
|
}
|
|
|
|
// parse & evaluate
|
|
|
|
inline void tokenize(const std::string& str, std::vector<std::string>& tokens) {
|
|
for (auto i = 0; i < str.size(); ++i) {
|
|
if (std::isspace(str[i])) {
|
|
continue;
|
|
}
|
|
if (is_par(str[i])) {
|
|
tokens.emplace_back(str.c_str() + i, 1);
|
|
continue;
|
|
}
|
|
auto s = i;
|
|
for (; !(std::isspace(str[i]) || is_par(str[i])); ++i);
|
|
tokens.emplace_back(str.c_str() + s, i - s);
|
|
i--;
|
|
}
|
|
}
|
|
|
|
inline std::shared_ptr<Expr> parse_atom(const std::vector<std::string>& tokens, unsigned long& cursor) {
|
|
auto token = tokens[cursor];
|
|
auto type = Type::Symbol;
|
|
if (std::isdigit(token[0]) || token[0] == '-') {
|
|
type = Type::Integer;
|
|
for (auto i = 1; i < token.size(); ++i) {
|
|
if (token[i] == '.') {
|
|
if (type == Type::Float) {
|
|
oops("parse failed, token: " + token);
|
|
}
|
|
type = Type::Float;
|
|
continue;
|
|
}
|
|
if (!std::isdigit(token[i])) {
|
|
oops("parse failed, token: " + token);
|
|
}
|
|
}
|
|
}
|
|
switch (type) {
|
|
case Type::Symbol:
|
|
return std::make_shared<Symbol>(token);
|
|
case Type::Integer:
|
|
return std::make_shared<Integer>(std::stol(token));
|
|
case Type::Float:
|
|
return std::make_shared<Float>(std::stod(token));
|
|
default:
|
|
break;
|
|
}
|
|
return nil;
|
|
}
|
|
|
|
inline std::shared_ptr<Expr> parse(const std::vector<std::string>& tokens, unsigned long& cursor) {
|
|
if (cursor >= tokens.size()) {
|
|
oops("parse error");
|
|
}
|
|
if (tokens[cursor] == "(") {
|
|
auto list = std::make_shared<List>();
|
|
while (tokens[++cursor] != ")") {
|
|
list->append(parse(tokens, cursor));
|
|
}
|
|
return list;
|
|
}
|
|
return parse_atom(tokens, cursor);
|
|
}
|
|
|
|
inline std::shared_ptr<Expr> eval(const std::shared_ptr<Expr>& expr, const std::shared_ptr<Env>& env) {
|
|
if (!expr) {
|
|
oops("syntax error");
|
|
return nil;
|
|
}
|
|
if (is_number(expr)) {
|
|
return expr;
|
|
}
|
|
if (is_symbol(expr)) {
|
|
return env->get(get_value<Symbol>(expr));
|
|
}
|
|
assert_type<Type::List>(expr);
|
|
auto value = get_value<List>(expr);
|
|
assert_type<Type::Symbol>(value[0]);
|
|
auto name = get_value<Symbol>(value[0]);
|
|
if (name == "quote") {
|
|
assert_len(expr, 2);
|
|
return value[1];
|
|
}
|
|
if (name == "def") {
|
|
assert_len(expr, 3);
|
|
assert_type<Type::Symbol>(value[1]);
|
|
auto symbol = get_value<Symbol>(value[1]);
|
|
if (!is_nil(env->get(symbol))) {
|
|
oops("symbol defined: " + symbol);
|
|
}
|
|
env->put(symbol, eval(value[2], env));
|
|
return nil;
|
|
}
|
|
if (name == "set!") {
|
|
assert_len(expr, 3);
|
|
assert_type<Type::Symbol>(value[1]);
|
|
auto symbol = get_value<Symbol>(value[1]);
|
|
env->put(symbol, eval(value[2], env));
|
|
return nil;
|
|
}
|
|
if (name == "prog") {
|
|
for (auto i = 1; i < value.size(); ++i) {
|
|
eval(value[i], env);
|
|
}
|
|
return nil;
|
|
}
|
|
if (name == "if") {
|
|
// (if (cond) (then body) (else body))
|
|
assert_len(expr, 4);
|
|
if (is_true(eval(value[1], env))) {
|
|
return eval(value[2], env);
|
|
} else {
|
|
return eval(value[3], env);
|
|
}
|
|
}
|
|
if (name == "while") {
|
|
// (while (cond) (loop body))
|
|
assert_len(expr, 3);
|
|
while (is_true(eval(value[1], env))) {
|
|
eval(value[2], env);
|
|
}
|
|
return nil;
|
|
}
|
|
if (name == "lambda") {
|
|
// (lambda (args) (body))
|
|
assert_len(expr, 3);
|
|
auto lambda = [expr, parent = env](const std::shared_ptr<List>& args) {
|
|
auto value = get_value<List>(expr);
|
|
auto symbols = get_value<List>(value[1]);
|
|
// arguments bind
|
|
if (symbols.size() != args->value().size()) {
|
|
oops("arguments error");
|
|
}
|
|
auto env = std::make_shared<Env>(parent);
|
|
for (auto i = 0; i < symbols.size(); ++i) {
|
|
auto symbol = get_value<Symbol>(symbols[i]);
|
|
env->put(symbol, args->value()[i]);
|
|
}
|
|
// eval body
|
|
return eval(value[2], env);
|
|
};
|
|
return std::make_shared<Callable>(std::move(lambda));
|
|
}
|
|
// function/lambda call
|
|
// (symbol arg1 arg2 arg3 ... )
|
|
auto callable = env->get(name);
|
|
if (!callable) {
|
|
oops("unknow symbol: " + name);
|
|
}
|
|
if (!is_type<Type::Callable>(callable)) {
|
|
oops("can't call symbol: " + name);
|
|
}
|
|
auto args = std::make_shared<List>();
|
|
for (auto i = 1; i < value.size(); ++i) {
|
|
args->append(eval(value[i], env));
|
|
}
|
|
return get_value<Callable>(callable)(args);
|
|
}
|
|
|
|
} // namespace lispoo
|
|
|