Merge bitcoin/bitcoin#30436: fix: Make TxidFromString() respect string_view length

09ce3501fa fix: Make TxidFromString() respect string_view length (Hodlinator)
01e314ce0a refactor: Change base_blob::SetHex() to take std::string_view (Hodlinator)
2f5577dc2e test: uint256 - Garbage suffixes and zero padding (Hodlinator)
f11f816800 refactor: Make uint256_tests no longer use deprecated BOOST_CHECK() (Hodlinator)
f0eeee2dc1 test: Add test for TxidFromString() behavior (Ryan Ofsky)

Pull request description:

  ### Problem

  Prior to this, `TxidFromString()` was passing `string_view::data()` into `uint256S()` which meant it would only receive the a naked `char*` pointer and potentially scan past the `string_view::length()` until it found a null terminator (or some other non-hex character).

  Appears to have been a fully dormant bug as callers were either passing a string literal or `std::string` directly to `TxidFromFromString()`, meaning a null terminator always existed at `pointer[length()]`. Bug existed since original merge of `TxidFromString()`.

  ### Solution

  Make `uint256S()` (and `base_blob::SetHex()`) take and operate on `std::string_view` instead of `const char*` and have `TxidFromString()` pass that in.

  (PR was prompted by comment in https://github.com/bitcoin/bitcoin/pull/30377#issuecomment-2208857200 (referring to https://github.com/bitcoin/bitcoin/pull/28922#discussion_r1404437378)).

ACKs for top commit:
  maflcko:
    re-ACK 09ce3501fa 🕓
  paplorinc:
    ACK 09ce3501fa
  ryanofsky:
    Code review ACK 09ce3501fa. I think the current code changes are about as small as you could make to fix the bug without introducing a string copy, and the surrounding test improvements are all very nice and welcome.

Tree-SHA512: c2c10551785fb6688d1e2492ba42a8eee4c19abbe8461bb0774d56a70c23cd6b0718d2641632890bee880c06202dee148126447dd2264eaed4f5fee7e1bcb581
This commit is contained in:
Ryan Ofsky 2024-07-23 14:00:03 -04:00
commit 7cc00bfc86
No known key found for this signature in database
GPG key ID: 46800E30FC748A66
8 changed files with 190 additions and 182 deletions

View file

@ -1028,4 +1028,12 @@ BOOST_AUTO_TEST_CASE(test_IsStandard)
} }
} }
BOOST_AUTO_TEST_CASE(test_TxidFromString)
{
// Make sure TxidFromString respects string_view length and stops reading at
// end of the substring.
BOOST_CHECK_EQUAL(TxidFromString(std::string_view("ABCD1234", 4)).ToString(),
"000000000000000000000000000000000000000000000000000000000000abcd");
}
BOOST_AUTO_TEST_SUITE_END() BOOST_AUTO_TEST_SUITE_END()

View file

