Sprout/sprout/checksum/sum.hpp

112 lines
3.6 KiB
C++
Raw Normal View History

2013-08-08 09:54:33 +00:00
/*=============================================================================
2014-01-08 07:48:12 +00:00
Copyright (c) 2011-2014 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)
=============================================================================*/
2012-08-08 12:59:47 +00:00
#ifndef SPROUT_CHECKSUM_SUM_HPP
#define SPROUT_CHECKSUM_SUM_HPP
#include <cstddef>
#include <cstdint>
#include <climits>
#include <type_traits>
#include <sprout/config.hpp>
2013-08-06 15:15:09 +00:00
#include <sprout/limits.hpp>
2012-08-08 12:59:47 +00:00
#include <sprout/container/functions.hpp>
#include <sprout/iterator/operation.hpp>
#include <sprout/iterator/bytes_iterator.hpp>
#include <sprout/numeric/accumulate.hpp>
#include <sprout/detail/integer.hpp>
namespace sprout {
static_assert(CHAR_BIT == 8, "CHAR_BIT == 8");
//
// sum_basic
//
template<std::size_t Bits = 8>
class sum_basic {
static_assert(Bits % 8 == 0, "Bits % 8 == 0");
public:
typedef typename sprout::detail::uint_t<Bits>::least value_type;
typedef sum_basic const const_type;
typedef typename std::conditional<
sizeof(std::size_t) < sizeof(value_type),
value_type,
std::size_t
>::type sum_type;
private:
sum_type sum_;
private:
template<typename InputIterator>
SPROUT_CONSTEXPR sum_type calc_sum(InputIterator first, InputIterator last) const {
2012-08-08 12:59:47 +00:00
return sprout::accumulate(
sprout::make_bytes_iterator(first),
sprout::make_bytes_iterator(last),
sum_
);
}
public:
2014-01-13 15:18:06 +00:00
SPROUT_CONSTEXPR sum_basic() SPROUT_DEFAULTED_DEFAULT_CONSTRUCTOR_DECL
sum_basic(sum_basic const&) = default;
2012-08-08 12:59:47 +00:00
explicit SPROUT_CONSTEXPR sum_basic(sum_type sum)
: sum_(sum)
{}
SPROUT_CXX14_CONSTEXPR void reset(sum_type new_sum = 0) {
2012-08-08 12:59:47 +00:00
sum_ = new_sum;
}
SPROUT_CONSTEXPR sum_basic const process_byte(std::uint8_t byte) const {
return sum_basic(sum_ + byte);
}
template<typename InputIterator>
SPROUT_CONSTEXPR sum_basic const process_block(InputIterator bytes_begin, InputIterator bytes_end) const {
2012-08-08 12:59:47 +00:00
return sum_basic(calc_sum(bytes_begin, bytes_end));
}
template<typename InputIterator>
SPROUT_CONSTEXPR sum_basic const process_bytes(InputIterator buffer, std::size_t byte_count) const {
2012-08-08 12:59:47 +00:00
return process_block(buffer, sprout::next(buffer, byte_count));
}
template<typename InputRange>
SPROUT_CONSTEXPR sum_basic const process_range(InputRange const& bytes_range) const {
2012-08-08 12:59:47 +00:00
return process_block(sprout::begin(bytes_range), sprout::end(bytes_range));
}
SPROUT_CXX14_CONSTEXPR void process_byte(std::uint8_t byte) {
2012-08-08 12:59:47 +00:00
sum_ += byte;
}
template<typename InputIterator>
SPROUT_CXX14_CONSTEXPR void process_block(InputIterator bytes_begin, InputIterator bytes_end) {
2012-08-08 12:59:47 +00:00
sum_ = calc_sum(bytes_begin, bytes_end);
}
template<typename InputIterator>
SPROUT_CXX14_CONSTEXPR void process_bytes(InputIterator buffer, std::size_t byte_count) {
2012-08-08 12:59:47 +00:00
process_block(buffer, sprout::next(buffer, byte_count));
}
template<typename InputRange>
SPROUT_CXX14_CONSTEXPR void process_range(InputRange const& bytes_range) {
2012-08-08 12:59:47 +00:00
process_block(sprout::begin(bytes_range), sprout::end(bytes_range));
}
SPROUT_CONSTEXPR value_type checksum() const {
2013-08-06 15:15:09 +00:00
return static_cast<value_type>(sum_ & sprout::numeric_limits<value_type>::max());
2012-08-08 12:59:47 +00:00
}
SPROUT_CONSTEXPR value_type operator()() const {
return checksum();
}
};
//
// sum8
// sum16
// sum32
//
typedef sprout::sum_basic<8> sum8;
typedef sprout::sum_basic<16> sum16;
typedef sprout::sum_basic<32> sum32;
} // namespace sprout
#endif // #ifndef SPROUT_CHECKSUM_SUM_HPP