26 namespace json_details {
29 inline constexpr unsigned json_option_bits_width = 0;
32 inline constexpr auto default_json_option_value = [] {
33 struct unknown_policy {};
34 return unknown_policy{ };
37 template<
typename Policy,
typename Options>
38 struct option_bits_start_impl;
40 template<
typename Policy,
typename... Options>
41 struct option_bits_start_impl<Policy, pack_list<Options...>> {
42 static constexpr auto idx =
43 daw::traits::pack_index_of_v<Policy, Options...>;
44 static_assert( idx >= 0,
"Policy is not registered" );
45 using tp_policies = pack_list<Options...>;
47 template<std::
size_t Pos,
int End>
48 static constexpr unsigned do_step( ) {
49 if constexpr( Pos >=
static_cast<std::size_t
>( End ) ) {
52 return json_option_bits_width<pack_element_t<Pos, tp_policies>>;
55 template<std::size_t... Is>
56 DAW_ATTRIB_INLINE
static constexpr unsigned
57 calc( std::index_sequence<Is...> ) {
58 return ( do_step<Is, idx>( ) + ... );
62 template<
typename... Options>
63 struct option_list_impl {
64 using type = pack_list<Options...>;
67 ( json_option_bits_width<Options> + ... + 0 ) <=
68 daw::bit_count_v<json_options_t>,
69 "The size of json_options_t is not large enough "
70 "to safely hold all the bits of state. Use a larger size." );
73 template<
typename Option>
74 inline constexpr bool is_option_flag = json_option_bits_width<Option> > 0;
76 template<
typename... Options>
77 inline constexpr bool are_option_flags =
78 ( is_option_flag<Options> and ... );
80 template<
typename Option,
typename Options>
81 inline constexpr unsigned basic_option_bits_start =
82 option_bits_start_impl<Option, Options>::template calc<>(
83 std::make_index_sequence<pack_size_v<Options>>{ } );
85 template<
typename... JsonOptions>
86 struct JsonOptionList;
88 template<
typename... OptionList,
typename Option>
90 set_bits_for( JsonOptionList<OptionList...>, Option e );
92 template<
typename... JsonOptions>
93 struct JsonOptionList {
94 using OptionList =
typename option_list_impl<JsonOptions...>::type;
96 template<
typename Option>
97 static inline constexpr unsigned option_bits_start =
98 basic_option_bits_start<Option, OptionList>;
100 template<
typename... ,
typename Option>
102 static_assert( is_option_flag<Option>,
103 "Only registered policy types are allowed" );
105 new_bits <<= option_bits_start<Option>;
110 ( set_bits_for<JsonOptions>(
111 default_json_option_value<JsonOptions> ) |
120 template<
typename... Options>
122 static_assert( json_details::are_option_flags<Options...>,
123 "Only registered option types are allowed" );
124 auto result = default_option_flag;
125 if constexpr(
sizeof...( Options ) > 0 ) {
126 result |= ( set_bits_for( options ) | ... );
132 template<
typename... OptionList,
typename Option>
133 constexpr void set_bits_in( JsonOptionList<OptionList...>,
135 static_assert( is_option_flag<Option>,
136 "Only registered policy types are allowed" );
137 auto new_bits =
static_cast<unsigned>( e );
138 constexpr unsigned mask = (1U << json_option_bits_width<Option>)-1U;
141 JsonOptionList<OptionList...>::template option_bits_start<Option>;
146 template<
typename... OptionList,
typename Option,
typename... Options>
150 static_assert( are_option_flags<Options...>,
151 "Only registered policy types are allowed" );
153 auto new_bits =
static_cast<unsigned>( pol );
154 constexpr unsigned mask = ( (1U << json_option_bits_width<Option>)-1U );
157 JsonOptionList<OptionList...>::template option_bits_start<Option>;
158 value &= ~( mask << JsonOptionList<
159 OptionList...>::template option_bits_start<Option> );
161 if constexpr(
sizeof...( Options ) > 0 ) {
162 return set_bits( JsonOptionList<OptionList...>{ }, value, pols... );
168 template<
typename... OptionList,
typename Option>
170 set_bits_for( JsonOptionList<OptionList...>, Option e ) {
171 return JsonOptionList<OptionList...>::set_bits_for( e );
174 template<
typename Option,
typename Result = Option,
175 typename... OptionList>
176 constexpr Result get_bits_for( JsonOptionList<OptionList...>,
178 static_assert( std::is_enum_v<Option>,
179 "Only enum options are allowed" );
180 static_assert( std::is_same_v<unsigned, std::underlying_type_t<Option>>,
181 "Looks like option was no specified correctly. "
182 "Underlying type should be unsigned" );
183 static_assert( is_option_flag<Option>,
184 "Only registered option types are allowed" );
185 constexpr unsigned mask =
186 ( 1U << (JsonOptionList<OptionList...>::template option_bits_start<
188 json_option_bits_width<Option>)) -
192 JsonOptionList<OptionList...>::template option_bits_start<Option>;
193 return static_cast<Result
>( Option{ value } );
196 template<
typename... OptionList,
typename... Options>
197 constexpr json_options_t json_option( JsonOptionList<OptionList...>,
198 Options... options ) {
199 return JsonOptionList<OptionList...>::options( options... );
203 struct default_option_flag_t;
205 template<
template<
class...>
class OptionList,
typename... Options>
206 struct default_option_flag_t<OptionList<Options...>> {
208 ( OptionList<Options...>::template set_bits_for<Options>(
209 default_json_option_value<Options> ) |
216 template<
typename OptionList>
218 default_option_flag_t<OptionList>::value;