1 | //===-- sanitizer_array_ref.h -----------------------------------*- C++ -*-===// |
2 | // |
3 | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
4 | // See https://llvm.org/LICENSE.txt for license information. |
5 | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
6 | // |
7 | //===----------------------------------------------------------------------===// |
8 | |
9 | #ifndef SANITIZER_ARRAY_REF_H |
10 | #define SANITIZER_ARRAY_REF_H |
11 | |
12 | #include "sanitizer_internal_defs.h" |
13 | |
14 | namespace __sanitizer { |
15 | |
16 | /// ArrayRef - Represent a constant reference to an array (0 or more elements |
17 | /// consecutively in memory), i.e. a start pointer and a length. It allows |
18 | /// various APIs to take consecutive elements easily and conveniently. |
19 | /// |
20 | /// This class does not own the underlying data, it is expected to be used in |
21 | /// situations where the data resides in some other buffer, whose lifetime |
22 | /// extends past that of the ArrayRef. For this reason, it is not in general |
23 | /// safe to store an ArrayRef. |
24 | /// |
25 | /// This is intended to be trivially copyable, so it should be passed by |
26 | /// value. |
27 | template <typename T> |
28 | class ArrayRef { |
29 | public: |
30 | constexpr ArrayRef() {} |
31 | constexpr ArrayRef(const T *begin, const T *end) : begin_(begin), end_(end) { |
32 | DCHECK(empty() || begin); |
33 | } |
34 | constexpr ArrayRef(const T *data, uptr length) |
35 | : ArrayRef(data, data + length) {} |
36 | template <uptr N> |
37 | constexpr ArrayRef(const T (&src)[N]) : ArrayRef(src, src + N) {} |
38 | template <typename C> |
39 | constexpr ArrayRef(const C &src) |
40 | : ArrayRef(src.data(), src.data() + src.size()) {} |
41 | ArrayRef(const T &one_elt) : ArrayRef(&one_elt, &one_elt + 1) {} |
42 | |
43 | const T *data() const { return empty() ? nullptr : begin_; } |
44 | |
45 | const T *begin() const { return begin_; } |
46 | const T *end() const { return end_; } |
47 | |
48 | bool empty() const { return begin_ == end_; } |
49 | |
50 | uptr size() const { return end_ - begin_; } |
51 | |
52 | /// equals - Check for element-wise equality. |
53 | bool equals(ArrayRef rhs) const { |
54 | if (size() != rhs.size()) |
55 | return false; |
56 | auto r = rhs.begin(); |
57 | for (auto &l : *this) { |
58 | if (!(l == *r)) |
59 | return false; |
60 | ++r; |
61 | } |
62 | return true; |
63 | } |
64 | |
65 | /// slice(n, m) - Chop off the first N elements of the array, and keep M |
66 | /// elements in the array. |
67 | ArrayRef<T> slice(uptr N, uptr M) const { |
68 | DCHECK_LE(N + M, size()); |
69 | return ArrayRef<T>(data() + N, M); |
70 | } |
71 | |
72 | /// slice(n) - Chop off the first N elements of the array. |
73 | ArrayRef<T> slice(uptr N) const { return slice(N, size() - N); } |
74 | |
75 | /// Drop the first \p N elements of the array. |
76 | ArrayRef<T> drop_front(uptr N = 1) const { |
77 | DCHECK_GE(size(), N); |
78 | return slice(N, size() - N); |
79 | } |
80 | |
81 | /// Drop the last \p N elements of the array. |
82 | ArrayRef<T> drop_back(uptr N = 1) const { |
83 | DCHECK_GE(size(), N); |
84 | return slice(0, size() - N); |
85 | } |
86 | |
87 | /// Return a copy of *this with only the first \p N elements. |
88 | ArrayRef<T> take_front(uptr N = 1) const { |
89 | if (N >= size()) |
90 | return *this; |
91 | return drop_back(N: size() - N); |
92 | } |
93 | |
94 | /// Return a copy of *this with only the last \p N elements. |
95 | ArrayRef<T> take_back(uptr N = 1) const { |
96 | if (N >= size()) |
97 | return *this; |
98 | return drop_front(N: size() - N); |
99 | } |
100 | |
101 | const T &operator[](uptr index) const { |
102 | DCHECK_LT(index, size()); |
103 | return begin_[index]; |
104 | } |
105 | |
106 | private: |
107 | const T *begin_ = nullptr; |
108 | const T *end_ = nullptr; |
109 | }; |
110 | |
111 | template <typename T> |
112 | inline bool operator==(ArrayRef<T> lhs, ArrayRef<T> rhs) { |
113 | return lhs.equals(rhs); |
114 | } |
115 | |
116 | template <typename T> |
117 | inline bool operator!=(ArrayRef<T> lhs, ArrayRef<T> rhs) { |
118 | return !(lhs == rhs); |
119 | } |
120 | |
121 | } // namespace __sanitizer |
122 | |
123 | #endif // SANITIZER_ARRAY_REF_H |
124 | |