mirror of
https://github.com/bolero-MURAKAMI/Sprout.git
synced 2024-11-25 00:43:44 +00:00
196 lines
6.2 KiB
C++
196 lines
6.2 KiB
C++
/*=============================================================================
|
|
Copyright (c) 2011-2019 Bolero MURAKAMI
|
|
https://github.com/bolero-MURAKAMI/Sprout
|
|
|
|
Distributed under the Boost Software License, Version 1.0. (See accompanying
|
|
file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
|
|
=============================================================================*/
|
|
#ifndef SPROUT_WEED_PARSER_NUMERIC_UINT_P_HPP
|
|
#define SPROUT_WEED_PARSER_NUMERIC_UINT_P_HPP
|
|
|
|
#include <sprout/config.hpp>
|
|
#include <sprout/workaround/std/cstddef.hpp>
|
|
#include <sprout/iterator/next.hpp>
|
|
#include <sprout/tuple/tuple.hpp>
|
|
#include <sprout/integer/integer_digits.hpp>
|
|
#include <sprout/type_traits/identity.hpp>
|
|
#include <sprout/weed/unused.hpp>
|
|
#include <sprout/weed/parser_result.hpp>
|
|
#include <sprout/weed/parser/parser_base.hpp>
|
|
#include <sprout/weed/detail/ndigits.hpp>
|
|
|
|
namespace sprout {
|
|
namespace weed {
|
|
//
|
|
// uint_p
|
|
//
|
|
template<
|
|
typename UIntType,
|
|
std::size_t Radix = 10,
|
|
std::size_t MinDigits = 1,
|
|
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 sprout::identity<UIntType>
|
|
{};
|
|
template<typename Context, typename Iterator>
|
|
struct result
|
|
: public sprout::identity<sprout::weed::parser_result<Iterator, typename attribute<Context, Iterator>::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;
|
|
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 == limit_digits,
|
|
typename result<Context, Iterator>::type
|
|
>::type call_1(
|
|
Iterator first, Iterator last,
|
|
Iterator temp_first,
|
|
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)
|
|
? 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)
|
|
;
|
|
}
|
|
template<std::size_t N, typename Context, typename Iterator, typename PResult>
|
|
SPROUT_CONSTEXPR typename std::enable_if<
|
|
N != limit_digits,
|
|
typename result<Context, Iterator>::type
|
|
>::type call_1(
|
|
Iterator first, Iterator last,
|
|
Iterator temp_first,
|
|
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)
|
|
? call_0<N, Context>(
|
|
sprout::next(first), last,
|
|
temp_first,
|
|
static_cast<UIntType>(t * Radix + sprout::tuples::get<0>(res)),
|
|
n
|
|
)
|
|
: N < MinDigits
|
|
? result_type(false, temp_first, attribute_type())
|
|
: result_type(true, first, t)
|
|
;
|
|
}
|
|
template<std::size_t N, typename Context, typename Iterator>
|
|
SPROUT_CONSTEXPR typename result<Context, Iterator>::type call_0(
|
|
Iterator first, Iterator last,
|
|
Iterator temp_first,
|
|
UIntType t,
|
|
std::size_t n
|
|
) const
|
|
{
|
|
typedef typename result<Context, Iterator>::type result_type;
|
|
typedef typename attribute<Context, Iterator>::type attribute_type;
|
|
return first != last
|
|
? call_1<N + 1, Context>(
|
|
first, last,
|
|
temp_first,
|
|
t,
|
|
n + 1,
|
|
sprout::weed::detail::from_ndigit<Radix, UIntType>(*first)
|
|
)
|
|
: N < MinDigits
|
|
? result_type(false, temp_first, attribute_type())
|
|
: result_type(true, first, t)
|
|
;
|
|
}
|
|
template<typename Context, typename Iterator, typename PResult>
|
|
SPROUT_CONSTEXPR typename result<Context, Iterator>::type call_i(
|
|
Iterator first, Iterator last,
|
|
Iterator temp_first,
|
|
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)
|
|
? call_0<1, Context>(
|
|
sprout::next(first), last,
|
|
temp_first,
|
|
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()(
|
|
Iterator first, Iterator last,
|
|
Context const&
|
|
) const
|
|
{
|
|
typedef typename result<Context, Iterator>::type result_type;
|
|
typedef typename attribute<Context, Iterator>::type attribute_type;
|
|
return first != last
|
|
? call<Context>(
|
|
first, last,
|
|
first
|
|
)
|
|
: result_type(false, first, attribute_type())
|
|
;
|
|
}
|
|
};
|
|
} // namespace weed
|
|
} // namespace sprout
|
|
|
|
#endif // #ifndef SPROUT_WEED_PARSER_NUMERIC_UINT_P_HPP
|