Reduce checkpoints’ effect on consensus. #5927

pull sipa wants to merge 1 commits into bitcoin:master from sipa:nolockin changing 6 files +15 −38
  1. sipa commented at 12:41 pm on March 19, 2015: member

    Instead of only checking height to decide whether to disable script checks, actually check whether a block is an ancestor of a checkpoint, up to which headers have been validated. This means that we don’t have to prevent accepting a side branch anymore - it will be safe, just less fast to do.

    We still need to prevent being fed a multitude of low-difficulty headers filling up our memory. The mechanism for that is unchanged for now: once a checkpoint is reached with headers, no headers chain branching off before that point are allowed anymore.

  2. sipa force-pushed on Mar 19, 2015
  3. jonasschnelli commented at 7:57 pm on March 19, 2015: contributor

    I’m getting EXC_BAD_ACCESS during setgenerate -> ConnectBlock().

    Maybe CBlockIndex* CBlockIndex::GetAncestor(int height) was called with a CBlockIndex with a pskip NULL Poiner?

     0* thread [#2](/bitcoin-bitcoin/2/): tid = 0x586dd, 0x00000001000518fe bitcoind`CBlockIndex::GetAncestor(this=0x000000010597cf38, height=0) + 190 at chain.cpp:89, stop reason = EXC_BAD_ACCESS (code=1, address=0x10)
     1    frame [#0](/bitcoin-bitcoin/0/): 0x00000001000518fe bitcoind`CBlockIndex::GetAncestor(this=0x000000010597cf38, height=0) + 190 at chain.cpp:89
     2   86               (heightSkip > height && !(heightSkipPrev < heightSkip - 2 &&
     3   87                                         heightSkipPrev >= height))) {
     4   88               // Only follow pskip if pprev->pskip isn't better than pskip->pprev.
     5-> 89               pindexWalk = pindexWalk->pskip;
     6   90               heightWalk = heightSkip;
     7   91           } else {
     8   92               pindexWalk = pindexWalk->pprev;
     9(lldb) bt
    10* thread [#2](/bitcoin-bitcoin/2/): tid = 0x586dd, 0x00000001000518fe bitcoind`CBlockIndex::GetAncestor(this=0x000000010597cf38, height=0) + 190 at chain.cpp:89, stop reason = EXC_BAD_ACCESS (code=1, address=0x10)
    11  * frame [#0](/bitcoin-bitcoin/0/): 0x00000001000518fe bitcoind`CBlockIndex::GetAncestor(this=0x000000010597cf38, height=0) + 190 at chain.cpp:89
    12    frame [#1](/bitcoin-bitcoin/1/): 0x0000000100163650 bitcoind`ConnectBlock(block=0x0000000101f04690, state=0x000000010597d5f0, pindex=0x000000010597cf38, view=0x000000010597cfc0, fJustCheck=true) + 592 at main.cpp:1761
    13    frame [#2](/bitcoin-bitcoin/2/): 0x00000001001730a7 bitcoind`TestBlockValidity(state=0x000000010
    
  4. sipa commented at 8:19 pm on March 19, 2015: member
    Thanks for doing the debugging job I should have done :)
  5. morcos commented at 10:15 pm on March 19, 2015: member
    Hmm.. So this requires #5875 to be safe I think. Was that the intention?
  6. sipa force-pushed on Mar 20, 2015
  7. sipa force-pushed on Mar 21, 2015
  8. sipa commented at 2:22 pm on March 21, 2015: member
    @morcos Well there are still measures in place to prevent us from accepting a long header-branch still, and we won’t accept blocks for which we don’t accept the header. Though the PR as-is indeed loosens some things, that are countered by #5875.
  9. morcos commented at 6:29 pm on March 21, 2015: member
    Ok, I think my concern is that before this change, when a node starts up, once it gets through enough IBD, it’s impossible to DOS it by just feeding it low difficulty full blocks, because those blocks would either have to be a fork before a checkpoint, or they wouldn’t connect back to a valid chain. After this change, I think it’s possible to push through a long low-difficulty fork before the node gets through the checkpoints. If the length is greater than the height of the last checkpoint, then from then on nothing prevents you from stuffing low difficulty blocks at this node forever until you fill up its disk or tie up its CPU. Once you’ve snuck in that low difficulty long fork at the beginning it’s usable any time after.
  10. sipa force-pushed on Mar 23, 2015
  11. sipa commented at 11:50 am on March 23, 2015: member
    @morcos You’re right, but just receiving headers up to the checkpoints suffices to prevent branches, which should be within seconds in normal circumstances (for now). Headers synchronization is however not very robust (it’s easy to have a peer that just don’t respond to headers request to delay the synchronization significantly), so I agree that it’s better to combine this with #5875.
  12. sipa force-pushed on Mar 23, 2015
  13. sipa commented at 1:43 pm on March 23, 2015: member
    @jonasschnelli Bug fixed.
  14. laanwj added the label Consensus on Mar 26, 2015
  15. sipa force-pushed on May 10, 2015
  16. sipa commented at 9:28 pm on May 10, 2015: member
    Rebased, and will rebase on #5975 if that gets merged first.
  17. Reduce checkpoints' effect on consensus.
    Instead of only checking height to decide whether to disable script checks,
    actually check whether a block is an ancestor of a checkpoint, up to which
    headers have been validated. This means that we don't have to prevent
    accepting a side branch anymore - it will be safe, just less fast to
    do.
    
    We still need to prevent being fed a multitude of low-difficulty headers
    filling up our memory. The mechanism for that is unchanged for now: once
    a checkpoint is reached with headers, no headers chain branching off before
    that point are allowed anymore.
    dce8360e44
  18. sipa force-pushed on May 13, 2015
  19. sipa commented at 8:43 pm on May 13, 2015: member
    @gavinandresen Bug fixed, I hope.
  20. laanwj commented at 2:57 pm on May 21, 2015: member
    Tested ACK (code reviewed, synced up to block 320700 with #5875 and #5927)
  21. jtimon commented at 11:52 pm on May 21, 2015: contributor
    ut ACK
  22. sdaftuar commented at 1:50 pm on May 22, 2015: member

    I’m concerned about the impact a malicious peer could have on a node starting up from scratch; in particular I think a malicious peer could partition a new node running this code from honest peers.

    In 0.10 and master, when a node starts up, it picks one peer to download the headers chain from. After the headers chain’s best block time is within 24 hours of now, the node will start to download headers and then blocks from the other peers.

    In master without this pull, if that sync peer is malicious or buggy and serves the node a bad chain, then once the node hits a checkpoint height, it’ll recognize it as bad and disconnect that peer. However, with this pull, the node could process a blockchain from the bad peer past the checkpoint heights. Upon connecting to other (honest) peers, those peers will send the node a getheaders message, and it will respond with the bad chain. Those honest peers would then disconnect the node for serving headers that violate a checkpoint, preventing the node from ever joining the honest network.

    FYI #6172 would limit the potential for this type of behavior by not responding to a getheaders until we’re past the checkpoint height, but it is insufficient protection for this PR, because that sync node just needs to serve more blocks than the checkpoint height to create a problem (ie once the bad chain gets past the criteria for exiting IsInitialBlockDownload, the node would start inv’ing bad chain blocks to peers as the tip is updated, triggering disconnection from any honest peers who may be connected, and also preventing new connections to honest peers who would send a getheaders message upon connection).

    I think the principle here is that if we are going to relax requirements so that we’d sometimes allow accepting blocks that might violate a currently-existing-checkpoint, then we first need to make peers be willing to stay connected to nodes that might serve them such blocks. That needs more thought, as we’d likely need to replace the current disconnect behavior with some other kind of DoS protection, and I’m not sure what that should look like.

    One idea I had was to make IsInitialBlockDownload have some knowledge of the amount of work on the most-work-chain (rather then just use the checkpointed block height), and set a node’s behavior to not deliver blocks (or headers) until exiting InitialBlockDownload – that way, any bad chains would have to have a lot of work in order to trick a node to start trying to serve it to others and face possible disconnection. I think this is doable, but it also feels like a workaround rather than a solution to the underlying issue, so perhaps there are better ideas out there.

  23. luke-jr commented at 2:01 am on June 2, 2015: member
    utACK
  24. laanwj commented at 8:32 am on June 10, 2015: member
    Going ahead to merge this (into master only). I agree @sdaftuar ’s issue should be addressed before the 0.12 release.
  25. laanwj merged this on Jun 10, 2015
  26. laanwj closed this on Jun 10, 2015

  27. laanwj referenced this in commit 8d9f0a6069 on Jun 10, 2015
  28. sdaftuar commented at 4:10 pm on October 29, 2015: member

    @sipa @laanwj I just remembered this pull, and I think the concern I raised earlier is still a potential issue for 0.12. Any thoughts on the best way to mitigate?

    One idea I had would be to not leave IBD (and therefore not announce new blocks or respond to a getheaders message) until your main chain builds off the last checkpointed block. When 0.12 is widely enough deployed, then in a future release we could eliminate that check (since 0.12 peers won’t disconnect each other for being on the non-checkpointed chain).

    An alternate idea would be to just hardcode in a value of required work on chainActive before we leave IBD, and we set that to something reasonably recent so that it would be expensive to attack. Again, when 0.12 is widely enough deployed, we can eliminate this hardcoded value and the associated check. But I don’t love the idea of introducing a new hardcoded value…

  29. morcos referenced this in commit 8bfda948fd on Mar 15, 2016
  30. morcos referenced this in commit 148e3ba757 on Mar 16, 2016
  31. rebroad referenced this in commit fe30e1b5bd on Dec 7, 2016
  32. zkbot referenced this in commit df07f9ad23 on Feb 15, 2017
  33. zkbot referenced this in commit ec069ce96c on Mar 3, 2017
  34. zkbot referenced this in commit 702eefc71a on Mar 3, 2017
  35. zkbot referenced this in commit 99c4c6de0c on Mar 3, 2017
  36. DrahtBot locked this on Sep 8, 2021

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: 2025-01-21 09:12 UTC

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