#ifndef _DEDEFS_HPP #define _DEDEFS_HPP /*------------------------------------------------------------------------- * drawElements C++ Base Library * ----------------------------- * * Copyright 2014 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * *//*! * \file * \brief Basic definitions. *//*--------------------------------------------------------------------*/ #include "deDefs.h" #include "deInt32.h" #if !defined(__cplusplus) # error "C++ is required" #endif #include namespace de { //! Compute absolute value of x. template inline T abs (T x) { return x < T(0) ? -x : x; } //! Get minimum of x and y. template inline T min (T x, T y) { return x <= y ? x : y; } //! Get maximum of x and y. template inline T max (T x, T y) { return x >= y ? x : y; } //! Clamp x in range a <= x <= b. template inline T clamp (T x, T a, T b) { DE_ASSERT(a <= b); return x < a ? a : (x > b ? b : x); } //! Test if x is in bounds a <= x < b. template inline bool inBounds (T x, T a, T b) { return a <= x && x < b; } //! Test if x is in range a <= x <= b. template inline bool inRange (T x, T a, T b) { return a <= x && x <= b; } //! Return T with low n bits set template inline T rightSetMask (T n) { DE_ASSERT(n < T(sizeof(T) * 8)); T one = T(1); return T((one << n) - one); } //! Return T with low n bits reset template inline T rightZeroMask (T n) { return T(~rightSetMask(n)); } //! Return T with high n bits set template inline T leftSetMask (T n) { const T tlen = T(sizeof(T) * 8); return T(~rightSetMask(tlen >= n ? tlen - n : T(0))); } //! Return T with high n bits reset template inline T leftZeroMask (T n) { return T(~leftSetMask(n)); } //! Round x up to a multiple of y. template inline T roundUp (T x, T y) { DE_ASSERT(y != T(0)); const T mod = x % y; return x + ((mod == T(0)) ? T(0) : (y - mod)); } //! Round x down to a multiple of y. template inline T roundDown (T x, T y) { DE_ASSERT(y != T(0)); return (x / y) * y; } //! Find the greatest common divisor of x and y. template T gcd (T x, T y) { DE_ASSERT(std::is_integral::value && std::is_unsigned::value); // Euclidean algorithm. while (y != T{0}) { T mod = x % y; x = y; y = mod; } return x; } //! Find the least common multiple of x and y. template T lcm (T x, T y) { DE_ASSERT(std::is_integral::value && std::is_unsigned::value); T prod = x * y; DE_ASSERT(x == 0 || prod / x == y); // Check overflow just in case. return (prod) / gcd(x, y); } //! Helper for DE_CHECK() macros. void throwRuntimeError (const char* message, const char* expr, const char* file, int line); //! Default deleter. template struct DefaultDeleter { inline DefaultDeleter (void) {} template inline DefaultDeleter (const DefaultDeleter&) {} template inline DefaultDeleter& operator= (const DefaultDeleter&) { return *this; } inline void operator() (T* ptr) const { delete ptr; } }; //! A deleter for arrays template struct ArrayDeleter { inline ArrayDeleter (void) {} template inline ArrayDeleter (const ArrayDeleter&) {} template inline ArrayDeleter& operator= (const ArrayDeleter&) { return *this; } inline void operator() (T* ptr) const { delete[] ptr; } }; //! Get required memory alignment for type template size_t alignOf (void) { struct PaddingCheck { deUint8 b; T t; }; return (size_t)DE_OFFSET_OF(PaddingCheck, t); } //! Similar to DE_LENGTH_OF_ARRAY but constexpr and without auxiliar helpers. template constexpr size_t arrayLength (const T (&)[N]) { return N; } //! Get least significant bit index inline int findLSB (deUint32 value) { return value ? deCtz32(value) : (-1); } //! Get most significant bit index inline int findMSB (deUint32 value) { return 31 - deClz32(value); } //! Get most significant bit index inline int findMSB (deInt32 value) { return (value < 0) ? findMSB(~(deUint32)value) : findMSB((deUint32)value); } } // de /*--------------------------------------------------------------------*//*! * \brief Throw runtime error if condition is not met. * \param X Condition to check. * * This macro throws std::runtime_error if condition X is not met. *//*--------------------------------------------------------------------*/ #define DE_CHECK_RUNTIME_ERR(X) do { if ((!deGetFalse() && (X)) ? DE_FALSE : DE_TRUE) ::de::throwRuntimeError(DE_NULL, #X, __FILE__, __LINE__); } while(deGetFalse()) /*--------------------------------------------------------------------*//*! * \brief Throw runtime error if condition is not met. * \param X Condition to check. * \param MSG Additional message to include in the exception. * * This macro throws std::runtime_error with message MSG if condition X is * not met. *//*--------------------------------------------------------------------*/ #define DE_CHECK_RUNTIME_ERR_MSG(X, MSG) do { if ((!deGetFalse() && (X)) ? DE_FALSE : DE_TRUE) ::de::throwRuntimeError(MSG, #X, __FILE__, __LINE__); } while(deGetFalse()) //! Get array start pointer. #define DE_ARRAY_BEGIN(ARR) (&(ARR)[0]) //! Get array end pointer. #define DE_ARRAY_END(ARR) (DE_ARRAY_BEGIN(ARR) + DE_LENGTH_OF_ARRAY(ARR)) //! Empty C++ compilation unit silencing. #if (DE_COMPILER == DE_COMPILER_MSC) # define DE_EMPTY_CPP_FILE namespace { deUint8 unused; } #else # define DE_EMPTY_CPP_FILE #endif // Warn if type is constructed, but left unused // // Used in types with non-trivial ctor/dtor but with ctor-dtor pair causing no (observable) // side-effects. // // \todo add attribute for GCC #if (DE_COMPILER == DE_COMPILER_CLANG) && defined(__has_attribute) # if __has_attribute(warn_unused) # define DE_WARN_UNUSED_TYPE __attribute__((warn_unused)) # else # define DE_WARN_UNUSED_TYPE # endif #else # define DE_WARN_UNUSED_TYPE #endif #endif // _DEDEFS_HPP