convert_impl.hpp
Go to the documentation of this file.
1 /* Copyright (C) 2022 Davide Faconti, Eurecat - All Rights Reserved
2 *
3 * Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"),
4 * to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense,
5 * and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
6 * The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
7 *
8 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
9 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
10 * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
11 */
12 
13 #pragma once
14 
15 #include <type_traits>
16 #include <cmath>
17 #include "simple_string.hpp"
18 
19 #undef max
20 #undef min
21 
22 namespace SafeAny
23 {
24 
25 namespace details
26 {
27 
28 template <typename BoolCondition>
29 using EnableIf = typename std::enable_if<BoolCondition::value, void>::type;
30 
31 template <typename T>
32 constexpr bool is_integer()
33 {
36 }
37 
38 template <typename T>
39 constexpr bool is_convertible_type()
40 {
41  return is_integer<T>() || std::is_floating_point<T>::value ||
44 }
45 
46 template <typename T>
47 constexpr bool is_convertible_to_bool()
48 {
49  return is_integer<T>() || std::is_floating_point<T>::value || std::is_enum<T>::value;
50 }
51 
52 template <typename T>
53 constexpr bool is_signed()
54 {
56 }
57 
58 template <typename T1, typename T2>
59 constexpr bool is_same()
60 {
62 }
63 
64 template <typename From, typename To>
65 inline void checkUpperLimit(const From& from)
66 {
67  if(from > static_cast<From>(std::numeric_limits<To>::max()))
68  {
69  throw std::runtime_error("Value outside the max numerical limit.");
70  }
71 }
72 
73 template <typename From, typename To>
74 inline void checkLowerLimit(const From& from)
75 {
76  if constexpr(std::is_same<To, bool>::value)
77  {
78  if(from != 0 && from != 1)
79  {
80  throw std::runtime_error("Implicit casting to bool is not allowed");
81  }
82  }
83  else if(from < std::numeric_limits<To>::min())
84  {
85  throw std::runtime_error("Value outside the lovest numerical limit.");
86  }
87 }
88 
89 template <typename From, typename To>
90 inline void checkTruncation(const From& from)
91 {
92  // Handle integer to floating point
93  if constexpr(std::is_integral_v<From> && std::is_floating_point_v<To>)
94  {
95  // Check if value can be represented exactly in the target type
96  constexpr uint64_t max_exact = (1LL << std::numeric_limits<double>::digits) - 1;
97  bool doesnt_fit = false;
98  if constexpr(!std::is_signed_v<From>)
99  {
100  doesnt_fit = static_cast<uint64_t>(from) > max_exact;
101  }
102  else
103  {
104  doesnt_fit = std::abs(static_cast<int64_t>(from)) > static_cast<int64_t>(max_exact);
105  }
106  if(doesnt_fit)
107  {
108  throw std::runtime_error("Loss of precision when converting a large integer number "
109  "to floating point:" +
110  std::to_string(from));
111  }
112  }
113  // Handle floating point to integer
114  else if constexpr(std::is_floating_point_v<From> && std::is_integral_v<To>)
115  {
116  if(from > static_cast<From>(std::numeric_limits<To>::max()) ||
117  from < static_cast<From>(std::numeric_limits<To>::lowest()) ||
118  from != std::nearbyint(from))
119  {
120  throw std::runtime_error("Invalid floating point to integer conversion");
121  }
122  }
123  // Handle other conversions
124  else
125  {
126  if(from > static_cast<From>(std::numeric_limits<To>::max()) ||
127  from < static_cast<From>(std::numeric_limits<To>::lowest()))
128  {
129  throw std::runtime_error("Value outside numeric limits");
130  }
131  To as_target = static_cast<To>(from);
132  From back_to_source = static_cast<From>(as_target);
133  if(from != back_to_source)
134  {
135  throw std::runtime_error("Value truncated in conversion");
136  }
137  }
138 }
139 
140 //----------------------- Implementation ----------------------------------------------
141 
142 template <typename SRC, typename DST>
143 void convertNumber(const SRC& source, DST& target)
144 {
145  static_assert(is_convertible_type<SRC>() && is_convertible_type<DST>(), "Not "
146  "convertible");
147 
148  constexpr bool both_integers = is_integer<SRC>() && is_integer<DST>();
149 
150  if constexpr(is_signed<SRC>() && !is_signed<DST>())
151  {
152  if(source < 0)
153  {
154  throw std::runtime_error("Value is negative and can't be converted to unsigned");
155  }
156  }
157  // these conversions are always safe:
158  // - same type
159  // - float -> double
160  if constexpr(is_same<SRC, DST>() || (is_same<SRC, float>() && is_same<DST, double>()))
161  {
162  // No check needed
163  target = static_cast<DST>(source);
164  }
165  else if constexpr(both_integers)
166  {
167  if constexpr(sizeof(SRC) == sizeof(DST) && !is_signed<SRC>() && is_signed<DST>())
168  {
169  checkUpperLimit<SRC, DST>(source);
170  }
171  // casting to a smaller number need to be check
172  else if constexpr(sizeof(SRC) > sizeof(DST))
173  {
174  if constexpr(is_signed<SRC>())
175  {
176  checkLowerLimit<SRC, DST>(source);
177  }
178  checkUpperLimit<SRC, DST>(source);
179  }
180  target = static_cast<DST>(source);
181  }
182  // special case: bool accept truncation
183  else if constexpr(is_convertible_to_bool<SRC>() && is_same<DST, bool>())
184  {
185  target = static_cast<DST>(source);
186  }
187  // casting to/from floating points might cause truncation.
188  else if constexpr(std::is_floating_point<SRC>::value ||
190  {
191  bool both_float =
193  // to avoid being too pedantic, let's accept casting between double and float
194  if(!both_float)
195  {
196  checkTruncation<SRC, DST>(source);
197  }
198  target = static_cast<DST>(source);
199  }
200 }
201 
202 } //end namespace details
203 } //end namespace SafeAny
SafeAny
Definition: convert_impl.hpp:22
SafeAny::details::checkUpperLimit
void checkUpperLimit(const From &from)
Definition: convert_impl.hpp:65
SafeAny::details::checkLowerLimit
void checkLowerLimit(const From &from)
Definition: convert_impl.hpp:74
SafeAny::details::convertNumber
void convertNumber(const SRC &source, DST &target)
Definition: convert_impl.hpp:143
SafeAny::details::is_same
constexpr bool is_same()
Definition: convert_impl.hpp:59
magic_enum::detail::value
constexpr E value(std::size_t i) noexcept
Definition: magic_enum.hpp:664
SafeAny::details::checkTruncation
void checkTruncation(const From &from)
Definition: convert_impl.hpp:90
simple_string.hpp
to_string
NLOHMANN_BASIC_JSON_TPL_DECLARATION std::string to_string(const NLOHMANN_BASIC_JSON_TPL &j)
user-defined to_string function for JSON values
Definition: json.hpp:24456
lexyd::digits
constexpr auto digits
Matches a non-empty list of digits.
Definition: digit.hpp:539
SafeAny::details::is_convertible_to_bool
constexpr bool is_convertible_to_bool()
Definition: convert_impl.hpp:47
SafeAny::details::is_convertible_type
constexpr bool is_convertible_type()
Definition: convert_impl.hpp:39
SafeAny::details::EnableIf
typename std::enable_if< BoolCondition::value, void >::type EnableIf
Definition: convert_impl.hpp:29
SafeAny::details::is_integer
constexpr bool is_integer()
Definition: convert_impl.hpp:32
SafeAny::details::is_signed
constexpr bool is_signed()
Definition: convert_impl.hpp:53


behaviortree_cpp_v4
Author(s): Davide Faconti
autogenerated on Fri Feb 28 2025 03:19:18