244 lines
7.5 KiB
C
244 lines
7.5 KiB
C
|
/*
|
||
|
* Copyright (C) 2016 Mark Roszko <mark.roszko@gmail.com>
|
||
|
* Copyright (C) 2016 QiEDA Developers
|
||
|
*
|
||
|
* This program is free software: you can redistribute it and/or modify
|
||
|
* it under the terms of the GNU General Public License as published by
|
||
|
* the Free Software Foundation, either version 2 of the License, or
|
||
|
* (at your option) any later version.
|
||
|
*
|
||
|
* This program is distributed in the hope that it will be useful,
|
||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
* GNU General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
*/
|
||
|
|
||
|
#ifndef SEXPR_H_
|
||
|
#define SEXPR_H_
|
||
|
|
||
|
#include <cstdint>
|
||
|
#include <string>
|
||
|
#include <vector>
|
||
|
#include "sexpr/isexprable.h"
|
||
|
#include "sexpr/sexpr_exception.h"
|
||
|
|
||
|
|
||
|
namespace SEXPR
|
||
|
{
|
||
|
using std::int32_t;
|
||
|
using std::int64_t;
|
||
|
|
||
|
enum SEXPR_TYPE
|
||
|
{
|
||
|
SEXPR_TYPE_LIST,
|
||
|
SEXPR_TYPE_ATOM_INTEGER,
|
||
|
SEXPR_TYPE_ATOM_DOUBLE,
|
||
|
SEXPR_TYPE_ATOM_STRING,
|
||
|
SEXPR_TYPE_ATOM_SYMBOL,
|
||
|
};
|
||
|
|
||
|
typedef std::vector<class SEXPR *> SEXPR_VECTOR;
|
||
|
|
||
|
class SEXPR
|
||
|
{
|
||
|
protected:
|
||
|
SEXPR_TYPE m_type;
|
||
|
SEXPR(SEXPR_TYPE type, size_t lineNumber);
|
||
|
SEXPR(SEXPR_TYPE type);
|
||
|
size_t m_lineNumber;
|
||
|
|
||
|
public:
|
||
|
virtual ~SEXPR() {};
|
||
|
bool IsList() const { return m_type == SEXPR_TYPE_LIST; }
|
||
|
bool IsSymbol() const { return m_type == SEXPR_TYPE_ATOM_SYMBOL; }
|
||
|
bool IsString() const { return m_type == SEXPR_TYPE_ATOM_STRING; }
|
||
|
bool IsDouble() const { return m_type == SEXPR_TYPE_ATOM_DOUBLE; }
|
||
|
bool IsInteger() const { return m_type == SEXPR_TYPE_ATOM_INTEGER; }
|
||
|
void AddChild(SEXPR* child);
|
||
|
SEXPR_VECTOR const * GetChildren() const;
|
||
|
SEXPR * GetChild(size_t idx) const;
|
||
|
size_t GetNumberOfChildren() const;
|
||
|
int64_t GetLongInteger() const;
|
||
|
int32_t GetInteger() const;
|
||
|
float GetFloat() const;
|
||
|
double GetDouble() const;
|
||
|
std::string const & GetString() const;
|
||
|
std::string const & GetSymbol() const;
|
||
|
SEXPR_LIST* GetList();
|
||
|
std::string AsString(size_t level = 0);
|
||
|
size_t GetLineNumber() { return m_lineNumber; }
|
||
|
};
|
||
|
|
||
|
struct SEXPR_INTEGER : public SEXPR
|
||
|
{
|
||
|
int64_t m_value;
|
||
|
SEXPR_INTEGER(int64_t value) : SEXPR(SEXPR_TYPE_ATOM_INTEGER), m_value(value) {};
|
||
|
SEXPR_INTEGER(int64_t value, int lineNumber) : SEXPR(SEXPR_TYPE_ATOM_INTEGER, lineNumber), m_value(value) {};
|
||
|
};
|
||
|
|
||
|
struct SEXPR_DOUBLE : public SEXPR
|
||
|
{
|
||
|
double m_value;
|
||
|
SEXPR_DOUBLE(double value) : SEXPR(SEXPR_TYPE_ATOM_DOUBLE), m_value(value) {};
|
||
|
SEXPR_DOUBLE(double value, int lineNumber) : SEXPR(SEXPR_TYPE_ATOM_DOUBLE, lineNumber), m_value(value) {};
|
||
|
};
|
||
|
|
||
|
struct SEXPR_STRING : public SEXPR
|
||
|
{
|
||
|
std::string m_value;
|
||
|
SEXPR_STRING(std::string value) : SEXPR(SEXPR_TYPE_ATOM_STRING), m_value(value) {};
|
||
|
SEXPR_STRING(std::string value, int lineNumber) : SEXPR(SEXPR_TYPE_ATOM_STRING, lineNumber), m_value(value) {};
|
||
|
};
|
||
|
|
||
|
struct SEXPR_SYMBOL : public SEXPR
|
||
|
{
|
||
|
std::string m_value;
|
||
|
SEXPR_SYMBOL(std::string value) : SEXPR(SEXPR_TYPE_ATOM_SYMBOL), m_value(value) {};
|
||
|
SEXPR_SYMBOL(std::string value, int lineNumber) : SEXPR(SEXPR_TYPE_ATOM_SYMBOL, lineNumber), m_value(value) {};
|
||
|
};
|
||
|
|
||
|
struct _OUT_STRING
|
||
|
{
|
||
|
bool _Symbol;
|
||
|
const std::string& _String;
|
||
|
};
|
||
|
|
||
|
inline _OUT_STRING AsSymbol(const std::string& str)
|
||
|
{
|
||
|
struct _OUT_STRING ret = { true, str };
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
inline _OUT_STRING AsString(const std::string& str)
|
||
|
{
|
||
|
struct _OUT_STRING ret = { false, str };
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
struct _IN_STRING
|
||
|
{
|
||
|
bool _Symbol;
|
||
|
std::string& _String;
|
||
|
|
||
|
};
|
||
|
|
||
|
inline _IN_STRING AsSymbol(std::string& str)
|
||
|
{
|
||
|
struct _IN_STRING ret = { true, str };
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
inline _IN_STRING AsString(std::string& str)
|
||
|
{
|
||
|
struct _IN_STRING ret = { false, str };
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
class SEXPR_SCAN_ARG {
|
||
|
friend class SEXPR_LIST;
|
||
|
public:
|
||
|
SEXPR_SCAN_ARG(int32_t* value) : type(INT) { u.int_value = value; }
|
||
|
SEXPR_SCAN_ARG(int64_t* value) : type(LONGINT) { u.lint_value = value; }
|
||
|
SEXPR_SCAN_ARG(double* value) : type(DOUBLE) { u.dbl_value = value; }
|
||
|
SEXPR_SCAN_ARG(std::string* value) : type(STRING) { u.str_value = value; }
|
||
|
SEXPR_SCAN_ARG(_IN_STRING& value) : type(SEXPR_STRING) { u.sexpr_str = &value; }
|
||
|
SEXPR_SCAN_ARG(const std::string* value) : type(STRING_COMP) { str_value = *value; }
|
||
|
SEXPR_SCAN_ARG(std::string value) : type(STRING_COMP) { str_value = value; }
|
||
|
SEXPR_SCAN_ARG(const char* value) : type(STRING_COMP) { str_value = value; }
|
||
|
|
||
|
private:
|
||
|
enum Type { INT, DOUBLE, STRING, LONGINT, STRING_COMP, SEXPR_STRING};
|
||
|
Type type;
|
||
|
union {
|
||
|
int64_t* lint_value;
|
||
|
int32_t* int_value;
|
||
|
double* dbl_value;
|
||
|
std::string* str_value;
|
||
|
_IN_STRING* sexpr_str;
|
||
|
} u;
|
||
|
|
||
|
std::string str_value;
|
||
|
};
|
||
|
|
||
|
class SEXPR_CHILDREN_ARG {
|
||
|
friend class SEXPR_LIST;
|
||
|
public:
|
||
|
SEXPR_CHILDREN_ARG(int32_t value) : type(INT) { u.int_value = value; }
|
||
|
SEXPR_CHILDREN_ARG(int64_t value) : type(LONGINT) { u.lint_value = value; }
|
||
|
SEXPR_CHILDREN_ARG(double value) : type(DOUBLE) { u.dbl_value = value; }
|
||
|
SEXPR_CHILDREN_ARG(std::string value) : type(STRING) { str_value = value; }
|
||
|
SEXPR_CHILDREN_ARG(const char* value) : type(STRING) { str_value = value; }
|
||
|
SEXPR_CHILDREN_ARG(const _OUT_STRING& value) : type(SEXPR_STRING) { str_value = value._String; u.symbol = value._Symbol; }
|
||
|
SEXPR_CHILDREN_ARG(SEXPR* ptr) : type(SEXPR_ATOM) { u.sexpr_ptr = ptr; }
|
||
|
|
||
|
private:
|
||
|
enum Type { INT, DOUBLE, STRING, LONGINT, SEXPR_STRING, SEXPR_ATOM };
|
||
|
Type type;
|
||
|
union {
|
||
|
int64_t lint_value;
|
||
|
int32_t int_value;
|
||
|
double dbl_value;
|
||
|
SEXPR* sexpr_ptr;
|
||
|
bool symbol;
|
||
|
} u;
|
||
|
std::string str_value;
|
||
|
};
|
||
|
|
||
|
class SEXPR_LIST : public SEXPR
|
||
|
{
|
||
|
public:
|
||
|
SEXPR_LIST() : SEXPR(SEXPR_TYPE_LIST), m_inStreamChild(0) {};
|
||
|
SEXPR_LIST(int lineNumber) : SEXPR(SEXPR_TYPE_LIST, lineNumber), m_inStreamChild(0) {};
|
||
|
|
||
|
template <typename... Args>
|
||
|
SEXPR_LIST(const Args&... args) : SEXPR(SEXPR_TYPE_LIST), m_inStreamChild(0)
|
||
|
{
|
||
|
AddChildren(args...);
|
||
|
};
|
||
|
|
||
|
SEXPR_VECTOR m_children;
|
||
|
|
||
|
template <typename... Args>
|
||
|
size_t Scan(const Args&... args)
|
||
|
{
|
||
|
SEXPR_SCAN_ARG arg_array[] = { args... };
|
||
|
return doScan(arg_array, sizeof...(Args));
|
||
|
}
|
||
|
|
||
|
template <typename... Args>
|
||
|
void AddChildren(const Args&... args)
|
||
|
{
|
||
|
SEXPR_CHILDREN_ARG arg_array[] = { args... };
|
||
|
doAddChildren(arg_array, sizeof...(Args));
|
||
|
}
|
||
|
|
||
|
virtual ~SEXPR_LIST();
|
||
|
|
||
|
friend SEXPR_LIST& operator<< (SEXPR_LIST& list, double value);
|
||
|
friend SEXPR_LIST& operator<< (SEXPR_LIST& list, float value);
|
||
|
friend SEXPR_LIST& operator<< (SEXPR_LIST& list, int64_t value);
|
||
|
friend SEXPR_LIST& operator<< (SEXPR_LIST& list, int32_t value);
|
||
|
friend SEXPR_LIST& operator<< (SEXPR_LIST& list, std::string value);
|
||
|
friend SEXPR_LIST& operator<< (SEXPR_LIST& list, const _OUT_STRING setting);
|
||
|
friend SEXPR_LIST& operator<< (SEXPR_LIST& list, const ISEXPRABLE& obj);
|
||
|
friend SEXPR_LIST& operator<< (SEXPR_LIST& list, SEXPR_LIST* list2);
|
||
|
friend SEXPR_LIST& operator<< (SEXPR_LIST& list, SEXPR* obj);
|
||
|
friend SEXPR_LIST& operator>> (SEXPR_LIST& input, ISEXPRABLE& obj);
|
||
|
friend SEXPR_LIST& operator>> (SEXPR_LIST& input, std::string& str);
|
||
|
friend SEXPR_LIST& operator>> (SEXPR_LIST& input, int32_t& inte);
|
||
|
friend SEXPR_LIST& operator>> (SEXPR_LIST& input, int64_t& inte);
|
||
|
friend SEXPR_LIST& operator>> (SEXPR_LIST& input, float& inte);
|
||
|
friend SEXPR_LIST& operator>> (SEXPR_LIST& input, double& inte);
|
||
|
friend SEXPR_LIST& operator>> (SEXPR_LIST& input, const _IN_STRING is);
|
||
|
private:
|
||
|
int m_inStreamChild;
|
||
|
size_t doScan(const SEXPR_SCAN_ARG *args, size_t num_args);
|
||
|
void doAddChildren(const SEXPR_CHILDREN_ARG *args, size_t num_args);
|
||
|
};
|
||
|
}
|
||
|
|
||
|
#endif
|