1 // Copyright (c) 2009-2010 Satoshi Nakamoto
2 // Copyright (c) 2009-2015 The Bitcoin Core developers
3 // Distributed under the MIT software license, see the accompanying
4 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
6 #ifndef BITCOIN_TEST_SCRIPTNUM10_H
7 #define BITCOIN_TEST_SCRIPTNUM10_H
17 class scriptnum10_error
: public std::runtime_error
20 explicit scriptnum10_error(const std::string
& str
) : std::runtime_error(str
) {}
26 * The ScriptNum implementation from Bitcoin Core 0.10.0, for cross-comparison.
30 explicit CScriptNum10(const int64_t& n
)
35 static const size_t nDefaultMaxNumSize
= 4;
37 explicit CScriptNum10(const std::vector
<unsigned char>& vch
, bool fRequireMinimal
,
38 const size_t nMaxNumSize
= nDefaultMaxNumSize
)
40 if (vch
.size() > nMaxNumSize
) {
41 throw scriptnum10_error("script number overflow");
43 if (fRequireMinimal
&& vch
.size() > 0) {
44 // Check that the number is encoded with the minimum possible
47 // If the most-significant-byte - excluding the sign bit - is zero
48 // then we're not minimal. Note how this test also rejects the
49 // negative-zero encoding, 0x80.
50 if ((vch
.back() & 0x7f) == 0) {
51 // One exception: if there's more than one byte and the most
52 // significant bit of the second-most-significant-byte is set
53 // it would conflict with the sign bit. An example of this case
54 // is +-255, which encode to 0xff00 and 0xff80 respectively.
56 if (vch
.size() <= 1 || (vch
[vch
.size() - 2] & 0x80) == 0) {
57 throw scriptnum10_error("non-minimally encoded script number");
61 m_value
= set_vch(vch
);
64 inline bool operator==(const int64_t& rhs
) const { return m_value
== rhs
; }
65 inline bool operator!=(const int64_t& rhs
) const { return m_value
!= rhs
; }
66 inline bool operator<=(const int64_t& rhs
) const { return m_value
<= rhs
; }
67 inline bool operator< (const int64_t& rhs
) const { return m_value
< rhs
; }
68 inline bool operator>=(const int64_t& rhs
) const { return m_value
>= rhs
; }
69 inline bool operator> (const int64_t& rhs
) const { return m_value
> rhs
; }
71 inline bool operator==(const CScriptNum10
& rhs
) const { return operator==(rhs
.m_value
); }
72 inline bool operator!=(const CScriptNum10
& rhs
) const { return operator!=(rhs
.m_value
); }
73 inline bool operator<=(const CScriptNum10
& rhs
) const { return operator<=(rhs
.m_value
); }
74 inline bool operator< (const CScriptNum10
& rhs
) const { return operator< (rhs
.m_value
); }
75 inline bool operator>=(const CScriptNum10
& rhs
) const { return operator>=(rhs
.m_value
); }
76 inline bool operator> (const CScriptNum10
& rhs
) const { return operator> (rhs
.m_value
); }
78 inline CScriptNum10
operator+( const int64_t& rhs
) const { return CScriptNum10(m_value
+ rhs
);}
79 inline CScriptNum10
operator-( const int64_t& rhs
) const { return CScriptNum10(m_value
- rhs
);}
80 inline CScriptNum10
operator+( const CScriptNum10
& rhs
) const { return operator+(rhs
.m_value
); }
81 inline CScriptNum10
operator-( const CScriptNum10
& rhs
) const { return operator-(rhs
.m_value
); }
83 inline CScriptNum10
& operator+=( const CScriptNum10
& rhs
) { return operator+=(rhs
.m_value
); }
84 inline CScriptNum10
& operator-=( const CScriptNum10
& rhs
) { return operator-=(rhs
.m_value
); }
86 inline CScriptNum10
operator-() const
88 assert(m_value
!= std::numeric_limits
<int64_t>::min());
89 return CScriptNum10(-m_value
);
92 inline CScriptNum10
& operator=( const int64_t& rhs
)
98 inline CScriptNum10
& operator+=( const int64_t& rhs
)
100 assert(rhs
== 0 || (rhs
> 0 && m_value
<= std::numeric_limits
<int64_t>::max() - rhs
) ||
101 (rhs
< 0 && m_value
>= std::numeric_limits
<int64_t>::min() - rhs
));
106 inline CScriptNum10
& operator-=( const int64_t& rhs
)
108 assert(rhs
== 0 || (rhs
> 0 && m_value
>= std::numeric_limits
<int64_t>::min() + rhs
) ||
109 (rhs
< 0 && m_value
<= std::numeric_limits
<int64_t>::max() + rhs
));
116 if (m_value
> std::numeric_limits
<int>::max())
117 return std::numeric_limits
<int>::max();
118 else if (m_value
< std::numeric_limits
<int>::min())
119 return std::numeric_limits
<int>::min();
123 std::vector
<unsigned char> getvch() const
125 return serialize(m_value
);
128 static std::vector
<unsigned char> serialize(const int64_t& value
)
131 return std::vector
<unsigned char>();
133 std::vector
<unsigned char> result
;
134 const bool neg
= value
< 0;
135 uint64_t absvalue
= neg
? -value
: value
;
139 result
.push_back(absvalue
& 0xff);
143 // - If the most significant byte is >= 0x80 and the value is positive, push a
144 // new zero-byte to make the significant byte < 0x80 again.
146 // - If the most significant byte is >= 0x80 and the value is negative, push a
147 // new 0x80 byte that will be popped off when converting to an integral.
149 // - If the most significant byte is < 0x80 and the value is negative, add
150 // 0x80 to it, since it will be subtracted and interpreted as a negative when
151 // converting to an integral.
153 if (result
.back() & 0x80)
154 result
.push_back(neg
? 0x80 : 0);
156 result
.back() |= 0x80;
162 static int64_t set_vch(const std::vector
<unsigned char>& vch
)
168 for (size_t i
= 0; i
!= vch
.size(); ++i
)
169 result
|= static_cast<int64_t>(vch
[i
]) << 8*i
;
171 // If the input vector's most significant byte is 0x80, remove it from
172 // the result's msb and return a negative.
173 if (vch
.back() & 0x80)
174 return -((int64_t)(result
& ~(0x80ULL
<< (8 * (vch
.size() - 1)))));
183 #endif // BITCOIN_TEST_BIGNUM_H