blaze lisp
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.

234 lines
4.1 KiB

2 years ago
/*
* Copyright (C) 2023 Riyyi
*
* SPDX-License-Identifier: MIT
*/
#include <cstdint> // int64_t
#include <memory>
#include <string>
2 years ago
#include "ast.h"
#include "environment.h"
#include "error.h"
#include "forward.h"
#include "printer.h"
#include "types.h"
2 years ago
namespace blaze {
Collection::Collection(const std::list<ValuePtr>& nodes)
: m_nodes(nodes)
{
}
Collection::Collection(ValueListConstIt begin, ValueListConstIt end)
: m_nodes(ValueList(begin, end))
{
}
void Collection::add(ValuePtr node)
2 years ago
{
if (node == nullptr) {
return;
}
2 years ago
m_nodes.push_back(node);
}
void Collection::addFront(ValuePtr node)
{
if (node == nullptr) {
return;
}
m_nodes.push_front(node);
}
ValueList Collection::rest() const
{
auto start = (m_nodes.size() > 0) ? std::next(m_nodes.begin()) : m_nodes.end();
return ValueList(start, m_nodes.end());
}
2 years ago
// -----------------------------------------
List::List(const std::list<ValuePtr>& nodes)
: Collection(nodes)
{
}
List::List(ValueListConstIt begin, ValueListConstIt end)
: Collection(begin, end)
{
}
// -----------------------------------------
Vector::Vector(const std::list<ValuePtr>& nodes)
: Collection(nodes)
{
}
Vector::Vector(ValueListConstIt begin, ValueListConstIt end)
: Collection(begin, end)
{
}
// -----------------------------------------
HashMap::HashMap(const Elements& elements)
: m_elements(elements)
{
}
void HashMap::add(const std::string& key, ValuePtr value)
{
if (value == nullptr) {
return;
}
m_elements.insert_or_assign(key, value);
}
void HashMap::add(ValuePtr key, ValuePtr value)
{
if (key == nullptr || value == nullptr) {
return;
}
m_elements.insert_or_assign(getKeyString(key), value);
}
void HashMap::remove(const std::string& key)
{
m_elements.erase(key);
}
void HashMap::remove(ValuePtr key)
{
if (key == nullptr) {
return;
}
m_elements.erase(getKeyString(key));
}
bool HashMap::exists(const std::string& key)
{
return m_elements.find(key) != m_elements.end();
}
bool HashMap::exists(ValuePtr key)
{
return exists(getKeyString(key));
}
ValuePtr HashMap::get(const std::string& key)
{
if (!exists(key)) {
return nullptr;
}
return m_elements[key];
}
ValuePtr HashMap::get(ValuePtr key)
{
return get(getKeyString(key));
}
std::string HashMap::getKeyString(ValuePtr key)
{
if (!is<String>(key.get()) && !is<Keyword>(key.get())) {
Error::the().add(format("wrong argument type: string or keyword, {}", key));
return {};
}
return is<String>(key.get())
? std::static_pointer_cast<String>(key)->data()
: std::static_pointer_cast<Keyword>(key)->keyword();
}
// -----------------------------------------
2 years ago
String::String(const std::string& data)
: m_data(data)
{
}
// -----------------------------------------
Keyword::Keyword(const std::string& data)
: m_data(std::string(1, 0x7f) + data) // 127
{
}
// -----------------------------------------
2 years ago
Number::Number(int64_t number)
: m_number(number)
{
}
// -----------------------------------------
Constant::Constant(State state)
: m_state(state)
{
}
Constant::Constant(bool state)
: m_state(state ? Constant::True : Constant::False)
2 years ago
{
}
// -----------------------------------------
Symbol::Symbol(const std::string& symbol)
: m_symbol(symbol)
{
}
// -----------------------------------------
Function::Function(const std::string& name, FunctionType function)
: m_name(name)
, m_function(function)
{
}
// -----------------------------------------
Lambda::Lambda(const std::vector<std::string>& bindings, ValuePtr body, EnvironmentPtr env)
: m_bindings(bindings)
, m_body(body)
, m_env(env)
{
}
Lambda::Lambda(std::shared_ptr<Lambda> that, bool is_macro)
: m_bindings(that->m_bindings)
, m_body(that->m_body)
, m_env(that->m_env)
, m_is_macro(is_macro)
{
}
// -----------------------------------------
Atom::Atom(ValuePtr pointer)
: m_value(pointer)
{
}
2 years ago
} // namespace blaze
// -----------------------------------------
void Formatter<blaze::ValuePtr>::format(Builder& builder, blaze::ValuePtr value) const
{
blaze::Printer printer;
return Formatter<std::string>::format(builder, printer.printNoErrorCheck(value));
}