summaryrefslogtreecommitdiff
path: root/src/parse/tokenstream.hpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/parse/tokenstream.hpp')
-rw-r--r--src/parse/tokenstream.hpp99
1 files changed, 99 insertions, 0 deletions
diff --git a/src/parse/tokenstream.hpp b/src/parse/tokenstream.hpp
new file mode 100644
index 00000000..c6e2de37
--- /dev/null
+++ b/src/parse/tokenstream.hpp
@@ -0,0 +1,99 @@
+/*
+ * MRustC - Rust Compiler
+ * - By John Hodge (Mutabah/thePowersGang)
+ *
+ * parse/tokenstream.hpp
+ * - Parser stream (TokenStream) header
+ */
+#pragma once
+
+#include <iostream>
+#include <vector>
+#include <span.hpp>
+#include <debug.hpp>
+#include <ident.hpp>
+#include "token.hpp"
+
+namespace AST {
+ class Module;
+ class MetaItems;
+}
+
+/// State the parser needs to pass down via a second channel.
+struct ParseState
+{
+ // Used for "for/if/while" to handle ambiguity
+ bool disallow_struct_literal = false;
+ // A debugging hook that disables expansion of macros
+ bool no_expand_macros = false;
+
+ ::AST::Module* module = nullptr;
+ ::AST::MetaItems* parent_attrs = nullptr;
+
+ ::AST::Module& get_current_mod() {
+ assert(this->module);
+ return *this->module;
+ }
+
+ friend ::std::ostream& operator<<(::std::ostream& os, const ParseState& ps) {
+ os << "ParseState {";
+ if(ps.disallow_struct_literal) os << " disallow_struct_literal";
+ if(ps.no_expand_macros) os << " no_expand_macros";
+ os << " }";
+ return os;
+ }
+};
+
+class TokenStream
+{
+ friend class TTLexer; // needs access to internals to know what was consumed
+
+ bool m_cache_valid;
+ Token m_cache;
+ ::std::vector<Token> m_lookahead;
+ ParseState m_parse_state;
+public:
+ TokenStream();
+ virtual ~TokenStream();
+ Token getToken();
+ void putback(Token tok);
+ eTokenType lookahead(unsigned int count);
+
+ virtual Position getPosition() const = 0;
+ virtual Ident::Hygiene getHygiene() const = 0;
+
+ ParseState& parse_state() { return m_parse_state; }
+
+ ProtoSpan start_span() const;
+ Span end_span(ProtoSpan ps) const;
+
+ Ident get_ident(Token tok) const;
+
+protected:
+ virtual Token realGetToken() = 0;
+private:
+ Token innerGetToken();
+};
+
+class SavedParseState
+{
+ TokenStream& m_lex;
+ ParseState m_state;
+public:
+ SavedParseState(TokenStream& lex, ParseState state):
+ m_lex(lex),
+ m_state(state)
+ {
+ }
+ ~SavedParseState()
+ {
+ DEBUG("Restoring " << m_state);
+ m_lex.parse_state() = m_state;
+ }
+};
+
+#define SET_MODULE(lex, mod) SavedParseState _sps(lex, lex.parse_state()); lex.parse_state().module = &(mod)
+#define SET_ATTRS(lex, attrs) SavedParseState _sps(lex, lex.parse_state()); lex.parse_state().parent_attrs = &(attrs)
+#define SET_PARSE_FLAG(lex, flag) SavedParseState _sps(lex, lex.parse_state()); lex.parse_state().flag = true
+#define CLEAR_PARSE_FLAG(lex, flag) SavedParseState _sps(lex, lex.parse_state()); lex.parse_state().flag = false
+#define CHECK_PARSE_FLAG(lex, flag) (lex.parse_state().flag == true)