Lazily update CBlockIndex entries when pruning #6118

pull sdaftuar wants to merge 1 commits into bitcoin:master from sdaftuar:lazy-blockindex-update changing 5 files +85 −65
  1. sdaftuar commented at 5:01 pm on May 8, 2015: member

    Rename BLOCK_HAVE_DATA and BLOCK_HAVE_UNDO to BLOCK_STORED_DATA/ BLOCK_STORED_UNDO.

    These status values now indicate that a block or its undo information has been stored on disk, but pruning could cause that data to no longer be present.

    HaveBlockData() uses vinfoBlockFile to determine whether we actually have block data for a given block, and updates the CBlockIndex passed in when BLOCK_STORED_DATA was set to true but the block file has been pruned.

    Also remove the iteration of mapBlockIndex in PruneOneBlockFile(), as we no longer need to update the CBlockIndex entries of the blocks referenced in a pruned file. @sipa Is this along the lines of what you had in mind?

  2. sdaftuar force-pushed on May 8, 2015
  3. jonasschnelli commented at 6:10 pm on May 8, 2015: contributor
    utACK. Plans to test this soon. Little indentation/tabs nit chain.h L88:90.
  4. sdaftuar force-pushed on May 8, 2015
  5. sdaftuar commented at 6:16 pm on May 8, 2015: member
    Nit fixed, thanks!
  6. jonasschnelli commented at 8:36 am on May 10, 2015: contributor

    Added this PR on top of master and left my pruned bitcoind mainnet node running over night. No issues. Just copied blocks/chainstate from a full-node to a fresh node and pruned to target 550MB. Worked as expected.

    Tested ACK.

  7. sdaftuar force-pushed on May 10, 2015
  8. sdaftuar force-pushed on May 10, 2015
  9. sipa commented at 9:44 pm on May 10, 2015: member
    Untested ACK. I don’t understand the Travis failure.
  10. sdaftuar force-pushed on May 13, 2015
  11. sdaftuar force-pushed on May 13, 2015
  12. sdaftuar force-pushed on May 13, 2015
  13. sdaftuar commented at 6:57 pm on May 13, 2015: member
    Rebased.
  14. laanwj commented at 10:50 am on May 15, 2015: member
    What is your reasoning behind this? From a conceptual point of view I like the flags to specify what they do now: does the block HAVE data or undo data.
  15. laanwj added the label Improvement on May 15, 2015
  16. sdaftuar commented at 1:41 pm on May 15, 2015: member

    I was trying to eliminate the need to iterate the entire mapBlockIndex whenever a file is pruned (since we currently lack a map from block files to blocks, we check each block to see if it’s in a blockfile being deleted). That seems like something that needs to be fixed at some point.

    One idea for addressing that would be to just maintain a (memory-only) map from block files to blockindex entries which we iterate when pruning; this would be straightforward, cpu efficient, and preserve existing HAVE_DATA semantics. However I think @sipa was concerned about the additional memory overhead of that approach (I believe it would be roughly an additional 10MB if you’re running in prune mode with a target high enough that no files are actually deleted – which is perhaps relevant if in the future most nodes may run with some level of pruning), and he suggested lazy updating instead.

    So I tried implementing the lazy-update approach to see how feasible (and/or cumbersome) this would be. It seems very efficient; the code changes to support it seem relatively minor; and I like the idea of not having to worry about updating state when block files are removed – that seems like it should make future pruning-related improvements easier to make. I do agree there’s a tradeoff, in that the current HAVE_DATA semantics seem cleaner versus the new STORED_DATA semantics, but it’s hard for me right now to say which semantics are better in the long run.

    I think that uncertainty would be an argument for waiting to see what the future of pruning code looks like and then deciding what solution makes most sense, except one thing I realized after coding this up is that if we decide to go in this direction with lazy updating of block index entries, then I think it would be better if we deploy this behavior with pruning in 0.11 than wait until 0.12 – otherwise there’s a backwards compatibility issue, where once your block index has been lazily updated, you can’t easily revert back to a version of pruning code that didn’t support that (at least with the present implementation).

    At any rate, I don’t feel strongly about which approach has the best tradeoffs, and if we end up not merging this for 0.11, then I’d say we may as well just wait until we implement sharding, and decide then what approach makes sense here.

  17. laanwj commented at 3:31 pm on May 15, 2015: member

    Is the overhead to have to iterate over the (in-memory) block index when a file is pruned significant?

    Muddying the database semantics to make a rare action slightly faster would be unfortunate. But I don’t feel strongly about it either.

    Also: this makes the ‘block has data’ check more involved and that is performed very often.

  18. in src/main.cpp: in fb10422601 outdated
    3469@@ -3460,8 +3470,10 @@ void static CheckBlockIndex()
    3470     CBlockIndex* pindexFirstNotScriptsValid = NULL; // Oldest ancestor of pindex which does not have BLOCK_VALID_SCRIPTS (regardless of being valid or not).
    3471     while (pindex != NULL) {
    3472         nNodes++;
    3473+        // Not calling HaveBlockData() directly, since that can change state.
    


    theuni commented at 4:25 pm on May 15, 2015:
    This comment points out that the HaveBlockData() name is very misleading. Could you change the name to indicate that it also updates the state?

    sdaftuar commented at 9:01 pm on May 15, 2015:

    Happy to change it, any suggestions? The function only changes state if in pruning mode, so I don’t want to go overboard with the description, and nothing concise is jumping out at me.

    I did notice the comment for the function doesn’t really emphasize the potentially state-changing behavior, I’ll start by rewording that for now to make it more obvious.

  19. sdaftuar commented at 4:28 pm on May 15, 2015: member

    @laanwj I believe it’s ~20ms to iterate mapBlockIndex. Arguably that is insignificant the way pruning works now, since it can only happen when we’re calling FlushStateToDisk() which is already an expensive operation.

    Regarding the efficiency of HaveBlockData(), I think that this implementation is very fast if there’s no lock contention and that seems to be the case presently; it’s unclear to me whether there could be lock contention to worry about down the road, in particular if the code were to change in the future so that the HaveBlockData() function is called from more places.

  20. morcos commented at 6:12 pm on May 15, 2015: member

    I think @sdaftuar has a point that if we’re ever going to implement lazy updating, the time to do it is now. But personally I think some sort of reverse lookup is eventually going to be needed. I implemented it to try it out and it’s fine, but I also think the simple 20ms iteration every time you delete a blockfile is good enough for now and we should just do the reverse lookup in whatever way makes sense with the sharding algorithm we end up with.

    Hmm, I was going to say the argument against the 20ms iteration is that once a day or so (when you’ve got an extra block file you can prune) then it’s possible (I think even probable) that you’re holding up block propagation. But come to think of it, that’s a problem with choosing when we’re pruning because even without the 20ms cost, you’re still doing a full chainstate flush then. If we think that’s problematic, maybe we can move around the pruning calls to only happen when we’re a bit less busy, it’ll take some testing though..

  21. Lazily update CBlockIndex entries when pruning
    Rename BLOCK_HAVE_DATA/ BLOCK_HAVE_UNDO to BLOCK_STORED_DATA/
    BLOCK_STORED_UNDO. These status values now indicate that a block or its undo
    information has been stored on disk, but pruning may cause that information to
    no longer be true.
    
    HaveBlockData() uses vinfoBlockFile to determine whether we actually have block
    data for a given block, and it updates the CBlockIndex passed in when
    STORED_DATA was set to true but the block file has been pruned.
    
    Also remove the iteration of mapBlockIndex in PruneOneBlockFile, as we no
    longer need to update the CBlockIndex entries of the blocks referenced in a
    pruned file.
    15d974823d
  22. sdaftuar force-pushed on May 15, 2015
  23. sdaftuar commented at 11:57 pm on May 15, 2015: member
    Updated a few comments and tried to make it clearer that HaveBlockData() can change state when pruning is enabled.
  24. sdaftuar commented at 8:21 pm on June 5, 2015: member
    Closing. This apparently need to be rebased, and it’s probably not worth it as I don’t think we’re going to merge for 0.11, and given the backwards compatibility issues this pull would generate if this were accepted later (see description above and #6148 (comment)), I think it’d be better for us to wait until a sharding solution is on the table and revisit this idea in that context.
  25. sdaftuar closed this on Jun 5, 2015

  26. MarcoFalke 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-22 03:12 UTC

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