| 1 | /*
 | 
| 2 |  * Souffle - A Datalog Compiler
 | 
| 3 |  * Copyright (c) 2022, The Souffle Developers. All rights reserved
 | 
| 4 |  * Licensed under the Universal Permissive License v 1.0 as shown at:
 | 
| 5 |  * - https://opensource.org/licenses/UPL
 | 
| 6 |  * - <souffle root>/licenses/SOUFFLE-UPL.txt
 | 
| 7 |  */
 | 
| 8 | 
 | 
| 9 | /**
 | 
| 10 |  * @file SymbolTableImpl.h
 | 
| 11 |  *
 | 
| 12 |  * SymbolTable definition
 | 
| 13 |  */
 | 
| 14 | 
 | 
| 15 | #pragma once
 | 
| 16 | 
 | 
| 17 | #include "souffle/SymbolTable.h"
 | 
| 18 | #include "souffle/datastructure/ConcurrentFlyweight.h"
 | 
| 19 | #include "souffle/utility/MiscUtil.h"
 | 
| 20 | #include "souffle/utility/ParallelUtil.h"
 | 
| 21 | #include "souffle/utility/StreamUtil.h"
 | 
| 22 | 
 | 
| 23 | #include <algorithm>
 | 
| 24 | #include <cstdlib>
 | 
| 25 | #include <deque>
 | 
| 26 | #include <initializer_list>
 | 
| 27 | #include <iostream>
 | 
| 28 | #include <memory>
 | 
| 29 | #include <string>
 | 
| 30 | #include <unordered_map>
 | 
| 31 | #include <utility>
 | 
| 32 | #include <vector>
 | 
| 33 | 
 | 
| 34 | namespace souffle {
 | 
| 35 | 
 | 
| 36 | /**
 | 
| 37 |  * @class SymbolTableImpl
 | 
| 38 |  *
 | 
| 39 |  * Implementation of the symbol table.
 | 
| 40 |  */
 | 
| 41 | class SymbolTableImpl : public SymbolTable, protected FlyweightImpl<std::string> {
 | 
| 42 | private:
 | 
| 43 |     using Base = FlyweightImpl<std::string>;
 | 
| 44 | 
 | 
| 45 | public:
 | 
| 46 |     class IteratorImpl : public SymbolTableIteratorInterface, private Base::iterator {
 | 
| 47 |     public:
 | 
| 48 |         IteratorImpl(Base::iterator&& it) : Base::iterator(it) {}
 | 
| 49 | 
 | 
| 50 |         IteratorImpl(const Base::iterator& it) : Base::iterator(it) {}
 | 
| 51 | 
 | 
| 52 |         const std::pair<const std::string, const std::size_t>& get() const {
 | 
| 53 |             return **this;
 | 
| 54 |         }
 | 
| 55 | 
 | 
| 56 |         bool equals(const SymbolTableIteratorInterface& other) {
 | 
| 57 |             return (*this) == static_cast<const IteratorImpl&>(other);
 | 
| 58 |         }
 | 
| 59 | 
 | 
| 60 |         SymbolTableIteratorInterface& incr() {
 | 
| 61 |             ++(*this);
 | 
| 62 |             return *this;
 | 
| 63 |         }
 | 
| 64 | 
 | 
| 65 |         std::unique_ptr<SymbolTableIteratorInterface> copy() const {
 | 
| 66 |             return std::make_unique<IteratorImpl>(*this);
 | 
| 67 |         }
 | 
| 68 |     };
 | 
| 69 | 
 | 
| 70 |     using iterator = SymbolTable::Iterator;
 | 
| 71 | 
 | 
| 72 |     /** @brief Construct a symbol table with the given number of concurrent access lanes. */
 | 
| 73 |     SymbolTableImpl(const std::size_t LaneCount = 1) : Base(LaneCount) {}
 | 
| 74 | 
 | 
| 75 |     /** @brief Construct a symbol table with the given initial symbols. */
 | 
| 76 |     SymbolTableImpl(std::initializer_list<std::string> symbols) : Base(1, symbols.size()) {
 | 
| 77 |         for (const auto& symbol : symbols) {
 | 
| 78 |             findOrInsert(symbol);
 | 
| 79 |         }
 | 
| 80 |     }
 | 
| 81 | 
 | 
| 82 |     /** @brief Construct a symbol table with the given number of concurrent access lanes and initial symbols.
 | 
| 83 |      */
 | 
| 84 |     SymbolTableImpl(const std::size_t LaneCount, std::initializer_list<std::string> symbols)
 | 
| 85 |             : Base(LaneCount, symbols.size()) {
 | 
| 86 |         for (const auto& symbol : symbols) {
 | 
| 87 |             findOrInsert(symbol);
 | 
| 88 |         }
 | 
| 89 |     }
 | 
| 90 | 
 | 
| 91 |     /**
 | 
| 92 |      * @brief Set the number of concurrent access lanes.
 | 
| 93 |      * This function is not thread-safe, do not call when other threads are using the datastructure.
 | 
| 94 |      */
 | 
| 95 |     void setNumLanes(const std::size_t NumLanes) {
 | 
| 96 |         Base::setNumLanes(NumLanes);
 | 
| 97 |     }
 | 
| 98 | 
 | 
| 99 |     iterator begin() const override {
 | 
| 100 |         return SymbolTable::Iterator(std::make_unique<IteratorImpl>(Base::begin()));
 | 
| 101 |     }
 | 
| 102 | 
 | 
| 103 |     iterator end() const override {
 | 
| 104 |         return SymbolTable::Iterator(std::make_unique<IteratorImpl>(Base::end()));
 | 
| 105 |     }
 | 
| 106 | 
 | 
| 107 |     bool weakContains(const std::string& symbol) const override {
 | 
| 108 |         return Base::weakContains(symbol);
 | 
| 109 |     }
 | 
| 110 | 
 | 
| 111 |     RamDomain encode(const std::string& symbol) override {
 | 
| 112 |         return Base::findOrInsert(symbol).first;
 | 
| 113 |     }
 | 
| 114 | 
 | 
| 115 |     const std::string& decode(const RamDomain index) const override {
 | 
| 116 |         return Base::fetch(index);
 | 
| 117 |     }
 | 
| 118 | 
 | 
| 119 |     RamDomain unsafeEncode(const std::string& symbol) override {
 | 
| 120 |         return encode(symbol);
 | 
| 121 |     }
 | 
| 122 | 
 | 
| 123 |     const std::string& unsafeDecode(const RamDomain index) const override {
 | 
| 124 |         return decode(index);
 | 
| 125 |     }
 | 
| 126 | 
 | 
| 127 |     std::pair<RamDomain, bool> findOrInsert(const std::string& symbol) override {
 | 
| 128 |         auto Res = Base::findOrInsert(symbol);
 | 
| 129 |         return std::make_pair(static_cast<RamDomain>(Res.first), Res.second);
 | 
| 130 |     }
 | 
| 131 | };
 | 
| 132 | 
 | 
| 133 | }  // namespace souffle
 |