aboutsummaryrefslogtreecommitdiff
path: root/objectstore/packed/delta_apply.go
diff options
context:
space:
mode:
authorGravatar Runxi Yu2026-02-21 20:21:55 +0800
committerGravatar Runxi Yu2026-02-21 20:21:55 +0800
commitb480c5c4dfaddb9b137f50b1a7651e3bce2ecb98 (patch)
tree4952eb430959bbfb5eb95fe0bbcf1675fff02d34 /objectstore/packed/delta_apply.go
parentobjectstore/packed: Best-effort touchCandidate (diff)
signatureNo signature
objectstore/packed: Improve delta base caching
Diffstat (limited to 'objectstore/packed/delta_apply.go')
-rw-r--r--objectstore/packed/delta_apply.go86
1 files changed, 77 insertions, 9 deletions
diff --git a/objectstore/packed/delta_apply.go b/objectstore/packed/delta_apply.go
index a067525d..6d5c736d 100644
--- a/objectstore/packed/delta_apply.go
+++ b/objectstore/packed/delta_apply.go
@@ -4,27 +4,33 @@ import (
"fmt"
deltaapply "codeberg.org/lindenii/furgit/format/delta/apply"
+ packfmt "codeberg.org/lindenii/furgit/format/pack"
"codeberg.org/lindenii/furgit/objecttype"
)
// deltaResolveContent resolves one object's content bytes from its pack location.
func (store *Store) deltaResolveContent(start location) (objecttype.Type, []byte, error) {
- plan, err := store.deltaPlanFor(start)
+ chain, err := store.deltaBuildChain(start)
if err != nil {
return objecttype.TypeInvalid, nil, err
}
+ return store.deltaResolveChain(chain)
+}
- baseType, out, err := store.deltaResolveBase(plan)
+// deltaResolveChain resolves one object chain into content bytes.
+func (store *Store) deltaResolveChain(chain deltaChain) (objecttype.Type, []byte, error) {
+ ty, out, nextDelta, err := store.deltaResolveChainStart(chain)
if err != nil {
return objecttype.TypeInvalid, nil, err
}
- for i := len(plan.frames) - 1; i >= 0; i-- {
- frame := plan.frames[i]
- pack, err := store.openPack(frame.packName)
+
+ for i := nextDelta; i >= 0; i-- {
+ node := chain.deltas[i]
+ pack, err := store.openPack(node.loc.packName)
if err != nil {
return objecttype.TypeInvalid, nil, err
}
- delta, err := inflateAt(pack, frame.dataOffset, -1)
+ delta, err := inflateAt(pack, node.dataOffset, -1)
if err != nil {
return objecttype.TypeInvalid, nil, err
}
@@ -32,13 +38,75 @@ func (store *Store) deltaResolveContent(start location) (objecttype.Type, []byte
if err != nil {
return objecttype.TypeInvalid, nil, err
}
+ store.cacheMu.Lock()
+ store.deltaCache.add(
+ deltaBaseKey{packName: node.loc.packName, offset: node.loc.offset},
+ ty,
+ out,
+ )
+ store.cacheMu.Unlock()
}
- if int64(len(out)) != plan.declaredSize {
+
+ if int64(len(out)) != chain.declaredSize {
return objecttype.TypeInvalid, nil, fmt.Errorf(
"objectstore/packed: resolved content size mismatch: got %d want %d",
len(out),
- plan.declaredSize,
+ chain.declaredSize,
+ )
+ }
+ if ty != chain.baseType {
+ return objecttype.TypeInvalid, nil, fmt.Errorf(
+ "objectstore/packed: resolved content type mismatch: got %d want %d",
+ ty,
+ chain.baseType,
+ )
+ }
+ return ty, out, nil
+}
+
+// deltaResolveChainStart finds the nearest cached chain node or inflates the
+// innermost base object. It returns the starting bytes and the next delta index
+// to apply in reverse order.
+func (store *Store) deltaResolveChainStart(chain deltaChain) (objecttype.Type, []byte, int, error) {
+ for i, node := range chain.deltas {
+ store.cacheMu.RLock()
+ ty, out, ok := store.deltaCache.get(
+ deltaBaseKey{packName: node.loc.packName, offset: node.loc.offset},
)
+ store.cacheMu.RUnlock()
+ if ok {
+ return ty, out, i - 1, nil
+ }
+ }
+
+ store.cacheMu.RLock()
+ ty, out, ok := store.deltaCache.get(
+ deltaBaseKey{packName: chain.baseLoc.packName, offset: chain.baseLoc.offset},
+ )
+ store.cacheMu.RUnlock()
+ if ok {
+ return ty, out, len(chain.deltas) - 1, nil
+ }
+
+ pack, meta, err := store.entryMetaAt(chain.baseLoc)
+ if err != nil {
+ return objecttype.TypeInvalid, nil, 0, err
}
- return baseType, out, nil
+ if !packfmt.IsBaseObjectType(meta.ty) {
+ return objecttype.TypeInvalid, nil, 0, fmt.Errorf("objectstore/packed: delta chain base is not a base object")
+ }
+ base, err := inflateAt(pack, meta.dataOffset, meta.size)
+ if err != nil {
+ return objecttype.TypeInvalid, nil, 0, err
+ }
+
+ store.cacheMu.Lock()
+ store.deltaCache.add(
+ deltaBaseKey{packName: chain.baseLoc.packName, offset: chain.baseLoc.offset},
+ meta.ty,
+ base,
+ )
+ store.cacheMu.Unlock()
+
+ return meta.ty, base, len(chain.deltas) - 1, nil
}