Add ConcatenationFile resizing

This commit is contained in:
Alex Barney 2019-03-26 19:41:26 -05:00
parent 1ee3841125
commit f570c56122
3 changed files with 105 additions and 39 deletions

View file

@ -1,25 +1,30 @@
using System; using System;
using System.Collections.Generic; using System.Collections.Generic;
using System.Diagnostics;
using System.Linq; using System.Linq;
namespace LibHac.IO namespace LibHac.IO
{ {
public class ConcatenationFile : FileBase public class ConcatenationFile : FileBase
{ {
private IFile[] Sources { get; } private IFileSystem BaseFileSystem { get; }
private long SplitFileSize { get; } private string FilePath { get; }
private List<IFile> Sources { get; }
private long SubFileSize { get; }
internal ConcatenationFile(IList<IFile> sources, long splitFileSize, OpenMode mode) internal ConcatenationFile(IFileSystem baseFileSystem, string path, IEnumerable<IFile> sources, long subFileSize, OpenMode mode)
{ {
Sources = sources.ToArray(); BaseFileSystem = baseFileSystem;
SplitFileSize = splitFileSize; FilePath = path;
Sources = sources.ToList();
SubFileSize = subFileSize;
Mode = mode; Mode = mode;
for (int i = 0; i < Sources.Length - 1; i++) for (int i = 0; i < Sources.Count - 1; i++)
{ {
if (Sources[i].GetSize() != SplitFileSize) if (Sources[i].GetSize() != SubFileSize)
{ {
throw new ArgumentException($"Source file must have size {splitFileSize}"); throw new ArgumentException($"Source file must have size {subFileSize}");
} }
} }
@ -34,11 +39,11 @@ namespace LibHac.IO
while (remaining > 0) while (remaining > 0)
{ {
int fileIndex = GetFileIndexFromOffset(offset); int fileIndex = GetSubFileIndexFromOffset(offset);
IFile file = Sources[fileIndex]; IFile file = Sources[fileIndex];
long fileOffset = offset - fileIndex * SplitFileSize; long fileOffset = offset - fileIndex * SubFileSize;
long fileEndOffset = Math.Min((fileIndex + 1) * SplitFileSize, GetSize()); long fileEndOffset = Math.Min((fileIndex + 1) * SubFileSize, GetSize());
int bytesToRead = (int)Math.Min(fileEndOffset - inPos, remaining); int bytesToRead = (int)Math.Min(fileEndOffset - inPos, remaining);
int bytesRead = file.Read(destination.Slice(outPos, bytesToRead), fileOffset); int bytesRead = file.Read(destination.Slice(outPos, bytesToRead), fileOffset);
@ -56,19 +61,19 @@ namespace LibHac.IO
{ {
ValidateWriteParams(source, offset); ValidateWriteParams(source, offset);
long inPos = offset; int inPos = 0;
int outPos = 0; long outPos = offset;
int remaining = source.Length; int remaining = source.Length;
while (remaining > 0) while (remaining > 0)
{ {
int fileIndex = GetFileIndexFromOffset(offset); int fileIndex = GetSubFileIndexFromOffset(outPos);
IFile file = Sources[fileIndex]; IFile file = Sources[fileIndex];
long fileOffset = offset - fileIndex * SplitFileSize; long fileOffset = outPos - fileIndex * SubFileSize;
long fileEndOffset = Math.Min((fileIndex + 1) * SplitFileSize, GetSize()); long fileEndOffset = Math.Min((fileIndex + 1) * SubFileSize, GetSize());
int bytesToWrite = (int)Math.Min(fileEndOffset - inPos, remaining); int bytesToWrite = (int)Math.Min(fileEndOffset - outPos, remaining);
file.Write(source.Slice(outPos, bytesToWrite), fileOffset); file.Write(source.Slice(inPos, bytesToWrite), fileOffset);
outPos += bytesToWrite; outPos += bytesToWrite;
inPos += bytesToWrite; inPos += bytesToWrite;
@ -98,12 +103,73 @@ namespace LibHac.IO
public override void SetSize(long size) public override void SetSize(long size)
{ {
throw new NotImplementedException(); long currentSize = GetSize();
if (currentSize == size) return;
int currentSubFileCount = QuerySubFileCount(currentSize, SubFileSize);
int newSubFileCount = QuerySubFileCount(size, SubFileSize);
if (size > currentSize)
{
IFile currentLastSubFile = Sources[currentSubFileCount - 1];
long newSubFileSize = QuerySubFileSize(currentSubFileCount - 1, size, SubFileSize);
currentLastSubFile.SetSize(newSubFileSize);
for (int i = currentSubFileCount; i < newSubFileCount; i++)
{
string newSubFilePath = ConcatenationFileSystem.GetSubFilePath(FilePath, i);
newSubFileSize = QuerySubFileSize(i, size, SubFileSize);
BaseFileSystem.CreateFile(newSubFilePath, newSubFileSize, CreateFileOptions.None);
Sources.Add(BaseFileSystem.OpenFile(newSubFilePath, Mode));
}
}
else
{
for (int i = currentSubFileCount - 1; i > newSubFileCount - 1; i--)
{
Sources[i].Dispose();
Sources.RemoveAt(i);
string subFilePath = ConcatenationFileSystem.GetSubFilePath(FilePath, i);
BaseFileSystem.DeleteFile(subFilePath);
} }
private int GetFileIndexFromOffset(long offset) long newLastFileSize = QuerySubFileSize(newSubFileCount - 1, size, SubFileSize);
Sources[newSubFileCount - 1].SetSize(newLastFileSize);
}
}
private int GetSubFileIndexFromOffset(long offset)
{ {
return (int)(offset / SplitFileSize); return (int)(offset / SubFileSize);
}
private static int QuerySubFileCount(long size, long subFileSize)
{
Debug.Assert(size >= 0);
Debug.Assert(subFileSize > 0);
if (size == 0) return 1;
return (int)Util.DivideByRoundUp(size, subFileSize);
}
private static long QuerySubFileSize(int subFileIndex, long totalSize, long subFileSize)
{
int subFileCount = QuerySubFileCount(totalSize, subFileSize);
Debug.Assert(subFileIndex < subFileCount);
if (subFileIndex + 1 == subFileCount)
{
long remainder = totalSize % subFileSize;
return remainder == 0 ? subFileSize : remainder;
}
return subFileSize;
} }
} }
} }

View file

@ -6,16 +6,16 @@ namespace LibHac.IO
{ {
public class ConcatenationFileSystem : IFileSystem public class ConcatenationFileSystem : IFileSystem
{ {
private const long DefaultSplitFileSize = 0xFFFF0000; // Hard-coded value used by FS private const long DefaultSubFileSize = 0xFFFF0000; // Hard-coded value used by FS
private IAttributeFileSystem BaseFileSystem { get; } private IAttributeFileSystem BaseFileSystem { get; }
private long SplitFileSize { get; } private long SubFileSize { get; }
public ConcatenationFileSystem(IAttributeFileSystem baseFileSystem) : this(baseFileSystem, DefaultSplitFileSize) { } public ConcatenationFileSystem(IAttributeFileSystem baseFileSystem) : this(baseFileSystem, DefaultSubFileSize) { }
public ConcatenationFileSystem(IAttributeFileSystem baseFileSystem, long splitFileSize) public ConcatenationFileSystem(IAttributeFileSystem baseFileSystem, long subFileSize)
{ {
BaseFileSystem = baseFileSystem; BaseFileSystem = baseFileSystem;
SplitFileSize = splitFileSize; SubFileSize = subFileSize;
} }
internal bool IsConcatenationFile(string path) internal bool IsConcatenationFile(string path)
@ -64,8 +64,8 @@ namespace LibHac.IO
for (int i = 0; remaining > 0; i++) for (int i = 0; remaining > 0; i++)
{ {
long fileSize = Math.Min(SplitFileSize, remaining); long fileSize = Math.Min(SubFileSize, remaining);
string fileName = GetSplitFilePath(path, i); string fileName = GetSubFilePath(path, i);
BaseFileSystem.CreateFile(fileName, fileSize, CreateFileOptions.None); BaseFileSystem.CreateFile(fileName, fileSize, CreateFileOptions.None);
@ -94,11 +94,11 @@ namespace LibHac.IO
BaseFileSystem.DeleteFile(path); BaseFileSystem.DeleteFile(path);
} }
int count = GetSplitFileCount(path); int count = GetSubFileCount(path);
for (int i = 0; i < count; i++) for (int i = 0; i < count; i++)
{ {
BaseFileSystem.DeleteFile(GetSplitFilePath(path, i)); BaseFileSystem.DeleteFile(GetSubFilePath(path, i));
} }
BaseFileSystem.DeleteDirectory(path); BaseFileSystem.DeleteDirectory(path);
@ -127,18 +127,18 @@ namespace LibHac.IO
return BaseFileSystem.OpenFile(path, mode); return BaseFileSystem.OpenFile(path, mode);
} }
int fileCount = GetSplitFileCount(path); int fileCount = GetSubFileCount(path);
var files = new List<IFile>(); var files = new List<IFile>();
for (int i = 0; i < fileCount; i++) for (int i = 0; i < fileCount; i++)
{ {
string filePath = GetSplitFilePath(path, i); string filePath = GetSubFilePath(path, i);
IFile file = BaseFileSystem.OpenFile(filePath, mode); IFile file = BaseFileSystem.OpenFile(filePath, mode);
files.Add(file); files.Add(file);
} }
return new ConcatenationFile(files, SplitFileSize, mode); return new ConcatenationFile(BaseFileSystem, path, files, SubFileSize, mode);
} }
public void RenameDirectory(string srcPath, string dstPath) public void RenameDirectory(string srcPath, string dstPath)
@ -197,11 +197,11 @@ namespace LibHac.IO
BaseFileSystem.Commit(); BaseFileSystem.Commit();
} }
private int GetSplitFileCount(string dirPath) private int GetSubFileCount(string dirPath)
{ {
int count = 0; int count = 0;
while (BaseFileSystem.FileExists(GetSplitFilePath(dirPath, count))) while (BaseFileSystem.FileExists(GetSubFilePath(dirPath, count)))
{ {
count++; count++;
} }
@ -209,19 +209,19 @@ namespace LibHac.IO
return count; return count;
} }
private static string GetSplitFilePath(string dirPath, int index) internal static string GetSubFilePath(string dirPath, int index)
{ {
return $"{dirPath}/{index:D2}"; return $"{dirPath}/{index:D2}";
} }
internal long GetConcatenationFileSize(string path) internal long GetConcatenationFileSize(string path)
{ {
int fileCount = GetSplitFileCount(path); int fileCount = GetSubFileCount(path);
long size = 0; long size = 0;
for (int i = 0; i < fileCount; i++) for (int i = 0; i < fileCount; i++)
{ {
size += BaseFileSystem.GetFileSize(GetSplitFilePath(path, i)); size += BaseFileSystem.GetFileSize(GetSubFilePath(path, i));
} }
return size; return size;

View file

@ -174,7 +174,7 @@ namespace LibHac.IO
public static FileAttributes ApplyNxAttributes(this FileAttributes attributes, NxFileAttributes nxAttributes) public static FileAttributes ApplyNxAttributes(this FileAttributes attributes, NxFileAttributes nxAttributes)
{ {
var nxAttributeBits = (FileAttributes)(((int)attributes & 3) << 4); var nxAttributeBits = (FileAttributes)(((int)nxAttributes & 3) << 4);
return attributes | nxAttributeBits; return attributes | nxAttributeBits;
} }
} }