// Copyright (c) 2001 Utrecht University (The Netherlands), // ETH Zurich (Switzerland), Freie Universitaet Berlin (Germany), // INRIA Sophia-Antipolis (France), Martin-Luther-University Halle-Wittenberg // (Germany), Max-Planck-Institute Saarbruecken (Germany), RISC Linz (Austria), // and Tel-Aviv University (Israel). All rights reserved. // // This file is part of CGAL (www.cgal.org); you can redistribute it and/or // modify it under the terms of the GNU Lesser General Public License as // published by the Free Software Foundation; version 2.1 of the License. // See the file LICENSE.LGPL distributed with CGAL. // // Licensees holding a valid commercial license may use this file in // accordance with the commercial license agreement provided with the software. // // This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE // WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. // // $Source: /CVSROOT/CGAL/Packages/Interval_arithmetic/include/CGAL/Filtered_predicate.h,v $ // $Revision: 1.9 $ $Date: 2003/10/21 12:17:23 $ // $Name: $ // // Author(s) : Sylvain Pion #ifndef CGAL_FILTER_PREDICATE_H #define CGAL_FILTER_PREDICATE_H #include #include CGAL_BEGIN_NAMESPACE // This template class is a wrapper that implements the filtering for any // predicate (dynamic filters with IA). // TODO : // - We should specialize it for the static filters versions when possible... // - each predicate in the default kernel should define a tag that says if it // wants to be filtered or not (=> all homogeneous predicate define this // tag). We could even test-suite that automatically. It makes a strong // new requirement on the kernel though... // Could be done with a traits mecanism ? // A default template could use the current IA, but other tags or whatever // could specify no filtering at all, or static filtering... // - same thing for constructions => virtual operator() ? // - similarly, constructions should have a tag saying if they can throw or // not, or we let all this up to the compiler optimizer to figure out ? // - Some caching should be done at the Point_2 level. template class Filtered_predicate { EP Exact_predicate; AP Approx_predicate; C2E To_Exact; C2F To_Approx; public: typedef typename AP::result_type result_type; typedef typename AP::Arity Arity; // Should be the same type as EP::result_type. Filtered_predicate() {} // These constructors are used for constructive predicates. // You should try to avoid constructive predicates, as they will construct // the exact values systematically (in the ctor), rather than lazily. template Filtered_predicate(const O &o1) : Exact_predicate(To_Exact(o1)), Approx_predicate(To_Approx(o1)) {} template Filtered_predicate(const O &o1, const O &o2) : Exact_predicate(To_Exact(o1), To_Exact(o2)), Approx_predicate(To_Approx(o1), To_Approx(o2)) {} template result_type operator()(const A1 &a1) const #ifndef CGAL_CFG_OUTOFLINE_TEMPLATE_MEMBER_DEFINITION_BUG ; #else { try { Protect_FPU_rounding P; return Approx_predicate(To_Approx(a1)); } catch (Interval_nt_advanced::unsafe_comparison) { Protect_FPU_rounding P(CGAL_FE_TONEAREST); return Exact_predicate(To_Exact(a1)); } } #endif template result_type operator()(const A1 &a1, const A2 &a2) const #ifndef CGAL_CFG_OUTOFLINE_TEMPLATE_MEMBER_DEFINITION_BUG ; #else { try { Protect_FPU_rounding P; return Approx_predicate(To_Approx(a1), To_Approx(a2)); } catch (Interval_nt_advanced::unsafe_comparison) { Protect_FPU_rounding P(CGAL_FE_TONEAREST); return Exact_predicate(To_Exact(a1), To_Exact(a2)); } } #endif template result_type operator()(const A1 &a1, const A2 &a2, const A3 &a3) const #ifndef CGAL_CFG_OUTOFLINE_TEMPLATE_MEMBER_DEFINITION_BUG ; #else { try { Protect_FPU_rounding P; return Approx_predicate(To_Approx(a1), To_Approx(a2), To_Approx(a3)); } catch (Interval_nt_advanced::unsafe_comparison) { Protect_FPU_rounding P(CGAL_FE_TONEAREST); return Exact_predicate(To_Exact(a1), To_Exact(a2), To_Exact(a3)); } } #endif template result_type operator()(const A1 &a1, const A2 &a2, const A3 &a3, const A4 &a4) const #ifndef CGAL_CFG_OUTOFLINE_TEMPLATE_MEMBER_DEFINITION_BUG ; #else { try { Protect_FPU_rounding P; return Approx_predicate(To_Approx(a1), To_Approx(a2), To_Approx(a3), To_Approx(a4)); } catch (Interval_nt_advanced::unsafe_comparison) { Protect_FPU_rounding P(CGAL_FE_TONEAREST); return Exact_predicate(To_Exact(a1), To_Exact(a2), To_Exact(a3), To_Exact(a4)); } } #endif template result_type operator()(const A1 &a1, const A2 &a2, const A3 &a3, const A4 &a4, const A5 &a5) const #ifndef CGAL_CFG_OUTOFLINE_TEMPLATE_MEMBER_DEFINITION_BUG ; #else { try { Protect_FPU_rounding P; return Approx_predicate(To_Approx(a1), To_Approx(a2), To_Approx(a3), To_Approx(a4), To_Approx(a5)); } catch (Interval_nt_advanced::unsafe_comparison) { Protect_FPU_rounding P(CGAL_FE_TONEAREST); return Exact_predicate(To_Exact(a1), To_Exact(a2), To_Exact(a3), To_Exact(a4), To_Exact(a5)); } } #endif template result_type operator()(const A1 &a1, const A2 &a2, const A3 &a3, const A4 &a4, const A5 &a5, const A6 &a6) const #ifndef CGAL_CFG_OUTOFLINE_TEMPLATE_MEMBER_DEFINITION_BUG ; #else { try { Protect_FPU_rounding P; return Approx_predicate(To_Approx(a1), To_Approx(a2), To_Approx(a3), To_Approx(a4), To_Approx(a5), To_Approx(a6)); } catch (Interval_nt_advanced::unsafe_comparison) { Protect_FPU_rounding P(CGAL_FE_TONEAREST); return Exact_predicate(To_Exact(a1), To_Exact(a2), To_Exact(a3), To_Exact(a4), To_Exact(a5), To_Exact(a6)); } } #endif template result_type operator()(const A1 &a1, const A2 &a2, const A3 &a3, const A4 &a4, const A5 &a5, const A6 &a6, const A7 &a7) const #ifndef CGAL_CFG_OUTOFLINE_TEMPLATE_MEMBER_DEFINITION_BUG ; #else { try { Protect_FPU_rounding P; return Approx_predicate(To_Approx(a1), To_Approx(a2), To_Approx(a3), To_Approx(a4), To_Approx(a5), To_Approx(a6), To_Approx(a7)); } catch (Interval_nt_advanced::unsafe_comparison) { Protect_FPU_rounding P(CGAL_FE_TONEAREST); return Exact_predicate(To_Exact(a1), To_Exact(a2), To_Exact(a3), To_Exact(a4), To_Exact(a5), To_Exact(a6), To_Exact(a7)); } } #endif // Idem for more than 7 arguments. Do it on demand. }; #ifndef CGAL_CFG_OUTOFLINE_TEMPLATE_MEMBER_DEFINITION_BUG template template typename Filtered_predicate::result_type Filtered_predicate:: operator()(const A1 &a1) const { try { Protect_FPU_rounding P; return Approx_predicate(To_Approx(a1)); } catch (Interval_nt_advanced::unsafe_comparison) { Protect_FPU_rounding P(CGAL_FE_TONEAREST); return Exact_predicate(To_Exact(a1)); } } template template typename Filtered_predicate::result_type Filtered_predicate:: operator()(const A1 &a1, const A2 &a2) const { try { Protect_FPU_rounding P; return Approx_predicate(To_Approx(a1), To_Approx(a2)); } catch (Interval_nt_advanced::unsafe_comparison) { Protect_FPU_rounding P(CGAL_FE_TONEAREST); return Exact_predicate(To_Exact(a1), To_Exact(a2)); } } template template typename Filtered_predicate::result_type Filtered_predicate:: operator()(const A1 &a1, const A2 &a2, const A3 &a3) const { try { Protect_FPU_rounding P; return Approx_predicate(To_Approx(a1), To_Approx(a2), To_Approx(a3)); } catch (Interval_nt_advanced::unsafe_comparison) { Protect_FPU_rounding P(CGAL_FE_TONEAREST); return Exact_predicate(To_Exact(a1), To_Exact(a2), To_Exact(a3)); } } template template typename Filtered_predicate::result_type Filtered_predicate:: operator()(const A1 &a1, const A2 &a2, const A3 &a3, const A4 &a4) const { try { Protect_FPU_rounding P; return Approx_predicate(To_Approx(a1), To_Approx(a2), To_Approx(a3), To_Approx(a4)); } catch (Interval_nt_advanced::unsafe_comparison) { Protect_FPU_rounding P(CGAL_FE_TONEAREST); return Exact_predicate(To_Exact(a1), To_Exact(a2), To_Exact(a3), To_Exact(a4)); } } template template typename Filtered_predicate::result_type Filtered_predicate:: operator()(const A1 &a1, const A2 &a2, const A3 &a3, const A4 &a4, const A5 &a5) const { try { Protect_FPU_rounding P; return Approx_predicate(To_Approx(a1), To_Approx(a2), To_Approx(a3), To_Approx(a4), To_Approx(a5)); } catch (Interval_nt_advanced::unsafe_comparison) { Protect_FPU_rounding P(CGAL_FE_TONEAREST); return Exact_predicate(To_Exact(a1), To_Exact(a2), To_Exact(a3), To_Exact(a4), To_Exact(a5)); } } template template typename Filtered_predicate::result_type Filtered_predicate:: operator()(const A1 &a1, const A2 &a2, const A3 &a3, const A4 &a4, const A5 &a5, const A6 &a6) const { try { Protect_FPU_rounding P; return Approx_predicate(To_Approx(a1), To_Approx(a2), To_Approx(a3), To_Approx(a4), To_Approx(a5), To_Approx(a6)); } catch (Interval_nt_advanced::unsafe_comparison) { Protect_FPU_rounding P(CGAL_FE_TONEAREST); return Exact_predicate(To_Exact(a1), To_Exact(a2), To_Exact(a3), To_Exact(a4), To_Exact(a5), To_Exact(a6)); } } #endif CGAL_END_NAMESPACE #endif // CGAL_FILTER_PREDICATE_H