sprout/weed/parser/numeric/int.hpp 追加

This commit is contained in:
bolero-MURAKAMI 2011-11-14 15:34:28 +09:00
parent ad94c0b247
commit 005f8a9a87
5 changed files with 329 additions and 23 deletions

View file

@ -17,19 +17,24 @@ namespace sprout {
// uint_p
//
template<
typename IntType,
std::size_t Radix,
typename UIntType,
std::size_t Radix = 10,
std::size_t MinDigits = 1,
std::size_t MaxDigits = sprout::integer_digits<IntType, Radix>::value
std::size_t MaxDigits = sprout::integer_digits<UIntType, Radix>::value
>
struct uint_p
: public sprout::weed::parser_base
{
private:
SPROUT_STATIC_CONSTEXPR std::size_t limit_digits = MaxDigits <= sprout::integer_digits<UIntType, Radix>::value
? MaxDigits
: sprout::integer_digits<UIntType, Radix>::value
;
public:
template<typename Context, typename Iterator>
struct attribute {
public:
typedef IntType type;
typedef UIntType type;
};
template<typename Context, typename Iterator>
struct result {
@ -37,26 +42,48 @@ namespace sprout {
typedef sprout::weed::parser_result<Iterator, typename attribute<Context, Iterator>::type> type;
};
private:
template<typename Context, typename Iterator>
SPROUT_CONSTEXPR typename result<Context, Iterator>::type make_result(
typename result<Context, Iterator>::type const& res,
int sign = 0
) const
{
typedef typename result<Context, Iterator>::type result_type;
typedef typename attribute<Context, Iterator>::type attribute_type;
return res.success() && sign < 0
? result_type(true, res.current(), -res.attr())
: res
;
}
template<std::size_t N, typename Context, typename Iterator, typename PResult>
SPROUT_CONSTEXPR typename std::enable_if<
N == MaxDigits,
N == limit_digits,
typename result<Context, Iterator>::type
>::type call_1(
Iterator first,
Iterator last,
Iterator temp_first,
IntType t,
UIntType t,
std::size_t n,
PResult const& res
) const
{
typedef typename result<Context, Iterator>::type result_type;
typedef typename attribute<Context, Iterator>::type attribute_type;
return sprout::tuples::get<1>(res)
? result_type(
true,
sprout::next(first),
static_cast<IntType>(t * Radix | sprout::tuples::get<0>(res))
)
? n < MaxDigits
? call_0<N - 1, Context>(
sprout::next(first),
last,
temp_first,
static_cast<UIntType>(t * Radix + sprout::tuples::get<0>(res)),
n
)
: result_type(
true,
sprout::next(first),
static_cast<UIntType>(t * Radix + sprout::tuples::get<0>(res))
)
: N < MinDigits
? result_type(false, temp_first, attribute_type())
: result_type(true, first, t)
@ -64,13 +91,14 @@ namespace sprout {
}
template<std::size_t N, typename Context, typename Iterator, typename PResult>
SPROUT_CONSTEXPR typename std::enable_if<
N != MaxDigits,
N != limit_digits,
typename result<Context, Iterator>::type
>::type call_1(
Iterator first,
Iterator last,
Iterator temp_first,
IntType t,
UIntType t,
std::size_t n,
PResult const& res
) const
{
@ -79,8 +107,10 @@ namespace sprout {
return sprout::tuples::get<1>(res)
? call_0<N, Context>(
sprout::next(first),
last, temp_first,
static_cast<IntType>(t * Radix | sprout::tuples::get<0>(res))
last,
temp_first,
static_cast<UIntType>(t * Radix + sprout::tuples::get<0>(res)),
n
)
: N < MinDigits
? result_type(false, temp_first, attribute_type())
@ -92,7 +122,8 @@ namespace sprout {
Iterator first,
Iterator last,
Iterator temp_first,
IntType t
UIntType t,
std::size_t n
) const
{
typedef typename result<Context, Iterator>::type result_type;
@ -103,7 +134,8 @@ namespace sprout {
last,
temp_first,
t,
sprout::weed::detail::from_ndigit<Radix, IntType>(*first)
n + 1,
sprout::weed::detail::from_ndigit<Radix, UIntType>(*first)
)
: N < MinDigits
? result_type(false, temp_first, attribute_type())
@ -111,7 +143,7 @@ namespace sprout {
;
}
template<typename Context, typename Iterator, typename PResult>
SPROUT_CONSTEXPR typename result<Context, Iterator>::type call(
SPROUT_CONSTEXPR typename result<Context, Iterator>::type call_i(
Iterator first,
Iterator last,
Iterator temp_first,
@ -125,11 +157,26 @@ namespace sprout {
sprout::next(first),
last,
temp_first,
sprout::tuples::get<0>(res)
sprout::tuples::get<0>(res),
1
)
: result_type(false, temp_first, attribute_type())
;
}
template<typename Context, typename Iterator>
SPROUT_CONSTEXPR typename result<Context, Iterator>::type call(
Iterator first,
Iterator last,
Iterator temp_first
) const
{
return call_i<Context>(
first,
last,
temp_first,
sprout::weed::detail::from_ndigit<Radix, UIntType>(*first)
);
}
public:
template<typename Context, typename Iterator>
SPROUT_CONSTEXPR typename result<Context, Iterator>::type operator()(
@ -144,8 +191,7 @@ namespace sprout {
? call<Context>(
first,
last,
first,
sprout::weed::detail::from_ndigit<Radix, IntType>(*first)
first
)
: result_type(false, first, attribute_type())
;
@ -155,3 +201,4 @@ namespace sprout {
} // namespace sprout
#endif // #ifndef SPROUT_WEED_PARSER_NUMERIC_UINT_P_HPP