blob: 455d9676e070add95faffee7fab4534ef325f0ac [file] [log] [blame]
// 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_