From 9efa549c57454bf3b8bb197d8a7e32c4bcd94276 Mon Sep 17 00:00:00 2001 From: Jeremy Powell Date: Thu, 21 Nov 2024 09:19:52 +1300 Subject: [PATCH] Detect mini FAT cycles with Floyds' algorithm --- OpenMcdf/MiniFatChainEnumerator.cs | 18 ++++++++++++++---- 1 file changed, 14 insertions(+), 4 deletions(-) diff --git a/OpenMcdf/MiniFatChainEnumerator.cs b/OpenMcdf/MiniFatChainEnumerator.cs index 77b565b..8704300 100644 --- a/OpenMcdf/MiniFatChainEnumerator.cs +++ b/OpenMcdf/MiniFatChainEnumerator.cs @@ -15,6 +15,7 @@ internal sealed class MiniFatChainEnumerator : ContextBase, IEnumerator uint index = uint.MaxValue; private uint current = uint.MaxValue; private long length = -1; + private uint slow = uint.MaxValue; // Floyd's cycle-finding algorithm public MiniFatChainEnumerator(RootContextSite rootContextSite, uint startSectorId) : base(rootContextSite) @@ -60,8 +61,8 @@ public bool MoveNext() } else if (!SectorType.IsFreeOrEndOfChain(current)) { - uint sectorId = Context.MiniFat[current]; - if (sectorId == SectorType.EndOfChain) + uint value = Context.MiniFat[current]; + if (value == SectorType.EndOfChain) { index = uint.MaxValue; current = uint.MaxValue; @@ -70,10 +71,18 @@ public bool MoveNext() uint nextIndex = index + 1; if (nextIndex > SectorType.Maximum) - throw new FileFormatException("Mini FAT chain is corrupt."); + throw new FileFormatException("Mini FAT chain length is greater than the maximum."); + + if (nextIndex % 2 == 0 && !SectorType.IsFreeOrEndOfChain(slow)) + { + // Slow might become free or end of chain while shrinking + slow = Context.MiniFat[slow]; + if (slow == value) + throw new FileFormatException("Mini FAT chain contains a loop."); + } index = nextIndex; - current = sectorId; + current = value; return true; } @@ -203,6 +212,7 @@ public void Reset() start = true; index = uint.MaxValue; current = uint.MaxValue; + slow = uint.MaxValue; } [ExcludeFromCodeCoverage]