@ -47,7 +47,7 @@ inline CTransactionRef create_placeholder_tx(size_t num_inputs, size_t num_outpu
// Create a Wtxid from a hex string // Create a Wtxid from a hex string
inline Wtxid WtxidFromString(std::string_view str) inline Wtxid WtxidFromString(std::string_view str)
{ {
return Wtxid::FromUint256(uint256S(str.data())); return Wtxid::FromUint256(uint256S(str));
} }
BOOST_FIXTURE_TEST_CASE(package_hash_tests, TestChain100Setup) BOOST_FIXTURE_TEST_CASE(package_hash_tests, TestChain100Setup)

View file

@ -58,13 +58,8 @@ static std::string ArrayToString(const unsigned char A[], unsigned int width)
return Stream.str(); return Stream.str();
} }
inline uint160 uint160S(const char *str) // Input is treated as little-endian.
{ inline uint160 uint160S(std::string_view str)
uint160 rv;
rv.SetHex(str);
return rv;
}
inline uint160 uint160S(const std::string& str)
{ {
uint160 rv; uint160 rv;
rv.SetHex(str); rv.SetHex(str);
@ -73,54 +68,57 @@ inline uint160 uint160S(const std::string& str)
BOOST_AUTO_TEST_CASE( basics ) // constructors, equality, inequality BOOST_AUTO_TEST_CASE( basics ) // constructors, equality, inequality
{ {
BOOST_CHECK(1 == 0+1);
// constructor uint256(vector<char>): // constructor uint256(vector<char>):
BOOST_CHECK(R1L.ToString() == ArrayToString(R1Array,32)); BOOST_CHECK_EQUAL(R1L.ToString(), ArrayToString(R1Array,32));
BOOST_CHECK(R1S.ToString() == ArrayToString(R1Array,20)); BOOST_CHECK_EQUAL(R1S.ToString(), ArrayToString(R1Array,20));
BOOST_CHECK(R2L.ToString() == ArrayToString(R2Array,32)); BOOST_CHECK_EQUAL(R2L.ToString(), ArrayToString(R2Array,32));
BOOST_CHECK(R2S.ToString() == ArrayToString(R2Array,20)); BOOST_CHECK_EQUAL(R2S.ToString(), ArrayToString(R2Array,20));
BOOST_CHECK(ZeroL.ToString() == ArrayToString(ZeroArray,32)); BOOST_CHECK_EQUAL(ZeroL.ToString(), ArrayToString(ZeroArray,32));
BOOST_CHECK(ZeroS.ToString() == ArrayToString(ZeroArray,20)); BOOST_CHECK_EQUAL(ZeroS.ToString(), ArrayToString(ZeroArray,20));
BOOST_CHECK(OneL.ToString() == ArrayToString(OneArray,32)); BOOST_CHECK_EQUAL(OneL.ToString(), ArrayToString(OneArray,32));
BOOST_CHECK(OneS.ToString() == ArrayToString(OneArray,20)); BOOST_CHECK_EQUAL(OneS.ToString(), ArrayToString(OneArray,20));
BOOST_CHECK(MaxL.ToString() == ArrayToString(MaxArray,32)); BOOST_CHECK_EQUAL(MaxL.ToString(), ArrayToString(MaxArray,32));
BOOST_CHECK(MaxS.ToString() == ArrayToString(MaxArray,20)); BOOST_CHECK_EQUAL(MaxS.ToString(), ArrayToString(MaxArray,20));
BOOST_CHECK(OneL.ToString() != ArrayToString(ZeroArray,32)); BOOST_CHECK_NE(OneL.ToString(), ArrayToString(ZeroArray,32));
BOOST_CHECK(OneS.ToString() != ArrayToString(ZeroArray,20)); BOOST_CHECK_NE(OneS.ToString(), ArrayToString(ZeroArray,20));
// == and != // == and !=
BOOST_CHECK(R1L != R2L && R1S != R2S); BOOST_CHECK_NE(R1L, R2L); BOOST_CHECK_NE(R1S, R2S);
BOOST_CHECK(ZeroL != OneL && ZeroS != OneS); BOOST_CHECK_NE(ZeroL, OneL); BOOST_CHECK_NE(ZeroS, OneS);
BOOST_CHECK(OneL != ZeroL && OneS != ZeroS); BOOST_CHECK_NE(OneL, ZeroL); BOOST_CHECK_NE(OneS, ZeroS);
BOOST_CHECK(MaxL != ZeroL && MaxS != ZeroS); BOOST_CHECK_NE(MaxL, ZeroL); BOOST_CHECK_NE(MaxS, ZeroS);
// String Constructor and Copy Constructor // String Constructor and Copy Constructor
BOOST_CHECK(uint256S("0x"+R1L.ToString()) == R1L); BOOST_CHECK_EQUAL(uint256S("0x"+R1L.ToString()), R1L);
BOOST_CHECK(uint256S("0x"+R2L.ToString()) == R2L); BOOST_CHECK_EQUAL(uint256S("0x"+R2L.ToString()), R2L);
BOOST_CHECK(uint256S("0x"+ZeroL.ToString()) == ZeroL); BOOST_CHECK_EQUAL(uint256S("0x"+ZeroL.ToString()), ZeroL);
BOOST_CHECK(uint256S("0x"+OneL.ToString()) == OneL); BOOST_CHECK_EQUAL(uint256S("0x"+OneL.ToString()), OneL);
BOOST_CHECK(uint256S("0x"+MaxL.ToString()) == MaxL); BOOST_CHECK_EQUAL(uint256S("0x"+MaxL.ToString()), MaxL);
BOOST_CHECK(uint256S(R1L.ToString()) == R1L); BOOST_CHECK_EQUAL(uint256S(R1L.ToString()), R1L);
BOOST_CHECK(uint256S(" 0x"+R1L.ToString()+" ") == R1L); BOOST_CHECK_EQUAL(uint256S(" 0x"+R1L.ToString()+" "), R1L);
BOOST_CHECK(uint256S("") == ZeroL); BOOST_CHECK_EQUAL(uint256S(" 0x"+R1L.ToString()+"-trash;%^& "), R1L);
BOOST_CHECK(R1L == uint256S(R1ArrayHex)); BOOST_CHECK_EQUAL(uint256S("\t \n \n \f\n\r\t\v\t 0x"+R1L.ToString()+" \t \n \n \f\n\r\t\v\t "), R1L);
BOOST_CHECK(uint256(R1L) == R1L); BOOST_CHECK_EQUAL(uint256S(""), ZeroL);
BOOST_CHECK(uint256(ZeroL) == ZeroL); BOOST_CHECK_EQUAL(R1L, uint256S(R1ArrayHex));
BOOST_CHECK(uint256(OneL) == OneL); BOOST_CHECK_EQUAL(uint256(R1L), R1L);
BOOST_CHECK_EQUAL(uint256(ZeroL), ZeroL);
BOOST_CHECK_EQUAL(uint256(OneL), OneL);
BOOST_CHECK(uint160S("0x"+R1S.ToString()) == R1S); BOOST_CHECK_EQUAL(uint160S("0x"+R1S.ToString()), R1S);
BOOST_CHECK(uint160S("0x"+R2S.ToString()) == R2S); BOOST_CHECK_EQUAL(uint160S("0x"+R2S.ToString()), R2S);
BOOST_CHECK(uint160S("0x"+ZeroS.ToString()) == ZeroS); BOOST_CHECK_EQUAL(uint160S("0x"+ZeroS.ToString()), ZeroS);
BOOST_CHECK(uint160S("0x"+OneS.ToString()) == OneS); BOOST_CHECK_EQUAL(uint160S("0x"+OneS.ToString()), OneS);
BOOST_CHECK(uint160S("0x"+MaxS.ToString()) == MaxS); BOOST_CHECK_EQUAL(uint160S("0x"+MaxS.ToString()), MaxS);
BOOST_CHECK(uint160S(R1S.ToString()) == R1S); BOOST_CHECK_EQUAL(uint160S(R1S.ToString()), R1S);
BOOST_CHECK(uint160S(" 0x"+R1S.ToString()+" ") == R1S); BOOST_CHECK_EQUAL(uint160S(" 0x"+R1S.ToString()+" "), R1S);
BOOST_CHECK(uint160S("") == ZeroS); BOOST_CHECK_EQUAL(uint160S(" 0x"+R1S.ToString()+"-trash;%^& "), R1S);
BOOST_CHECK(R1S == uint160S(R1ArrayHex)); BOOST_CHECK_EQUAL(uint160S(" \t \n \n \f\n\r\t\v\t 0x"+R1S.ToString()+" \t \n \n \f\n\r\t\v\t"), R1S);
BOOST_CHECK_EQUAL(uint160S(""), ZeroS);
BOOST_CHECK_EQUAL(R1S, uint160S(R1ArrayHex));
BOOST_CHECK(uint160(R1S) == R1S); BOOST_CHECK_EQUAL(uint160(R1S), R1S);
BOOST_CHECK(uint160(ZeroS) == ZeroS); BOOST_CHECK_EQUAL(uint160(ZeroS), ZeroS);
BOOST_CHECK(uint160(OneS) == OneS); BOOST_CHECK_EQUAL(uint160(OneS), OneS);
} }
BOOST_AUTO_TEST_CASE( comparison ) // <= >= < > BOOST_AUTO_TEST_CASE( comparison ) // <= >= < >
@ -129,140 +127,147 @@ BOOST_AUTO_TEST_CASE( comparison ) // <= >= < >
for (int i = 255; i >= 0; --i) { for (int i = 255; i >= 0; --i) {
uint256 TmpL; uint256 TmpL;
*(TmpL.begin() + (i>>3)) |= 1<<(7-(i&7)); *(TmpL.begin() + (i>>3)) |= 1<<(7-(i&7));
BOOST_CHECK( LastL < TmpL ); BOOST_CHECK_LT(LastL, TmpL);
LastL = TmpL; LastL = TmpL;
} }
BOOST_CHECK( ZeroL < R1L ); BOOST_CHECK_LT(ZeroL, R1L);
BOOST_CHECK( R2L < R1L ); BOOST_CHECK_LT(R2L, R1L);
BOOST_CHECK( ZeroL < OneL ); BOOST_CHECK_LT(ZeroL, OneL);
BOOST_CHECK( OneL < MaxL ); BOOST_CHECK_LT(OneL, MaxL);
BOOST_CHECK( R1L < MaxL ); BOOST_CHECK_LT(R1L, MaxL);
BOOST_CHECK( R2L < MaxL ); BOOST_CHECK_LT(R2L, MaxL);
uint160 LastS; uint160 LastS;
for (int i = 159; i >= 0; --i) { for (int i = 159; i >= 0; --i) {
uint160 TmpS; uint160 TmpS;
*(TmpS.begin() + (i>>3)) |= 1<<(7-(i&7)); *(TmpS.begin() + (i>>3)) |= 1<<(7-(i&7));
BOOST_CHECK( LastS < TmpS ); BOOST_CHECK_LT(LastS, TmpS);
LastS = TmpS; LastS = TmpS;
} }
BOOST_CHECK( ZeroS < R1S ); BOOST_CHECK_LT(ZeroS, R1S);
BOOST_CHECK( R2S < R1S ); BOOST_CHECK_LT(R2S, R1S);
BOOST_CHECK( ZeroS < OneS ); BOOST_CHECK_LT(ZeroS, OneS);
BOOST_CHECK( OneS < MaxS ); BOOST_CHECK_LT(OneS, MaxS);
BOOST_CHECK( R1S < MaxS ); BOOST_CHECK_LT(R1S, MaxS);
BOOST_CHECK( R2S < MaxS ); BOOST_CHECK_LT(R2S, MaxS);
// Verify hex strings are little-endian
BOOST_CHECK_LT(uint256S("2000000000000000000000000000000000000000000000000000000000000001"),
uint256S("1000000000000000000000000000000000000000000000000000000000000002"));
} }
BOOST_AUTO_TEST_CASE( methods ) // GetHex SetHex begin() end() size() GetLow64 GetSerializeSize, Serialize, Unserialize BOOST_AUTO_TEST_CASE( methods ) // GetHex SetHex begin() end() size() GetLow64 GetSerializeSize, Serialize, Unserialize
{ {
BOOST_CHECK(R1L.GetHex() == R1L.ToString()); BOOST_CHECK_EQUAL(R1L.GetHex(), R1L.ToString());
BOOST_CHECK(R2L.GetHex() == R2L.ToString()); BOOST_CHECK_EQUAL(R2L.GetHex(), R2L.ToString());
BOOST_CHECK(OneL.GetHex() == OneL.ToString()); BOOST_CHECK_EQUAL(OneL.GetHex(), OneL.ToString());
BOOST_CHECK(MaxL.GetHex() == MaxL.ToString()); BOOST_CHECK_EQUAL(MaxL.GetHex(), MaxL.ToString());
uint256 TmpL(R1L); uint256 TmpL(R1L);
BOOST_CHECK(TmpL == R1L); BOOST_CHECK_EQUAL(TmpL, R1L);
TmpL.SetHex(R2L.ToString()); BOOST_CHECK(TmpL == R2L); // Verify previous values don't persist when setting to truncated string.
TmpL.SetHex(ZeroL.ToString()); BOOST_CHECK(TmpL == uint256()); TmpL.SetHex("21");
BOOST_CHECK_EQUAL(TmpL.ToString(), "0000000000000000000000000000000000000000000000000000000000000021");
TmpL.SetHex(R2L.ToString()); BOOST_CHECK_EQUAL(TmpL, R2L);
TmpL.SetHex(ZeroL.ToString()); BOOST_CHECK_EQUAL(TmpL, uint256());
TmpL.SetHex(R1L.ToString()); TmpL.SetHex(R1L.ToString());
BOOST_CHECK(memcmp(R1L.begin(), R1Array, 32)==0); BOOST_CHECK_EQUAL_COLLECTIONS(R1L.begin(), R1L.end(), R1Array, R1Array + R1L.size());
BOOST_CHECK(memcmp(TmpL.begin(), R1Array, 32)==0); BOOST_CHECK_EQUAL_COLLECTIONS(TmpL.begin(), TmpL.end(), R1Array, R1Array + TmpL.size());
BOOST_CHECK(memcmp(R2L.begin(), R2Array, 32)==0); BOOST_CHECK_EQUAL_COLLECTIONS(R2L.begin(), R2L.end(), R2Array, R2Array + R2L.size());
BOOST_CHECK(memcmp(ZeroL.begin(), ZeroArray, 32)==0); BOOST_CHECK_EQUAL_COLLECTIONS(ZeroL.begin(), ZeroL.end(), ZeroArray, ZeroArray + ZeroL.size());
BOOST_CHECK(memcmp(OneL.begin(), OneArray, 32)==0); BOOST_CHECK_EQUAL_COLLECTIONS(OneL.begin(), OneL.end(), OneArray, OneArray + OneL.size());
BOOST_CHECK(R1L.size() == sizeof(R1L)); BOOST_CHECK_EQUAL(R1L.size(), sizeof(R1L));
BOOST_CHECK(sizeof(R1L) == 32); BOOST_CHECK_EQUAL(sizeof(R1L), 32);
BOOST_CHECK(R1L.size() == 32); BOOST_CHECK_EQUAL(R1L.size(), 32);
BOOST_CHECK(R2L.size() == 32); BOOST_CHECK_EQUAL(R2L.size(), 32);
BOOST_CHECK(ZeroL.size() == 32); BOOST_CHECK_EQUAL(ZeroL.size(), 32);
BOOST_CHECK(MaxL.size() == 32); BOOST_CHECK_EQUAL(MaxL.size(), 32);
BOOST_CHECK(R1L.begin() + 32 == R1L.end()); BOOST_CHECK_EQUAL(R1L.begin() + 32, R1L.end());
BOOST_CHECK(R2L.begin() + 32 == R2L.end()); BOOST_CHECK_EQUAL(R2L.begin() + 32, R2L.end());
BOOST_CHECK(OneL.begin() + 32 == OneL.end()); BOOST_CHECK_EQUAL(OneL.begin() + 32, OneL.end());
BOOST_CHECK(MaxL.begin() + 32 == MaxL.end()); BOOST_CHECK_EQUAL(MaxL.begin() + 32, MaxL.end());
BOOST_CHECK(TmpL.begin() + 32 == TmpL.end()); BOOST_CHECK_EQUAL(TmpL.begin() + 32, TmpL.end());
BOOST_CHECK(GetSerializeSize(R1L) == 32); BOOST_CHECK_EQUAL(GetSerializeSize(R1L), 32);
BOOST_CHECK(GetSerializeSize(ZeroL) == 32); BOOST_CHECK_EQUAL(GetSerializeSize(ZeroL), 32);
DataStream ss{}; DataStream ss{};
ss << R1L; ss << R1L;
BOOST_CHECK(ss.str() == std::string(R1Array,R1Array+32)); BOOST_CHECK_EQUAL(ss.str(), std::string(R1Array,R1Array+32));
ss >> TmpL; ss >> TmpL;
BOOST_CHECK(R1L == TmpL); BOOST_CHECK_EQUAL(R1L, TmpL);
ss.clear(); ss.clear();
ss << ZeroL; ss << ZeroL;
BOOST_CHECK(ss.str() == std::string(ZeroArray,ZeroArray+32)); BOOST_CHECK_EQUAL(ss.str(), std::string(ZeroArray,ZeroArray+32));
ss >> TmpL; ss >> TmpL;
BOOST_CHECK(ZeroL == TmpL); BOOST_CHECK_EQUAL(ZeroL, TmpL);
ss.clear(); ss.clear();
ss << MaxL; ss << MaxL;
BOOST_CHECK(ss.str() == std::string(MaxArray,MaxArray+32)); BOOST_CHECK_EQUAL(ss.str(), std::string(MaxArray,MaxArray+32));
ss >> TmpL; ss >> TmpL;
BOOST_CHECK(MaxL == TmpL); BOOST_CHECK_EQUAL(MaxL, TmpL);
ss.clear(); ss.clear();
BOOST_CHECK(R1S.GetHex() == R1S.ToString()); BOOST_CHECK_EQUAL(R1S.GetHex(), R1S.ToString());
BOOST_CHECK(R2S.GetHex() == R2S.ToString()); BOOST_CHECK_EQUAL(R2S.GetHex(), R2S.ToString());
BOOST_CHECK(OneS.GetHex() == OneS.ToString()); BOOST_CHECK_EQUAL(OneS.GetHex(), OneS.ToString());
BOOST_CHECK(MaxS.GetHex() == MaxS.ToString()); BOOST_CHECK_EQUAL(MaxS.GetHex(), MaxS.ToString());
uint160 TmpS(R1S); uint160 TmpS(R1S);
BOOST_CHECK(TmpS == R1S); BOOST_CHECK_EQUAL(TmpS, R1S);
TmpS.SetHex(R2S.ToString()); BOOST_CHECK(TmpS == R2S); TmpS.SetHex(R2S.ToString()); BOOST_CHECK_EQUAL(TmpS, R2S);
TmpS.SetHex(ZeroS.ToString()); BOOST_CHECK(TmpS == uint160()); TmpS.SetHex(ZeroS.ToString()); BOOST_CHECK_EQUAL(TmpS, uint160());
TmpS.SetHex(R1S.ToString()); TmpS.SetHex(R1S.ToString());
BOOST_CHECK(memcmp(R1S.begin(), R1Array, 20)==0); BOOST_CHECK_EQUAL_COLLECTIONS(R1S.begin(), R1S.end(), R1Array, R1Array + R1S.size());
BOOST_CHECK(memcmp(TmpS.begin(), R1Array, 20)==0); BOOST_CHECK_EQUAL_COLLECTIONS(TmpS.begin(), TmpS.end(), R1Array, R1Array + TmpS.size());
BOOST_CHECK(memcmp(R2S.begin(), R2Array, 20)==0); BOOST_CHECK_EQUAL_COLLECTIONS(R2S.begin(), R2S.end(), R2Array, R2Array + R2S.size());
BOOST_CHECK(memcmp(ZeroS.begin(), ZeroArray, 20)==0); BOOST_CHECK_EQUAL_COLLECTIONS(ZeroS.begin(), ZeroS.end(), ZeroArray, ZeroArray + ZeroS.size());
BOOST_CHECK(memcmp(OneS.begin(), OneArray, 20)==0); BOOST_CHECK_EQUAL_COLLECTIONS(OneS.begin(), OneS.end(), OneArray, OneArray + OneS.size());
BOOST_CHECK(R1S.size() == sizeof(R1S)); BOOST_CHECK_EQUAL(R1S.size(), sizeof(R1S));
BOOST_CHECK(sizeof(R1S) == 20); BOOST_CHECK_EQUAL(sizeof(R1S), 20);
BOOST_CHECK(R1S.size() == 20); BOOST_CHECK_EQUAL(R1S.size(), 20);
BOOST_CHECK(R2S.size() == 20); BOOST_CHECK_EQUAL(R2S.size(), 20);
BOOST_CHECK(ZeroS.size() == 20); BOOST_CHECK_EQUAL(ZeroS.size(), 20);
BOOST_CHECK(MaxS.size() == 20); BOOST_CHECK_EQUAL(MaxS.size(), 20);
BOOST_CHECK(R1S.begin() + 20 == R1S.end()); BOOST_CHECK_EQUAL(R1S.begin() + 20, R1S.end());
BOOST_CHECK(R2S.begin() + 20 == R2S.end()); BOOST_CHECK_EQUAL(R2S.begin() + 20, R2S.end());
BOOST_CHECK(OneS.begin() + 20 == OneS.end()); BOOST_CHECK_EQUAL(OneS.begin() + 20, OneS.end());
BOOST_CHECK(MaxS.begin() + 20 == MaxS.end()); BOOST_CHECK_EQUAL(MaxS.begin() + 20, MaxS.end());
BOOST_CHECK(TmpS.begin() + 20 == TmpS.end()); BOOST_CHECK_EQUAL(TmpS.begin() + 20, TmpS.end());
BOOST_CHECK(GetSerializeSize(R1S) == 20); BOOST_CHECK_EQUAL(GetSerializeSize(R1S), 20);
BOOST_CHECK(GetSerializeSize(ZeroS) == 20); BOOST_CHECK_EQUAL(GetSerializeSize(ZeroS), 20);
ss << R1S; ss << R1S;
BOOST_CHECK(ss.str() == std::string(R1Array,R1Array+20)); BOOST_CHECK_EQUAL(ss.str(), std::string(R1Array,R1Array+20));
ss >> TmpS; ss >> TmpS;
BOOST_CHECK(R1S == TmpS); BOOST_CHECK_EQUAL(R1S, TmpS);
ss.clear(); ss.clear();
ss << ZeroS; ss << ZeroS;
BOOST_CHECK(ss.str() == std::string(ZeroArray,ZeroArray+20)); BOOST_CHECK_EQUAL(ss.str(), std::string(ZeroArray,ZeroArray+20));
ss >> TmpS; ss >> TmpS;
BOOST_CHECK(ZeroS == TmpS); BOOST_CHECK_EQUAL(ZeroS, TmpS);
ss.clear(); ss.clear();
ss << MaxS; ss << MaxS;
BOOST_CHECK(ss.str() == std::string(MaxArray,MaxArray+20)); BOOST_CHECK_EQUAL(ss.str(), std::string(MaxArray,MaxArray+20));
ss >> TmpS; ss >> TmpS;
BOOST_CHECK(MaxS == TmpS); BOOST_CHECK_EQUAL(MaxS, TmpS);
ss.clear(); ss.clear();
} }
BOOST_AUTO_TEST_CASE( conversion ) BOOST_AUTO_TEST_CASE( conversion )
{ {
BOOST_CHECK(ArithToUint256(UintToArith256(ZeroL)) == ZeroL); BOOST_CHECK_EQUAL(ArithToUint256(UintToArith256(ZeroL)), ZeroL);
BOOST_CHECK(ArithToUint256(UintToArith256(OneL)) == OneL); BOOST_CHECK_EQUAL(ArithToUint256(UintToArith256(OneL)), OneL);
BOOST_CHECK(ArithToUint256(UintToArith256(R1L)) == R1L); BOOST_CHECK_EQUAL(ArithToUint256(UintToArith256(R1L)), R1L);
BOOST_CHECK(ArithToUint256(UintToArith256(R2L)) == R2L); BOOST_CHECK_EQUAL(ArithToUint256(UintToArith256(R2L)), R2L);
BOOST_CHECK(UintToArith256(ZeroL) == 0); BOOST_CHECK_EQUAL(UintToArith256(ZeroL), 0);
BOOST_CHECK(UintToArith256(OneL) == 1); BOOST_CHECK_EQUAL(UintToArith256(OneL), 1);
BOOST_CHECK(ArithToUint256(0) == ZeroL); BOOST_CHECK_EQUAL(ArithToUint256(0), ZeroL);
BOOST_CHECK(ArithToUint256(1) == OneL); BOOST_CHECK_EQUAL(ArithToUint256(1), OneL);
BOOST_CHECK(arith_uint256(UintToArith256(uint256S(R1L.GetHex()))) == UintToArith256(R1L)); BOOST_CHECK_EQUAL(arith_uint256(UintToArith256(uint256S(R1L.GetHex()))), UintToArith256(R1L));
BOOST_CHECK(arith_uint256(UintToArith256(uint256S(R2L.GetHex()))) == UintToArith256(R2L)); BOOST_CHECK_EQUAL(arith_uint256(UintToArith256(uint256S(R2L.GetHex()))), UintToArith256(R2L));
BOOST_CHECK(R1L.GetHex() == UintToArith256(R1L).GetHex()); BOOST_CHECK_EQUAL(R1L.GetHex(), UintToArith256(R1L).GetHex());
BOOST_CHECK(R2L.GetHex() == UintToArith256(R2L).GetHex()); BOOST_CHECK_EQUAL(R2L.GetHex(), UintToArith256(R2L).GetHex());
} }
BOOST_AUTO_TEST_CASE( operator_with_self ) BOOST_AUTO_TEST_CASE( operator_with_self )
@ -285,13 +290,13 @@ BOOST_AUTO_TEST_CASE( operator_with_self )
#endif #endif
arith_uint256 v = UintToArith256(uint256S("02")); arith_uint256 v = UintToArith256(uint256S("02"));
v *= v; v *= v;
BOOST_CHECK(v == UintToArith256(uint256S("04"))); BOOST_CHECK_EQUAL(v, UintToArith256(uint256S("04")));
v /= v; v /= v;
BOOST_CHECK(v == UintToArith256(uint256S("01"))); BOOST_CHECK_EQUAL(v, UintToArith256(uint256S("01")));
v += v; v += v;
BOOST_CHECK(v == UintToArith256(uint256S("02"))); BOOST_CHECK_EQUAL(v, UintToArith256(uint256S("02")));
v -= v; v -= v;
BOOST_CHECK(v == UintToArith256(uint256S("0"))); BOOST_CHECK_EQUAL(v, UintToArith256(uint256S("0")));
#if defined(__clang__) #if defined(__clang__)
# pragma clang diagnostic pop # pragma clang diagnostic pop
#endif #endif

