2020-08-11 12:15:03 +00:00
|
|
|
|
using System;
|
2020-08-11 13:41:30 +00:00
|
|
|
|
using System.Buffers.Binary;
|
2020-08-11 12:15:03 +00:00
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.IO;
|
2020-08-11 13:41:30 +00:00
|
|
|
|
using System.Runtime.Versioning;
|
2020-08-11 12:15:03 +00:00
|
|
|
|
using System.Text;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using ICSharpCode.SharpZipLib.Zip.Compression.Streams;
|
|
|
|
|
using K4os.Compression.LZ4.Streams;
|
|
|
|
|
using Wabbajack.Common;
|
|
|
|
|
using File = Alphaleonis.Win32.Filesystem.File;
|
|
|
|
|
|
|
|
|
|
namespace Compression.BSA
|
|
|
|
|
{
|
|
|
|
|
public class FileRecord : IFile
|
|
|
|
|
{
|
2020-08-11 13:41:30 +00:00
|
|
|
|
public const int HeaderLength = 0x10;
|
2020-08-11 12:15:03 +00:00
|
|
|
|
|
2020-08-11 13:41:30 +00:00
|
|
|
|
private readonly ReadOnlyMemorySlice<byte> _headerData;
|
|
|
|
|
internal readonly int _index;
|
|
|
|
|
internal readonly int _overallIndex;
|
|
|
|
|
internal readonly FileNameBlock _nameBlock;
|
|
|
|
|
internal readonly Lazy<string> _name;
|
|
|
|
|
internal Lazy<(uint Size, uint OnDisk, uint Original)> _size;
|
|
|
|
|
|
|
|
|
|
public ulong Hash => BinaryPrimitives.ReadUInt64LittleEndian(_headerData);
|
|
|
|
|
protected uint RawSize => BinaryPrimitives.ReadUInt32LittleEndian(_headerData.Slice(0x8));
|
|
|
|
|
public uint Offset => BinaryPrimitives.ReadUInt32LittleEndian(_headerData.Slice(0xC));
|
|
|
|
|
public string Name => _name.Value;
|
|
|
|
|
public uint Size => _size.Value.Size;
|
|
|
|
|
|
|
|
|
|
public bool FlipCompression => (RawSize & (0x1 << 30)) > 0;
|
|
|
|
|
|
|
|
|
|
internal FolderRecord Folder { get; }
|
|
|
|
|
internal BSAReader BSA => Folder.BSA;
|
|
|
|
|
|
|
|
|
|
internal FileRecord(
|
|
|
|
|
FolderRecord folderRecord,
|
|
|
|
|
ReadOnlyMemorySlice<byte> data,
|
|
|
|
|
int index,
|
|
|
|
|
int overallIndex,
|
|
|
|
|
FileNameBlock nameBlock)
|
2020-08-11 12:15:03 +00:00
|
|
|
|
{
|
|
|
|
|
_index = index;
|
2020-08-11 13:41:30 +00:00
|
|
|
|
_overallIndex = overallIndex;
|
|
|
|
|
_headerData = data;
|
|
|
|
|
_nameBlock = nameBlock;
|
2020-08-11 12:15:03 +00:00
|
|
|
|
Folder = folderRecord;
|
2020-08-11 13:41:30 +00:00
|
|
|
|
_name = new Lazy<string>(GetName, System.Threading.LazyThreadSafetyMode.PublicationOnly);
|
2020-08-11 12:15:03 +00:00
|
|
|
|
|
2020-08-11 13:41:30 +00:00
|
|
|
|
// Will be replaced if CopyDataTo is called before value is created
|
|
|
|
|
_size = new Lazy<(uint Size, uint OnDisk, uint Original)>(
|
|
|
|
|
mode: System.Threading.LazyThreadSafetyMode.ExecutionAndPublication,
|
|
|
|
|
valueFactory: () =>
|
|
|
|
|
{
|
|
|
|
|
using var rdr = BSA.GetStream();
|
|
|
|
|
rdr.BaseStream.Position = Offset;
|
|
|
|
|
return ReadSize(rdr);
|
|
|
|
|
});
|
2020-08-11 12:15:03 +00:00
|
|
|
|
}
|
|
|
|
|
|
2020-08-11 13:41:30 +00:00
|
|
|
|
public RelativePath Path => new RelativePath(string.IsNullOrEmpty(Folder.Name) ? Name : Folder.Name + "\\" + Name, skipValidation: true);
|
2020-08-11 12:15:03 +00:00
|
|
|
|
|
|
|
|
|
public bool Compressed
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
2020-08-11 13:41:30 +00:00
|
|
|
|
if (FlipCompression) return !BSA.CompressedByDefault;
|
|
|
|
|
return BSA.CompressedByDefault;
|
2020-08-11 12:15:03 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public FileStateObject State => new BSAFileStateObject(this);
|
|
|
|
|
|
|
|
|
|
public async ValueTask CopyDataTo(Stream output)
|
|
|
|
|
{
|
2020-09-05 14:01:32 +00:00
|
|
|
|
await using var in_file = await BSA._streamFactory.GetStream().ConfigureAwait(false);
|
2020-08-11 12:15:03 +00:00
|
|
|
|
using var rdr = new BinaryReader(in_file);
|
2020-08-11 13:41:30 +00:00
|
|
|
|
rdr.BaseStream.Position = Offset;
|
|
|
|
|
|
|
|
|
|
(uint Size, uint OnDisk, uint Original) size = ReadSize(rdr);
|
|
|
|
|
if (!_size.IsValueCreated)
|
|
|
|
|
{
|
|
|
|
|
_size = new Lazy<(uint Size, uint OnDisk, uint Original)>(value: size);
|
|
|
|
|
}
|
2020-08-11 12:15:03 +00:00
|
|
|
|
|
2020-08-11 13:41:30 +00:00
|
|
|
|
if (BSA.HeaderType == VersionType.SSE)
|
2020-08-11 12:15:03 +00:00
|
|
|
|
{
|
|
|
|
|
if (Compressed)
|
|
|
|
|
{
|
|
|
|
|
using var r = LZ4Stream.Decode(rdr.BaseStream);
|
2020-08-11 13:41:30 +00:00
|
|
|
|
await r.CopyToLimitAsync(output, size.Original).ConfigureAwait(false);
|
2020-08-11 12:15:03 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2020-08-11 13:41:30 +00:00
|
|
|
|
await rdr.BaseStream.CopyToLimitAsync(output, size.OnDisk).ConfigureAwait(false);
|
2020-08-11 12:15:03 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (Compressed)
|
|
|
|
|
{
|
|
|
|
|
await using var z = new InflaterInputStream(rdr.BaseStream);
|
2020-08-11 13:41:30 +00:00
|
|
|
|
await z.CopyToLimitAsync(output, size.Original).ConfigureAwait(false);
|
2020-08-11 12:15:03 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
2020-08-11 13:41:30 +00:00
|
|
|
|
await rdr.BaseStream.CopyToLimitAsync(output, size.OnDisk).ConfigureAwait(false);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private string GetName()
|
|
|
|
|
{
|
|
|
|
|
var names = _nameBlock.Names.Value;
|
|
|
|
|
return names[_overallIndex].ReadStringTerm(BSA.HeaderType);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private (uint Size, uint OnDisk, uint Original) ReadSize(BinaryReader rdr)
|
|
|
|
|
{
|
|
|
|
|
uint size = RawSize;
|
|
|
|
|
if (FlipCompression)
|
|
|
|
|
size = size ^ (0x1 << 30);
|
|
|
|
|
|
|
|
|
|
if (Compressed)
|
|
|
|
|
size -= 4;
|
|
|
|
|
|
|
|
|
|
byte nameBlobOffset;
|
|
|
|
|
if (BSA.HasNameBlobs)
|
|
|
|
|
{
|
|
|
|
|
nameBlobOffset = rdr.ReadByte();
|
|
|
|
|
// Just skip, not using
|
|
|
|
|
rdr.BaseStream.Position += nameBlobOffset;
|
2020-08-25 01:34:57 +00:00
|
|
|
|
// Minus one more for the size of the name blob offset size
|
|
|
|
|
nameBlobOffset++;
|
2020-08-11 13:41:30 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
nameBlobOffset = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint originalSize;
|
|
|
|
|
if (Compressed)
|
|
|
|
|
{
|
|
|
|
|
originalSize = rdr.ReadUInt32();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
originalSize = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
uint onDiskSize = size - nameBlobOffset;
|
|
|
|
|
if (Compressed)
|
|
|
|
|
{
|
|
|
|
|
return (Size: originalSize, OnDisk: onDiskSize, Original: originalSize);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return (Size: onDiskSize, OnDisk: onDiskSize, Original: originalSize);
|
2020-08-11 12:15:03 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Dump(Action<string> print)
|
|
|
|
|
{
|
2020-08-11 13:41:30 +00:00
|
|
|
|
print($"Name: {Name}");
|
|
|
|
|
print($"Offset: {Offset}");
|
|
|
|
|
print($"Raw Size: {RawSize}");
|
2020-08-11 12:15:03 +00:00
|
|
|
|
print($"Index: {_index}");
|
|
|
|
|
}
|
2020-09-05 14:01:32 +00:00
|
|
|
|
|
|
|
|
|
public async ValueTask<IStreamFactory> GetStreamFactory()
|
|
|
|
|
{
|
|
|
|
|
var ms = new MemoryStream();
|
|
|
|
|
await CopyDataTo(ms);
|
|
|
|
|
ms.Position = 0;
|
|
|
|
|
return new MemoryStreamFactory(ms);
|
|
|
|
|
}
|
2020-08-11 12:15:03 +00:00
|
|
|
|
}
|
|
|
|
|
}
|