CNL  2.0.2 (development)
Compositional Numeric Library
policy.h
1 
2 // Copyright John McFarlane 2019.
3 // Distributed under the Boost Software License, Version 1.0.
4 // (See accompanying file ../LICENSE_1_0.txt or copy at
5 // http://www.boost.org/LICENSE_1_0.txt)
6 
7 #if !defined(CNL_IMPL_ELASTIC_TAG_POLICY_H)
8 #define CNL_IMPL_ELASTIC_TAG_POLICY_H
9 
10 #include "../custom_operator/op.h"
11 
12 #include <algorithm>
13 
15 namespace cnl {
16  namespace _impl {
17  template<binary_arithmetic_op Operation, int LhsDigits, bool LhsIsSigned, int RhsDigits, bool RhsIsSigned>
18  struct policy;
19 
20  template<int LhsDigits, bool LhsIsSigned, int RhsDigits, bool RhsIsSigned>
21  struct policy<_impl::add_op, LhsDigits, LhsIsSigned, RhsDigits, RhsIsSigned> {
22  static constexpr int digits = std::max(LhsDigits, RhsDigits) + 1;
23  static constexpr bool is_signed = LhsIsSigned || RhsIsSigned;
24  };
25 
26  template<int LhsDigits, bool LhsIsSigned, int RhsDigits, bool RhsIsSigned>
27  struct policy<_impl::subtract_op, LhsDigits, LhsIsSigned, RhsDigits, RhsIsSigned> {
28  static constexpr int digits =
29  std::max(LhsDigits, RhsDigits) + (LhsIsSigned | RhsIsSigned);
30  static constexpr bool is_signed = true;
31  };
32 
33  template<int LhsDigits, bool LhsIsSigned, int RhsDigits, bool RhsIsSigned>
34  struct policy<_impl::multiply_op, LhsDigits, LhsIsSigned, RhsDigits, RhsIsSigned> {
35  static constexpr auto contribution(int operand_digits)
36  {
37  return operand_digits == 1 ? 0 : operand_digits;
38  }
39  static constexpr int digits = std::max(1, contribution(LhsDigits) + contribution(RhsDigits));
40  static constexpr bool is_signed = LhsIsSigned || RhsIsSigned;
41  };
42 
43  template<int LhsDigits, bool LhsIsSigned, int RhsDigits, bool RhsIsSigned>
44  struct policy<_impl::divide_op, LhsDigits, LhsIsSigned, RhsDigits, RhsIsSigned> {
45  static constexpr int digits = LhsDigits;
46  static constexpr bool is_signed = LhsIsSigned || RhsIsSigned;
47  };
48 
49  template<int LhsDigits, bool LhsIsSigned, int RhsDigits, bool RhsIsSigned>
50  struct policy<_impl::modulo_op, LhsDigits, LhsIsSigned, RhsDigits, RhsIsSigned> {
51  static constexpr int digits = std::min(LhsDigits, RhsDigits);
52  static constexpr bool is_signed = LhsIsSigned || RhsIsSigned;
53  };
54 
55  template<int LhsDigits, bool LhsIsSigned, int RhsDigits, bool RhsIsSigned>
56  struct policy<_impl::bitwise_or_op, LhsDigits, LhsIsSigned, RhsDigits, RhsIsSigned> {
57  static constexpr int digits = std::max(LhsDigits, RhsDigits);
58  static constexpr bool is_signed = LhsIsSigned || RhsIsSigned;
59  };
60 
61  template<int LhsDigits, bool LhsIsSigned, int RhsDigits, bool RhsIsSigned>
62  struct policy<_impl::bitwise_and_op, LhsDigits, LhsIsSigned, RhsDigits, RhsIsSigned> {
63  static constexpr int digits = std::min(LhsDigits, RhsDigits);
64  static constexpr bool is_signed = LhsIsSigned || RhsIsSigned;
65  };
66 
67  template<int LhsDigits, bool LhsIsSigned, int RhsDigits, bool RhsIsSigned>
68  struct policy<_impl::bitwise_xor_op, LhsDigits, LhsIsSigned, RhsDigits, RhsIsSigned> {
69  static constexpr int digits = std::max(LhsDigits, RhsDigits);
70  static constexpr bool is_signed = LhsIsSigned || RhsIsSigned;
71  };
72  }
73 }
74 
75 #endif // CNL_IMPL_ELASTIC_TAG_POLICY_H
cnl
compositional numeric library
Definition: abort.h:15
std::min
T min(T... args)
std::max
T max(T... args)