1*795d594fSAndroid Build Coastguard Worker /* 2*795d594fSAndroid Build Coastguard Worker * Copyright (C) 2015 The Android Open Source Project 3*795d594fSAndroid Build Coastguard Worker * 4*795d594fSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License"); 5*795d594fSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License. 6*795d594fSAndroid Build Coastguard Worker * You may obtain a copy of the License at 7*795d594fSAndroid Build Coastguard Worker * 8*795d594fSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0 9*795d594fSAndroid Build Coastguard Worker * 10*795d594fSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software 11*795d594fSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS, 12*795d594fSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*795d594fSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and 14*795d594fSAndroid Build Coastguard Worker * limitations under the License. 15*795d594fSAndroid Build Coastguard Worker */ 16*795d594fSAndroid Build Coastguard Worker 17*795d594fSAndroid Build Coastguard Worker #ifndef ART_LIBARTBASE_BASE_ARRAY_SLICE_H_ 18*795d594fSAndroid Build Coastguard Worker #define ART_LIBARTBASE_BASE_ARRAY_SLICE_H_ 19*795d594fSAndroid Build Coastguard Worker 20*795d594fSAndroid Build Coastguard Worker #include <ostream> 21*795d594fSAndroid Build Coastguard Worker #include "bit_utils.h" 22*795d594fSAndroid Build Coastguard Worker #include "casts.h" 23*795d594fSAndroid Build Coastguard Worker #include "iteration_range.h" 24*795d594fSAndroid Build Coastguard Worker #include "length_prefixed_array.h" 25*795d594fSAndroid Build Coastguard Worker #include "stride_iterator.h" 26*795d594fSAndroid Build Coastguard Worker 27*795d594fSAndroid Build Coastguard Worker namespace art { 28*795d594fSAndroid Build Coastguard Worker 29*795d594fSAndroid Build Coastguard Worker // An ArraySlice is an abstraction over an array or a part of an array of a particular type. It does 30*795d594fSAndroid Build Coastguard Worker // bounds checking and can be made from several common array-like structures in Art. 31*795d594fSAndroid Build Coastguard Worker template <typename T> 32*795d594fSAndroid Build Coastguard Worker class ArraySlice { 33*795d594fSAndroid Build Coastguard Worker public: 34*795d594fSAndroid Build Coastguard Worker using value_type = T; 35*795d594fSAndroid Build Coastguard Worker using reference = T&; 36*795d594fSAndroid Build Coastguard Worker using const_reference = const T&; 37*795d594fSAndroid Build Coastguard Worker using pointer = T*; 38*795d594fSAndroid Build Coastguard Worker using const_pointer = const T*; 39*795d594fSAndroid Build Coastguard Worker using iterator = StrideIterator<T>; 40*795d594fSAndroid Build Coastguard Worker using const_iterator = StrideIterator<const T>; 41*795d594fSAndroid Build Coastguard Worker using reverse_iterator = std::reverse_iterator<iterator>; 42*795d594fSAndroid Build Coastguard Worker using const_reverse_iterator = std::reverse_iterator<const_iterator>; 43*795d594fSAndroid Build Coastguard Worker using difference_type = ptrdiff_t; 44*795d594fSAndroid Build Coastguard Worker using size_type = size_t; 45*795d594fSAndroid Build Coastguard Worker 46*795d594fSAndroid Build Coastguard Worker // Create an empty array slice. ArraySlice()47*795d594fSAndroid Build Coastguard Worker ArraySlice() : array_(nullptr), size_(0), element_size_(0) {} 48*795d594fSAndroid Build Coastguard Worker 49*795d594fSAndroid Build Coastguard Worker // Create an array slice of the first 'length' elements of the array, with each element being 50*795d594fSAndroid Build Coastguard Worker // element_size bytes long. 51*795d594fSAndroid Build Coastguard Worker ArraySlice(T* array, 52*795d594fSAndroid Build Coastguard Worker size_t length, 53*795d594fSAndroid Build Coastguard Worker size_t element_size = sizeof(T)) array_(array)54*795d594fSAndroid Build Coastguard Worker : array_(array), 55*795d594fSAndroid Build Coastguard Worker size_(dchecked_integral_cast<uint32_t>(length)), 56*795d594fSAndroid Build Coastguard Worker element_size_(element_size) { 57*795d594fSAndroid Build Coastguard Worker DCHECK(array_ != nullptr || length == 0); 58*795d594fSAndroid Build Coastguard Worker } 59*795d594fSAndroid Build Coastguard Worker 60*795d594fSAndroid Build Coastguard Worker ArraySlice(LengthPrefixedArray<T>* lpa, 61*795d594fSAndroid Build Coastguard Worker size_t element_size = sizeof(T), 62*795d594fSAndroid Build Coastguard Worker size_t alignment = alignof(T)) 63*795d594fSAndroid Build Coastguard Worker : ArraySlice( 64*795d594fSAndroid Build Coastguard Worker lpa != nullptr && lpa->size() != 0 ? &lpa->At(0, element_size, alignment) : nullptr, 65*795d594fSAndroid Build Coastguard Worker lpa != nullptr ? lpa->size() : 0, 66*795d594fSAndroid Build Coastguard Worker element_size) {} 67*795d594fSAndroid Build Coastguard Worker ArraySlice(const ArraySlice<T>&) = default; 68*795d594fSAndroid Build Coastguard Worker ArraySlice(ArraySlice<T>&&) noexcept = default; 69*795d594fSAndroid Build Coastguard Worker ArraySlice<T>& operator=(const ArraySlice<T>&) = default; 70*795d594fSAndroid Build Coastguard Worker ArraySlice<T>& operator=(ArraySlice<T>&&) noexcept = default; 71*795d594fSAndroid Build Coastguard Worker 72*795d594fSAndroid Build Coastguard Worker // Iterators. begin()73*795d594fSAndroid Build Coastguard Worker iterator begin() { return iterator(&AtUnchecked(0), element_size_); } begin()74*795d594fSAndroid Build Coastguard Worker const_iterator begin() const { return const_iterator(&AtUnchecked(0), element_size_); } cbegin()75*795d594fSAndroid Build Coastguard Worker const_iterator cbegin() const { return const_iterator(&AtUnchecked(0), element_size_); } end()76*795d594fSAndroid Build Coastguard Worker StrideIterator<T> end() { return StrideIterator<T>(&AtUnchecked(size_), element_size_); } end()77*795d594fSAndroid Build Coastguard Worker const_iterator end() const { return const_iterator(&AtUnchecked(size_), element_size_); } cend()78*795d594fSAndroid Build Coastguard Worker const_iterator cend() const { return const_iterator(&AtUnchecked(size_), element_size_); } rbegin()79*795d594fSAndroid Build Coastguard Worker reverse_iterator rbegin() { return reverse_iterator(end()); } rbegin()80*795d594fSAndroid Build Coastguard Worker const_reverse_iterator rbegin() const { return const_reverse_iterator(end()); } crbegin()81*795d594fSAndroid Build Coastguard Worker const_reverse_iterator crbegin() const { return const_reverse_iterator(cend()); } rend()82*795d594fSAndroid Build Coastguard Worker reverse_iterator rend() { return reverse_iterator(begin()); } rend()83*795d594fSAndroid Build Coastguard Worker const_reverse_iterator rend() const { return const_reverse_iterator(begin()); } crend()84*795d594fSAndroid Build Coastguard Worker const_reverse_iterator crend() const { return const_reverse_iterator(cbegin()); } 85*795d594fSAndroid Build Coastguard Worker 86*795d594fSAndroid Build Coastguard Worker // Size. size()87*795d594fSAndroid Build Coastguard Worker size_type size() const { return size_; } empty()88*795d594fSAndroid Build Coastguard Worker bool empty() const { return size() == 0u; } 89*795d594fSAndroid Build Coastguard Worker 90*795d594fSAndroid Build Coastguard Worker // Element access. NOTE: Not providing at() and data(). 91*795d594fSAndroid Build Coastguard Worker 92*795d594fSAndroid Build Coastguard Worker reference operator[](size_t index) { 93*795d594fSAndroid Build Coastguard Worker DCHECK_LT(index, size_); 94*795d594fSAndroid Build Coastguard Worker return AtUnchecked(index); 95*795d594fSAndroid Build Coastguard Worker } 96*795d594fSAndroid Build Coastguard Worker 97*795d594fSAndroid Build Coastguard Worker const_reference operator[](size_t index) const { 98*795d594fSAndroid Build Coastguard Worker DCHECK_LT(index, size_); 99*795d594fSAndroid Build Coastguard Worker return AtUnchecked(index); 100*795d594fSAndroid Build Coastguard Worker } 101*795d594fSAndroid Build Coastguard Worker front()102*795d594fSAndroid Build Coastguard Worker reference front() { 103*795d594fSAndroid Build Coastguard Worker DCHECK(!empty()); 104*795d594fSAndroid Build Coastguard Worker return (*this)[0]; 105*795d594fSAndroid Build Coastguard Worker } 106*795d594fSAndroid Build Coastguard Worker front()107*795d594fSAndroid Build Coastguard Worker const_reference front() const { 108*795d594fSAndroid Build Coastguard Worker DCHECK(!empty()); 109*795d594fSAndroid Build Coastguard Worker return (*this)[0]; 110*795d594fSAndroid Build Coastguard Worker } 111*795d594fSAndroid Build Coastguard Worker back()112*795d594fSAndroid Build Coastguard Worker reference back() { 113*795d594fSAndroid Build Coastguard Worker DCHECK(!empty()); 114*795d594fSAndroid Build Coastguard Worker return (*this)[size_ - 1u]; 115*795d594fSAndroid Build Coastguard Worker } 116*795d594fSAndroid Build Coastguard Worker back()117*795d594fSAndroid Build Coastguard Worker const_reference back() const { 118*795d594fSAndroid Build Coastguard Worker DCHECK(!empty()); 119*795d594fSAndroid Build Coastguard Worker return (*this)[size_ - 1u]; 120*795d594fSAndroid Build Coastguard Worker } 121*795d594fSAndroid Build Coastguard Worker SubArray(size_type pos)122*795d594fSAndroid Build Coastguard Worker ArraySlice<T> SubArray(size_type pos) { 123*795d594fSAndroid Build Coastguard Worker return SubArray(pos, size() - pos); 124*795d594fSAndroid Build Coastguard Worker } 125*795d594fSAndroid Build Coastguard Worker SubArray(size_type pos)126*795d594fSAndroid Build Coastguard Worker ArraySlice<const T> SubArray(size_type pos) const { 127*795d594fSAndroid Build Coastguard Worker return SubArray(pos, size() - pos); 128*795d594fSAndroid Build Coastguard Worker } 129*795d594fSAndroid Build Coastguard Worker SubArray(size_type pos,size_type length)130*795d594fSAndroid Build Coastguard Worker ArraySlice<T> SubArray(size_type pos, size_type length) { 131*795d594fSAndroid Build Coastguard Worker DCHECK_LE(pos, size()); 132*795d594fSAndroid Build Coastguard Worker DCHECK_LE(length, size() - pos); 133*795d594fSAndroid Build Coastguard Worker return ArraySlice<T>(&AtUnchecked(pos), length, element_size_); 134*795d594fSAndroid Build Coastguard Worker } 135*795d594fSAndroid Build Coastguard Worker SubArray(size_type pos,size_type length)136*795d594fSAndroid Build Coastguard Worker ArraySlice<const T> SubArray(size_type pos, size_type length) const { 137*795d594fSAndroid Build Coastguard Worker DCHECK_LE(pos, size()); 138*795d594fSAndroid Build Coastguard Worker DCHECK_LE(length, size() - pos); 139*795d594fSAndroid Build Coastguard Worker return ArraySlice<const T>(&AtUnchecked(pos), length, element_size_); 140*795d594fSAndroid Build Coastguard Worker } 141*795d594fSAndroid Build Coastguard Worker ElementSize()142*795d594fSAndroid Build Coastguard Worker size_t ElementSize() const { 143*795d594fSAndroid Build Coastguard Worker return element_size_; 144*795d594fSAndroid Build Coastguard Worker } 145*795d594fSAndroid Build Coastguard Worker Contains(const T * element)146*795d594fSAndroid Build Coastguard Worker bool Contains(const T* element) const { 147*795d594fSAndroid Build Coastguard Worker return &AtUnchecked(0) <= element && element < &AtUnchecked(size_) && 148*795d594fSAndroid Build Coastguard Worker ((reinterpret_cast<uintptr_t>(element) - 149*795d594fSAndroid Build Coastguard Worker reinterpret_cast<uintptr_t>(&AtUnchecked(0))) % element_size_) == 0; 150*795d594fSAndroid Build Coastguard Worker } 151*795d594fSAndroid Build Coastguard Worker OffsetOf(const T * element)152*795d594fSAndroid Build Coastguard Worker size_t OffsetOf(const T* element) const { 153*795d594fSAndroid Build Coastguard Worker DCHECK(Contains(element)); 154*795d594fSAndroid Build Coastguard Worker // Since it's possible element_size_ != sizeof(T) we cannot just use pointer arithmatic 155*795d594fSAndroid Build Coastguard Worker uintptr_t base_ptr = reinterpret_cast<uintptr_t>(&AtUnchecked(0)); 156*795d594fSAndroid Build Coastguard Worker uintptr_t obj_ptr = reinterpret_cast<uintptr_t>(element); 157*795d594fSAndroid Build Coastguard Worker return (obj_ptr - base_ptr) / element_size_; 158*795d594fSAndroid Build Coastguard Worker } 159*795d594fSAndroid Build Coastguard Worker 160*795d594fSAndroid Build Coastguard Worker private: AtUnchecked(size_t index)161*795d594fSAndroid Build Coastguard Worker T& AtUnchecked(size_t index) { 162*795d594fSAndroid Build Coastguard Worker return *reinterpret_cast<T*>(reinterpret_cast<uintptr_t>(array_) + index * element_size_); 163*795d594fSAndroid Build Coastguard Worker } 164*795d594fSAndroid Build Coastguard Worker AtUnchecked(size_t index)165*795d594fSAndroid Build Coastguard Worker const T& AtUnchecked(size_t index) const { 166*795d594fSAndroid Build Coastguard Worker return *reinterpret_cast<T*>(reinterpret_cast<uintptr_t>(array_) + index * element_size_); 167*795d594fSAndroid Build Coastguard Worker } 168*795d594fSAndroid Build Coastguard Worker 169*795d594fSAndroid Build Coastguard Worker T* array_; 170*795d594fSAndroid Build Coastguard Worker size_t size_; 171*795d594fSAndroid Build Coastguard Worker size_t element_size_; 172*795d594fSAndroid Build Coastguard Worker }; 173*795d594fSAndroid Build Coastguard Worker 174*795d594fSAndroid Build Coastguard Worker template<typename T> 175*795d594fSAndroid Build Coastguard Worker std::ostream& operator<<(std::ostream& os, const ArraySlice<T>& ts) { 176*795d594fSAndroid Build Coastguard Worker bool first = true; 177*795d594fSAndroid Build Coastguard Worker os << "["; 178*795d594fSAndroid Build Coastguard Worker for (const T& t : ts) { 179*795d594fSAndroid Build Coastguard Worker if (!first) { os << ", "; } 180*795d594fSAndroid Build Coastguard Worker first = false; 181*795d594fSAndroid Build Coastguard Worker os << t; 182*795d594fSAndroid Build Coastguard Worker } 183*795d594fSAndroid Build Coastguard Worker os << "]"; 184*795d594fSAndroid Build Coastguard Worker return os; 185*795d594fSAndroid Build Coastguard Worker } 186*795d594fSAndroid Build Coastguard Worker 187*795d594fSAndroid Build Coastguard Worker } // namespace art 188*795d594fSAndroid Build Coastguard Worker 189*795d594fSAndroid Build Coastguard Worker #endif // ART_LIBARTBASE_BASE_ARRAY_SLICE_H_ 190