// Copyright 2015 Google Inc. All rights reserved // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #ifndef SYMTAB_H_ #define SYMTAB_H_ #include <bitset> #include <string> #include <vector> #include "string_piece.h" using namespace std; extern vector<string*>* g_symbols; class Symtab; class Var; class Symbol { public: explicit Symbol() : v_(-1) {} const string& str() const { return *((*g_symbols)[v_]); } const char* c_str() const { return str().c_str(); } bool empty() const { return !v_; } int val() const { return v_; } char get(size_t i) const { const string& s = str(); if (i >= s.size()) return 0; return s[i]; } bool IsValid() const { return v_ >= 0; } Var* PeekGlobalVar() const; Var* GetGlobalVar() const; void SetGlobalVar(Var* v, bool is_override = false, bool* readonly = nullptr) const; private: explicit Symbol(int v); int v_; friend class Symtab; friend class SymbolSet; }; /* A set of symbols represented as bitmap indexed by Symbol's ordinal value. */ class SymbolSet { public: SymbolSet() : low_(0), high_(0) {} /* Returns true if Symbol belongs to this set. */ bool exists(Symbol sym) const { size_t bit_nr = static_cast<size_t>(sym.val()); return sym.IsValid() && bit_nr >= low_ && bit_nr < high_ && bits_[(bit_nr - low_) / 64][(bit_nr - low_) % 64]; } /* Adds Symbol to this set. */ void insert(Symbol sym) { if (!sym.IsValid()) { return; } size_t bit_nr = static_cast<size_t>(sym.val()); if (bit_nr < low_ || bit_nr >= high_) { resize(bit_nr); } bits_[(bit_nr - low_) / 64][(bit_nr - low_) % 64] = true; } /* Returns the number of Symbol's in this set. */ size_t size() const { size_t n = 0; for (auto const& bitset : bits_) { n += bitset.count(); } return n; } /* Allow using foreach. * E.g., * SymbolSet symbol_set; * for (auto const& symbol: symbol_set) { ... } */ class iterator { const SymbolSet* bitset_; size_t pos_; iterator(const SymbolSet* bitset, size_t pos) : bitset_(bitset), pos_(pos) {} /* Proceed to the next Symbol. */ void next() { size_t bit_nr = (pos_ > bitset_->low_) ? pos_ - bitset_->low_ : 0; while (bit_nr < (bitset_->high_ - bitset_->low_)) { if ((bit_nr % 64) == 0 && !bitset_->bits_[bit_nr / 64].any()) { bit_nr += 64; continue; } if (bitset_->bits_[bit_nr / 64][bit_nr % 64]) { break; } ++bit_nr; } pos_ = bitset_->low_ + bit_nr; } public: iterator& operator++() { if (pos_ < bitset_->high_) { ++pos_; next(); } return *this; } bool operator==(iterator other) const { return bitset_ == other.bitset_ && pos_ == other.pos_; } bool operator!=(iterator other) const { return !(*this == other); } Symbol operator*() { return Symbol(pos_); } friend class SymbolSet; }; iterator begin() const { iterator it(this, low_); it.next(); return it; } iterator end() const { return iterator(this, high_); } private: friend class iterator; /* Ensure that given bit number is in [low_, high_) */ void resize(size_t bit_nr) { size_t new_low = bit_nr & ~63; size_t new_high = (bit_nr + 64) & ~63; if (bits_.empty()) { high_ = low_ = new_low; } if (new_low > low_) { new_low = low_; } if (new_high <= high_) { new_high = high_; } if (new_low == low_) { bits_.resize((new_high - new_low) / 64); } else { std::vector<std::bitset<64> > newbits((new_high - new_low) / 64); std::copy(bits_.begin(), bits_.end(), newbits.begin() + (low_ - new_low) / 64); bits_.swap(newbits); } low_ = new_low; high_ = new_high; } /* Keep only the (aligned) range where at least one bit has been set. * E.g., if we only ever set bits 65 and 141, |low_| will be 64, |high_| * will be 192, and |bits_| will have 2 elements. */ size_t low_; size_t high_; std::vector<std::bitset<64> > bits_; }; class ScopedGlobalVar { public: ScopedGlobalVar(Symbol name, Var* var); ~ScopedGlobalVar(); private: Symbol name_; Var* orig_; }; inline bool operator==(const Symbol& x, const Symbol& y) { return x.val() == y.val(); } inline bool operator<(const Symbol& x, const Symbol& y) { return x.val() < y.val(); } namespace std { template <> struct hash<Symbol> { size_t operator()(const Symbol& s) const { return s.val(); } }; } // namespace std extern Symbol kEmptySym; extern Symbol kShellSym; extern Symbol kKatiReadonlySym; void InitSymtab(); void QuitSymtab(); Symbol Intern(StringPiece s); string JoinSymbols(const vector<Symbol>& syms, const char* sep); #endif // SYMTAB_H_