2016-08-01 05:25:37 +00:00
|
|
|
/*
|
2017-02-02 10:06:28 +00:00
|
|
|
This file is part of solidity.
|
2016-08-01 05:25:37 +00:00
|
|
|
|
2017-02-02 10:06:28 +00:00
|
|
|
solidity is free software: you can redistribute it and/or modify
|
2016-08-01 05:25:37 +00:00
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
2017-02-02 10:06:28 +00:00
|
|
|
solidity is distributed in the hope that it will be useful,
|
2016-08-01 05:25:37 +00:00
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
2017-02-02 10:06:28 +00:00
|
|
|
along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
2016-08-01 05:25:37 +00:00
|
|
|
*/
|
|
|
|
/** @file FixedHash.h
|
|
|
|
* @author Gav Wood <i@gavwood.com>
|
|
|
|
* @date 2014
|
|
|
|
*
|
|
|
|
* The FixedHash fixed-size "hash" container type.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2020-01-06 10:52:23 +00:00
|
|
|
#include <libsolutil/CommonData.h>
|
2017-08-25 10:07:02 +00:00
|
|
|
|
|
|
|
#include <boost/functional/hash.hpp>
|
|
|
|
#include <boost/io/ios_state.hpp>
|
|
|
|
|
2016-08-01 05:25:37 +00:00
|
|
|
#include <array>
|
|
|
|
#include <cstdint>
|
|
|
|
#include <algorithm>
|
|
|
|
|
2019-12-11 16:31:36 +00:00
|
|
|
namespace solidity::util
|
2016-08-01 05:25:37 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
/// Fixed-size raw-byte array container type, with an API optimised for storing hashes.
|
|
|
|
/// Transparently converts to/from the corresponding arithmetic type; this will
|
|
|
|
/// assume the data contained in the hash is big-endian.
|
|
|
|
template <unsigned N>
|
|
|
|
class FixedHash
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
/// The corresponding arithmetic type.
|
|
|
|
using Arith = boost::multiprecision::number<boost::multiprecision::cpp_int_backend<N * 8, N * 8, boost::multiprecision::unsigned_magnitude, boost::multiprecision::unchecked, void>>;
|
|
|
|
|
|
|
|
/// The size of the container.
|
|
|
|
enum { size = N };
|
|
|
|
|
|
|
|
/// Method to convert from a string.
|
|
|
|
enum ConstructFromStringType { FromHex, FromBinary };
|
|
|
|
|
|
|
|
/// Method to convert from a string.
|
|
|
|
enum ConstructFromHashType { AlignLeft, AlignRight, FailIfDifferent };
|
|
|
|
|
|
|
|
/// Construct an empty hash.
|
2017-08-21 14:42:17 +00:00
|
|
|
explicit FixedHash() { m_data.fill(0); }
|
2016-08-01 05:25:37 +00:00
|
|
|
|
|
|
|
/// Construct from another hash, filling with zeroes or cropping as necessary.
|
|
|
|
template <unsigned M> explicit FixedHash(FixedHash<M> const& _h, ConstructFromHashType _t = AlignLeft) { m_data.fill(0); unsigned c = std::min(M, N); for (unsigned i = 0; i < c; ++i) m_data[_t == AlignRight ? N - 1 - i : i] = _h[_t == AlignRight ? M - 1 - i : i]; }
|
|
|
|
|
|
|
|
/// Convert from the corresponding arithmetic type.
|
|
|
|
FixedHash(Arith const& _arith) { toBigEndian(_arith, m_data); }
|
|
|
|
|
|
|
|
/// Explicitly construct, copying from a byte array.
|
2020-06-05 11:36:03 +00:00
|
|
|
explicit FixedHash(bytes const& _array, ConstructFromHashType _sizeMismatchBehavior = FailIfDifferent)
|
|
|
|
{
|
|
|
|
if (_array.size() == N)
|
|
|
|
memcpy(m_data.data(), _array.data(), _array.size());
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_data.fill(0);
|
|
|
|
if (_sizeMismatchBehavior != FailIfDifferent)
|
|
|
|
{
|
|
|
|
auto bytesToCopy = std::min<size_t>(_array.size(), N);
|
|
|
|
for (size_t i = 0; i < bytesToCopy; ++i)
|
|
|
|
if (_sizeMismatchBehavior == AlignRight)
|
|
|
|
m_data[N - 1 - i] = _array[_array.size() - 1 - i];
|
|
|
|
else
|
|
|
|
m_data[i] = _array[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-08-01 05:25:37 +00:00
|
|
|
|
|
|
|
/// Explicitly construct, copying from a byte array.
|
2020-06-05 11:36:03 +00:00
|
|
|
explicit FixedHash(bytesConstRef _b, ConstructFromHashType _t = FailIfDifferent)
|
|
|
|
{
|
|
|
|
if (_b.size() == N)
|
|
|
|
memcpy(m_data.data(), _b.data(), std::min<size_t>(_b.size(), N));
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_data.fill(0);
|
|
|
|
if (_t != FailIfDifferent)
|
|
|
|
{
|
|
|
|
auto c = std::min<size_t>(_b.size(), N);
|
|
|
|
for (size_t i = 0; i < c; ++i)
|
|
|
|
if (_t == AlignRight)
|
|
|
|
m_data[N - 1 - i] = _b[_b.size() - 1 - i];
|
|
|
|
else
|
|
|
|
m_data[i] = _b[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-08-01 05:25:37 +00:00
|
|
|
|
|
|
|
/// Explicitly construct, copying from a string.
|
2019-12-11 16:31:36 +00:00
|
|
|
explicit FixedHash(std::string const& _s, ConstructFromStringType _t = FromHex, ConstructFromHashType _ht = FailIfDifferent): FixedHash(_t == FromHex ? fromHex(_s, WhenError::Throw) : solidity::util::asBytes(_s), _ht) {}
|
2016-08-01 05:25:37 +00:00
|
|
|
|
|
|
|
/// Convert to arithmetic type.
|
|
|
|
operator Arith() const { return fromBigEndian<Arith>(m_data); }
|
|
|
|
|
|
|
|
// The obvious comparison operators.
|
|
|
|
bool operator==(FixedHash const& _c) const { return m_data == _c.m_data; }
|
|
|
|
bool operator!=(FixedHash const& _c) const { return m_data != _c.m_data; }
|
2017-08-25 10:10:28 +00:00
|
|
|
/// Required to sort objects of this type or use them as map keys.
|
2016-08-01 05:25:37 +00:00
|
|
|
bool operator<(FixedHash const& _c) const { for (unsigned i = 0; i < N; ++i) if (m_data[i] < _c.m_data[i]) return true; else if (m_data[i] > _c.m_data[i]) return false; return false; }
|
|
|
|
|
|
|
|
/// @returns a particular byte from the hash.
|
2018-11-07 11:04:46 +00:00
|
|
|
uint8_t& operator[](unsigned _i) { return m_data[_i]; }
|
2016-08-01 05:25:37 +00:00
|
|
|
/// @returns a particular byte from the hash.
|
2018-11-07 11:04:46 +00:00
|
|
|
uint8_t operator[](unsigned _i) const { return m_data[_i]; }
|
2016-08-01 05:25:37 +00:00
|
|
|
|
|
|
|
/// @returns the hash as a user-readable hex string.
|
2018-12-05 19:34:27 +00:00
|
|
|
std::string hex() const { return toHex(asBytes()); }
|
2016-08-01 05:25:37 +00:00
|
|
|
|
|
|
|
/// @returns a mutable byte vector_ref to the object's data.
|
|
|
|
bytesRef ref() { return bytesRef(m_data.data(), N); }
|
|
|
|
|
|
|
|
/// @returns a constant byte vector_ref to the object's data.
|
|
|
|
bytesConstRef ref() const { return bytesConstRef(m_data.data(), N); }
|
|
|
|
|
|
|
|
/// @returns a mutable byte pointer to the object's data.
|
2018-11-07 11:04:46 +00:00
|
|
|
uint8_t* data() { return m_data.data(); }
|
2016-08-01 05:25:37 +00:00
|
|
|
|
|
|
|
/// @returns a constant byte pointer to the object's data.
|
2018-11-07 11:04:46 +00:00
|
|
|
uint8_t const* data() const { return m_data.data(); }
|
2016-08-01 05:25:37 +00:00
|
|
|
|
|
|
|
/// @returns a copy of the object's data as a byte vector.
|
|
|
|
bytes asBytes() const { return bytes(data(), data() + N); }
|
|
|
|
|
|
|
|
private:
|
2018-11-07 11:04:46 +00:00
|
|
|
std::array<uint8_t, N> m_data; ///< The binary data.
|
2016-08-01 05:25:37 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/// Stream I/O for the FixedHash class.
|
|
|
|
template <unsigned N>
|
|
|
|
inline std::ostream& operator<<(std::ostream& _out, FixedHash<N> const& _h)
|
|
|
|
{
|
2017-08-22 12:55:01 +00:00
|
|
|
boost::io::ios_all_saver guard(_out);
|
2016-08-01 05:25:37 +00:00
|
|
|
_out << std::noshowbase << std::hex << std::setfill('0');
|
|
|
|
for (unsigned i = 0; i < N; ++i)
|
|
|
|
_out << std::setw(2) << (int)_h[i];
|
|
|
|
_out << std::dec;
|
|
|
|
return _out;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Common types of FixedHash.
|
|
|
|
using h256 = FixedHash<32>;
|
|
|
|
using h160 = FixedHash<20>;
|
|
|
|
|
|
|
|
}
|