2021-01-17 08:30:51 +01:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using LibHac.Fs;
|
|
|
|
|
using LibHac.FsSystem;
|
|
|
|
|
using LibHac.FsSystem.Save;
|
|
|
|
|
using LibHac.Tests.Fs;
|
|
|
|
|
using Xunit;
|
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
namespace LibHac.Tests.FsSystem;
|
|
|
|
|
|
|
|
|
|
public class BufferedStorageTests
|
2021-01-17 08:30:51 +01:00
|
|
|
|
{
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Fact]
|
|
|
|
|
public void Write_SingleBlock_CanReadBack()
|
2021-01-17 08:30:51 +01:00
|
|
|
|
{
|
2021-11-14 20:08:57 +01:00
|
|
|
|
byte[] buffer = new byte[0x18000];
|
|
|
|
|
byte[] workBuffer = new byte[0x18000];
|
|
|
|
|
var bufferManager = new FileSystemBufferManager();
|
|
|
|
|
Assert.Success(bufferManager.Initialize(5, buffer, 0x4000, workBuffer));
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
byte[] storageBuffer = new byte[0x80000];
|
|
|
|
|
var baseStorage = new SubStorage(new MemoryStorage(storageBuffer), 0, storageBuffer.Length);
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
var bufferedStorage = new BufferedStorage();
|
|
|
|
|
Assert.Success(bufferedStorage.Initialize(baseStorage, bufferManager, 0x4000, 4));
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
byte[] writeBuffer = new byte[0x400];
|
|
|
|
|
byte[] readBuffer = new byte[0x400];
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
writeBuffer.AsSpan().Fill(0xAA);
|
|
|
|
|
Assert.Success(bufferedStorage.Write(0x10000, writeBuffer));
|
|
|
|
|
Assert.Success(bufferedStorage.Read(0x10000, readBuffer));
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
Assert.Equal(writeBuffer, readBuffer);
|
|
|
|
|
}
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
public class AccessTestConfig
|
|
|
|
|
{
|
|
|
|
|
public int[] SizeClassProbs { get; set; }
|
|
|
|
|
public int[] SizeClassMaxSizes { get; set; }
|
|
|
|
|
public int[] TaskProbs { get; set; }
|
|
|
|
|
public int[] AccessTypeProbs { get; set; }
|
|
|
|
|
public ulong RngSeed { get; set; }
|
|
|
|
|
public int FrequentAccessBlockCount { get; set; }
|
|
|
|
|
public int BlockSize { get; set; }
|
|
|
|
|
public int StorageCacheCount { get; set; }
|
|
|
|
|
public bool EnableBulkRead { get; set; }
|
|
|
|
|
public int StorageSize { get; set; }
|
|
|
|
|
public int HeapSize { get; set; }
|
|
|
|
|
public int HeapBlockSize { get; set; }
|
|
|
|
|
public int BufferManagerCacheCount { get; set; }
|
|
|
|
|
}
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
public static AccessTestConfig[] AccessTestConfigs =
|
|
|
|
|
{
|
2021-01-17 08:30:51 +01:00
|
|
|
|
new()
|
|
|
|
|
{
|
|
|
|
|
SizeClassProbs = new[] {50, 50, 5},
|
|
|
|
|
SizeClassMaxSizes = new[] {0x4000, 0x80000, 0x800000}, // 4 KB, 512 KB, 8 MB
|
|
|
|
|
TaskProbs = new[] {50, 50, 1}, // Read, Write, Flush
|
|
|
|
|
AccessTypeProbs = new[] {10, 10, 5}, // Random, Sequential, Frequent block
|
|
|
|
|
RngSeed = 35467,
|
|
|
|
|
FrequentAccessBlockCount = 6,
|
|
|
|
|
BlockSize = 0x4000,
|
|
|
|
|
StorageCacheCount = 40,
|
|
|
|
|
EnableBulkRead = true,
|
|
|
|
|
StorageSize = 0x1000000,
|
|
|
|
|
HeapSize = 0x180000,
|
|
|
|
|
HeapBlockSize = 0x4000,
|
|
|
|
|
BufferManagerCacheCount = 50
|
|
|
|
|
},
|
|
|
|
|
new()
|
|
|
|
|
{
|
|
|
|
|
SizeClassProbs = new[] {50, 50, 5},
|
|
|
|
|
SizeClassMaxSizes = new[] {0x4000, 0x80000, 0x800000}, // 4 KB, 512 KB, 8 MB
|
|
|
|
|
TaskProbs = new[] {50, 50, 1}, // Read, Write, Flush
|
|
|
|
|
AccessTypeProbs = new[] {10, 10, 5}, // Random, Sequential, Frequent block
|
|
|
|
|
RngSeed = 6548433,
|
|
|
|
|
FrequentAccessBlockCount = 6,
|
|
|
|
|
BlockSize = 0x4000,
|
|
|
|
|
StorageCacheCount = 40,
|
|
|
|
|
EnableBulkRead = false,
|
|
|
|
|
StorageSize = 0x1000000,
|
|
|
|
|
HeapSize = 0x180000,
|
|
|
|
|
HeapBlockSize = 0x4000,
|
|
|
|
|
BufferManagerCacheCount = 50
|
|
|
|
|
},
|
|
|
|
|
new()
|
|
|
|
|
{
|
|
|
|
|
SizeClassProbs = new[] {50, 50, 0},
|
|
|
|
|
SizeClassMaxSizes = new[] {0x4000, 0x80000, 0x800000}, // 4 KB, 512 KB, 8 MB
|
|
|
|
|
TaskProbs = new[] {50, 0, 0},
|
|
|
|
|
AccessTypeProbs = new[] {10, 10, 5}, // Random, Sequential, Frequent block
|
|
|
|
|
RngSeed = 756478,
|
|
|
|
|
FrequentAccessBlockCount = 16,
|
|
|
|
|
BlockSize = 0x4000,
|
|
|
|
|
StorageCacheCount = 8,
|
|
|
|
|
EnableBulkRead = true,
|
|
|
|
|
StorageSize = 0x1000000,
|
|
|
|
|
HeapSize = 0xE00000,
|
|
|
|
|
HeapBlockSize = 0x4000,
|
|
|
|
|
BufferManagerCacheCount = 0x400
|
|
|
|
|
},
|
|
|
|
|
new()
|
|
|
|
|
{
|
|
|
|
|
SizeClassProbs = new[] {50, 50, 0},
|
|
|
|
|
SizeClassMaxSizes = new[] {0x4000, 0x80000, 0x800000}, // 4 KB, 512 KB, 8 MB
|
|
|
|
|
TaskProbs = new[] {50, 0, 0},
|
|
|
|
|
AccessTypeProbs = new[] {0, 0, 5}, // Random, Sequential, Frequent block
|
|
|
|
|
RngSeed = 38197549,
|
|
|
|
|
FrequentAccessBlockCount = 16,
|
|
|
|
|
BlockSize = 0x4000,
|
|
|
|
|
StorageCacheCount = 16,
|
|
|
|
|
EnableBulkRead = false,
|
|
|
|
|
StorageSize = 0x1000000,
|
|
|
|
|
HeapSize = 0xE00000,
|
|
|
|
|
HeapBlockSize = 0x4000,
|
|
|
|
|
BufferManagerCacheCount = 0x400
|
|
|
|
|
},
|
|
|
|
|
new()
|
|
|
|
|
{
|
|
|
|
|
SizeClassProbs = new[] {50, 50, 0},
|
|
|
|
|
SizeClassMaxSizes = new[] {0x4000, 0x80000, 0x800000}, // 4 KB, 512 KB, 8 MB
|
|
|
|
|
TaskProbs = new[] {50, 50, 1}, // Read, Write, Flush
|
|
|
|
|
AccessTypeProbs = new[] {10, 10, 5}, // Random, Sequential, Frequent block
|
|
|
|
|
RngSeed = 567365,
|
|
|
|
|
FrequentAccessBlockCount = 6,
|
|
|
|
|
BlockSize = 0x4000,
|
|
|
|
|
StorageCacheCount = 8,
|
|
|
|
|
EnableBulkRead = false,
|
|
|
|
|
StorageSize = 0x100000,
|
|
|
|
|
HeapSize = 0x180000,
|
|
|
|
|
HeapBlockSize = 0x4000,
|
|
|
|
|
BufferManagerCacheCount = 50
|
|
|
|
|
},
|
|
|
|
|
new()
|
|
|
|
|
{
|
|
|
|
|
SizeClassProbs = new[] {50, 50, 0},
|
|
|
|
|
SizeClassMaxSizes = new[] {0x4000, 0x80000, 0x800000}, // 4 KB, 512 KB, 8 MB
|
|
|
|
|
TaskProbs = new[] {50, 50, 1}, // Read, Write, Flush
|
|
|
|
|
AccessTypeProbs = new[] {10, 10, 5}, // Random, Sequential, Frequent block
|
|
|
|
|
RngSeed = 949365,
|
|
|
|
|
FrequentAccessBlockCount = 6,
|
|
|
|
|
BlockSize = 0x4000,
|
|
|
|
|
StorageCacheCount = 8,
|
|
|
|
|
EnableBulkRead = false,
|
|
|
|
|
StorageSize = 0x100000,
|
|
|
|
|
HeapSize = 0x180000,
|
|
|
|
|
HeapBlockSize = 0x4000,
|
|
|
|
|
BufferManagerCacheCount = 50
|
|
|
|
|
},
|
|
|
|
|
new()
|
|
|
|
|
{
|
|
|
|
|
SizeClassProbs = new[] {50, 50, 10},
|
|
|
|
|
SizeClassMaxSizes = new[] {0x4000, 0x80000, 0x800000}, // 4 KB, 512 KB, 8 MB
|
|
|
|
|
TaskProbs = new[] {50, 50, 1}, // Read, Write, Flush
|
|
|
|
|
AccessTypeProbs = new[] {10, 10, 5}, // Random, Sequential, Frequent block
|
|
|
|
|
RngSeed = 670670,
|
|
|
|
|
FrequentAccessBlockCount = 16,
|
|
|
|
|
BlockSize = 0x4000,
|
|
|
|
|
StorageCacheCount = 8,
|
|
|
|
|
EnableBulkRead = true,
|
|
|
|
|
StorageSize = 0x1000000,
|
|
|
|
|
HeapSize = 0xE00000,
|
|
|
|
|
HeapBlockSize = 0x4000,
|
|
|
|
|
BufferManagerCacheCount = 0x400
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
private static TheoryData<T> CreateTheoryData<T>(IEnumerable<T> items)
|
|
|
|
|
{
|
|
|
|
|
var output = new TheoryData<T>();
|
|
|
|
|
|
|
|
|
|
foreach (T item in items)
|
2021-01-17 08:30:51 +01:00
|
|
|
|
{
|
2021-11-14 20:08:57 +01:00
|
|
|
|
output.Add(item);
|
|
|
|
|
}
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
return output;
|
|
|
|
|
}
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
public static TheoryData<AccessTestConfig> AccessTestTheoryData = CreateTheoryData(AccessTestConfigs);
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
[Theory]
|
|
|
|
|
[MemberData(nameof(AccessTestTheoryData))]
|
|
|
|
|
public void ReadWrite_AccessCorrectnessTestAgainstMemoryStorage(AccessTestConfig config)
|
|
|
|
|
{
|
|
|
|
|
int orderMax = FileSystemBuddyHeap.QueryOrderMax((nuint)config.HeapSize, (nuint)config.HeapBlockSize);
|
|
|
|
|
int workBufferSize = (int)FileSystemBuddyHeap.QueryWorkBufferSize(orderMax);
|
|
|
|
|
byte[] workBuffer = GC.AllocateArray<byte>(workBufferSize, true);
|
|
|
|
|
byte[] heapBuffer = new byte[config.HeapSize];
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
var bufferManager = new FileSystemBufferManager();
|
|
|
|
|
Assert.Success(bufferManager.Initialize(config.BufferManagerCacheCount, heapBuffer, config.HeapBlockSize, workBuffer));
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
byte[] memoryStorageArray = new byte[config.StorageSize];
|
|
|
|
|
byte[] bufferedStorageArray = new byte[config.StorageSize];
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
var memoryStorage = new MemoryStorage(memoryStorageArray);
|
|
|
|
|
var baseBufferedStorage = new SubStorage(new MemoryStorage(bufferedStorageArray), 0, bufferedStorageArray.Length);
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
var bufferedStorage = new BufferedStorage();
|
|
|
|
|
Assert.Success(bufferedStorage.Initialize(baseBufferedStorage, bufferManager, config.BlockSize, config.StorageCacheCount));
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
if (config.EnableBulkRead)
|
|
|
|
|
{
|
|
|
|
|
bufferedStorage.EnableBulkRead();
|
|
|
|
|
}
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
var memoryStorageEntry = new StorageTester.Entry(memoryStorage, memoryStorageArray);
|
|
|
|
|
var bufferedStorageEntry = new StorageTester.Entry(bufferedStorage, bufferedStorageArray);
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
var testerConfig = new StorageTester.Configuration()
|
|
|
|
|
{
|
|
|
|
|
Entries = new[] { memoryStorageEntry, bufferedStorageEntry },
|
|
|
|
|
SizeClassProbs = config.SizeClassProbs,
|
|
|
|
|
SizeClassMaxSizes = config.SizeClassMaxSizes,
|
|
|
|
|
TaskProbs = config.TaskProbs,
|
|
|
|
|
AccessTypeProbs = config.AccessTypeProbs,
|
|
|
|
|
RngSeed = config.RngSeed,
|
|
|
|
|
FrequentAccessBlockCount = config.FrequentAccessBlockCount
|
|
|
|
|
};
|
2021-01-17 08:30:51 +01:00
|
|
|
|
|
2021-11-14 20:08:57 +01:00
|
|
|
|
var tester = new StorageTester(testerConfig);
|
|
|
|
|
tester.Run(0x100);
|
2021-01-17 08:30:51 +01:00
|
|
|
|
}
|
|
|
|
|
}
|