1
0
Fork 0
mirror of https://github.com/bolero-MURAKAMI/Sprout synced 2024-11-12 21:09:01 +00:00
Sprout/sprout/weed/parser/numeric/uint_p.hpp

197 lines
6.2 KiB
C++
Raw Normal View History

2013-08-08 09:54:33 +00:00
/*=============================================================================
2016-02-25 09:48:28 +00:00
Copyright (c) 2011-2016 Bolero MURAKAMI
2013-08-08 09:54:33 +00:00
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)
=============================================================================*/
2011-11-14 04:22:04 +00:00
#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>
2011-11-14 04:22:04 +00:00
#include <sprout/iterator/next.hpp>
#include <sprout/tuple/tuple.hpp>
#include <sprout/integer/integer_digits.hpp>
2013-03-25 03:43:47 +00:00
#include <sprout/type_traits/identity.hpp>
2011-11-14 04:22:04 +00:00
#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,
2011-11-14 04:22:04 +00:00
std::size_t MinDigits = 1,
std::size_t MaxDigits = sprout::integer_digits<UIntType, Radix>::value
2011-11-14 04:22:04 +00:00
>
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
;
2011-11-14 04:22:04 +00:00
public:
template<typename Context, typename Iterator>
2013-03-25 03:43:47 +00:00
struct attribute
: public sprout::identity<UIntType>
{};
2011-11-14 04:22:04 +00:00
template<typename Context, typename Iterator>
2013-03-25 03:43:47 +00:00
struct result
: public sprout::identity<sprout::weed::parser_result<Iterator, typename attribute<Context, Iterator>::type> >
{};
2011-11-14 04:22:04 +00:00
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
;
}
2011-11-14 04:22:04 +00:00
template<std::size_t N, typename Context, typename Iterator, typename PResult>
SPROUT_CONSTEXPR typename std::enable_if<
N == limit_digits,
2011-11-14 04:22:04 +00:00
typename result<Context, Iterator>::type
>::type call_1(
2016-04-01 14:37:48 +00:00
Iterator first, Iterator last,
2011-11-14 04:22:04 +00:00
Iterator temp_first,
UIntType t,
std::size_t n,
2011-11-14 04:22:04 +00:00
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>(
2016-04-01 14:37:48 +00:00
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))
)
2011-11-14 04:22:04 +00:00
: 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,
2011-11-14 04:22:04 +00:00
typename result<Context, Iterator>::type
>::type call_1(
2016-04-01 14:37:48 +00:00
Iterator first, Iterator last,
2011-11-14 04:22:04 +00:00
Iterator temp_first,
UIntType t,
std::size_t n,
2011-11-14 04:22:04 +00:00
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>(
2016-04-01 14:37:48 +00:00
sprout::next(first), last,
temp_first,
static_cast<UIntType>(t * Radix + sprout::tuples::get<0>(res)),
n
2011-11-14 04:22:04 +00:00
)
: 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(
2016-04-01 14:37:48 +00:00
Iterator first, Iterator last,
2011-11-14 04:22:04 +00:00
Iterator temp_first,
UIntType t,
std::size_t n
2011-11-14 04:22:04 +00:00
) 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>(
2016-04-01 14:37:48 +00:00
first, last,
2011-11-14 04:22:04 +00:00
temp_first,
t,
n + 1,
sprout::weed::detail::from_ndigit<Radix, UIntType>(*first)
2011-11-14 04:22:04 +00:00
)
: 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(
2016-04-01 14:37:48 +00:00
Iterator first, Iterator last,
2011-11-14 04:22:04 +00:00
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>(
2016-04-01 14:37:48 +00:00
sprout::next(first), last,
2011-11-14 04:22:04 +00:00
temp_first,
sprout::tuples::get<0>(res),
1
2011-11-14 04:22:04 +00:00
)
: result_type(false, temp_first, attribute_type())
;
}
template<typename Context, typename Iterator>
SPROUT_CONSTEXPR typename result<Context, Iterator>::type call(
2016-04-01 14:37:48 +00:00
Iterator first, Iterator last,
Iterator temp_first
) const
{
return call_i<Context>(
2016-04-01 14:37:48 +00:00
first, last,
temp_first,
sprout::weed::detail::from_ndigit<Radix, UIntType>(*first)
);
}
2011-11-14 04:22:04 +00:00
public:
template<typename Context, typename Iterator>
SPROUT_CONSTEXPR typename result<Context, Iterator>::type operator()(
2016-04-01 14:37:48 +00:00
Iterator first, Iterator last,
2011-11-14 04:22:04 +00:00
Context const&
) const
{
typedef typename result<Context, Iterator>::type result_type;
typedef typename attribute<Context, Iterator>::type attribute_type;
return first != last
? call<Context>(
2016-04-01 14:37:48 +00:00
first, last,
first
2011-11-14 04:22:04 +00:00
)
: result_type(false, first, attribute_type())
;
}
};
} // namespace weed
} // namespace sprout
#endif // #ifndef SPROUT_WEED_PARSER_NUMERIC_UINT_P_HPP