| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298 |
- // Boost.Geometry (aka GGL, Generic Geometry Library)
- // Copyright (c) 2007-2015 Barend Gehrels, Amsterdam, the Netherlands.
- // Copyright (c) 2017 Adam Wulkiewicz, Lodz, Poland.
- // This file was modified by Oracle on 2013-2016.
- // Modifications copyright (c) 2013-2016 Oracle and/or its affiliates.
- // Contributed and/or modified by Adam Wulkiewicz, on behalf of Oracle
- // Use, modification and distribution is subject to the Boost Software License,
- // Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
- // http://www.boost.org/LICENSE_1_0.txt)
- #ifndef BOOST_GEOMETRY_ALGORITHMS_DETAIL_RELATE_RESULT_HPP
- #define BOOST_GEOMETRY_ALGORITHMS_DETAIL_RELATE_RESULT_HPP
- #include <cstddef>
- #include <cstring>
- #include <boost/mpl/assert.hpp>
- #include <boost/mpl/at.hpp>
- #include <boost/mpl/begin.hpp>
- #include <boost/mpl/deref.hpp>
- #include <boost/mpl/end.hpp>
- #include <boost/mpl/is_sequence.hpp>
- #include <boost/mpl/next.hpp>
- #include <boost/static_assert.hpp>
- #include <boost/throw_exception.hpp>
- #include <boost/tuple/tuple.hpp>
- #include <boost/type_traits/integral_constant.hpp>
- #include <boost/geometry/core/assert.hpp>
- #include <boost/geometry/core/coordinate_dimension.hpp>
- #include <boost/geometry/core/exception.hpp>
- #include <boost/geometry/util/condition.hpp>
- namespace boost { namespace geometry {
- #ifndef DOXYGEN_NO_DETAIL
- namespace detail { namespace relate {
- enum field { interior = 0, boundary = 1, exterior = 2 };
- // TODO: IF THE RESULT IS UPDATED WITH THE MAX POSSIBLE VALUE FOR SOME PAIR OF GEOEMTRIES
- // THE VALUE ALREADY STORED MUSN'T BE CHECKED
- // update() calls chould be replaced with set() in those cases
- // but for safety reasons (STATIC_ASSERT) we should check if parameter D is valid and set() doesn't do that
- // so some additional function could be added, e.g. set_dim()
- // --------------- MATRIX ----------------
- // matrix
- template <std::size_t Height, std::size_t Width = Height>
- class matrix
- {
- public:
- typedef char value_type;
- typedef std::size_t size_type;
- typedef const char * const_iterator;
- typedef const_iterator iterator;
- static const std::size_t static_width = Width;
- static const std::size_t static_height = Height;
- static const std::size_t static_size = Width * Height;
-
- inline matrix()
- {
- ::memset(m_array, 'F', static_size);
- }
- template <field F1, field F2>
- inline char get() const
- {
- BOOST_STATIC_ASSERT(F1 < Height && F2 < Width);
- static const std::size_t index = F1 * Width + F2;
- BOOST_STATIC_ASSERT(index < static_size);
- return m_array[index];
- }
- template <field F1, field F2, char V>
- inline void set()
- {
- BOOST_STATIC_ASSERT(F1 < Height && F2 < Width);
- static const std::size_t index = F1 * Width + F2;
- BOOST_STATIC_ASSERT(index < static_size);
- m_array[index] = V;
- }
- inline char operator[](std::size_t index) const
- {
- BOOST_GEOMETRY_ASSERT(index < static_size);
- return m_array[index];
- }
- inline const_iterator begin() const
- {
- return m_array;
- }
- inline const_iterator end() const
- {
- return m_array + static_size;
- }
- inline static std::size_t size()
- {
- return static_size;
- }
-
- inline const char * data() const
- {
- return m_array;
- }
- inline std::string str() const
- {
- return std::string(m_array, static_size);
- }
- private:
- char m_array[static_size];
- };
- // matrix_handler
- template <typename Matrix>
- class matrix_handler
- {
- public:
- typedef Matrix result_type;
- static const bool interrupt = false;
- matrix_handler()
- {}
- result_type const& result() const
- {
- return m_matrix;
- }
- result_type const& matrix() const
- {
- return m_matrix;
- }
- result_type & matrix()
- {
- return m_matrix;
- }
- template <field F1, field F2, char D>
- inline bool may_update() const
- {
- BOOST_STATIC_ASSERT('0' <= D && D <= '9');
- char const c = m_matrix.template get<F1, F2>();
- return D > c || c > '9';
- }
- template <field F1, field F2, char V>
- inline void set()
- {
- static const bool in_bounds = F1 < Matrix::static_height
- && F2 < Matrix::static_width;
- typedef boost::integral_constant<bool, in_bounds> in_bounds_t;
- set_dispatch<F1, F2, V>(in_bounds_t());
- }
- template <field F1, field F2, char D>
- inline void update()
- {
- static const bool in_bounds = F1 < Matrix::static_height
- && F2 < Matrix::static_width;
- typedef boost::integral_constant<bool, in_bounds> in_bounds_t;
- update_dispatch<F1, F2, D>(in_bounds_t());
- }
- private:
- template <field F1, field F2, char V>
- inline void set_dispatch(integral_constant<bool, true>)
- {
- static const std::size_t index = F1 * Matrix::static_width + F2;
- BOOST_STATIC_ASSERT(index < Matrix::static_size);
- BOOST_STATIC_ASSERT(('0' <= V && V <= '9') || V == 'T' || V == 'F');
- m_matrix.template set<F1, F2, V>();
- }
- template <field F1, field F2, char V>
- inline void set_dispatch(integral_constant<bool, false>)
- {}
- template <field F1, field F2, char D>
- inline void update_dispatch(integral_constant<bool, true>)
- {
- static const std::size_t index = F1 * Matrix::static_width + F2;
- BOOST_STATIC_ASSERT(index < Matrix::static_size);
- BOOST_STATIC_ASSERT('0' <= D && D <= '9');
- char const c = m_matrix.template get<F1, F2>();
- if ( D > c || c > '9')
- m_matrix.template set<F1, F2, D>();
- }
- template <field F1, field F2, char D>
- inline void update_dispatch(integral_constant<bool, false>)
- {}
- Matrix m_matrix;
- };
- // --------------- RUN-TIME MASK ----------------
- // run-time mask
- template <std::size_t Height, std::size_t Width = Height>
- class mask
- {
- public:
- static const std::size_t static_width = Width;
- static const std::size_t static_height = Height;
- static const std::size_t static_size = Width * Height;
- inline mask(const char * s)
- {
- char * it = m_array;
- char * const last = m_array + static_size;
- for ( ; it != last && *s != '\0' ; ++it, ++s )
- {
- char c = *s;
- check_char(c);
- *it = c;
- }
- if ( it != last )
- {
- ::memset(it, '*', last - it);
- }
- }
- inline mask(const char * s, std::size_t count)
- {
- if ( count > static_size )
- {
- count = static_size;
- }
- if ( count > 0 )
- {
- std::for_each(s, s + count, check_char);
- ::memcpy(m_array, s, count);
- }
- if ( count < static_size )
- {
- ::memset(m_array + count, '*', static_size - count);
- }
- }
- template <field F1, field F2>
- inline char get() const
- {
- BOOST_STATIC_ASSERT(F1 < Height && F2 < Width);
- static const std::size_t index = F1 * Width + F2;
- BOOST_STATIC_ASSERT(index < static_size);
- return m_array[index];
- }
- private:
- static inline void check_char(char c)
- {
- bool const is_valid = c == '*' || c == 'T' || c == 'F'
- || ( c >= '0' && c <= '9' );
- if ( !is_valid )
- {
- BOOST_THROW_EXCEPTION(geometry::invalid_input_exception());
- }
- }
- char m_array[static_size];
- };
- // interrupt()
- template <typename Mask, bool InterruptEnabled>
- struct interrupt_dispatch
- {
- template <field F1, field F2, char V>
- static inline bool apply(Mask const&)
- {
- return false;
- }
- };
- template <typename Mask>
- struct interrupt_dispatch<Mask, true>
- {
- template <field F1, field F2, char V>
- static inline bool apply(Mask const& mask)
- {
- char m = mask.template get<F1, F2>();
- return check_element<V>(m);
- }
- template <char V>
- static inline bool check_element(char m)
- {
- if ( BOOST_GEOMETRY_CONDITION(V >= '0' && V <= '9') )
- {
- return m == 'F' || ( m < V && m >= '0' && m <= '9' );
- }
- else if ( BOOST_GEOMETRY_CONDITION(V == 'T') )
- {
- return m == 'F';
- }
- return false;
- }
- };
- template <typename Masks, int I = 0, int N = boost::tuples::length<Masks>::value>
- struct interrupt_dispatch_tuple
- {
- template <field F1, field F2, char V>
- static inline bool apply(Masks const& masks)
- {
- typedef typename boost::tuples::element<I, Masks>::type mask_type;
- mask_type const& mask = boost::get<I>(masks);
- return interrupt_dispatch<mask_type, true>::template apply<F1, F2, V>(mask)
- && interrupt_dispatch_tuple<Masks, I+1>::template apply<F1, F2, V>(masks);
- }
- };
- template <typename Masks, int N>
- struct interrupt_dispatch_tuple<Masks, N, N>
- {
- template <field F1, field F2, char V>
- static inline bool apply(Masks const& )
- {
- return true;
- }
- };
- //template <typename T0, typename T1, typename T2, typename T3, typename T4,
- // typename T5, typename T6, typename T7, typename T8, typename T9>
- //struct interrupt_dispatch<boost::tuple<T0, T1, T2, T3, T4, T5, T6, T7, T8, T9>, true>
- //{
- // typedef boost::tuple<T0, T1, T2, T3, T4, T5, T6, T7, T8, T9> mask_type;
- // template <field F1, field F2, char V>
- // static inline bool apply(mask_type const& mask)
- // {
- // return interrupt_dispatch_tuple<mask_type>::template apply<F1, F2, V>(mask);
- // }
- //};
- template <typename Head, typename Tail>
- struct interrupt_dispatch<boost::tuples::cons<Head, Tail>, true>
- {
- typedef boost::tuples::cons<Head, Tail> mask_type;
- template <field F1, field F2, char V>
- static inline bool apply(mask_type const& mask)
- {
- return interrupt_dispatch_tuple<mask_type>::template apply<F1, F2, V>(mask);
- }
- };
- template <field F1, field F2, char V, bool InterruptEnabled, typename Mask>
- inline bool interrupt(Mask const& mask)
- {
- return interrupt_dispatch<Mask, InterruptEnabled>
- ::template apply<F1, F2, V>(mask);
- }
- // may_update()
- template <typename Mask>
- struct may_update_dispatch
- {
- template <field F1, field F2, char D, typename Matrix>
- static inline bool apply(Mask const& mask, Matrix const& matrix)
- {
- BOOST_STATIC_ASSERT('0' <= D && D <= '9');
- char const m = mask.template get<F1, F2>();
-
- if ( m == 'F' )
- {
- return true;
- }
- else if ( m == 'T' )
- {
- char const c = matrix.template get<F1, F2>();
- return c == 'F'; // if it's T or between 0 and 9, the result will be the same
- }
- else if ( m >= '0' && m <= '9' )
- {
- char const c = matrix.template get<F1, F2>();
- return D > c || c > '9';
- }
- return false;
- }
- };
- template <typename Masks, int I = 0, int N = boost::tuples::length<Masks>::value>
- struct may_update_dispatch_tuple
- {
- template <field F1, field F2, char D, typename Matrix>
- static inline bool apply(Masks const& masks, Matrix const& matrix)
- {
- typedef typename boost::tuples::element<I, Masks>::type mask_type;
- mask_type const& mask = boost::get<I>(masks);
- return may_update_dispatch<mask_type>::template apply<F1, F2, D>(mask, matrix)
- || may_update_dispatch_tuple<Masks, I+1>::template apply<F1, F2, D>(masks, matrix);
- }
- };
- template <typename Masks, int N>
- struct may_update_dispatch_tuple<Masks, N, N>
- {
- template <field F1, field F2, char D, typename Matrix>
- static inline bool apply(Masks const& , Matrix const& )
- {
- return false;
- }
- };
- //template <typename T0, typename T1, typename T2, typename T3, typename T4,
- // typename T5, typename T6, typename T7, typename T8, typename T9>
- //struct may_update_dispatch< boost::tuple<T0, T1, T2, T3, T4, T5, T6, T7, T8, T9> >
- //{
- // typedef boost::tuple<T0, T1, T2, T3, T4, T5, T6, T7, T8, T9> mask_type;
- // template <field F1, field F2, char D, typename Matrix>
- // static inline bool apply(mask_type const& mask, Matrix const& matrix)
- // {
- // return may_update_dispatch_tuple<mask_type>::template apply<F1, F2, D>(mask, matrix);
- // }
- //};
- template <typename Head, typename Tail>
- struct may_update_dispatch< boost::tuples::cons<Head, Tail> >
- {
- typedef boost::tuples::cons<Head, Tail> mask_type;
- template <field F1, field F2, char D, typename Matrix>
- static inline bool apply(mask_type const& mask, Matrix const& matrix)
- {
- return may_update_dispatch_tuple<mask_type>::template apply<F1, F2, D>(mask, matrix);
- }
- };
- template <field F1, field F2, char D, typename Mask, typename Matrix>
- inline bool may_update(Mask const& mask, Matrix const& matrix)
- {
- return may_update_dispatch<Mask>
- ::template apply<F1, F2, D>(mask, matrix);
- }
- // check_matrix()
- template <typename Mask>
- struct check_dispatch
- {
- template <typename Matrix>
- static inline bool apply(Mask const& mask, Matrix const& matrix)
- {
- return per_one<interior, interior>(mask, matrix)
- && per_one<interior, boundary>(mask, matrix)
- && per_one<interior, exterior>(mask, matrix)
- && per_one<boundary, interior>(mask, matrix)
- && per_one<boundary, boundary>(mask, matrix)
- && per_one<boundary, exterior>(mask, matrix)
- && per_one<exterior, interior>(mask, matrix)
- && per_one<exterior, boundary>(mask, matrix)
- && per_one<exterior, exterior>(mask, matrix);
- }
- template <field F1, field F2, typename Matrix>
- static inline bool per_one(Mask const& mask, Matrix const& matrix)
- {
- const char mask_el = mask.template get<F1, F2>();
- const char el = matrix.template get<F1, F2>();
- if ( mask_el == 'F' )
- {
- return el == 'F';
- }
- else if ( mask_el == 'T' )
- {
- return el == 'T' || ( el >= '0' && el <= '9' );
- }
- else if ( mask_el >= '0' && mask_el <= '9' )
- {
- return el == mask_el;
- }
- return true;
- }
- };
- template <typename Masks, int I = 0, int N = boost::tuples::length<Masks>::value>
- struct check_dispatch_tuple
- {
- template <typename Matrix>
- static inline bool apply(Masks const& masks, Matrix const& matrix)
- {
- typedef typename boost::tuples::element<I, Masks>::type mask_type;
- mask_type const& mask = boost::get<I>(masks);
- return check_dispatch<mask_type>::apply(mask, matrix)
- || check_dispatch_tuple<Masks, I+1>::apply(masks, matrix);
- }
- };
- template <typename Masks, int N>
- struct check_dispatch_tuple<Masks, N, N>
- {
- template <typename Matrix>
- static inline bool apply(Masks const&, Matrix const&)
- {
- return false;
- }
- };
- //template <typename T0, typename T1, typename T2, typename T3, typename T4,
- // typename T5, typename T6, typename T7, typename T8, typename T9>
- //struct check_dispatch< boost::tuple<T0, T1, T2, T3, T4, T5, T6, T7, T8, T9> >
- //{
- // typedef boost::tuple<T0, T1, T2, T3, T4, T5, T6, T7, T8, T9> mask_type;
- // template <typename Matrix>
- // static inline bool apply(mask_type const& mask, Matrix const& matrix)
- // {
- // return check_dispatch_tuple<mask_type>::apply(mask, matrix);
- // }
- //};
- template <typename Head, typename Tail>
- struct check_dispatch< boost::tuples::cons<Head, Tail> >
- {
- typedef boost::tuples::cons<Head, Tail> mask_type;
- template <typename Matrix>
- static inline bool apply(mask_type const& mask, Matrix const& matrix)
- {
- return check_dispatch_tuple<mask_type>::apply(mask, matrix);
- }
- };
- template <typename Mask, typename Matrix>
- inline bool check_matrix(Mask const& mask, Matrix const& matrix)
- {
- return check_dispatch<Mask>::apply(mask, matrix);
- }
- // matrix_width
- template <typename MatrixOrMask>
- struct matrix_width
- {
- static const std::size_t value = MatrixOrMask::static_width;
- };
- template <typename Tuple,
- int I = 0,
- int N = boost::tuples::length<Tuple>::value>
- struct matrix_width_tuple
- {
- static const std::size_t
- current = matrix_width<typename boost::tuples::element<I, Tuple>::type>::value;
- static const std::size_t
- next = matrix_width_tuple<Tuple, I+1>::value;
- static const std::size_t
- value = current > next ? current : next;
- };
- template <typename Tuple, int N>
- struct matrix_width_tuple<Tuple, N, N>
- {
- static const std::size_t value = 0;
- };
- template <typename Head, typename Tail>
- struct matrix_width< boost::tuples::cons<Head, Tail> >
- {
- static const std::size_t
- value = matrix_width_tuple< boost::tuples::cons<Head, Tail> >::value;
- };
- // mask_handler
- template <typename Mask, bool Interrupt>
- class mask_handler
- : private matrix_handler
- <
- relate::matrix<matrix_width<Mask>::value>
- >
- {
- typedef matrix_handler
- <
- relate::matrix<matrix_width<Mask>::value>
- > base_t;
- public:
- typedef bool result_type;
- bool interrupt;
- inline explicit mask_handler(Mask const& m)
- : interrupt(false)
- , m_mask(m)
- {}
- result_type result() const
- {
- return !interrupt
- && check_matrix(m_mask, base_t::matrix());
- }
- template <field F1, field F2, char D>
- inline bool may_update() const
- {
- return detail::relate::may_update<F1, F2, D>(
- m_mask, base_t::matrix()
- );
- }
- template <field F1, field F2, char V>
- inline void set()
- {
- if ( relate::interrupt<F1, F2, V, Interrupt>(m_mask) )
- {
- interrupt = true;
- }
- else
- {
- base_t::template set<F1, F2, V>();
- }
- }
- template <field F1, field F2, char V>
- inline void update()
- {
- if ( relate::interrupt<F1, F2, V, Interrupt>(m_mask) )
- {
- interrupt = true;
- }
- else
- {
- base_t::template update<F1, F2, V>();
- }
- }
- private:
- Mask const& m_mask;
- };
- // --------------- COMPILE-TIME MASK ----------------
- // static_check_characters
- template
- <
- typename Seq,
- typename First = typename boost::mpl::begin<Seq>::type,
- typename Last = typename boost::mpl::end<Seq>::type
- >
- struct static_check_characters
- : static_check_characters
- <
- Seq,
- typename boost::mpl::next<First>::type
- >
- {
- typedef typename boost::mpl::deref<First>::type type;
- static const char value = type::value;
- static const bool is_valid = (value >= '0' && value <= '9')
- || value == 'T' || value == 'F' || value == '*';
- BOOST_MPL_ASSERT_MSG((is_valid),
- INVALID_STATIC_MASK_CHARACTER,
- (type));
- };
- template <typename Seq, typename Last>
- struct static_check_characters<Seq, Last, Last>
- {};
- // static_mask
- template
- <
- typename Seq,
- std::size_t Height,
- std::size_t Width = Height
- >
- struct static_mask
- {
- static const std::size_t static_width = Width;
- static const std::size_t static_height = Height;
- static const std::size_t static_size = Width * Height;
- BOOST_STATIC_ASSERT(
- std::size_t(boost::mpl::size<Seq>::type::value) == static_size);
-
- template <detail::relate::field F1, detail::relate::field F2>
- struct static_get
- {
- BOOST_STATIC_ASSERT(std::size_t(F1) < static_height);
- BOOST_STATIC_ASSERT(std::size_t(F2) < static_width);
- static const char value
- = boost::mpl::at_c<Seq, F1 * static_width + F2>::type::value;
- };
- private:
- // check static_mask characters
- enum { mask_check = sizeof(static_check_characters<Seq>) };
- };
- // static_should_handle_element
- template <typename StaticMask, field F1, field F2, bool IsSequence>
- struct static_should_handle_element_dispatch
- {
- static const char mask_el = StaticMask::template static_get<F1, F2>::value;
- static const bool value = mask_el == 'F'
- || mask_el == 'T'
- || ( mask_el >= '0' && mask_el <= '9' );
- };
- template <typename First, typename Last, field F1, field F2>
- struct static_should_handle_element_sequence
- {
- typedef typename boost::mpl::deref<First>::type StaticMask;
- static const bool value
- = static_should_handle_element_dispatch
- <
- StaticMask,
- F1, F2,
- boost::mpl::is_sequence<StaticMask>::value
- >::value
- || static_should_handle_element_sequence
- <
- typename boost::mpl::next<First>::type,
- Last,
- F1, F2
- >::value;
- };
- template <typename Last, field F1, field F2>
- struct static_should_handle_element_sequence<Last, Last, F1, F2>
- {
- static const bool value = false;
- };
- template <typename StaticMask, field F1, field F2>
- struct static_should_handle_element_dispatch<StaticMask, F1, F2, true>
- {
- static const bool value
- = static_should_handle_element_sequence
- <
- typename boost::mpl::begin<StaticMask>::type,
- typename boost::mpl::end<StaticMask>::type,
- F1, F2
- >::value;
- };
- template <typename StaticMask, field F1, field F2>
- struct static_should_handle_element
- {
- static const bool value
- = static_should_handle_element_dispatch
- <
- StaticMask,
- F1, F2,
- boost::mpl::is_sequence<StaticMask>::value
- >::value;
- };
- // static_interrupt
- template <typename StaticMask, char V, field F1, field F2, bool InterruptEnabled, bool IsSequence>
- struct static_interrupt_dispatch
- {
- static const bool value = false;
- };
- template <typename StaticMask, char V, field F1, field F2, bool IsSequence>
- struct static_interrupt_dispatch<StaticMask, V, F1, F2, true, IsSequence>
- {
- static const char mask_el = StaticMask::template static_get<F1, F2>::value;
- static const bool value
- = ( V >= '0' && V <= '9' ) ?
- ( mask_el == 'F' || ( mask_el < V && mask_el >= '0' && mask_el <= '9' ) ) :
- ( ( V == 'T' ) ? mask_el == 'F' : false );
- };
- template <typename First, typename Last, char V, field F1, field F2>
- struct static_interrupt_sequence
- {
- typedef typename boost::mpl::deref<First>::type StaticMask;
- static const bool value
- = static_interrupt_dispatch
- <
- StaticMask,
- V, F1, F2,
- true,
- boost::mpl::is_sequence<StaticMask>::value
- >::value
- && static_interrupt_sequence
- <
- typename boost::mpl::next<First>::type,
- Last,
- V, F1, F2
- >::value;
- };
- template <typename Last, char V, field F1, field F2>
- struct static_interrupt_sequence<Last, Last, V, F1, F2>
- {
- static const bool value = true;
- };
- template <typename StaticMask, char V, field F1, field F2>
- struct static_interrupt_dispatch<StaticMask, V, F1, F2, true, true>
- {
- static const bool value
- = static_interrupt_sequence
- <
- typename boost::mpl::begin<StaticMask>::type,
- typename boost::mpl::end<StaticMask>::type,
- V, F1, F2
- >::value;
- };
- template <typename StaticMask, char V, field F1, field F2, bool EnableInterrupt>
- struct static_interrupt
- {
- static const bool value
- = static_interrupt_dispatch
- <
- StaticMask,
- V, F1, F2,
- EnableInterrupt,
- boost::mpl::is_sequence<StaticMask>::value
- >::value;
- };
- // static_may_update
- template <typename StaticMask, char D, field F1, field F2, bool IsSequence>
- struct static_may_update_dispatch
- {
- static const char mask_el = StaticMask::template static_get<F1, F2>::value;
- static const int version
- = mask_el == 'F' ? 0
- : mask_el == 'T' ? 1
- : mask_el >= '0' && mask_el <= '9' ? 2
- : 3;
- template <typename Matrix>
- static inline bool apply(Matrix const& matrix)
- {
- return apply_dispatch(matrix, integral_constant<int, version>());
- }
- // mask_el == 'F'
- template <typename Matrix>
- static inline bool apply_dispatch(Matrix const& , integral_constant<int, 0>)
- {
- return true;
- }
- // mask_el == 'T'
- template <typename Matrix>
- static inline bool apply_dispatch(Matrix const& matrix, integral_constant<int, 1>)
- {
- char const c = matrix.template get<F1, F2>();
- return c == 'F'; // if it's T or between 0 and 9, the result will be the same
- }
- // mask_el >= '0' && mask_el <= '9'
- template <typename Matrix>
- static inline bool apply_dispatch(Matrix const& matrix, integral_constant<int, 2>)
- {
- char const c = matrix.template get<F1, F2>();
- return D > c || c > '9';
- }
- // else
- template <typename Matrix>
- static inline bool apply_dispatch(Matrix const&, integral_constant<int, 3>)
- {
- return false;
- }
- };
- template <typename First, typename Last, char D, field F1, field F2>
- struct static_may_update_sequence
- {
- typedef typename boost::mpl::deref<First>::type StaticMask;
- template <typename Matrix>
- static inline bool apply(Matrix const& matrix)
- {
- return static_may_update_dispatch
- <
- StaticMask,
- D, F1, F2,
- boost::mpl::is_sequence<StaticMask>::value
- >::apply(matrix)
- || static_may_update_sequence
- <
- typename boost::mpl::next<First>::type,
- Last,
- D, F1, F2
- >::apply(matrix);
- }
- };
- template <typename Last, char D, field F1, field F2>
- struct static_may_update_sequence<Last, Last, D, F1, F2>
- {
- template <typename Matrix>
- static inline bool apply(Matrix const& /*matrix*/)
- {
- return false;
- }
- };
- template <typename StaticMask, char D, field F1, field F2>
- struct static_may_update_dispatch<StaticMask, D, F1, F2, true>
- {
- template <typename Matrix>
- static inline bool apply(Matrix const& matrix)
- {
- return static_may_update_sequence
- <
- typename boost::mpl::begin<StaticMask>::type,
- typename boost::mpl::end<StaticMask>::type,
- D, F1, F2
- >::apply(matrix);
- }
- };
- template <typename StaticMask, char D, field F1, field F2>
- struct static_may_update
- {
- template <typename Matrix>
- static inline bool apply(Matrix const& matrix)
- {
- return static_may_update_dispatch
- <
- StaticMask,
- D, F1, F2,
- boost::mpl::is_sequence<StaticMask>::value
- >::apply(matrix);
- }
- };
- // static_check_matrix
- template <typename StaticMask, bool IsSequence>
- struct static_check_dispatch
- {
- template <typename Matrix>
- static inline bool apply(Matrix const& matrix)
- {
- return per_one<interior, interior>::apply(matrix)
- && per_one<interior, boundary>::apply(matrix)
- && per_one<interior, exterior>::apply(matrix)
- && per_one<boundary, interior>::apply(matrix)
- && per_one<boundary, boundary>::apply(matrix)
- && per_one<boundary, exterior>::apply(matrix)
- && per_one<exterior, interior>::apply(matrix)
- && per_one<exterior, boundary>::apply(matrix)
- && per_one<exterior, exterior>::apply(matrix);
- }
-
- template <field F1, field F2>
- struct per_one
- {
- static const char mask_el = StaticMask::template static_get<F1, F2>::value;
- static const int version
- = mask_el == 'F' ? 0
- : mask_el == 'T' ? 1
- : mask_el >= '0' && mask_el <= '9' ? 2
- : 3;
- template <typename Matrix>
- static inline bool apply(Matrix const& matrix)
- {
- const char el = matrix.template get<F1, F2>();
- return apply_dispatch(el, integral_constant<int, version>());
- }
- // mask_el == 'F'
- static inline bool apply_dispatch(char el, integral_constant<int, 0>)
- {
- return el == 'F';
- }
- // mask_el == 'T'
- static inline bool apply_dispatch(char el, integral_constant<int, 1>)
- {
- return el == 'T' || ( el >= '0' && el <= '9' );
- }
- // mask_el >= '0' && mask_el <= '9'
- static inline bool apply_dispatch(char el, integral_constant<int, 2>)
- {
- return el == mask_el;
- }
- // else
- static inline bool apply_dispatch(char /*el*/, integral_constant<int, 3>)
- {
- return true;
- }
- };
- };
- template <typename First, typename Last>
- struct static_check_sequence
- {
- typedef typename boost::mpl::deref<First>::type StaticMask;
- template <typename Matrix>
- static inline bool apply(Matrix const& matrix)
- {
- return static_check_dispatch
- <
- StaticMask,
- boost::mpl::is_sequence<StaticMask>::value
- >::apply(matrix)
- || static_check_sequence
- <
- typename boost::mpl::next<First>::type,
- Last
- >::apply(matrix);
- }
- };
- template <typename Last>
- struct static_check_sequence<Last, Last>
- {
- template <typename Matrix>
- static inline bool apply(Matrix const& /*matrix*/)
- {
- return false;
- }
- };
- template <typename StaticMask>
- struct static_check_dispatch<StaticMask, true>
- {
- template <typename Matrix>
- static inline bool apply(Matrix const& matrix)
- {
- return static_check_sequence
- <
- typename boost::mpl::begin<StaticMask>::type,
- typename boost::mpl::end<StaticMask>::type
- >::apply(matrix);
- }
- };
- template <typename StaticMask>
- struct static_check_matrix
- {
- template <typename Matrix>
- static inline bool apply(Matrix const& matrix)
- {
- return static_check_dispatch
- <
- StaticMask,
- boost::mpl::is_sequence<StaticMask>::value
- >::apply(matrix);
- }
- };
- // static_mask_handler
- template <typename StaticMask, bool Interrupt>
- class static_mask_handler
- : private matrix_handler< matrix<3> >
- {
- typedef matrix_handler< relate::matrix<3> > base_type;
- public:
- typedef bool result_type;
- bool interrupt;
- inline static_mask_handler()
- : interrupt(false)
- {}
- inline explicit static_mask_handler(StaticMask const& /*dummy*/)
- : interrupt(false)
- {}
- result_type result() const
- {
- return (!Interrupt || !interrupt)
- && static_check_matrix<StaticMask>::apply(base_type::matrix());
- }
- template <field F1, field F2, char D>
- inline bool may_update() const
- {
- return static_may_update<StaticMask, D, F1, F2>::
- apply(base_type::matrix());
- }
- template <field F1, field F2>
- static inline bool expects()
- {
- return static_should_handle_element<StaticMask, F1, F2>::value;
- }
- template <field F1, field F2, char V>
- inline void set()
- {
- static const bool interrupt_c = static_interrupt<StaticMask, V, F1, F2, Interrupt>::value;
- static const bool should_handle = static_should_handle_element<StaticMask, F1, F2>::value;
- static const int version = interrupt_c ? 0
- : should_handle ? 1
- : 2;
- set_dispatch<F1, F2, V>(integral_constant<int, version>());
- }
- template <field F1, field F2, char V>
- inline void update()
- {
- static const bool interrupt_c = static_interrupt<StaticMask, V, F1, F2, Interrupt>::value;
- static const bool should_handle = static_should_handle_element<StaticMask, F1, F2>::value;
- static const int version = interrupt_c ? 0
- : should_handle ? 1
- : 2;
- update_dispatch<F1, F2, V>(integral_constant<int, version>());
- }
- private:
- // Interrupt && interrupt
- template <field F1, field F2, char V>
- inline void set_dispatch(integral_constant<int, 0>)
- {
- interrupt = true;
- }
- // else should_handle
- template <field F1, field F2, char V>
- inline void set_dispatch(integral_constant<int, 1>)
- {
- base_type::template set<F1, F2, V>();
- }
- // else
- template <field F1, field F2, char V>
- inline void set_dispatch(integral_constant<int, 2>)
- {}
- // Interrupt && interrupt
- template <field F1, field F2, char V>
- inline void update_dispatch(integral_constant<int, 0>)
- {
- interrupt = true;
- }
- // else should_handle
- template <field F1, field F2, char V>
- inline void update_dispatch(integral_constant<int, 1>)
- {
- base_type::template update<F1, F2, V>();
- }
- // else
- template <field F1, field F2, char V>
- inline void update_dispatch(integral_constant<int, 2>)
- {}
- };
- // --------------- UTIL FUNCTIONS ----------------
- // set
- template <field F1, field F2, char V, typename Result>
- inline void set(Result & res)
- {
- res.template set<F1, F2, V>();
- }
- template <field F1, field F2, char V, bool Transpose>
- struct set_dispatch
- {
- template <typename Result>
- static inline void apply(Result & res)
- {
- res.template set<F1, F2, V>();
- }
- };
- template <field F1, field F2, char V>
- struct set_dispatch<F1, F2, V, true>
- {
- template <typename Result>
- static inline void apply(Result & res)
- {
- res.template set<F2, F1, V>();
- }
- };
- template <field F1, field F2, char V, bool Transpose, typename Result>
- inline void set(Result & res)
- {
- set_dispatch<F1, F2, V, Transpose>::apply(res);
- }
- // update
- template <field F1, field F2, char D, typename Result>
- inline void update(Result & res)
- {
- res.template update<F1, F2, D>();
- }
- template <field F1, field F2, char D, bool Transpose>
- struct update_result_dispatch
- {
- template <typename Result>
- static inline void apply(Result & res)
- {
- update<F1, F2, D>(res);
- }
- };
- template <field F1, field F2, char D>
- struct update_result_dispatch<F1, F2, D, true>
- {
- template <typename Result>
- static inline void apply(Result & res)
- {
- update<F2, F1, D>(res);
- }
- };
- template <field F1, field F2, char D, bool Transpose, typename Result>
- inline void update(Result & res)
- {
- update_result_dispatch<F1, F2, D, Transpose>::apply(res);
- }
- // may_update
- template <field F1, field F2, char D, typename Result>
- inline bool may_update(Result const& res)
- {
- return res.template may_update<F1, F2, D>();
- }
- template <field F1, field F2, char D, bool Transpose>
- struct may_update_result_dispatch
- {
- template <typename Result>
- static inline bool apply(Result const& res)
- {
- return may_update<F1, F2, D>(res);
- }
- };
- template <field F1, field F2, char D>
- struct may_update_result_dispatch<F1, F2, D, true>
- {
- template <typename Result>
- static inline bool apply(Result const& res)
- {
- return may_update<F2, F1, D>(res);
- }
- };
- template <field F1, field F2, char D, bool Transpose, typename Result>
- inline bool may_update(Result const& res)
- {
- return may_update_result_dispatch<F1, F2, D, Transpose>::apply(res);
- }
- // result_dimension
- template <typename Geometry>
- struct result_dimension
- {
- BOOST_STATIC_ASSERT(geometry::dimension<Geometry>::value >= 0);
- static const char value
- = ( geometry::dimension<Geometry>::value <= 9 ) ?
- ( '0' + geometry::dimension<Geometry>::value ) :
- 'T';
- };
- }} // namespace detail::relate
- #endif // DOXYGEN_NO_DETAIL
- }} // namespace boost::geometry
- #endif // BOOST_GEOMETRY_ALGORITHMS_DETAIL_RELATE_RESULT_HPP
|