From e25782f602fca13b9d6dd812dfdc7facab52f0bd Mon Sep 17 00:00:00 2001 From: Badrish Chandramouli Date: Mon, 6 May 2024 17:37:06 -0700 Subject: [PATCH] Fix flush with generic allocator so that we do not write invalid data to disk in rare cases (record was invalid when making initial image, but (#916) --- cs/src/core/Allocator/GenericAllocator.cs | 32 +++++++++++++++++------ 1 file changed, 24 insertions(+), 8 deletions(-) diff --git a/cs/src/core/Allocator/GenericAllocator.cs b/cs/src/core/Allocator/GenericAllocator.cs index c117d32b6..f5fa98e45 100644 --- a/cs/src/core/Allocator/GenericAllocator.cs +++ b/cs/src/core/Allocator/GenericAllocator.cs @@ -452,29 +452,46 @@ private void WriteAsync(long flushPage, ulong alignedDestinationAddres } long endPosition = 0; - for (int i=start/recordSize; i>(recordPtr); + AddressInfo* key_address = null, value_address = null; + + // Zero out object reference addresses (AddressInfo) in the planned disk image + if (KeyHasObjects()) + { + key_address = GetKeyAddressInfo((long)recordPtr); + *key_address = default; + } + if (ValueHasObjects()) + { + value_address = GetValueAddressInfo((long)recordPtr); + *value_address = default; + } + + // Now fill in AddressInfo data for the valid records + if (!record.info.Invalid) { var address = (flushPage << LogPageSizeBits) + i * recordSize; - if (address < fuzzyStartLogicalAddress || !src[i].info.IsInNewVersion) + if (address < fuzzyStartLogicalAddress || !record.info.IsInNewVersion) { if (KeyHasObjects()) { long pos = ms.Position; keySerializer.Serialize(ref src[i].key); - var key_address = GetKeyAddressInfo((long)(buffer.aligned_pointer + i * recordSize)); key_address->Address = pos; key_address->Size = (int)(ms.Position - pos); addr.Add((long)key_address); endPosition = pos + key_address->Size; } - if (ValueHasObjects() && !src[i].info.Tombstone) + if (ValueHasObjects() && !record.info.Tombstone) { long pos = ms.Position; valueSerializer.Serialize(ref src[i].value); - var value_address = GetValueAddressInfo((long)(buffer.aligned_pointer + i * recordSize)); value_address->Address = pos; value_address->Size = (int)(ms.Position - pos); addr.Add((long)value_address); @@ -484,7 +501,6 @@ private void WriteAsync(long flushPage, ulong alignedDestinationAddres else { // Mark v+1 records as invalid to avoid deserializing them on recovery - ref var record = ref Unsafe.AsRef>(buffer.aligned_pointer + i * recordSize); record.info.SetInvalid(); } } @@ -846,7 +862,7 @@ public void Deserialize(byte *raw, long ptr, long untilptr, Record[] stream.Seek(streamStartPos + key_addr->Address - start_addr, SeekOrigin.Begin); } - keySerializer.Deserialize(out src[ptr/recordSize].key); + keySerializer.Deserialize(out src[ptr / recordSize].key); } else {