1 //
2 // Copyright 2002 The ANGLE Project Authors. All rights reserved.
3 // Use of this source code is governed by a BSD-style license that can be
4 // found in the LICENSE file.
5 //
6
7 #ifndef COMPILER_TRANSLATOR_COMMON_H_
8 #define COMPILER_TRANSLATOR_COMMON_H_
9
10 #include <stdio.h>
11 #include <limits>
12 #include <map>
13 #include <sstream>
14 #include <string>
15 #include <unordered_map>
16 #include <vector>
17
18 #include "common/angleutils.h"
19 #include "common/debug.h"
20 #include "common/third_party/smhasher/src/PMurHash.h"
21 #include "compiler/translator/PoolAlloc.h"
22
23 namespace sh
24 {
25
26 struct TSourceLoc
27 {
28 int first_file;
29 int first_line;
30 int last_file;
31 int last_line;
32 };
33
34 constexpr TSourceLoc kNoSourceLoc{-1, -1, -1, -1};
35
36 //
37 // Put POOL_ALLOCATOR_NEW_DELETE in base classes to make them use this scheme.
38 //
39 #define POOL_ALLOCATOR_NEW_DELETE \
40 void *operator new(size_t s) { return GetGlobalPoolAllocator()->allocate(s); } \
41 void *operator new(size_t, void *_Where) { return (_Where); } \
42 void operator delete(void *) {} \
43 void operator delete(void *, void *) {} \
44 void *operator new[](size_t s) { return GetGlobalPoolAllocator()->allocate(s); } \
45 void *operator new[](size_t, void *_Where) { return (_Where); } \
46 void operator delete[](void *) {} \
47 void operator delete[](void *, void *) {}
48
49 //
50 // Pool version of string.
51 //
52 typedef pool_allocator<char> TStringAllocator;
53 typedef std::basic_string<char, std::char_traits<char>, TStringAllocator> TString;
54 typedef std::basic_ostringstream<char, std::char_traits<char>, TStringAllocator> TStringStream;
55
56 //
57 // Persistent memory. Should only be used for strings that survive across compiles.
58 //
59 using TPersistString = std::string;
60 using TPersistStringStream = std::ostringstream;
61
62 //
63 // Pool allocator versions of vectors, lists, and maps
64 //
65 template <class T>
66 class TVector : public std::vector<T, pool_allocator<T>>
67 {
68 public:
69 POOL_ALLOCATOR_NEW_DELETE
70
71 typedef typename std::vector<T, pool_allocator<T>>::size_type size_type;
TVector()72 TVector() : std::vector<T, pool_allocator<T>>() {}
TVector(const pool_allocator<T> & a)73 TVector(const pool_allocator<T> &a) : std::vector<T, pool_allocator<T>>(a) {}
TVector(size_type i)74 TVector(size_type i) : std::vector<T, pool_allocator<T>>(i) {}
TVector(size_type i,const T & value)75 TVector(size_type i, const T &value) : std::vector<T, pool_allocator<T>>(i, value) {}
76 template <typename InputIt>
TVector(InputIt first,InputIt last)77 TVector(InputIt first, InputIt last) : std::vector<T, pool_allocator<T>>(first, last)
78 {}
TVector(std::initializer_list<T> init)79 TVector(std::initializer_list<T> init) : std::vector<T, pool_allocator<T>>(init) {}
80 };
81
82 template <class K, class D, class H = std::hash<K>, class CMP = std::equal_to<K>>
83 class TUnorderedMap : public std::unordered_map<K, D, H, CMP, pool_allocator<std::pair<const K, D>>>
84 {
85 public:
86 POOL_ALLOCATOR_NEW_DELETE
87 typedef pool_allocator<std::pair<const K, D>> tAllocator;
88
TUnorderedMap()89 TUnorderedMap() : std::unordered_map<K, D, H, CMP, tAllocator>() {}
90 // use correct two-stage name lookup supported in gcc 3.4 and above
TUnorderedMap(const tAllocator & a)91 TUnorderedMap(const tAllocator &a)
92 : std::unordered_map<K, D, H, CMP, tAllocator>(
93 std::unordered_map<K, D, H, CMP, tAllocator>::key_compare(),
94 a)
95 {}
96 };
97
98 template <class K, class D, class CMP = std::less<K>>
99 class TMap : public std::map<K, D, CMP, pool_allocator<std::pair<const K, D>>>
100 {
101 public:
102 POOL_ALLOCATOR_NEW_DELETE
103 typedef pool_allocator<std::pair<const K, D>> tAllocator;
104
TMap()105 TMap() : std::map<K, D, CMP, tAllocator>() {}
106 // use correct two-stage name lookup supported in gcc 3.4 and above
TMap(const tAllocator & a)107 TMap(const tAllocator &a)
108 : std::map<K, D, CMP, tAllocator>(std::map<K, D, CMP, tAllocator>::key_compare(), a)
109 {}
110 };
111
112 // Basic implementation of C++20's span for use with pool-allocated containers (TVector) or static
113 // arrays. This is used by the array sizes member of TType to allow arrayed types to be
114 // constexpr-constructed.
115 // See the reference for std::span here: https://en.cppreference.com/w/cpp/container/span
116 template <typename T>
117 class TSpan
118 {
119 public:
120 typedef size_t size_type;
121
TSpan()122 constexpr TSpan() {}
TSpan(T * ptr,size_type size)123 constexpr TSpan(T *ptr, size_type size) : mData(ptr), mSize(size) {}
124
TSpan(const TSpan & that)125 constexpr TSpan(const TSpan &that) : mData(that.mData), mSize(that.mSize) {}
126 constexpr TSpan &operator=(const TSpan &that)
127 {
128 mData = that.mData;
129 mSize = that.mSize;
130 return *this;
131 }
132
133 // Note: the pointer is taken out of the TVector because TVector's memory is pool allocated,
134 // so the memory will live on even if the TVector is destroyed.
135 template <typename S>
TSpan(const TVector<S> & vec)136 TSpan(const TVector<S> &vec) : mData(vec.data()), mSize(vec.size())
137 {}
138 template <typename S>
139 TSpan &operator=(const TVector<S> &vec)
140 {
141 mData = vec.data();
142 mSize = vec.size();
143 return *this;
144 }
145
146 constexpr bool operator==(const TSpan &that) const
147 {
148 if (mSize != that.mSize)
149 {
150 return false;
151 }
152
153 if (mData == that.mData)
154 {
155 return true;
156 }
157
158 for (size_type index = 0; index < mSize; ++index)
159 {
160 if (mData[index] != that.mData[index])
161 {
162 return false;
163 }
164 }
165
166 return true;
167 }
168 constexpr bool operator!=(const TSpan &that) const { return !(*this == that); }
169
data()170 constexpr T *data() const { return mData; }
size()171 constexpr size_type size() const { return mSize; }
empty()172 constexpr bool empty() const { return mSize == 0; }
173
174 constexpr T &operator[](size_type index) const { return mData[index]; }
front()175 constexpr T &front() const { return mData[0]; }
back()176 constexpr T &back() const { return mData[mSize - 1]; }
177
begin()178 constexpr T *begin() const { return mData; }
end()179 constexpr T *end() const { return mData + mSize; }
180
rbegin()181 constexpr std::reverse_iterator<T *> rbegin() const
182 {
183 return std::make_reverse_iterator(end());
184 }
rend()185 constexpr std::reverse_iterator<T *> rend() const
186 {
187 return std::make_reverse_iterator(begin());
188 }
189
first(size_type count)190 constexpr TSpan first(size_type count) const
191 {
192 ASSERT(count <= mSize);
193 return count == 0 ? TSpan() : TSpan(mData, count);
194 }
last(size_type count)195 constexpr TSpan last(size_type count) const
196 {
197 ASSERT(count <= mSize);
198 return count == 0 ? TSpan() : TSpan(mData + mSize - count, count);
199 }
subspan(size_type offset,size_type count)200 constexpr TSpan subspan(size_type offset, size_type count) const
201 {
202 ASSERT(offset + count <= mSize);
203 return count == 0 ? TSpan() : TSpan(mData + offset, count);
204 }
205
206 private:
207 T *mData = nullptr;
208 size_t mSize = 0;
209 };
210
211 // Integer to TString conversion
212 template <typename T>
str(T i)213 inline TString str(T i)
214 {
215 ASSERT(std::numeric_limits<T>::is_integer);
216 char buffer[((8 * sizeof(T)) / 3) + 3];
217 const char *formatStr = std::numeric_limits<T>::is_signed ? "%d" : "%u";
218 snprintf(buffer, sizeof(buffer), formatStr, i);
219 return buffer;
220 }
221
222 // Allocate a char array in the global memory pool. str must be a null terminated string. strLength
223 // is the length without the null terminator.
AllocatePoolCharArray(const char * str,size_t strLength)224 inline const char *AllocatePoolCharArray(const char *str, size_t strLength)
225 {
226 size_t requiredSize = strLength + 1;
227 char *buffer = static_cast<char *>(GetGlobalPoolAllocator()->allocate(requiredSize));
228 memcpy(buffer, str, requiredSize);
229 ASSERT(buffer[strLength] == '\0');
230 return buffer;
231 }
232
233 // Initialize a new stream which must be imbued with the classic locale
234 template <typename T>
InitializeStream()235 T InitializeStream()
236 {
237 T stream;
238 stream.imbue(std::locale::classic());
239 return stream;
240 }
241
242 } // namespace sh
243
244 namespace std
245 {
246 template <>
247 struct hash<sh::TString>
248 {
249 size_t operator()(const sh::TString &s) const
250 {
251 return angle::PMurHash32(0, s.data(), static_cast<int>(s.length()));
252 }
253 };
254 } // namespace std
255
256 #endif // COMPILER_TRANSLATOR_COMMON_H_
257