///////////////////////////////////////////////////////////////////////////// // Copyright (c) Electronic Arts Inc. All rights reserved. ///////////////////////////////////////////////////////////////////////////// /////////////////////////////////////////////////////////////////////////////// // Implements a templated array class as per the C++ standard TR1 (technical // report 1, which is a list of proposed C++ library amendments). // The primary distinctions between this array and TR1 array are: // - array::size_type is defined as eastl_size_t instead of size_t in order // to save memory and run faster on 64 bit systems. /////////////////////////////////////////////////////////////////////////////// #ifndef EASTL_ARRAY_H #define EASTL_ARRAY_H #include #include #include #include #include #if EASTL_EXCEPTIONS_ENABLED EA_DISABLE_ALL_VC_WARNINGS() #include // std::out_of_range, std::length_error. EA_RESTORE_ALL_VC_WARNINGS() #endif #if defined(EA_PRAGMA_ONCE_SUPPORTED) #pragma once // Some compilers (e.g. VC++) benefit significantly from using this. We've measured 3-4% build speed improvements in apps as a result. #endif namespace eastl { /////////////////////////////////////////////////////////////////////// /// array /// /// Implements a templated array class as per the C++ standard TR1. /// This class allows you to use a built-in C style array like an STL vector. /// It does not let you change its size, as it is just like a C built-in array. /// Our implementation here strives to remove function call nesting, as that /// makes it hard for us to profile debug builds due to function call overhead. /// Note that this is intentionally a struct with public data, as per the /// C++ standard update proposal requirements. /// /// Example usage: /// array a = { { 0, 1, 2, 3, 4 } }; // Strict compilers such as GCC require the double brackets. /// a[2] = 4; /// for(array::iterator i = a.begin(); i < a.end(); ++i) /// *i = 0; /// template struct array { public: typedef array this_type; typedef T value_type; typedef value_type& reference; typedef const value_type& const_reference; typedef value_type* iterator; typedef const value_type* const_iterator; typedef eastl::reverse_iterator reverse_iterator; typedef eastl::reverse_iterator const_reverse_iterator; typedef eastl_size_t size_type; // See config.h for the definition of eastl_size_t, which defaults to size_t. typedef ptrdiff_t difference_type; public: enum { count = N }; // Note that the member data is intentionally public. // This allows for aggregate initialization of the // object (e.g. array a = { 0, 3, 2, 4 }; ) value_type mValue[N ? N : 1]; public: // We intentionally provide no constructor, destructor, or assignment operator. void fill(const value_type& value); // Unlike the swap function for other containers, array::swap takes linear time, // may exit via an exception, and does not cause iterators to become associated with the other container. void swap(this_type& x) EA_NOEXCEPT_IF(eastl::is_nothrow_swappable::value); EA_CPP14_CONSTEXPR iterator begin() EA_NOEXCEPT; EA_CPP14_CONSTEXPR const_iterator begin() const EA_NOEXCEPT; EA_CPP14_CONSTEXPR const_iterator cbegin() const EA_NOEXCEPT; EA_CPP14_CONSTEXPR iterator end() EA_NOEXCEPT; EA_CPP14_CONSTEXPR const_iterator end() const EA_NOEXCEPT; EA_CPP14_CONSTEXPR const_iterator cend() const EA_NOEXCEPT; EA_CPP14_CONSTEXPR reverse_iterator rbegin() EA_NOEXCEPT; EA_CPP14_CONSTEXPR const_reverse_iterator rbegin() const EA_NOEXCEPT; EA_CPP14_CONSTEXPR const_reverse_iterator crbegin() const EA_NOEXCEPT; EA_CPP14_CONSTEXPR reverse_iterator rend() EA_NOEXCEPT; EA_CPP14_CONSTEXPR const_reverse_iterator rend() const EA_NOEXCEPT; EA_CPP14_CONSTEXPR const_reverse_iterator crend() const EA_NOEXCEPT; EA_CPP14_CONSTEXPR bool empty() const EA_NOEXCEPT; EA_CPP14_CONSTEXPR size_type size() const EA_NOEXCEPT; EA_CPP14_CONSTEXPR size_type max_size() const EA_NOEXCEPT; EA_CPP14_CONSTEXPR T* data() EA_NOEXCEPT; EA_CPP14_CONSTEXPR const T* data() const EA_NOEXCEPT; EA_CPP14_CONSTEXPR reference operator[](size_type i); EA_CPP14_CONSTEXPR const_reference operator[](size_type i) const; EA_CPP14_CONSTEXPR const_reference at(size_type i) const; EA_CPP14_CONSTEXPR reference at(size_type i); EA_CPP14_CONSTEXPR reference front(); EA_CPP14_CONSTEXPR const_reference front() const; EA_CPP14_CONSTEXPR reference back(); EA_CPP14_CONSTEXPR const_reference back() const; bool validate() const; int validate_iterator(const_iterator i) const; }; // class array /////////////////////////////////////////////////////////////////////////// // template deduction guides /////////////////////////////////////////////////////////////////////////// #ifdef __cpp_deduction_guides template array(T, U...) -> array; #endif /////////////////////////////////////////////////////////////////////// // array /////////////////////////////////////////////////////////////////////// template inline void array::fill(const value_type& value) { eastl::fill_n(&mValue[0], N, value); } template inline void array::swap(this_type& x) EA_NOEXCEPT_IF(eastl::is_nothrow_swappable::value) { eastl::swap_ranges(&mValue[0], &mValue[N], &x.mValue[0]); } template EA_CPP14_CONSTEXPR inline typename array::iterator array::begin() EA_NOEXCEPT { return &mValue[0]; } template EA_CPP14_CONSTEXPR inline typename array::const_iterator array::begin() const EA_NOEXCEPT { return &mValue[0]; } template EA_CPP14_CONSTEXPR inline typename array::const_iterator array::cbegin() const EA_NOEXCEPT { return &mValue[0]; } template EA_CPP14_CONSTEXPR inline typename array::iterator array::end() EA_NOEXCEPT { return &mValue[N]; } template EA_CPP14_CONSTEXPR inline typename array::const_iterator array::end() const EA_NOEXCEPT { return &mValue[N]; } template EA_CPP14_CONSTEXPR inline typename array::const_iterator array::cend() const EA_NOEXCEPT { return &mValue[N]; } template EA_CPP14_CONSTEXPR inline typename array::reverse_iterator array::rbegin() EA_NOEXCEPT { return reverse_iterator(&mValue[N]); } template EA_CPP14_CONSTEXPR inline typename array::const_reverse_iterator array::rbegin() const EA_NOEXCEPT { return const_reverse_iterator(&mValue[N]); } template EA_CPP14_CONSTEXPR inline typename array::const_reverse_iterator array::crbegin() const EA_NOEXCEPT { return const_reverse_iterator(&mValue[N]); } template EA_CPP14_CONSTEXPR inline typename array::reverse_iterator array::rend() EA_NOEXCEPT { return reverse_iterator(&mValue[0]); } template EA_CPP14_CONSTEXPR inline typename array::const_reverse_iterator array::rend() const EA_NOEXCEPT { return const_reverse_iterator(static_cast(&mValue[0])); } template EA_CPP14_CONSTEXPR inline typename array::const_reverse_iterator array::crend() const EA_NOEXCEPT { return const_reverse_iterator(static_cast(&mValue[0])); } template EA_CPP14_CONSTEXPR inline typename array::size_type array::size() const EA_NOEXCEPT { return (size_type)N; } template EA_CPP14_CONSTEXPR inline typename array::size_type array::max_size() const EA_NOEXCEPT { return (size_type)N; } template EA_CPP14_CONSTEXPR inline bool array::empty() const EA_NOEXCEPT { return (N == 0); } template EA_CPP14_CONSTEXPR inline typename array::reference array::operator[](size_type i) { #if EASTL_ASSERT_ENABLED if(EASTL_UNLIKELY(i >= N)) EASTL_FAIL_MSG("array::operator[] -- out of range"); #endif EA_ANALYSIS_ASSUME(i < N); return mValue[i]; } template EA_CPP14_CONSTEXPR inline typename array::const_reference array::operator[](size_type i) const { #if EASTL_ASSERT_ENABLED if(EASTL_UNLIKELY(i >= N)) EASTL_FAIL_MSG("array::operator[] -- out of range"); #endif EA_ANALYSIS_ASSUME(i < N); return mValue[i]; } template EA_CPP14_CONSTEXPR inline typename array::reference array::front() { #if EASTL_ASSERT_ENABLED if(EASTL_UNLIKELY(empty())) // We don't allow the user to reference an empty container. EASTL_FAIL_MSG("array::front -- empty array"); #endif return mValue[0]; } template EA_CPP14_CONSTEXPR inline typename array::const_reference array::front() const { #if EASTL_ASSERT_ENABLED if(EASTL_UNLIKELY(empty())) // We don't allow the user to reference an empty container. EASTL_FAIL_MSG("array::front -- empty array"); #endif return mValue[0]; } template EA_CPP14_CONSTEXPR inline typename array::reference array::back() { #if EASTL_ASSERT_ENABLED if(EASTL_UNLIKELY(empty())) // We don't allow the user to reference an empty container. EASTL_FAIL_MSG("array::back -- empty array"); #endif return mValue[N - 1]; } template EA_CPP14_CONSTEXPR inline typename array::const_reference array::back() const { #if EASTL_ASSERT_ENABLED if(EASTL_UNLIKELY(empty())) // We don't allow the user to reference an empty container. EASTL_FAIL_MSG("array::back -- empty array"); #endif return mValue[N - 1]; } template EA_CPP14_CONSTEXPR inline T* array::data() EA_NOEXCEPT { return mValue; } template EA_CPP14_CONSTEXPR inline const T* array::data() const EA_NOEXCEPT { return mValue; } template EA_CPP14_CONSTEXPR inline typename array::const_reference array::at(size_type i) const { #if EASTL_EXCEPTIONS_ENABLED if(EASTL_UNLIKELY(i >= N)) throw std::out_of_range("array::at -- out of range"); #elif EASTL_ASSERT_ENABLED if(EASTL_UNLIKELY(i >= N)) EASTL_FAIL_MSG("array::at -- out of range"); #endif EA_ANALYSIS_ASSUME(i < N); return static_cast(mValue[i]); } template EA_CPP14_CONSTEXPR inline typename array::reference array::at(size_type i) { #if EASTL_EXCEPTIONS_ENABLED if(EASTL_UNLIKELY(i >= N)) throw std::out_of_range("array::at -- out of range"); #elif EASTL_ASSERT_ENABLED if(EASTL_UNLIKELY(i >= N)) EASTL_FAIL_MSG("array::at -- out of range"); #endif EA_ANALYSIS_ASSUME(i < N); return static_cast(mValue[i]); } template inline bool array::validate() const { return true; // There is nothing to do. } template inline int array::validate_iterator(const_iterator i) const { if(i >= mValue) { if(i < (mValue + N)) return (isf_valid | isf_current | isf_can_dereference); if(i <= (mValue + N)) return (isf_valid | isf_current); } return isf_none; } /////////////////////////////////////////////////////////////////////// // global operators /////////////////////////////////////////////////////////////////////// template EA_CPP14_CONSTEXPR inline bool operator==(const array& a, const array& b) { return eastl::equal(&a.mValue[0], &a.mValue[N], &b.mValue[0]); } template EA_CPP14_CONSTEXPR inline bool operator<(const array& a, const array& b) { return eastl::lexicographical_compare(&a.mValue[0], &a.mValue[N], &b.mValue[0], &b.mValue[N]); } template EA_CPP14_CONSTEXPR inline bool operator!=(const array& a, const array& b) { return !eastl::equal(&a.mValue[0], &a.mValue[N], &b.mValue[0]); } template EA_CPP14_CONSTEXPR inline bool operator>(const array& a, const array& b) { return eastl::lexicographical_compare(&b.mValue[0], &b.mValue[N], &a.mValue[0], &a.mValue[N]); } template EA_CPP14_CONSTEXPR inline bool operator<=(const array& a, const array& b) { return !eastl::lexicographical_compare(&b.mValue[0], &b.mValue[N], &a.mValue[0], &a.mValue[N]); } template EA_CPP14_CONSTEXPR inline bool operator>=(const array& a, const array& b) { return !eastl::lexicographical_compare(&a.mValue[0], &a.mValue[N], &b.mValue[0], &b.mValue[N]); } template inline void swap(array& a, array& b) { eastl::swap_ranges(&a.mValue[0], &a.mValue[N], &b.mValue[0]); } /////////////////////////////////////////////////////////////////////// // to_array /////////////////////////////////////////////////////////////////////// namespace internal { template EA_CONSTEXPR auto to_array(T (&a)[N], index_sequence) { return eastl::array, N>{{a[I]...}}; } template EA_CONSTEXPR auto to_array(T (&&a)[N], index_sequence) { return eastl::array, N>{{eastl::move(a[I])...}}; } } template EA_CONSTEXPR eastl::array, N> to_array(T (&a)[N]) { static_assert(eastl::is_constructible_v, "element type T must be copy-initializable"); static_assert(!eastl::is_array_v, "passing multidimensional arrays to to_array is ill-formed"); return internal::to_array(a, eastl::make_index_sequence{}); } template EA_CONSTEXPR eastl::array, N> to_array(T (&&a)[N]) { static_assert(eastl::is_move_constructible_v, "element type T must be move-constructible"); static_assert(!eastl::is_array_v, "passing multidimensional arrays to to_array is ill-formed"); return internal::to_array(eastl::move(a), eastl::make_index_sequence{}); } } // namespace eastl #endif // Header include guard