P4C
The P4 Compiler
Loading...
Searching...
No Matches
stringref.h
1/*
2Copyright 2013-present Barefoot Networks, Inc.
3
4Licensed under the Apache License, Version 2.0 (the "License");
5you may not use this file except in compliance with the License.
6You may obtain a copy of the License at
7
8 http://www.apache.org/licenses/LICENSE-2.0
9
10Unless required by applicable law or agreed to in writing, software
11distributed under the License is distributed on an "AS IS" BASIS,
12WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13See the License for the specific language governing permissions and
14limitations under the License.
15*/
16
17#ifndef LIB_STRINGREF_H_
18#define LIB_STRINGREF_H_
19
20#include <cstdlib>
21#include <cstring>
22#include <iostream>
23#include <string>
24
25#include "config.h"
26#include "cstring.h"
27
28#if !HAVE_MEMRCHR
29static inline void *memrchr(const char *s, int c, size_t n) {
30 for (auto *p = s + n - 1; p >= s; --p)
31 if (*p == c) return const_cast<char *>(p);
32 return nullptr;
33}
34#endif
35
36/* A StringRef is a *TEMPORARY* reference to a string held in memory managed by some other
37 * object. As such, it becomes dangling (invalid) when that other object goes away, so needs
38 * to be used with care. StringRefs should in general have short lifetimes, and not be
39 * stored in other long-lived objects. */
40
41struct [[deprecated("Use std::string_view instead")]] StringRef {
42 const char *p;
43 size_t len;
44 StringRef() : p(0), len(0) {}
45 StringRef(const char *str, size_t l) : p(str), len(l) {}
46 StringRef(const char *str) : p(str), len(str ? strlen(str) : 0) {} // NOLINT
47 StringRef(const std::string &str) : p(str.data()), len(str.size()) {} // NOLINT
48 StringRef(cstring str) : p(str.c_str()), len(str.size()) {} // NOLINT
49 void clear() {
50 p = 0;
51 len = 0;
52 }
53 StringRef(const StringRef &a) : p(a.p), len(a.len) {}
54 // avoid clang-tidy complaining about assignment that is actually safe
55 // NOLINTBEGIN(bugprone-unhandled-self-assignment)
56 StringRef &operator=(const StringRef &a) {
57 p = a.p;
58 len = a.len;
59 return *this;
60 }
61 // NOLINTEND(bugprone-unhandled-self-assignment)
62 explicit operator bool() const { return p != 0; }
63
64 bool operator==(const StringRef &a) const {
65 return p ? (a.p && len == a.len && (!len || !memcmp(p, a.p, len))) : !a.p;
66 }
67 bool operator!=(const StringRef &a) const { return !operator==(a); }
68 bool operator==(const std::string &a) const { return operator==(StringRef(a)); }
69 bool operator==(const char *a) const {
70 return p ? (a && (!len || !strncmp(p, a, len)) && !a[len]) : !a;
71 }
72 bool operator==(cstring a) const { return operator==(a.c_str()); }
73 template <class T>
74 bool operator!=(T a) const {
75 return !(*this == a);
76 }
77 bool isNullOrEmpty() const { return p == 0 || len == 0; }
78
79 int compare(const StringRef &a) const {
80 if (!p) return a.p ? -1 : 0;
81 if (!a.p) return 1;
82 int rv = memcmp(p, a.p, std::min(len, a.len));
83 if (!rv && len != a.len) rv = len < a.len ? -1 : 1;
84 return rv;
85 }
86 int compare(const std::string &a) const { return compare(StringRef(a)); }
87 int compare(const char *a) const {
88 if (!p) return a ? -1 : 0;
89 if (!a) return 1;
90 int rv = strncmp(p, a, len);
91 if (!rv && a[len]) rv = -1;
92 return rv;
93 }
94 int compare(cstring a) const { return compare(a.c_str()); }
95 template <class T>
96 bool operator<(T a) const {
97 return compare(a) < 0;
98 }
99 template <class T>
100 bool operator<=(T a) const {
101 return compare(a) <= 0;
102 }
103 template <class T>
104 bool operator>(T a) const {
105 return compare(a) > 0;
106 }
107 template <class T>
108 bool operator>=(T a) const {
109 return compare(a) >= 0;
110 }
111
112 explicit operator std::string() const { return std::string(p, len); }
113 operator cstring() const { return std::string(p, len); }
114 cstring toString() const { return std::string(p, len); }
115 std::string string() const { return std::string(p, len); }
116 StringRef &operator+=(size_t i) {
117 if (len < i) {
118 p = 0;
119 len = 0;
120 } else {
121 p += i;
122 len -= i;
123 }
124 return *this;
125 }
126 StringRef &operator++() {
127 p++;
128 if (len)
129 len--;
130 else
131 p = 0;
132 return *this;
133 } // NOLINT
134 StringRef operator++(int) {
135 StringRef rv(*this);
136 ++*this;
137 return rv;
138 }
139 StringRef &operator--() {
140 if (len)
141 len--;
142 else
143 p = 0;
144 return *this;
145 } // NOLINT
146 StringRef operator--(int) {
147 StringRef rv(*this);
148 --*this;
149 return rv;
150 }
151 char operator[](size_t i) const { return i < len ? p[i] : 0; }
152 char operator*() const { return len ? *p : 0; }
153 StringRef operator+(size_t i) const {
154 StringRef rv(*this);
155 rv += i;
156 return rv;
157 }
158 StringRef &trim(const char *white = " \t\r\n") {
159 while (len > 0 && strchr(white, *p)) {
160 p++;
161 len--;
162 }
163 while (len > 0 && strchr(white, p[len - 1])) {
164 len--;
165 }
166 return *this;
167 }
168 bool trimCR() {
169 bool rv = false;
170 while (len > 0 && p[len - 1] == '\r') {
171 rv = true;
172 len--;
173 }
174 return rv;
175 }
176 StringRef trim(const char *white = " \t\r\n") const {
177 StringRef rv(*this);
178 rv.trim(white);
179 return rv;
180 }
181 const char *begin() const { return p; }
182 const char *end() const { return p + len; }
183 const char *find(char ch) const {
184 return p ? static_cast<const char *>(memchr(p, ch, len)) : p;
185 }
186 const char *findlast(char ch) const {
187 return p ? static_cast<const char *>(memrchr(p, ch, len)) : p;
188 }
189 const char *find(const char *set) const {
190 if (!p) return 0;
191 size_t off = strcspn(p, set);
192 return off >= len ? 0 : p + off;
193 }
194 const char *findstr(StringRef sub) {
195 if (sub.len < 1) return p;
196 const char *s = begin(), *e = end();
197 while (s < e && (s = static_cast<const char *>(memchr(s, *sub.p, e - s)))) {
198 if (sub.len > (size_t)(e - s)) return nullptr;
199 if (!memcmp(s, sub.p, sub.len)) return s;
200 s++;
201 }
202 return nullptr;
203 }
204 StringRef before(const char *s) const {
205 return (size_t)(s - p) <= len ? StringRef(p, s - p) : StringRef();
206 }
207 StringRef after(const char *s) const {
208 return (size_t)(s - p) <= len ? StringRef(s, p + len - s) : StringRef();
209 }
210 StringRef substr(size_t start, size_t length) const {
211 if (len <= start) return 0;
212 if (len <= length) return StringRef(p + start, len - start);
213 return StringRef(p + start, length);
214 }
215 class Split;
216 Split split(char) const;
217 Split split(const char *) const;
218};
219
220template <class T>
221inline auto operator==(T a, const StringRef &b) -> decltype(b.operator==(a)) {
222 return b == a;
223}
224template <class T>
225inline auto operator!=(T a, const StringRef &b) -> decltype(b.operator!=(a)) {
226 return b != a;
227}
228template <class T>
229inline auto operator>=(T a, const StringRef &b) -> decltype(b.operator<=(a)) {
230 return b <= a;
231}
232template <class T>
233inline auto operator>(T a, const StringRef &b) -> decltype(b.operator<(a)) {
234 return b < a;
235}
236template <class T>
237inline auto operator<=(T a, const StringRef &b) -> decltype(b.operator>=(a)) {
238 return b >= a;
239}
240template <class T>
241inline auto operator<(T a, const StringRef &b) -> decltype(b.operator>(a)) {
242 return b > a;
243}
244
245inline std::ostream &operator<<(std::ostream &os, const StringRef &a) {
246 return a.len ? os.write(a.p, a.len) : os;
247}
248inline std::string &operator+=(std::string &s, const StringRef &a) {
249 return a.len ? s.append(a.p, a.len) : s;
250}
251inline std::string operator+(const StringRef &s, const StringRef &a) {
252 std::string rv(s);
253 rv += a;
254 return rv;
255}
256inline std::string operator+(const std::string &s, const StringRef &a) {
257 std::string rv(s);
258 rv += a;
259 return rv;
260}
261inline std::string operator+(const StringRef &s, const std::string &a) {
262 std::string rv(s);
263 rv += a;
264 return rv;
265}
266inline std::string operator+(const char *s, const StringRef &a) {
267 std::string rv(s);
268 rv += a;
269 return rv;
270}
271inline std::string operator+(const StringRef &s, const char *a) {
272 std::string rv(s);
273 rv += a;
274 return rv;
275}
276inline std::string operator+(cstring s, const StringRef &a) {
277 std::string rv(s);
278 rv += a;
279 return rv;
280}
281inline std::string operator+(const StringRef &s, cstring a) {
282 std::string rv(s);
283 rv += a;
284 return rv;
285}
286
288 StringRef rest;
289 const char *set;
290 const char *ptr;
291 friend struct StringRef;
292 Split(StringRef r, const char *s, const char *p) : rest(r), set(s), ptr(p) {}
293
294 public:
295 Split begin() const { return *this; }
296 Split end() const { return Split(StringRef(), nullptr, nullptr); }
297 StringRef operator*() const { return ptr ? rest.before(ptr) : rest; }
298 Split &operator++() {
299 if (ptr) {
300 rest = rest.after(ptr + 1);
301 ptr = set ? rest.find(set) : rest.find(*ptr);
302 } else {
303 rest.clear();
304 }
305 return *this;
306 }
307 bool operator==(const Split &a) const { return rest == a.rest && ptr == a.ptr; }
308 bool operator!=(const Split &a) const { return rest != a.rest || ptr != a.ptr; }
309};
310
311inline StringRef::Split StringRef::split(char ch) const { return Split(*this, nullptr, find(ch)); }
312inline StringRef::Split StringRef::split(const char *s) const { return Split(*this, s, find(s)); }
313
314#endif /* LIB_STRINGREF_H_ */
Definition stringref.h:287
Definition cstring.h:80
Definition stringref.h:41