refactor: remove CBlockIndex copy construction #25311

pull jamesob wants to merge 1 commits into bitcoin:master from jamesob:2022-06-cblockindex-delete-copy changing 3 files +31 −17
  1. jamesob commented at 4:10 pm on June 8, 2022: member

    Copy construction of CBlockIndex objects is a footgun because of the wide use of equality-by-pointer comparison in the code base. There are also potential lifetime confusions of using copied instances, since there are recursive pointer members (e.g. pprev).

    (See also #24008 (review))

    We can’t just delete the copy constructors because they are used for derived classes (CDiskBlockIndex), so we mark them protected.

  2. laanwj added the label UTXO Db and Indexes on Jun 8, 2022
  3. jamesob force-pushed on Jun 8, 2022
  4. jamesob commented at 5:01 pm on June 8, 2022: member

    This can be tested trivially by adding a construction-by-copy somewhere and verifying the compiler rejects it:

     0% make
     1Making all in src
     2make[1]: Entering directory '/home/james/src/bitcoin/src'
     3make[2]: Entering directory '/home/james/src/bitcoin/src'
     4make[3]: Entering directory '/home/james/src/bitcoin'
     5make[3]: Leaving directory '/home/james/src/bitcoin'
     6  CXX      libbitcoin_node_a-validation.o
     7  CXX      libbitcoin_util_a-clientversion.o
     8  AR       libbitcoin_util.a
     9  CXXLD    bitcoin-util
    10  CXXLD    bitcoin-wallet
    11  CXXLD    bitcoin-cli
    12  CXXLD    bitcoin-tx
    13validation.cpp:4170:29: error: calling a protected constructor of class 'CBlockIndex'
    14            auto newblock = CBlockIndex(*block);
    15                            ^
    16./chain.h:369:5: note: declared protected here
    17    CBlockIndex(const CBlockIndex&) = default;
    18    ^
    191 error generated.
    
  5. in src/test/fuzz/pow.cpp:60 in 28e019236b outdated
    56@@ -57,7 +57,7 @@ FUZZ_TARGET_INIT(pow, initialize_pow)
    57             } else {
    58                 current_block.nChainWork = ConsumeArithUInt256(fuzzed_data_provider);
    59             }
    60-            blocks.push_back(current_block);
    61+            blocks.push_back(&current_block);
    


    maflcko commented at 5:01 pm on June 8, 2022:
    Pretty sure this will later-on result in a use-after-free?

    jamesob commented at 5:40 pm on June 8, 2022:
    Argh, good catch. Thought I’d checked for that. Fixed.

    maflcko commented at 5:44 pm on June 8, 2022:
    Hmm, the CI fuzz run initially passed on this test: https://cirrus-ci.com/task/5646660507271168?logs=ci#L3420 . I wonder why it didn’t catch that.

    maflcko commented at 8:00 am on June 9, 2022:

    valgrind doesn’t catch this either. Looks like the compiler will just optimize this UB away and set all objects to the same pointer, even if they are different?

    This is mildly shocking, as it makes it impossible to catch this kind of bug with any tool.


    jamesob commented at 3:01 pm on June 9, 2022:
    That’s scary. Glad my fuzz-via-idiocy technique is yielding more novel info about C++ compilers.
  6. maflcko commented at 5:03 pm on June 8, 2022: member
    Concept ACK
  7. in src/chain.h:370 in 28e019236b outdated
    365+    //! instance.
    366+    //!
    367+    //! We declare these protected instead of simply deleting them so that
    368+    //! CDiskBlockIndex can reuse copy construction.
    369+    CBlockIndex(const CBlockIndex&) = default;
    370+    CBlockIndex& operator=(const CBlockIndex&) = default;
    


    jonatack commented at 5:09 pm on June 8, 2022:

    Consider using the rule of five? https://isocpp.github.io/CppCoreGuidelines/CppCoreGuidelines#c21-if-you-define-or-delete-any-copy-move-or-destructor-function-define-or-delete-them-all. Example in #22481.

    Not sure (haven’t tested), would it be better to define these separately in CBlockIndex (= delete) and CDiskBlockIndex (= default)?


    jamesob commented at 6:03 pm on June 8, 2022:

    Rule of 5

    Sounds good, fixed.

    Not sure (haven’t tested), would it be better to define these separately in CBlockIndex (= delete) and CDiskBlockIndex (= default)?

    This doesn’t work because to get a default copy constructor, the signature needs to look like T(const T&) = default;; because the CDiskBlockIndex constructor in question has a reference to a CBlockIndex pointer, we can’t make use of a default copy constructor for CDiskBlockIndex. If we define one, we have to spell out all members from CBlockIndex, which is brittle.


    ajtowns commented at 3:37 am on June 13, 2022:
    operator= isn’t needed as far as I can see; is there any reason to default rather than delete it?

    jamesob commented at 7:02 pm on September 7, 2022:

    operator= isn’t needed as far as I can see

    Good point, done.

  8. jonatack commented at 5:09 pm on June 8, 2022: contributor
    Concept ACK
  9. jamesob force-pushed on Jun 8, 2022
  10. jamesob force-pushed on Jun 8, 2022
  11. in src/chain.h:376 in 62f6e49b20 outdated
    371+    CBlockIndex(CBlockIndex&&) = delete;
    372+    CBlockIndex& operator=(CBlockIndex&&) = delete;
    373+
    374+public:
    375+    // Here to satisfy the "rule of 5," but must be public.
    376+    virtual ~CBlockIndex() = default;
    


    ryanofsky commented at 5:07 pm on June 9, 2022:

    In commit “remove CBlockIndex copy construction” (62f6e49b20162f7987a446dc3f20d6ebff61bf54)

    This looks like it is the only virtual method so it is going to add a vtable pointer to every CBlockIndex instance and waste memory. (I also don’t see a reason to declare a destructor here, “rule of 5” notwithstanding. If there is a reason, more helpful IMO to just say what the reason is instead of citing a 537 page guidelines document!)


    jamesob commented at 7:54 pm on June 9, 2022:
    Ha, well put. Pushed.
  12. ryanofsky changes_requested
  13. ryanofsky commented at 5:09 pm on June 9, 2022: contributor
    Conditional code review ACK 62f6e49b20162f7987a446dc3f20d6ebff61bf54 if the virtual method is dropped
  14. jamesob force-pushed on Jun 9, 2022
  15. jonatack commented at 3:06 pm on June 10, 2022: contributor

    IIUC correctly, the issue here is that of (or similar to) slicing: a pointer to a derived class implicitly converts to one to its public base class, and with copy operations this becomes a subtle footgun. Looking through various resources, two solutions appear to be:

    • disallow copying of the base class by deleting the copy operations or making them private or protected, and providing a clone() function if needed to allow a user to request a non-slicing copy, or
    • disallow converting a pointer to the derived class to a pointer to a base by making the base class a private or protected base

    This PR opts for the first solution, which seems fine. This approach is also suggested at the bottom of https://en.cppreference.com/w/cpp/language/rule_of_three and in the guideline “A polymorphic class should suppress public copy/move”.

    The last push removing the public virtual destructor in the base class involves some tradeoffs that might be discussed. Not declaring it reduces the size of CBlockIndex and CDiskBlockIndex from 152 to 144 and 184 to 176 bytes, respectively, a saving of 8 bytes due to not needing a hidden vptr class data member pointer (as well as avoiding a vtbl + pointer entry and dynamic dispatch in general). In most cases, those additional costs are considered relatively low, so declaring destructors virtual is a good practice as a safety mechanism for unintended polymorphic destruction (destructing derived class objects through pointers or references of base class types).

    Not declaring the virtual destructor is also only possible due to issues with the current design. Inherited non-virtual functions should never be redefined to avoid inconsistent behavior (Scott Meyers, Effective C++, Item 36), yet that is what we’re currently doing with GetBlockHash() and ToString(), either from oversight or as a hack to avoid making them virtual. Items 34 and 35 in that book suggest alternatives (Template Pattern/NVI Idiom, Strategy Patterns), but as a start it may be prudent and less footgunny for future reviewers and bug avoidance to consider keeping the explicit virtual public destructor in the base and making those two functions virtual in CBlockIndex and override in the derived CDiskBlockIndex.

    If the overhead of dynamic dispatch is unacceptable, maybe there is a better/safer design than the current one, but if we intend for CBlockIndex to be inherited from, consider if its destructor and redefined methods ought to be virtual. If CBlockIndex is not intended to be inherited from, it can be marked as final to prevent that in the first place, without imposing any other use restrictions on the class itself.

    I’m not an expert, so have been trying to understand this area better from researching it. A couple related resources:

    Scott Meyers, Effective C++, Item 8: Polymorphic base classes should declare virtual destructors. If a class has any virtual functions, it should have a virtual destructor. Classes not designed to be base classes or not designed to be used polymorphically should not declare virtual destructors.

    Stroustroup, https://www.stroustrup.com/C++11FAQ.html#default2, I strongly recommend that if you declare one of these five functions, you explicitly declare all. Think of copying, moving, and destruction as closely related operations, rather than individual operations that you can freely mix and match - you can specify arbitrary combinations, but only a few combinations make sense semantically.

    Herb Sutter, http://www.gotw.ca/publications/mill18.htm, Guideline 4: A base class destructor should be either public and virtual, or protected and nonvirtual. / Don’t derive from concrete classes. Or, as Scott Meyers puts it in Item 33 of More Effective C++, “Make non-leaf classes abstract.”

  16. maflcko commented at 3:14 pm on June 10, 2022: member

    IIUC correctly, the issue here is that of (or similar to) slicing: a pointer to a derived class implicitly converts to one to its public base class, and with copy operations this becomes a subtle footgun

    This may or may not be a problem, but was not the primary motivation I mentioned. The motivation is simply that creating an exact copy will (obviously) create a new memory location where the copy lives (aka pointer). This is problematic when the value of the pointer is used to check for equality/unequality of objects.

    For example,

    0bool IsTip(const CBlockIndex& b) { return &b == ::ActiveTip(); }
    1int main() {
    2 const CBlockIndex tip{*Assert(::ActiveTip())};
    3 Assert(IsTip(&tip));
    4}
    

    will fail.

  17. ryanofsky commented at 4:12 pm on June 10, 2022: contributor

    Appreciate the thorough response jonatack , and agree declaring a virtual destructor could prevent slicing issues if virtual methods were added in the future, and also agree that use of inheritance in CBlockIndex is pretty sketchy, with overridden nonvirtual methods being a sign of that.

    I may be biased because I tend to think of classes with virtual methods designed to be overridden as pretty different things from simple classes that just encapsulate some state, but I wouldn’t think it is a good tradeoff to make class destructors virtual by default just because there is a chance someone may inherit from the class and try to use it polymorphically in the future. I think whoever is doing this later should be able to spot problems and that in the worst case “class has virtual method but non-virtual destructor” compiler warnings should help them.

    It’s hard to think about future-bugs that don’t yet exist in a complex language like c++, and I think c++ guidelines are good for generating ideas about bugs that may happen. But after you decide to prevent a bug, it should be possible to say what the bug is without citing guidelines as an authority. It is easy to go in circles thinking about guidelines abstractly and I think it is better to keep things concrete.

  18. ryanofsky approved
  19. ryanofsky commented at 4:14 pm on June 10, 2022: contributor
    Code review ACK 0497d76a944182b739d9abfd963c032a7d1dbd8, only change since last review is CBlockIndex stops having a vtable again
  20. jonatack commented at 3:10 pm on June 11, 2022: contributor

    The last sentence of the commit message needs to be updated, otherwise tested ACK 0497d76a944182b739d9abfd963c032a7d1dbd89 code review, verified that this change fixes both slicing and the failing pointer equality check of an exact copy with the following diff partly derived from #25311 (comment):

     0--- a/src/test/validation_chainstatemanager_tests.cpp
     1+++ b/src/test/validation_chainstatemanager_tests.cpp
     2@@ -24,6 +24,23 @@ using node::SnapshotMetadata;
     3
     4+static bool IsTip(const CBlockIndex& b, const ChainstateManager& m)
     5+{
     6+    return &b == m.ActiveTip();
     7+}
     8+
     9+static void naive(CBlockIndex* p)
    10+{
    11+    CBlockIndex b = *p; // may slice: invokes CBlockIndex:CBlockIndex(const CBlockIndex&), runs on master, raises a build error after this change
    12+}
    13+
    14+static void user()
    15+{
    16+    CDiskBlockIndex d;
    17+    naive(&d);
    18+    CBlockIndex bb = d; // slices: invokes CBlockIndex:CBlockIndex(const CBlockIndex&) rather than CDiskBlockIndex:CDiskBlockIndex(const CDiskBlockIndex&), runs on master, raises a build error after this change
    19+}
    20+
    21 //! Basic tests for ChainstateManager.
    22@@ -310,6 +327,11 @@ BOOST_FIXTURE_TEST_CASE(chainstatemanager_activate_snapshot, TestChain100Setup)
    23     BOOST_CHECK_EQUAL(
    24         *chainman.ActiveChainstate().m_from_snapshot_blockhash,
    25         loaded_snapshot_blockhash);
    26+
    27+    // Pointer equality sanity check after exact copy.
    28+    const CBlockIndex index{*Assert(chainman.ActiveTip())}; // runs on master, raises a build error after this change
    29+    BOOST_CHECK(IsTip(index, chainman)); // fails on master
    30+    user(); // to avoid -Werror for unused function
    31 }
    
  21. jonatack commented at 2:30 pm on June 12, 2022: contributor
    @ryanofsky Good points. To follow up on our discussion, I wrote a failing test and implementation of a non-virtual public interface that calls private virtual implementation methods, i.e. the Template pattern via the Non-Virtual Interface (NVI) idiom, that makes the test pass in https://github.com/jonatack/bitcoin/commits/CBlockIndex-CDiskBlockIndex-class-design. Then, based on our discussion here, opted to propose a simpler change in #25349 that doesn’t add a vptr or vtable.
  22. in src/test/fuzz/pow.cpp:28 in 0497d76a94 outdated
    24@@ -25,17 +25,18 @@ FUZZ_TARGET_INIT(pow, initialize_pow)
    25 {
    26     FuzzedDataProvider fuzzed_data_provider(buffer.data(), buffer.size());
    27     const Consensus::Params& consensus_params = Params().GetConsensus();
    28-    std::vector<CBlockIndex> blocks;
    


    ajtowns commented at 3:32 am on June 13, 2022:
    This code is already buggy if the vector were to ever be moved while being resized? Could have been a std::array<CBlockIndex, 10000> perhaps.
  23. in src/chain.h:367 in 0497d76a94 outdated
    363+    //! of simplifying lifetime considerations due to attributes like pprev and
    364+    //! pskip, which are at risk of becoming dangling pointers in a copied
    365+    //! instance.
    366+    //!
    367+    //! We declare these protected instead of simply deleting them so that
    368+    //! CDiskBlockIndex can reuse copy construction.
    


    ajtowns commented at 4:30 am on June 13, 2022:

    I think CDiskBlockIndex::ToString is only used in the fuzz test; and renaming GetBlockHash to CalculateBlockHash would help avoid the “virtual” concerns:

    0    uint256 GetBlockHash() const = delete;
    1    uint256 CalculateBlockHash() const
    2    {
    3        CBlockHeader block;
    4        block.nVersion = nVersion;
    5        ...
    6    }
    7 
    8    std::string ToString() const = delete;
    9};
    

    Another alternative might be:

     0class CDiskBlockIndex
     1{
     2private:
     3    CBlockIndex m_cbi;
     4public:
     5    uint256 hashPrev;
     6    CDiskBlockIndex(const CBlockIndex* pindex)
     7    {
     8        m_cbi.pprev = pindex->pprev;
     9        m_cbi.nHeight = pindex->nHeight;
    10        ...
    11    }
    12    void SetCBlockIndex(CBlockIndex* pindex) const
    13    {
    14        pindex->nHeight = m_cbi.nHeight;
    15        ...
    16    }
    17    uint256 CalculateBlockHash() const;
    18    SERIALIZE_METHODS(...) { ... }
    19};
    

    ie, defining the “copy” code explicitly, which would let you move the explicit copy code in CBlockTreeDB::LoadBlockIndexGuts() into diskindex.Set(pindexNew);. At least they’re in one place, that way, and you can prevent any implicit copies of CBlockIndex?

    I’m not seeing any other ways of improving on this though – you can’t have CDiskBlockIndex use a reference to a CBlockIndex, because the ref has to be non-const for deserialization, and const for serialization, and it can’t simultaneously be both; and you can’t move de/serialization into CBlockIndex because it needs to store hashPrev on deser, and doesn’t have anywhere to do so…


    jonatack commented at 5:37 am on June 13, 2022:

    I think CDiskBlockIndex::ToString is only used in the fuzz test; and renaming GetBlockHash to CalculateBlockHash would help avoid the “virtual” concerns

    Yes, done in #25349 mentioned in my comment above.


    jamesob commented at 7:03 pm on September 7, 2022:
    As far as I can tell, this feedback thread has been addressed by @jonatack’s merged change - but let me know if not @ajtowns.
  24. ajtowns commented at 4:35 am on June 13, 2022: contributor
    Concept ACK
  25. DrahtBot commented at 8:55 am on June 15, 2022: contributor

    The following sections might be updated with supplementary metadata relevant to reviewers and maintainers.

    Reviews

    See the guideline for information on the review process.

    Type Reviewers
    ACK MarcoFalke, ajtowns
    Stale ACK ryanofsky, jonatack

    If your review is incorrectly listed, please react with 👎 to this comment and the bot will ignore it on the next update.

    Conflicts

    No conflicts as of last run.

  26. DrahtBot added the label Needs rebase on Jul 25, 2022
  27. jonatack commented at 4:34 pm on July 25, 2022: contributor
    @jamesob update with @ajtowns review feedback? The rebase should be trivial.
  28. jamesob commented at 7:29 pm on July 25, 2022: member
    Thanks for the ping @jonatack, will get back to this in the next day or so.
  29. jamesob force-pushed on Sep 7, 2022
  30. DrahtBot removed the label Needs rebase on Sep 7, 2022
  31. ajtowns commented at 1:25 am on September 8, 2022: contributor

    Delete move constructors and declare the destructor to satisfy the “rule of 5.”

    Code doesn’t seem to actually declare the destructor. Maybe:

    0-    CBlockIndex()
    1-    {
    2-    }
    3+    CBlockIndex() = default;
    4+    ~CBlockIndex() = default;
    
  32. in src/test/fuzz/chain.cpp:17 in 5b84eb8760 outdated
    13@@ -14,28 +14,32 @@
    14 FUZZ_TARGET(chain)
    15 {
    16     FuzzedDataProvider fuzzed_data_provider(buffer.data(), buffer.size());
    17-    std::optional<CDiskBlockIndex> disk_block_index = ConsumeDeserializable<CDiskBlockIndex>(fuzzed_data_provider);
    18-    if (!disk_block_index) {
    19+    const std::vector<uint8_t> index_buff = ConsumeRandomLengthByteVector(fuzzed_data_provider);
    


    maflcko commented at 4:08 pm on September 8, 2022:

    why?

    If there is a reason for the change, it would be good to make it in a separate commit, or explain why it needs to be done in this commit.

    If there is no reason, it would be better to not make the change.


    jamesob commented at 6:20 pm on September 9, 2022:
    Huh, weird, can’t remember why I did this but as you suggest it does compile without. Removed.
  33. jamesob force-pushed on Sep 9, 2022
  34. jamesob commented at 6:20 pm on September 9, 2022: member
    Pushed feedback from @ajtowns, @MarcoFalke. Thanks guys.
  35. achow101 commented at 7:30 pm on October 12, 2022: member
  36. in src/test/fuzz/pow.cpp:37 in 89fcfc872e outdated
    34         if (!block_header) {
    35             continue;
    36         }
    37-        CBlockIndex current_block{*block_header};
    38+        blocks.push_back(std::make_unique<CBlockIndex>(*block_header));
    39+        CBlockIndex& current_block = *blocks.back();
    


    maflcko commented at 7:21 pm on December 15, 2022:
    0        CBlockIndex& current_block{*blocks.emplace_back(std::make_unique<CBlockIndex>(*block_header))};
    

    nit, can write in one line

  37. in src/test/miner_tests.cpp:83 in 89fcfc872e outdated
    82+static std::unique_ptr<CBlockIndex> CreateBlockIndex(int nHeight, CBlockIndex* active_chain_tip) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
    83 {
    84-    CBlockIndex index;
    85-    index.nHeight = nHeight;
    86-    index.pprev = active_chain_tip;
    87+    std::unique_ptr<CBlockIndex> index = std::make_unique<CBlockIndex>();
    


    maflcko commented at 7:26 pm on December 15, 2022:
    0    auto index{std::make_unique<CBlockIndex>()};
    

    nit: No need to repeat the same type redundantly in one line

  38. maflcko approved
  39. maflcko commented at 7:28 pm on December 15, 2022: member

    Nice.

    Left two nits. Happy to re-ACK, or leave them ignored.

    review ACK 89fcfc872eebd8a177561189575f67da27752cb1 🍲

    Signature:

     0-----BEGIN PGP SIGNED MESSAGE-----
     1Hash: SHA512
     2
     3review ACK 89fcfc872eebd8a177561189575f67da27752cb1 🍲
     4-----BEGIN PGP SIGNATURE-----
     5
     6iQGzBAEBCgAdFiEE+rVPoUahrI9sLGYTzit1aX5ppUgFAlwqrYAACgkQzit1aX5p
     7pUiSNQwAicS4GLjofIlttbHg6E36/ZFqeRcuM7JGwoRO/dX3cpVFtBCB+3G5rObS
     8RN/AEZ5eWyELNicp2/q+1kJf0LyBkqPXLMVDWqT4gf54fVCdX26orqGUqmHTl8HY
     9Cxw8wi+x48YTb/+iLPpbpFqLNoE82oypyKNDBCR8O8AOt1fKeIoNI0ZbvgFWTGjy
    10mYYN+Noi2MPWW9aa6Q7Y2LL/bdpMfLttj7E6Gt+XTTxpeX5IrogR7tGpUbBIbuBv
    11a+Px6BXakK+nOweLGnOH0lYcEJOqFIqWyqvP6PUevGSgtYlZm6Z+X7ZFp/btWj4q
    12VJIB7Dn/g8XhBm3Up9YGEqtfsbOZlihhovoBp24nXOk8BGb4WsaKJzCPqku7o9Px
    13Ee7w44W+cfH8OKs/3g0+2z0wwSqzA5JLxxpezhabiUclfKGaxJhbczWHJxWTktRr
    1491p8KV+MdDaers1ZI5QOXs+eKrsSP4CjdiQnMMGgrw/ua1IbTmjntKSlwY7/Vq4a
    15kjYKqGut
    16=d+cV
    17-----END PGP SIGNATURE-----
    
  40. maflcko removed the label UTXO Db and Indexes on Dec 15, 2022
  41. maflcko renamed this:
    remove CBlockIndex copy construction
    refactor: remove CBlockIndex copy construction
    on Dec 15, 2022
  42. DrahtBot added the label Refactoring on Dec 15, 2022
  43. maflcko added the label UTXO Db and Indexes on Dec 15, 2022
  44. jamesob force-pushed on Dec 15, 2022
  45. remove CBlockIndex copy construction
    Copy construction of CBlockIndex objects is a footgun because of the
    wide use of equality-by-pointer comparison in the code base. There are
    also potential lifetime confusions of using copied instances, since
    there are recursive pointer references (e.g. pprev).
    
    We can't just delete the copy constructors because they are used for
    derived classes (CDiskBlockIndex), so we mark them protected.
    
    Delete move constructors and declare the destructor to satisfy the
    "rule of 5."
    36c201feb7
  46. maflcko approved
  47. maflcko commented at 7:57 pm on December 15, 2022: member

    re-ACK 36c201feb74bbb87d22bd956373dbbb9c47fb7e7 🏻

    Signature:

     0-----BEGIN PGP SIGNED MESSAGE-----
     1Hash: SHA512
     2
     3re-ACK 36c201feb74bbb87d22bd956373dbbb9c47fb7e7  🏻
     4-----BEGIN PGP SIGNATURE-----
     5
     6iQGzBAEBCgAdFiEE+rVPoUahrI9sLGYTzit1aX5ppUgFAlwqrYAACgkQzit1aX5p
     7pUgR+wwAl9CSAv7TffHzPRCAHtkQzy2bADeuYb13/tGZmht477Ny9r3DNVPI2TRU
     8r7rwlAH836ssp4Xqxq/y2EIrerKJ7nY1GS9rdELrdJjUMRpa3rVDgG0fLPcp5cKU
     9or0XHvezB1a4PO8CAeiSuM9tavWQki+2CntslLEofVXcqcp/HZbl9tbENdtIZu/z
    10rxHgaZfFSJ+B1a4JmXzybj56SSX1AFrLLbyBX61rNSAqVlHmJDmzRGjJrcK1P13m
    11l0c8H8MeeHNFxw+cV7gB6bca9jDw56oHX+bDswCYMUARUevCZME50x0ny9d6EgWN
    12LSJ1QQGvk+5GgXlbK8vEbsRnYzGKi/M8GOlSTJV0c1jk2eloWXmkclkPKdf+V50a
    13V7T12GjTgnuQtDJcX0hpJ9VSP4EQ+9qTmzdjFLsavx2Ad/kRbrZh/Dck3iXOPcJZ
    14WpJ3QkwXPW62hfSXW6ZKZwBDShLr8eCfC+JAyyidmuSbVjTLL0Xu0Z8K9KPtv8eD
    15wXNA+JYi
    16=9oo/
    17-----END PGP SIGNATURE-----
    
  48. fanquake requested review from ryanofsky on Dec 16, 2022
  49. fanquake requested review from ajtowns on Dec 16, 2022
  50. ajtowns commented at 8:05 am on December 18, 2022: contributor
    ACK 36c201feb74bbb87d22bd956373dbbb9c47fb7e7 - code review only
  51. ajtowns approved
  52. fanquake merged this on Dec 19, 2022
  53. fanquake closed this on Dec 19, 2022

  54. sidhujag referenced this in commit 3ef761fbd9 on Dec 19, 2022
  55. bitcoin locked this on Dec 19, 2023

github-metadata-mirror

This is a metadata mirror of the GitHub repository bitcoin/bitcoin. This site is not affiliated with GitHub. Content is generated from a GitHub metadata backup.
generated: 2024-12-03 15:12 UTC

This site is hosted by @0xB10C
More mirrored repositories can be found on mirror.b10c.me