X-Git-Url: https://gitweb.ps.run/toc/blobdiff_plain/8aeae09e74b46ca52866f22b48f55fecdf27b849..7f83e1b208e87e3808b268303bb633a8fda203f5:/src/toc.h diff --git a/src/toc.h b/src/toc.h index 0c55466..43ab1ba 100644 --- a/src/toc.h +++ b/src/toc.h @@ -4,56 +4,80 @@ #include #include "repr.h" +#include "typeInfo.h" template -std::ostream & operator<< (std::ostream & out, const std::vector & v) { - bool comma = false; - for (auto t : v) { - if (comma) out << ", "; - else comma = true; - out << t; +std::string vectorStr (const std::vector & v, const std::string & separator, bool end = false) +{ + std::stringstream sstr; + + bool putSeparator = false; + for (auto t : v) + { + if (putSeparator) sstr << separator; + else putSeparator = true; + sstr << t; } - return out; + if (end && !v.empty()) + sstr << separator; + + return sstr.str(); } std::ostream & operator<< (std::ostream & out, const Type & t); std::ostream & operator<< (std::ostream & out, const Variable & v); std::ostream & operator<< (std::ostream & out, const Body & b); -std::ostream & operator<< (std::ostream & out, const UnaryOperatorExpr & o); -std::ostream & operator<< (std::ostream & out, const BinaryOperatorExpr & o); -std::ostream & operator<< (std::ostream & out, const TernaryOperatorExpr & o); std::ostream & operator<< (std::ostream & out, const Expr & e); std::ostream & operator<< (std::ostream & out, const Stmt & s); void tocFunction (std::ostream & out, const Function & f, bool stub); void tocStruct (std::ostream & out, const Struct & s, bool stub); void tocProgram (std::ostream & out, const Program & p); +void tocNamespace (std::ostream & out, const Namespace & n, bool stub); static const int TAB_WIDTH = 2; static int indentation = 0; -static void indent(std::ostream & out, int change = 0) { +static void indent(std::ostream & out, int change = 0) +{ indentation += change; out << std::string(indentation, ' '); } -std::ostream & operator<< (std::ostream & out, const Type & t) { - out << t.name; +static std::vector namespaces; +static std::string namespacePrefix() { + std::stringstream sstr; + for (auto n : namespaces) + { + sstr << n << "_"; + } + return sstr.str(); +} + +static Program globalPrg; + +std::ostream & operator<< (std::ostream & out, const Type & t) +{ + out << vectorStr(t.namespacePrefixes, "_", true) << t.name; return out; } -std::ostream & operator<< (std::ostream & out, const Variable & v) { +std::ostream & operator<< (std::ostream & out, const Variable & v) +{ out << v.type << " "; std::stringstream sstr; std::string s = v.name; - for (auto m = v.type.modifiers.rbegin(); m != v.type.modifiers.rend(); m++) { - if (m->type == TypeModifierType::Pointer) { + for (auto m = v.type.modifiers.rbegin(); m != v.type.modifiers.rend(); m++) + { + if (m->type == TypeModifierType::Pointer) + { sstr.str(std::string()); sstr << "*(" << s << ")"; s = sstr.str(); } - else { + else + { sstr.str(std::string()); sstr << "(" << s << ")["; if (m->_staticArray) @@ -66,19 +90,22 @@ std::ostream & operator<< (std::ostream & out, const Variable & v) { return out; } -std::ostream & operator<< (std::ostream & out, const Body & b) { +std::ostream & operator<< (std::ostream & out, const Body & b) +{ indent(out); out << "{\n"; indentation += 2; - for (auto v : b.variables) { + for (auto v : b.ctx->variables) + { indent(out); out << v << ";\n"; } out << "\n"; - for (auto s : b.statements) { + for (auto s : b.statements) + { indent(out); out << s << "\n"; } @@ -88,65 +115,59 @@ std::ostream & operator<< (std::ostream & out, const Body & b) { return out; } -std::ostream & operator<< (std::ostream & out, const UnaryOperatorExpr & o) { - if (o.type == UnaryOperatorType::IncrementPost || o.type == UnaryOperatorType::DecrementPost) { - out << UnaryOperatorTypeStrings[(int)o.type] << *o.expr; - } - else { - out << *o.expr << UnaryOperatorTypeStrings[(int)o.type]; - } - - return out; -} -std::ostream & operator<< (std::ostream & out, const BinaryOperatorExpr & o) { - out << *o.lexpr << " " << BinaryOperatorTypeStrings[(int)o.type] << " " << *o.rexpr; - - return out; -} -std::ostream & operator<< (std::ostream & out, const TernaryOperatorExpr & o) { - out << *o.lexpr << " ? " << *o.rexprTrue << " : " << *o.rexprFalse; - - return out; -} -std::ostream & operator<< (std::ostream & out, const Expr & e) { - if (e.parenthesized) - out << "("; - - switch (e.type) { +std::ostream & operator<< (std::ostream & out, const Expr & e) +{ + switch (e.type) + { case ExprType::Func: - out << e._func.functionName << "(" << e._func.arguments << ")"; break; + out << vectorStr(e._func.namespacePrefixes, "_", true) << e._func.functionName << "(" << vectorStr(e._func.arguments, ", ") << ")"; break; + case ExprType::Method: + { + TypeInfo ti = typeExpr(globalPrg, namespaces, *e._method.expr); + out << + vectorStr(ti.type.namespacePrefixes, "_", true) << + ti.type.name << "_" << e._method.methodName << "(" << *e._method.expr << vectorStr(e._method.arguments, ", ") << ")"; break; + } case ExprType::Lit: /**/ if (e._lit.type == LitType::Int) out << e._lit._int; else if (e._lit.type == LitType::Decimal) out << e._lit._decimal; else if (e._lit.type == LitType::String) out << e._lit._string; else if (e._lit.type == LitType::Bool) out << e._lit._bool; break; - case ExprType::Identifier: - out << e._identifier.name; break; - case ExprType::Brackets: - out << *e._brackets.lexpr << "[" << *e._brackets.rexpr << "]"; break; + case ExprType::Paren: + out << "(" << e._paren.expr << ")"; break; case ExprType::Dot: - out << *e._dot.expr << "." << e._dot.ident.name; break; - case ExprType::UnaryOperator: - out << e._unaryOperator; break; - case ExprType::BinaryOperator: - out << e._binaryOperator; break; - case ExprType::TernaryOperator: - out << e._ternaryOperator; break; + out << *e._dot.expr << "." << e._dot.identifier; break; + case ExprType::PrefixOp: + out << PrefixOperatorTypeStrings[(int)e._prefixOp.type] << *e._prefixOp.expr; break; + case ExprType::PostfixOp: + out << *e._postfixOp.expr << PostfixOperatorTypeStrings[(int)e._postfixOp.type]; break; + case ExprType::BinaryOp: + out << *e._binaryOp.lexpr << + " " << BinaryOperatorTypeStrings[(int)e._binaryOp.type] << " " << + *e._binaryOp.rexpr; break; + case ExprType::TernaryOp: + out << *e._ternaryOp.lexpr << + " ? " << *e._ternaryOp.rexprTrue << + " : " << *e._ternaryOp.rexprFalse; break; + case ExprType::Bracket: + out << *e._brackets.lexpr << "[" << *e._brackets.rexpr << "]"; break; + case ExprType::Identifier: + out << vectorStr(e._identifier.namespacePrefixes, "_", true) << e._identifier.identifier; break; } - - if (e.parenthesized) - out << ")"; return out; } -std::ostream & operator<< (std::ostream & out, const Stmt & s) { - switch (s.type) { +std::ostream & operator<< (std::ostream & out, const Stmt & s) +{ + switch (s.type) + { case StmtType::If: out << "if (" << s._if.condition << ")\n" << s._if.body; break; case StmtType::Switch: - out << "switch (" << s._switch.ident.name << ")\n{\n"; - for (auto c : s._switch.cases) { + out << "switch (" << s._switch.ident << ")\n{\n"; + for (auto c : s._switch.cases) + { indent(out, 2); out << "case " << *c.expr << ": " << c.body << "break;"; } @@ -155,14 +176,14 @@ std::ostream & operator<< (std::ostream & out, const Stmt & s) { break; case StmtType::For: out << "for (" << - s._for.varName << " = " << *s._for.initValue << "; " << + s._for.init << "; " << *s._for.condition << "; " << *s._for.action << ")\n" << s._for.body; break; case StmtType::While: out << "while (" << s._while.condition << ")\n" << s._while.body; break; case StmtType::Assign: - out << s._assign.name << " = " << s._assign.expr << ";"; break; + out << s._assign.lexpr << " = " << s._assign.rexpr << ";"; break; case StmtType::Return: out << "return " << s._return.expr << ";"; break; case StmtType::Expr: @@ -173,32 +194,50 @@ std::ostream & operator<< (std::ostream & out, const Stmt & s) { } -void tocFunction (std::ostream & out, const Function & f, bool stub) { - out << f.returnType << " " << f.name << " (" << f.parameters << ")"; +void tocFunction (std::ostream & out, const Function & f, bool stub) +{ + out << f.returnType << " " << namespacePrefix() << f.name << " (" << vectorStr(f.parameters, ", ") << ")"; - if (stub) { + if (stub) + { out << ";\n"; } - else { + else + { out << "\n" << f.body; } } -void tocStruct (std::ostream & out, const Struct & s, bool stub) { - out << "struct " << s.name; - if (stub) { +void tocStruct (std::ostream & out, const Struct & s, bool stub) +{ + out << "struct " << namespacePrefix() << s.name; + if (stub) + { out << ";\n"; - for (auto m : s.methods) { - m.parameters.insert(m.parameters.begin(), {"this", {s.name, {{TypeModifierType::Pointer, false, -1}}}}); - out << m.returnType << " " << - s.name << "_" << m.name << - " (" << m.parameters << ");\n"; + for (auto m : s.methods) + { + Function f = m; + + f.parameters.insert(f.parameters.begin(), + {"this", + { + namespaces, + s.name, + { + {TypeModifierType::Pointer, false, -1} + } + } + }); + out << f.returnType << " " << + namespacePrefix() << s.name << "_" << f.name << + " (" << vectorStr(f.parameters, ", ") << ");\n"; } return; } out << "\n{\n"; indentation += 2; - for (auto m : s.members) { + for (auto m : s.members) + { indent(out); out << m << ";\n"; } @@ -206,26 +245,80 @@ void tocStruct (std::ostream & out, const Struct & s, bool stub) { indent(out, -2); out << "};\n"; - for (auto m : s.methods) { - m.parameters.insert(m.parameters.begin(), {"this", {s.name, {{TypeModifierType::Pointer, false, -1}}}}); - out << m.returnType << " " << s.name << "_" << m.name << " (" << m.parameters << ")\n" << m.body; + for (auto m : s.methods) + { + Function f = m; + f.parameters.insert(f.parameters.begin(), + {"this", + { + namespaces, + s.name, + { + {TypeModifierType::Pointer, false, -1} + } + } + }); + out << f.returnType << " " << + namespacePrefix() << s.name << "_" << f.name << + " (" << vectorStr(f.parameters, ", ") << ")\n" << f.body; } } -void tocProgram (std::ostream & out, const Program & p) { - for (auto s : p.structs) { +void tocProgram (std::ostream & out, const Program & p) +{ + globalPrg = p; + for (auto n : p.namespaces) + { + tocNamespace(out, n, true); + } + for (auto s : p.structs) + { tocStruct(out, s, true); } - for (auto f : p.functions) { + for (auto f : p.functions) + { tocFunction(out, f, true); } - for (auto v : p.variables) { + for (auto v : p.ctx->variables) + { out << v << ";\n"; } - for (auto s : p.structs) { + for (auto n : p.namespaces) + { + tocNamespace(out, n, false); + } + for (auto s : p.structs) + { tocStruct(out, s, false); } - for (auto f : p.functions) { + for (auto f : p.functions) + { tocFunction(out, f, false); } } + + +void tocNamespace (std::ostream & out, const Namespace & n, bool stub) +{ + namespaces.push_back(n.name); + if (!stub) + { + for (auto v : n.ctx->variables) + { + out << v << ";\n"; + } + } + for (auto n : n.namespaces) + { + tocNamespace(out, n, stub); + } + for (auto s : n.structs) + { + tocStruct(out, s, stub); + } + for (auto f : n.functions) + { + tocFunction(out, f, stub); + } + namespaces.pop_back(); +} \ No newline at end of file