View file

@ -79,6 +79,18 @@ const std::function<std::string(const char*)> G_TRANSLATION_FUN = nullptr;
/** Random context to get unique temp data dirs. Separate from g_insecure_rand_ctx, which can be seeded from a const env var */ /** Random context to get unique temp data dirs. Separate from g_insecure_rand_ctx, which can be seeded from a const env var */
static FastRandomContext g_insecure_rand_ctx_temp_path; static FastRandomContext g_insecure_rand_ctx_temp_path;
std::ostream& operator<<(std::ostream& os, const arith_uint256& num)
{
os << ArithToUint256(num).ToString();
return os;
}
std::ostream& operator<<(std::ostream& os, const uint160& num)
{
os << num.ToString();
return os;
}
std::ostream& operator<<(std::ostream& os, const uint256& num) std::ostream& operator<<(std::ostream& os, const uint256& num)
{ {
os << num.ToString(); os << num.ToString();

View file

@ -24,6 +24,7 @@
#include <type_traits> #include <type_traits>
#include <vector> #include <vector>
class arith_uint256;
class CFeeRate; class CFeeRate;
class Chainstate; class Chainstate;
class FastRandomContext; class FastRandomContext;
@ -236,7 +237,9 @@ std::unique_ptr<T> MakeNoLogFileContext(const ChainType chain_type = ChainType::
CBlock getBlock13b8a(); CBlock getBlock13b8a();
// define an implicit conversion here so that uint256 may be used directly in BOOST_CHECK_* // Make types usable in BOOST_CHECK_*
std::ostream& operator<<(std::ostream& os, const arith_uint256& num);
std::ostream& operator<<(std::ostream& os, const uint160& num);
std::ostream& operator<<(std::ostream& os, const uint256& num); std::ostream& operator<<(std::ostream& os, const uint256& num);
/** /**

View file

@ -18,39 +18,31 @@ std::string base_blob<BITS>::GetHex() const
} }
template <unsigned int BITS> template <unsigned int BITS>
void base_blob<BITS>::SetHex(const char* psz) void base_blob<BITS>::SetHex(const std::string_view str)
{ {
std::fill(m_data.begin(), m_data.end(), 0); std::fill(m_data.begin(), m_data.end(), 0);
// skip leading spaces const auto trimmed = util::RemovePrefixView(util::TrimStringView(str), "0x");
while (IsSpace(*psz))
psz++;
// skip 0x // Note: if we are passed a greater number of digits than would fit as bytes
if (psz[0] == '0' && ToLower(psz[1]) == 'x') // in m_data, we will be discarding the leftmost ones.
psz += 2; // str="12bc" in a WIDTH=1 m_data => m_data[] == "\0xbc", not "0x12".
// hex string to uint
size_t digits = 0; size_t digits = 0;
while (::HexDigit(psz[digits]) != -1) for (const char c : trimmed) {
digits++; if (::HexDigit(c) == -1) break;
++digits;
}
unsigned char* p1 = m_data.data(); unsigned char* p1 = m_data.data();
unsigned char* pend = p1 + WIDTH; unsigned char* pend = p1 + WIDTH;
while (digits > 0 && p1 < pend) { while (digits > 0 && p1 < pend) {
*p1 = ::HexDigit(psz[--digits]); *p1 = ::HexDigit(trimmed[--digits]);
if (digits > 0) { if (digits > 0) {
*p1 |= ((unsigned char)::HexDigit(psz[--digits]) << 4); *p1 |= ((unsigned char)::HexDigit(trimmed[--digits]) << 4);
p1++; p1++;
} }
} }
} }
template <unsigned int BITS>
void base_blob<BITS>::SetHex(const std::string& str)
{
SetHex(str.c_str());
}
template <unsigned int BITS> template <unsigned int BITS>
std::string base_blob<BITS>::ToString() const std::string base_blob<BITS>::ToString() const
{ {
@ -60,14 +52,12 @@ std::string base_blob<BITS>::ToString() const
// Explicit instantiations for base_blob<160> // Explicit instantiations for base_blob<160>
template std::string base_blob<160>::GetHex() const; template std::string base_blob<160>::GetHex() const;
template std::string base_blob<160>::ToString() const; template std::string base_blob<160>::ToString() const;
template void base_blob<160>::SetHex(const char*); template void base_blob<160>::SetHex(std::string_view);
template void base_blob<160>::SetHex(const std::string&);
// Explicit instantiations for base_blob<256> // Explicit instantiations for base_blob<256>
template std::string base_blob<256>::GetHex() const; template std::string base_blob<256>::GetHex() const;
template std::string base_blob<256>::ToString() const; template std::string base_blob<256>::ToString() const;
template void base_blob<256>::SetHex(const char*); template void base_blob<256>::SetHex(std::string_view);
template void base_blob<256>::SetHex(const std::string&);
const uint256 uint256::ZERO(0); const uint256 uint256::ZERO(0);
const uint256 uint256::ONE(1); const uint256 uint256::ONE(1);

View file

@ -57,9 +57,9 @@ public:
friend constexpr bool operator!=(const base_blob& a, const base_blob& b) { return a.Compare(b) != 0; } friend constexpr bool operator!=(const base_blob& a, const base_blob& b) { return a.Compare(b) != 0; }
friend constexpr bool operator<(const base_blob& a, const base_blob& b) { return a.Compare(b) < 0; } friend constexpr bool operator<(const base_blob& a, const base_blob& b) { return a.Compare(b) < 0; }
// Hex string representations are little-endian.
std::string GetHex() const; std::string GetHex() const;
void SetHex(const char* psz); void SetHex(std::string_view str);
void SetHex(const std::string& str);
std::string ToString() const; std::string ToString() const;
constexpr const unsigned char* data() const { return m_data.data(); } constexpr const unsigned char* data() const { return m_data.data(); }
@ -112,21 +112,11 @@ public:
static const uint256 ONE; static const uint256 ONE;
}; };
/* uint256 from const char *. /* uint256 from std::string_view, treated as little-endian.
* This is a separate function because the constructor uint256(const char*) can result * This is not a uint256 constructor because of historical fears of uint256(0)
* in dangerously catching uint256(0). * resolving to a NULL string and crashing.
*/ */
inline uint256 uint256S(const char *str) inline uint256 uint256S(std::string_view str)
{
uint256 rv;
rv.SetHex(str);
return rv;
}
/* uint256 from std::string.
* This is a separate function because the constructor uint256(const std::string &str) can result
* in dangerously catching uint256(0) via std::string(const char*).
*/
inline uint256 uint256S(const std::string& str)
{ {
uint256 rv; uint256 rv;
rv.SetHex(str); rv.SetHex(str);

View file

@ -68,7 +68,7 @@ using Wtxid = transaction_identifier<true>;
inline Txid TxidFromString(std::string_view str) inline Txid TxidFromString(std::string_view str)
{ {
return Txid::FromUint256(uint256S(str.data())); return Txid::FromUint256(uint256S(str));
} }
#endif // BITCOIN_UTIL_TRANSACTION_IDENTIFIER_H #endif // BITCOIN_UTIL_TRANSACTION_IDENTIFIER_H