/* * 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. * * Copyright 2010-2016 Danny Robson */ #include "extent.hpp" #include "debug.hpp" #include "maths.hpp" #include #include using util::extent; using util::extent_range; /////////////////////////////////////////////////////////////////////////////// template extent::extent (vector _v) { std::transform (std::begin (_v), std::end (_v), this->begin (), [] (auto i) { // using std::abs gives unsigned abs warnings under clang. this tricks // it sufficiently to quiet the warnings. return i < 0 ? -i : i; }); } /////////////////////////////////////////////////////////////////////////////// template extent extent::expanded (util::vector mag) const { return *this + mag; } //----------------------------------------------------------------------------- template extent extent::expanded (T t) const { return *this + util::vector {t}; } //----------------------------------------------------------------------------- template extent extent::contracted (vector t) const { return *this - t; } //----------------------------------------------------------------------------- template extent extent::contracted (T t) const { return *this - vector {t}; } /////////////////////////////////////////////////////////////////////////////// template bool extent::empty (void) const { return almost_zero (area()); } /////////////////////////////////////////////////////////////////////////////// template extent_range::extent_range (extent _target): m_target (_target) { ; } //----------------------------------------------------------------------------- template typename extent_range::iterator extent_range::begin (void) const { return iterator (m_target, util::point (0)); } //----------------------------------------------------------------------------- template typename extent_range::iterator extent_range::end (void) const { util::point cursor (0); cursor[S-1] = m_target[S-1]; return iterator (m_target, cursor); } /////////////////////////////////////////////////////////////////////////////// template extent_range::iterator::iterator (extent _target, util::point _cursor): m_cursor (_cursor), m_target (_target) { static_assert (std::is_integral::value, "range stepping size is ill-defined for non-integral types"); } //----------------------------------------------------------------------------- template util::point extent_range::iterator::operator* (void) const { return m_cursor; } //----------------------------------------------------------------------------- template typename extent_range::iterator& extent_range::iterator::operator++ (void) { ++m_cursor[0]; for (size_t i = 0; i < S - 1; ++i) { if (m_cursor[i] != m_target[i]) break; m_cursor[i] = 0; m_cursor[i+1]++; continue; } return *this; } //----------------------------------------------------------------------------- template bool extent_range::iterator::operator== (const iterator &rhs) const { return m_cursor == rhs.m_cursor; } //----------------------------------------------------------------------------- template bool extent_range::iterator::operator!= (const iterator &rhs) const { return m_cursor != rhs.m_cursor; } /////////////////////////////////////////////////////////////////////////////// namespace util::debug { template struct validator> { static bool is_valid (const extent &e) { return std::all_of (std::begin (e.data), std::end (e.data), [] (auto i) { return i >= 0; }); } }; } //----------------------------------------------------------------------------- #define INSTANTIATE_S_T(S,T) \ template struct util::extent; \ template bool util::debug::is_valid (const extent&); \ template struct util::debug::validator>; #define INSTANTIATE(T) \ INSTANTIATE_S_T(1,T) \ INSTANTIATE_S_T(2,T) \ INSTANTIATE_S_T(3,T) #define INSTANTIATE_INT(T) \ template struct util::extent_range<1,T>; \ template struct util::extent_range<2,T>; \ template struct util::extent_range<3,T>; INSTANTIATE( int16_t) INSTANTIATE( int32_t) INSTANTIATE( int64_t) INSTANTIATE(uint16_t) INSTANTIATE(uint32_t) INSTANTIATE(uint64_t) INSTANTIATE(float) INSTANTIATE(double) INSTANTIATE_INT( int16_t) INSTANTIATE_INT( int32_t) INSTANTIATE_INT( int64_t) INSTANTIATE_INT(uint16_t) INSTANTIATE_INT(uint32_t) INSTANTIATE_INT(uint64_t)