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.
sipa force-pushed
on Mar 19, 2015
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?
sipa
commented at 8:19 pm on March 19, 2015:
member
Thanks for doing the debugging job I should have done :)
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?
sipa force-pushed
on Mar 20, 2015
sipa force-pushed
on Mar 21, 2015
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.
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.
sipa force-pushed
on Mar 23, 2015
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.
sipa force-pushed
on Mar 23, 2015
sipa
commented at 1:43 pm on March 23, 2015:
member
Rebased, and will rebase on #5975 if that gets merged first.
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.
laanwj
commented at 2:57 pm on May 21, 2015:
member
Tested ACK (code reviewed, synced up to block 320700 with #5875 and #5927)
jtimon
commented at 11:52 pm on May 21, 2015:
contributor
ut ACK
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.
luke-jr
commented at 2:01 am on June 2, 2015:
member
utACK
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.
laanwj merged this
on Jun 10, 2015
laanwj closed this
on Jun 10, 2015
laanwj referenced this in commit
8d9f0a6069
on Jun 10, 2015
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…
morcos referenced this in commit
8bfda948fd
on Mar 15, 2016
morcos referenced this in commit
148e3ba757
on Mar 16, 2016
rebroad referenced this in commit
fe30e1b5bd
on Dec 7, 2016
zkbot referenced this in commit
df07f9ad23
on Feb 15, 2017
zkbot referenced this in commit
ec069ce96c
on Mar 3, 2017
zkbot referenced this in commit
702eefc71a
on Mar 3, 2017
zkbot referenced this in commit
99c4c6de0c
on Mar 3, 2017
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