/root/bitcoin/src/primitives/transaction_identifier.h
Line | Count | Source |
1 | | // Copyright (c) 2023-present The Bitcoin Core developers |
2 | | // Distributed under the MIT software license, see the accompanying |
3 | | // file COPYING or https://opensource.org/license/mit. |
4 | | |
5 | | #ifndef BITCOIN_PRIMITIVES_TRANSACTION_IDENTIFIER_H |
6 | | #define BITCOIN_PRIMITIVES_TRANSACTION_IDENTIFIER_H |
7 | | |
8 | | #include <attributes.h> |
9 | | #include <uint256.h> |
10 | | #include <util/types.h> |
11 | | |
12 | | #include <compare> |
13 | | #include <cstddef> |
14 | | #include <optional> |
15 | | #include <string> |
16 | | #include <string_view> |
17 | | #include <tuple> |
18 | | #include <type_traits> |
19 | | #include <variant> |
20 | | |
21 | | /** transaction_identifier represents the two canonical transaction identifier |
22 | | * types (txid, wtxid).*/ |
23 | | template <bool has_witness> |
24 | | class transaction_identifier |
25 | | { |
26 | | uint256 m_wrapped; |
27 | | |
28 | | // Note: Use FromUint256 externally instead. |
29 | 444M | transaction_identifier(const uint256& wrapped) : m_wrapped{wrapped} {}transaction_identifier<false>::transaction_identifier(uint256 const&) Line | Count | Source | 29 | 212M | transaction_identifier(const uint256& wrapped) : m_wrapped{wrapped} {} |
transaction_identifier<true>::transaction_identifier(uint256 const&) Line | Count | Source | 29 | 232M | transaction_identifier(const uint256& wrapped) : m_wrapped{wrapped} {} |
|
30 | | |
31 | 1.37G | constexpr int Compare(const transaction_identifier<has_witness>& other) const { return m_wrapped.Compare(other.m_wrapped); }transaction_identifier<false>::Compare(transaction_identifier<false> const&) const Line | Count | Source | 31 | 1.15G | constexpr int Compare(const transaction_identifier<has_witness>& other) const { return m_wrapped.Compare(other.m_wrapped); } |
transaction_identifier<true>::Compare(transaction_identifier<true> const&) const Line | Count | Source | 31 | 221M | constexpr int Compare(const transaction_identifier<has_witness>& other) const { return m_wrapped.Compare(other.m_wrapped); } |
|
32 | | template <typename Other> |
33 | | constexpr int Compare(const Other& other) const |
34 | | { |
35 | | static_assert(ALWAYS_FALSE<Other>, "Forbidden comparison type"); |
36 | | return 0; |
37 | | } |
38 | | |
39 | | public: |
40 | 301M | transaction_identifier() : m_wrapped{} {}transaction_identifier<false>::transaction_identifier() Line | Count | Source | 40 | 289M | transaction_identifier() : m_wrapped{} {} |
transaction_identifier<true>::transaction_identifier() Line | Count | Source | 40 | 12.1M | transaction_identifier() : m_wrapped{} {} |
|
41 | | consteval explicit transaction_identifier(std::string_view hex_str) : m_wrapped{uint256{hex_str}} {} |
42 | | |
43 | | template <typename Other> |
44 | 703M | bool operator==(const Other& other) const { return Compare(other) == 0; }bool transaction_identifier<false>::operator==<transaction_identifier<false>>(transaction_identifier<false> const&) const Line | Count | Source | 44 | 596M | bool operator==(const Other& other) const { return Compare(other) == 0; } |
bool transaction_identifier<true>::operator==<transaction_identifier<true>>(transaction_identifier<true> const&) const Line | Count | Source | 44 | 106M | bool operator==(const Other& other) const { return Compare(other) == 0; } |
|
45 | | template <typename Other> |
46 | 673M | std::strong_ordering operator<=>(const Other& other) const { return Compare(other) <=> 0; }std::strong_ordering transaction_identifier<false>::operator<=><transaction_identifier<false>>(transaction_identifier<false> const&) const Line | Count | Source | 46 | 558M | std::strong_ordering operator<=>(const Other& other) const { return Compare(other) <=> 0; } |
std::strong_ordering transaction_identifier<true>::operator<=><transaction_identifier<true>>(transaction_identifier<true> const&) const Line | Count | Source | 46 | 114M | std::strong_ordering operator<=>(const Other& other) const { return Compare(other) <=> 0; } |
|
47 | | |
48 | 2.02G | const uint256& ToUint256() const LIFETIMEBOUND { return m_wrapped; }transaction_identifier<false>::ToUint256() const Line | Count | Source | 48 | 1.76G | const uint256& ToUint256() const LIFETIMEBOUND { return m_wrapped; } |
transaction_identifier<true>::ToUint256() const Line | Count | Source | 48 | 267M | const uint256& ToUint256() const LIFETIMEBOUND { return m_wrapped; } |
|
49 | 444M | static transaction_identifier FromUint256(const uint256& id) { return {id}; }transaction_identifier<false>::FromUint256(uint256 const&) Line | Count | Source | 49 | 212M | static transaction_identifier FromUint256(const uint256& id) { return {id}; } |
transaction_identifier<true>::FromUint256(uint256 const&) Line | Count | Source | 49 | 232M | static transaction_identifier FromUint256(const uint256& id) { return {id}; } |
|
50 | | |
51 | | /** Wrapped `uint256` methods. */ |
52 | 653M | constexpr bool IsNull() const { return m_wrapped.IsNull(); } |
53 | 36.8M | constexpr void SetNull() { m_wrapped.SetNull(); } |
54 | | static std::optional<transaction_identifier> FromHex(std::string_view hex) |
55 | 0 | { |
56 | 0 | auto u{uint256::FromHex(hex)}; |
57 | 0 | if (!u) return std::nullopt; |
58 | 0 | return FromUint256(*u); |
59 | 0 | } Unexecuted instantiation: transaction_identifier<false>::FromHex(std::basic_string_view<char, std::char_traits<char>>) Unexecuted instantiation: transaction_identifier<true>::FromHex(std::basic_string_view<char, std::char_traits<char>>) |
60 | 0 | std::string GetHex() const { return m_wrapped.GetHex(); }Unexecuted instantiation: transaction_identifier<false>::GetHex[abi:cxx11]() const Unexecuted instantiation: transaction_identifier<true>::GetHex[abi:cxx11]() const |
61 | 10.9M | std::string ToString() const { return m_wrapped.ToString(); }transaction_identifier<false>::ToString[abi:cxx11]() const Line | Count | Source | 61 | 6.05M | std::string ToString() const { return m_wrapped.ToString(); } |
transaction_identifier<true>::ToString[abi:cxx11]() const Line | Count | Source | 61 | 4.90M | std::string ToString() const { return m_wrapped.ToString(); } |
|
62 | | static constexpr auto size() { return decltype(m_wrapped)::size(); } |
63 | | constexpr const std::byte* data() const { return reinterpret_cast<const std::byte*>(m_wrapped.data()); } |
64 | 4.10M | constexpr const std::byte* begin() const { return reinterpret_cast<const std::byte*>(m_wrapped.begin()); }transaction_identifier<true>::begin() const Line | Count | Source | 64 | 4.10M | constexpr const std::byte* begin() const { return reinterpret_cast<const std::byte*>(m_wrapped.begin()); } |
Unexecuted instantiation: transaction_identifier<false>::begin() const |
65 | 1.80M | constexpr const std::byte* end() const { return reinterpret_cast<const std::byte*>(m_wrapped.end()); } |
66 | 971M | template <typename Stream> void Serialize(Stream& s) const { m_wrapped.Serialize(s); }void transaction_identifier<false>::Serialize<ParamsStream<SizeComputer&, TransactionSerParams>>(ParamsStream<SizeComputer&, TransactionSerParams>&) const Line | Count | Source | 66 | 443M | template <typename Stream> void Serialize(Stream& s) const { m_wrapped.Serialize(s); } |
void transaction_identifier<false>::Serialize<ParamsStream<VectorWriter&, TransactionSerParams>>(ParamsStream<VectorWriter&, TransactionSerParams>&) const Line | Count | Source | 66 | 22.5M | template <typename Stream> void Serialize(Stream& s) const { m_wrapped.Serialize(s); } |
Unexecuted instantiation: void transaction_identifier<false>::Serialize<ParamsStream<DataStream&, TransactionSerParams>>(ParamsStream<DataStream&, TransactionSerParams>&) const void transaction_identifier<false>::Serialize<DataStream>(DataStream&) const Line | Count | Source | 66 | 120M | template <typename Stream> void Serialize(Stream& s) const { m_wrapped.Serialize(s); } |
Unexecuted instantiation: void transaction_identifier<false>::Serialize<ParamsStream<ParamsStream<VectorWriter&, TransactionSerParams>&, TransactionSerParams>>(ParamsStream<ParamsStream<VectorWriter&, TransactionSerParams>&, TransactionSerParams>&) const Unexecuted instantiation: void transaction_identifier<false>::Serialize<AutoFile>(AutoFile&) const void transaction_identifier<false>::Serialize<HashWriter>(HashWriter&) const Line | Count | Source | 66 | 1.10M | template <typename Stream> void Serialize(Stream& s) const { m_wrapped.Serialize(s); } |
void transaction_identifier<false>::Serialize<ParamsStream<BufferedWriter<AutoFile>&, TransactionSerParams>>(ParamsStream<BufferedWriter<AutoFile>&, TransactionSerParams>&) const Line | Count | Source | 66 | 30.3M | template <typename Stream> void Serialize(Stream& s) const { m_wrapped.Serialize(s); } |
Unexecuted instantiation: void transaction_identifier<false>::Serialize<ParamsStream<AutoFile&, TransactionSerParams>>(ParamsStream<AutoFile&, TransactionSerParams>&) const void transaction_identifier<true>::Serialize<HashWriter>(HashWriter&) const Line | Count | Source | 66 | 1.20M | template <typename Stream> void Serialize(Stream& s) const { m_wrapped.Serialize(s); } |
void transaction_identifier<false>::Serialize<ParamsStream<HashWriter&, TransactionSerParams>>(ParamsStream<HashWriter&, TransactionSerParams>&) const Line | Count | Source | 66 | 350M | template <typename Stream> void Serialize(Stream& s) const { m_wrapped.Serialize(s); } |
|
67 | 21.6M | template <typename Stream> void Unserialize(Stream& s) { m_wrapped.Unserialize(s); }void transaction_identifier<false>::Unserialize<ParamsStream<SpanReader&, TransactionSerParams>>(ParamsStream<SpanReader&, TransactionSerParams>&) Line | Count | Source | 67 | 248k | template <typename Stream> void Unserialize(Stream& s) { m_wrapped.Unserialize(s); } |
Unexecuted instantiation: void transaction_identifier<false>::Unserialize<SpanReader>(SpanReader&) Unexecuted instantiation: void transaction_identifier<false>::Unserialize<DataStream>(DataStream&) void transaction_identifier<false>::Unserialize<ParamsStream<DataStream&, TransactionSerParams>>(ParamsStream<DataStream&, TransactionSerParams>&) Line | Count | Source | 67 | 21.4M | template <typename Stream> void Unserialize(Stream& s) { m_wrapped.Unserialize(s); } |
Unexecuted instantiation: void transaction_identifier<false>::Unserialize<ParamsStream<AutoFile&, TransactionSerParams>>(ParamsStream<AutoFile&, TransactionSerParams>&) Unexecuted instantiation: void transaction_identifier<false>::Unserialize<AutoFile>(AutoFile&) Unexecuted instantiation: void transaction_identifier<false>::Unserialize<ParamsStream<BufferedFile&, TransactionSerParams>>(ParamsStream<BufferedFile&, TransactionSerParams>&) |
68 | | }; |
69 | | |
70 | | /** Txid commits to all transaction fields except the witness. */ |
71 | | using Txid = transaction_identifier<false>; |
72 | | /** Wtxid commits to all transaction fields including the witness. */ |
73 | | using Wtxid = transaction_identifier<true>; |
74 | | |
75 | | template <typename T> |
76 | | concept TxidOrWtxid = std::is_same_v<T, Txid> || std::is_same_v<T, Wtxid>; |
77 | | |
78 | | class GenTxid : public std::variant<Txid, Wtxid> |
79 | | { |
80 | | public: |
81 | | using variant::variant; |
82 | | |
83 | 5.90M | bool IsWtxid() const { return std::holds_alternative<Wtxid>(*this); } |
84 | | |
85 | | const uint256& ToUint256() const LIFETIMEBOUND |
86 | 263M | { |
87 | 263M | return std::visit([](const auto& id) -> const uint256& { return id.ToUint256(); }, *this);uint256 const& GenTxid::ToUint256() const::'lambda'(auto const&)::operator()<transaction_identifier<false>>(auto const&) const Line | Count | Source | 87 | 243M | return std::visit([](const auto& id) -> const uint256& { return id.ToUint256(); }, *this); |
uint256 const& GenTxid::ToUint256() const::'lambda'(auto const&)::operator()<transaction_identifier<true>>(auto const&) const Line | Count | Source | 87 | 20.2M | return std::visit([](const auto& id) -> const uint256& { return id.ToUint256(); }, *this); |
|
88 | 263M | } |
89 | | |
90 | | friend auto operator<=>(const GenTxid& a, const GenTxid& b) |
91 | 2.60M | { |
92 | | // Use a reference for read-only access to the hash, avoiding a copy that might not be optimized away. |
93 | 2.60M | return std::tuple<bool, const uint256&>(a.IsWtxid(), a.ToUint256()) <=> std::tuple<bool, const uint256&>(b.IsWtxid(), b.ToUint256()); |
94 | 2.60M | } |
95 | | }; |
96 | | |
97 | | #endif // BITCOIN_PRIMITIVES_TRANSACTION_IDENTIFIER_H |