mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-01-18 21:35:24 +01:00
a7d6826f60
In this commit, we eliminate some code duplication by removing the old `HashMutex` struct as it just duplicates all the code with a different type (uint64 and hash). We then make the main Mutex struct take a type param, so the key can be parametrized when the struct is instantiated.
74 lines
2.1 KiB
Go
74 lines
2.1 KiB
Go
package blockcache
|
|
|
|
import (
|
|
"github.com/btcsuite/btcd/btcutil"
|
|
"github.com/btcsuite/btcd/chaincfg/chainhash"
|
|
"github.com/btcsuite/btcd/wire"
|
|
"github.com/lightninglabs/neutrino"
|
|
"github.com/lightninglabs/neutrino/cache"
|
|
"github.com/lightninglabs/neutrino/cache/lru"
|
|
"github.com/lightningnetwork/lnd/lntypes"
|
|
"github.com/lightningnetwork/lnd/multimutex"
|
|
)
|
|
|
|
// BlockCache is an lru cache for blocks.
|
|
type BlockCache struct {
|
|
Cache *lru.Cache[wire.InvVect, *neutrino.CacheableBlock]
|
|
HashMutex *multimutex.Mutex[lntypes.Hash]
|
|
}
|
|
|
|
// NewBlockCache creates a new BlockCache with the given maximum capacity.
|
|
func NewBlockCache(capacity uint64) *BlockCache {
|
|
return &BlockCache{
|
|
Cache: lru.NewCache[wire.InvVect, *neutrino.CacheableBlock](
|
|
capacity,
|
|
),
|
|
HashMutex: multimutex.NewMutex[lntypes.Hash](),
|
|
}
|
|
}
|
|
|
|
// GetBlock first checks to see if the BlockCache already contains the block
|
|
// with the given hash. If it does then the block is fetched from the cache and
|
|
// returned. Otherwise the getBlockImpl function is used in order to fetch the
|
|
// new block and then it is stored in the block cache and returned.
|
|
func (bc *BlockCache) GetBlock(hash *chainhash.Hash,
|
|
getBlockImpl func(hash *chainhash.Hash) (*wire.MsgBlock,
|
|
error)) (*wire.MsgBlock, error) {
|
|
|
|
bc.HashMutex.Lock(lntypes.Hash(*hash))
|
|
defer bc.HashMutex.Unlock(lntypes.Hash(*hash))
|
|
|
|
// Create an inv vector for getting the block.
|
|
inv := wire.NewInvVect(wire.InvTypeWitnessBlock, hash)
|
|
|
|
// Check if the block corresponding to the given hash is already
|
|
// stored in the blockCache and return it if it is.
|
|
cacheBlock, err := bc.Cache.Get(*inv)
|
|
if err != nil && err != cache.ErrElementNotFound {
|
|
return nil, err
|
|
}
|
|
if cacheBlock != nil {
|
|
return cacheBlock.MsgBlock(), nil
|
|
}
|
|
|
|
// Fetch the block from the chain backends.
|
|
block, err := getBlockImpl(hash)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Add the new block to blockCache. If the Cache is at its maximum
|
|
// capacity then the LFU item will be evicted in favour of this new
|
|
// block.
|
|
_, err = bc.Cache.Put(
|
|
*inv, &neutrino.CacheableBlock{
|
|
Block: btcutil.NewBlock(block),
|
|
},
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return block, nil
|
|
}
|