2017-07-26 15:25:29 +10:00
|
|
|
/*
|
2018-08-04 15:14:06 +10:00
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
2017-07-26 15:25:29 +10:00
|
|
|
*
|
|
|
|
* Copyright 2017 Danny Robson <danny@nerdcruft.net>
|
|
|
|
*/
|
|
|
|
|
2018-01-10 17:19:39 +11:00
|
|
|
#include "parse.hpp"
|
|
|
|
|
|
|
|
#include "cast.hpp"
|
2017-07-26 15:25:29 +10:00
|
|
|
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <stdexcept>
|
|
|
|
|
|
|
|
using util::parse;
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
template <typename T> struct c_traits;
|
|
|
|
template <> struct c_traits<long> { static constexpr auto func = strtol; };
|
|
|
|
template <> struct c_traits<long long> { static constexpr auto func = strtoll; };
|
|
|
|
template <> struct c_traits<unsigned long> { static constexpr auto func = strtoul; };
|
|
|
|
template <> struct c_traits<unsigned long long> { static constexpr auto func = strtoull; };
|
|
|
|
|
|
|
|
template <> struct c_traits<float> { static constexpr auto func = strtof; };
|
|
|
|
template <> struct c_traits<double> { static constexpr auto func = strtod; };
|
|
|
|
template <> struct c_traits<long double> { static constexpr auto func = strtold; };
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
T
|
|
|
|
c_iparse (const char *first, const char *last)
|
|
|
|
{
|
|
|
|
auto tail = const_cast<char *> (last);
|
|
|
|
auto val = c_traits<T>::func (first, &tail, 0);
|
|
|
|
|
|
|
|
if (tail != last)
|
|
|
|
throw std::invalid_argument ("unable to parse");
|
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
template <typename T>
|
|
|
|
T
|
|
|
|
c_fparse (const char *first, const char *last)
|
|
|
|
{
|
|
|
|
auto tail = const_cast<char *> (last);
|
|
|
|
auto val = c_traits<T>::func (first, &tail);
|
|
|
|
|
|
|
|
if (tail != last)
|
|
|
|
throw std::invalid_argument ("unable to parse");
|
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
|
|
#define C_PARSE(T, KLASS) \
|
|
|
|
template <> \
|
|
|
|
T \
|
2017-12-18 14:50:10 +11:00
|
|
|
util::parse<T> (util::view<const char *> str) \
|
2017-07-26 15:25:29 +10:00
|
|
|
{ \
|
2017-12-18 14:50:10 +11:00
|
|
|
return c_ ## KLASS ## parse<T> (std::cbegin (str), std::cend (str)); \
|
2017-07-26 15:25:29 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
C_PARSE(long, i)
|
|
|
|
C_PARSE(long long, i)
|
|
|
|
C_PARSE(unsigned long, i)
|
|
|
|
C_PARSE(unsigned long long, i)
|
|
|
|
|
|
|
|
C_PARSE(float, f)
|
|
|
|
C_PARSE(double, f)
|
|
|
|
C_PARSE(long double, f)
|
2017-12-19 18:12:35 +11:00
|
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
int
|
|
|
|
util::parse<int> (util::view<const char*> str)
|
|
|
|
{
|
|
|
|
auto intermediate = util::parse<long> (str);
|
2018-01-16 15:11:15 +11:00
|
|
|
return util::cast::lossless<int> (intermediate);
|
2017-12-19 18:12:35 +11:00
|
|
|
}
|
2018-03-27 16:11:04 +11:00
|
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
unsigned
|
|
|
|
util::parse<unsigned> (util::view<const char*> str)
|
|
|
|
{
|
|
|
|
return util::cast::narrow<unsigned> (parse<unsigned long> (str));
|
|
|
|
}
|
2018-07-27 18:28:30 +10:00
|
|
|
|
|
|
|
|
|
|
|
template <>
|
|
|
|
unsigned short
|
|
|
|
util::parse<unsigned short> (util::view<const char*> str)
|
|
|
|
{
|
|
|
|
return util::cast::narrow<unsigned short> (parse<unsigned long> (str));
|
|
|
|
}
|