pluto-src 0.1.1+0.10.4

Sources of Pluto (Lua 5.4 dialect) and logic to build it.
Documentation
#include "Mixed.hpp"

#include <ostream>

#include "Canvas.hpp"
#include "Exception.hpp"
#include "parse_tree.hpp"
#include "SharedPtr.hpp"

NAMESPACE_SOUP
{
	Mixed::Mixed(const Mixed& b)
		: type(b.type)
	{
		switch (type)
		{
		case NONE:
			break;

		case AST_BLOCK:
			SOUP_THROW(Exception("Can't copy this type"));

		case INT:
		case UINT:
			val = b.val;
			break;

		case STRING:
		case FUNC:
		case VAR_NAME:
			val = reinterpret_cast<uint64_t>(new std::string(*reinterpret_cast<std::string*>(b.val)));
			break;

		case MIXED_SP_MIXED_MAP:
			val = reinterpret_cast<uint64_t>(new std::unordered_map<Mixed, SharedPtr<Mixed>>(b.getMixedSpMixedMap()));
			break;

		case CANVAS:
			val = reinterpret_cast<uint64_t>(new Canvas(*reinterpret_cast<Canvas*>(b.val)));
			break;
		}
	}

	Mixed::Mixed(std::unordered_map<Mixed, SharedPtr<Mixed>>&& val)
		: type(MIXED_SP_MIXED_MAP), val(reinterpret_cast<uint64_t>(new std::unordered_map<Mixed, SharedPtr<Mixed>>(std::move(val))))
	{
	}

	Mixed::Mixed(astBlock* val)
		: type(AST_BLOCK), val(reinterpret_cast<uint64_t>(val))
	{
	}

	Mixed::Mixed(Canvas&& val)
		: type(CANVAS), val(reinterpret_cast<uint64_t>(new Canvas(std::move(val))))
	{
	}

	void Mixed::release()
	{
		switch (type)
		{
		case NONE:
		case INT:
		case UINT:
			break;

		case STRING:
		case FUNC:
		case VAR_NAME:
			delete reinterpret_cast<std::string*>(val);
			break;

		case MIXED_SP_MIXED_MAP:
			delete reinterpret_cast<std::unordered_map<Mixed, SharedPtr<Mixed>>*>(val);
			break;

		case AST_BLOCK:
			delete reinterpret_cast<astBlock*>(val);
			break;

		case CANVAS:
			delete reinterpret_cast<Canvas*>(val);
			break;
		}
	}

	std::ostream& operator<<(std::ostream& os, const Mixed& v)
	{
		os << v.toString();
		return os;
	}

	const char* Mixed::getTypeName(Type t) noexcept
	{
		switch (t)
		{
		default:
			break;

		case INT:
			return "int";

		case UINT:
			return "uint";

		case STRING:
			return "string";

		case FUNC:
			return "func";

		case VAR_NAME:
			return "var name";

		case AST_BLOCK:
			return "astBlock";
		}
		return "complex type";
	}

	const char* Mixed::getTypeName() const noexcept
	{
		return getTypeName(type);
	}

	std::string Mixed::toString(const std::string& ast_block_prefix) const noexcept
	{
		if (type == INT)
		{
			return std::to_string((int64_t)val);
		}
		if (type == UINT)
		{
			return std::to_string(val);
		}
		if (type == STRING)
		{
			return *reinterpret_cast<std::string*>(val);
		}
		if (type == AST_BLOCK)
		{
			return reinterpret_cast<astBlock*>(val)->toString(ast_block_prefix);
		}
		return {};
	}

	std::string Mixed::toStringWithFallback() const noexcept
	{
		if (auto str = toString(); !str.empty())
		{
			return str;
		}
		std::string str(1, '[');
		str.append(getTypeName());
		str.push_back(']');
		return str;
	}

	void Mixed::assertType(Type e) const
	{
		SOUP_IF_UNLIKELY (type != e)
		{
			std::string str = "Expected Mixed to hold ";
			str.append(getTypeName(e));
			str.append(", found ");
			str.append(getTypeName());
			SOUP_THROW(Exception(std::move(str)));
		}
	}

	int64_t Mixed::getInt() const
	{
		assertType(INT);
		return (int64_t)val;
	}

	uint64_t Mixed::getUInt() const
	{
		assertType(UINT);
		return val;
	}

	std::string& Mixed::getString() const
	{
		assertType(STRING);
		return *reinterpret_cast<std::string*>(val);
	}

	std::string& Mixed::getFunc() const
	{
		assertType(FUNC);
		return *reinterpret_cast<std::string*>(val);
	}

	std::string& Mixed::getVarName() const
	{
		assertType(VAR_NAME);
		return *reinterpret_cast<std::string*>(val);
	}

	std::unordered_map<Mixed, SharedPtr<Mixed>>& Mixed::getMixedSpMixedMap() const
	{
		assertType(MIXED_SP_MIXED_MAP);
		return *reinterpret_cast<std::unordered_map<Mixed, SharedPtr<Mixed>>*>(val);
	}

	astBlock& Mixed::getAstBlock() const
	{
		assertType(AST_BLOCK);
		return *reinterpret_cast<astBlock*>(val);
	}

	Canvas& Mixed::getCanvas() const
	{
		assertType(CANVAS);
		return *reinterpret_cast<Canvas*>(val);
	}
}