Fixed two bugs in wire:

* Integer conversion checks in src/wire/read.h
  * Missing "boolean" function in wire::writer and derived types
This commit is contained in:
Lee Clagett 2022-10-08 12:08:42 -04:00 committed by Leee *!* Clagett
parent e8b1734a43
commit 96a97103a7
5 changed files with 53 additions and 24 deletions

View file

@ -73,6 +73,12 @@ namespace wire
return buf;
}
void json_writer::boolean(const bool source)
{
formatter_.Bool(source);
check_flush();
}
void json_writer::integer(const int source)
{
formatter_.Int(source);

View file

@ -85,6 +85,8 @@ namespace wire
//! \return Null-terminated buffer containing uint as decimal ascii
static std::array<char, uint_to_string_size> to_string(std::uintmax_t) noexcept;
void boolean(bool) override final;
void integer(int) override final;
void integer(std::intmax_t) override final;

View file

@ -35,9 +35,12 @@ void wire::reader::increment_depth()
WIRE_DLOG_THROW_(error::schema::maximum_depth);
}
[[noreturn]] void wire::integer::throw_exception(std::intmax_t source, std::intmax_t min)
[[noreturn]] void wire::integer::throw_exception(std::intmax_t source, std::intmax_t min, std::uintmax_t max)
{
WIRE_DLOG_THROW(error::schema::larger_integer, source << " given when " << min << " is minimum permitted");
if (source < 0)
WIRE_DLOG_THROW(error::schema::larger_integer, source << " given when " << min << " is minimum permitted");
else
throw_exception(std::uintmax_t(source), max);
}
[[noreturn]] void wire::integer::throw_exception(std::uintmax_t source, std::uintmax_t max)
{

View file

@ -167,32 +167,43 @@ namespace wire
namespace integer
{
[[noreturn]] void throw_exception(std::intmax_t source, std::intmax_t min);
[[noreturn]] void throw_exception(std::uintmax_t source, std::uintmax_t max);
[[noreturn]] void throw_exception(std::intmax_t value, std::intmax_t min, std::uintmax_t max);
[[noreturn]] void throw_exception(std::uintmax_t value, std::uintmax_t max);
template<typename Target, typename U>
inline bool will_overflow(const U val) noexcept
{
using t_limit = std::numeric_limits<Target>;
using s_limit = std::numeric_limits<U>;
static_assert(t_limit::is_integer, "target must be integer");
static_assert(s_limit::is_integer, "source must be integer");
/* Adapted from:
https://blog.reverberate.org/2012/12/testing-for-integer-overflow-in-c-and-c.html */
if (t_limit::is_signed)
{
using im_limit = std::numeric_limits<std::intmax_t>;
return
(!s_limit::is_signed && std::uintmax_t(val) > std::uintmax_t(im_limit::max())) ||
std::intmax_t(val) < std::intmax_t(t_limit::min()) ||
std::intmax_t(val) > std::intmax_t(t_limit::max());
}
return val < 0 || std::uintmax_t(val) > std::uintmax_t(t_limit::max());
}
template<typename Target, typename U>
inline Target convert_to(const U source)
{
using common = typename std::common_type<Target, U>::type;
static constexpr const Target target_min = std::numeric_limits<Target>::min();
static constexpr const Target target_max = std::numeric_limits<Target>::max();
/* After optimizations, this is:
* 1 check for unsigned -> unsigned (uint, uint)
* 2 checks for signed -> signed (int, int)
* 2 checks for signed -> unsigned-- (
* 1 check for unsigned -> signed (uint, uint)
Put `WIRE_DLOG_THROW` in cpp to reduce code/ASM duplication. Do not
remove first check, signed values can be implicitly converted to
unsigned in some checks. */
if (!std::numeric_limits<Target>::is_signed && source < 0)
throw_exception(std::intmax_t(source), std::intmax_t(0));
else if (common(source) < common(target_min))
throw_exception(std::intmax_t(source), std::intmax_t(target_min));
else if (common(target_max) < common(source))
throw_exception(std::uintmax_t(source), std::uintmax_t(target_max));
using limit = std::numeric_limits<Target>;
if (will_overflow<Target>(source))
{
if (std::numeric_limits<U>::is_signed)
throw_exception(source, limit::min(), limit::max());
else
throw_exception(source, limit::max());
}
return Target(source);
}
}

View file

@ -47,6 +47,8 @@ namespace wire
virtual ~writer() noexcept;
virtual void boolean(bool) = 0;
virtual void integer(int) = 0;
virtual void integer(std::intmax_t) = 0;
@ -78,6 +80,11 @@ namespace wire
// leave in header, compiler can de-virtualize when final type is given
inline void write_bytes(writer& dest, const bool source)
{
dest.boolean(source);
}
inline void write_bytes(writer& dest, const int source)
{
dest.integer(source);