diff --git a/CHANGELOG.md b/CHANGELOG.md index 4e53377bd..c95b062ba 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,6 +1,8 @@ ### Changelog -#### Version - 3.6.2.0 - TBD +#### Version - 3.7.0.0 - TBD +* Added Starfield support + * Note: Hashes were added earlier, but the earlier version was not fully compatible due to Wabbajack extracting the BA2 archives incorrectly. This has been fixed. * Updated GameFinder dependency * Updated WebView dependency * Updated other dependencies diff --git a/Wabbajack.App.Wpf/Properties/PublishProfiles/FolderProfile.pubxml b/Wabbajack.App.Wpf/Properties/PublishProfiles/FolderProfile.pubxml new file mode 100644 index 000000000..acb3e3351 --- /dev/null +++ b/Wabbajack.App.Wpf/Properties/PublishProfiles/FolderProfile.pubxml @@ -0,0 +1,18 @@ + + + + + Release + Any CPU + bin\Release\Publish + FileSystem + <_TargetId>Folder + net8.0-windows + win-x64 + true + false + false + + \ No newline at end of file diff --git a/Wabbajack.CLI/Properties/launchSettings.json b/Wabbajack.CLI/Properties/launchSettings.json new file mode 100644 index 000000000..be1633035 --- /dev/null +++ b/Wabbajack.CLI/Properties/launchSettings.json @@ -0,0 +1,7 @@ +{ + "profiles": { + "Wabbajack.CLI": { + "commandName": "Project" + } + } +} \ No newline at end of file diff --git a/Wabbajack.Compression.BSA.Test/CompressionTests.cs b/Wabbajack.Compression.BSA.Test/CompressionTests.cs index ead5ee50f..d881891ea 100644 --- a/Wabbajack.Compression.BSA.Test/CompressionTests.cs +++ b/Wabbajack.Compression.BSA.Test/CompressionTests.cs @@ -55,6 +55,7 @@ public async Task CanRecreateBSAs(string name, AbsolutePath path) { if (name == "tes4.bsa") return; // not sure why is is failing + var reader = await BSADispatch.Open(path); var dataStates = await reader.Files @@ -78,7 +79,6 @@ await dataStates.PDoAll( var rebuiltStream = new MemoryStream(); await build.Build(rebuiltStream, CancellationToken.None); - rebuiltStream.Position = 0; var reader2 = await BSADispatch.Open(new MemoryStreamFactory(rebuiltStream, path, path.LastModifiedUtc())); await reader.Files.Zip(reader2.Files) diff --git a/Wabbajack.Compression.BSA/FO4Archive/Builder.cs b/Wabbajack.Compression.BSA/BA2Archive/Builder.cs similarity index 88% rename from Wabbajack.Compression.BSA/FO4Archive/Builder.cs rename to Wabbajack.Compression.BSA/BA2Archive/Builder.cs index 7116fdb7a..ca13f4493 100644 --- a/Wabbajack.Compression.BSA/FO4Archive/Builder.cs +++ b/Wabbajack.Compression.BSA/BA2Archive/Builder.cs @@ -10,7 +10,7 @@ using Wabbajack.DTOs.BSA.FileStates; using Wabbajack.Paths.IO; -namespace Wabbajack.Compression.BSA.FO4Archive; +namespace Wabbajack.Compression.BSA.BA2Archive; public class Builder : IBuilder { @@ -33,7 +33,7 @@ public async ValueTask AddFile(AFile state, Stream src, CancellationToken token) break; case BA2EntryType.DX10: - var resultdx10 = await DX10FileEntryBuilder.Create((BA2DX10File)state, src, _slab, token); + var resultdx10 = await DX10FileEntryBuilder.Create((BA2DX10File)state, src, _slab, _state.Compression == 3, token); lock (_entries) { _entries.Add(resultdx10); @@ -59,6 +59,13 @@ public async ValueTask Build(Stream fs, CancellationToken token) bw.Write((uint) _entries.Count); var tableOffsetLoc = bw.BaseStream.Position; bw.Write((ulong) 0); + if(_state.Version == 2 || _state.Version == 3) + { + bw.Write(_state.Unknown1); + bw.Write(_state.Unknown2); + if (_state.Version == 3) + bw.Write(_state.Compression); + } foreach (var entry in _entries) entry.WriteHeader(bw, token); diff --git a/Wabbajack.Compression.BSA/BA2Archive/ChunkBuilder.cs b/Wabbajack.Compression.BSA/BA2Archive/ChunkBuilder.cs new file mode 100644 index 000000000..979067957 --- /dev/null +++ b/Wabbajack.Compression.BSA/BA2Archive/ChunkBuilder.cs @@ -0,0 +1,93 @@ +using System; +using System.IO; +using System.Threading; +using System.Threading.Tasks; +using ICSharpCode.SharpZipLib.Zip.Compression; +using ICSharpCode.SharpZipLib.Zip.Compression.Streams; +using K4os.Compression.LZ4; +using K4os.Compression.LZ4.Encoders; +using Wabbajack.Common; +using Wabbajack.DTOs.BSA.FileStates; +using Wabbajack.DTOs.GitHub; + +namespace Wabbajack.Compression.BSA.BA2Archive; + +public class ChunkBuilder +{ + private BA2Chunk _chunk; + private Stream _dataSlab; + private long _offsetOffset; + private uint _packSize; + + public static async Task Create(BA2DX10File state, BA2Chunk chunk, Stream src, + DiskSlabAllocator slab, bool useLz4Compression, CancellationToken token) + { + var builder = new ChunkBuilder {_chunk = chunk}; + + if (!chunk.Compressed) + { + builder._dataSlab = slab.Allocate(chunk.FullSz); + await src.CopyToLimitAsync(builder._dataSlab, (int) chunk.FullSz, token); + } + else + { + if (!useLz4Compression) + { + var deflater = new Deflater(Deflater.BEST_COMPRESSION); + await using var ms = new MemoryStream(); + await using (var ds = new DeflaterOutputStream(ms, deflater)) + { + ds.IsStreamOwner = false; + await src.CopyToLimitAsync(ds, (int)chunk.FullSz, token); + } + + builder._dataSlab = slab.Allocate(ms.Length); + ms.Position = 0; + await ms.CopyToLimitAsync(builder._dataSlab, (int)ms.Length, token); + builder._packSize = (uint)ms.Length; + } + else + { + byte[] full = new byte[chunk.FullSz]; + await using (var copyStream = new MemoryStream()) + { + await src.CopyToLimitAsync(copyStream, (int)chunk.FullSz, token); + full = copyStream.ToArray(); + } + var maxOutput = LZ4Codec.MaximumOutputSize((int)chunk.FullSz); + byte[] compressed = new byte[maxOutput]; + int compressedSize = LZ4Codec.Encode(full, 0, full.Length, compressed, 0, compressed.Length, LZ4Level.L12_MAX); + var ms = new MemoryStream(compressed, 0, compressedSize); + builder._dataSlab = slab.Allocate(compressedSize); + ms.Position = 0; + await ms.CopyToLimitAsync(builder._dataSlab, compressedSize, token); + builder._packSize = (uint)compressedSize; + } + } + + builder._dataSlab.Position = 0; + + return builder; + } + + public void WriteHeader(BinaryWriter bw) + { + _offsetOffset = bw.BaseStream.Position; + bw.Write((ulong) 0); + bw.Write(_packSize); + bw.Write(_chunk.FullSz); + bw.Write(_chunk.StartMip); + bw.Write(_chunk.EndMip); + bw.Write(_chunk.Align); + } + + public async ValueTask WriteData(BinaryWriter bw, CancellationToken token) + { + var pos = bw.BaseStream.Position; + bw.BaseStream.Position = _offsetOffset; + bw.Write((ulong) pos); + bw.BaseStream.Position = pos; + await _dataSlab.CopyToLimitAsync(bw.BaseStream, (int) _dataSlab.Length, token); + await _dataSlab.DisposeAsync(); + } +} \ No newline at end of file diff --git a/Wabbajack.Compression.BSA/BA2Archive/DX10Entry.cs b/Wabbajack.Compression.BSA/BA2Archive/DX10Entry.cs new file mode 100644 index 000000000..00c883123 --- /dev/null +++ b/Wabbajack.Compression.BSA/BA2Archive/DX10Entry.cs @@ -0,0 +1,179 @@ +using System; +using System.Collections.Generic; +using System.IO; +using System.Linq; +using System.Runtime.InteropServices; +using System.Text; +using System.Threading; +using System.Threading.Tasks; +using DirectXTex; +using ICSharpCode.SharpZipLib.Zip.Compression; +using K4os.Compression.LZ4; +using K4os.Compression.LZ4.Streams; +using Wabbajack.Common; +using Wabbajack.Compression.BSA.BA2Archive; +using Wabbajack.DTOs.BSA.FileStates; +using Wabbajack.DTOs.Streams; +using Wabbajack.Paths; + +namespace Wabbajack.Compression.BSA.BA2Archive; + +public class DX10Entry : IBA2FileEntry +{ + private readonly Reader _bsa; + private ushort _chunkHdrLen; + private List _chunks; + private uint _dirHash; + private string _extension; + private byte _format; + private ushort _height; + private int _index; + private uint _nameHash; + private byte _numChunks; + private byte _numMips; + private ushort _unk16; + private byte _unk8; + private ushort _width; + private readonly byte _isCubemap; + private readonly byte _tileMode; + + public DX10Entry(Reader ba2Reader, int idx) + { + _bsa = ba2Reader; + var _rdr = ba2Reader._rdr; + _nameHash = _rdr.ReadUInt32(); + FullPath = _nameHash.ToString("X"); + _extension = Encoding.UTF8.GetString(_rdr.ReadBytes(4)); + _dirHash = _rdr.ReadUInt32(); + _unk8 = _rdr.ReadByte(); + _numChunks = _rdr.ReadByte(); + _chunkHdrLen = _rdr.ReadUInt16(); + _height = _rdr.ReadUInt16(); + _width = _rdr.ReadUInt16(); + _numMips = _rdr.ReadByte(); + _format = _rdr.ReadByte(); + _isCubemap = _rdr.ReadByte(); + _tileMode = _rdr.ReadByte(); + _index = idx; + + _chunks = Enumerable.Range(0, _numChunks) + .Select(_ => new TextureChunk(_rdr)) + .ToList(); + } + private DirectXTexUtility.TexMetadata? _metadata = null; + + public DirectXTexUtility.TexMetadata Metadata + { + get + { + if (_metadata == null) + _metadata = DirectXTexUtility.GenerateMetadata(_width, _height, _numMips, (DirectXTexUtility.DXGIFormat)_format, _isCubemap == 1); + return (DirectXTexUtility.TexMetadata)_metadata; + } + } + + private uint _headerSize = 0; + public uint HeaderSize + { + get + { + if (_headerSize > 0) + return _headerSize; + uint size = 0; + size += (uint)Marshal.SizeOf(DirectXTexUtility.DDSHeader.DDSMagic); + size += (uint)Marshal.SizeOf(); + var pixelFormat = DirectXTexUtility.GetPixelFormat(Metadata); + var hasDx10Header = DirectXTexUtility.HasDx10Header(pixelFormat); + if (hasDx10Header) + size += (uint)Marshal.SizeOf(); + + return _headerSize = size; + } + } + + public string FullPath { get; set; } + + public RelativePath Path => FullPath.ToRelativePath(); + public uint Size => (uint)_chunks.Sum(f => f._fullSz) + HeaderSize; + + public AFile State => new BA2DX10File + { + Path = Path, + NameHash = _nameHash, + Extension = _extension, + DirHash = _dirHash, + Unk8 = _unk8, + ChunkHdrLen = _chunkHdrLen, + Height = _height, + Width = _width, + NumMips = _numMips, + PixelFormat = _format, + IsCubeMap = _isCubemap, + TileMode = _tileMode, + Index = _index, + Chunks = _chunks.Select(ch => new BA2Chunk + { + FullSz = ch._fullSz, + StartMip = ch._startMip, + EndMip = ch._endMip, + Align = ch._align, + Compressed = ch._packSz != 0 + }).ToArray() + }; + + public async ValueTask CopyDataTo(Stream output, CancellationToken token) + { + var bw = new BinaryWriter(output); + + WriteHeader(bw); + + await using var fs = await _bsa._streamFactory.GetStream(); + using var br = new BinaryReader(fs); + foreach (var chunk in _chunks) + { + var full = new byte[chunk._fullSz]; + var isCompressed = chunk._packSz != 0; + + br.BaseStream.Seek((long)chunk._offset, SeekOrigin.Begin); + + if (!isCompressed) + { + await br.BaseStream.ReadAsync(full, token); + } + else + { + var compressed = new byte[chunk._packSz]; + await br.BaseStream.ReadAsync(compressed, token); + if (_bsa._compression == 3) + { + LZ4Codec.PartialDecode(compressed, full); + } + else + { + var inflater = new Inflater(); + inflater.SetInput(compressed); + inflater.Inflate(full); + } + } + + await bw.BaseStream.WriteAsync(full, token); + } + } + + + public async ValueTask GetStreamFactory(CancellationToken token) + { + var ms = new MemoryStream(); + await CopyDataTo(ms, token); + ms.Position = 0; + return new MemoryStreamFactory(ms, Path, _bsa._streamFactory.LastModifiedUtc); + } + + private void WriteHeader(BinaryWriter bw) + { + DirectXTexUtility.GenerateDDSHeader(Metadata, DirectXTexUtility.DDSFlags.FORCEDX10EXTMISC2, out var header, out var header10); + var headerBytes = DirectXTexUtility.EncodeDDSHeader(header, header10); + bw.Write(headerBytes); + } +} + diff --git a/Wabbajack.Compression.BSA/FO4Archive/DX10FileEntryBuilder.cs b/Wabbajack.Compression.BSA/BA2Archive/DX10FileEntryBuilder.cs similarity index 62% rename from Wabbajack.Compression.BSA/FO4Archive/DX10FileEntryBuilder.cs rename to Wabbajack.Compression.BSA/BA2Archive/DX10FileEntryBuilder.cs index 86045356d..0af601d9a 100644 --- a/Wabbajack.Compression.BSA/FO4Archive/DX10FileEntryBuilder.cs +++ b/Wabbajack.Compression.BSA/BA2Archive/DX10FileEntryBuilder.cs @@ -1,18 +1,20 @@ +using DirectXTex; using System.Collections.Generic; using System.IO; +using System.Runtime.InteropServices; using System.Text; using System.Threading; using System.Threading.Tasks; -using Compression.BSA; using Wabbajack.DTOs.BSA.FileStates; using Wabbajack.DTOs.Texture; -namespace Wabbajack.Compression.BSA.FO4Archive; +namespace Wabbajack.Compression.BSA.BA2Archive; public class DX10FileEntryBuilder : IFileBuilder { private List _chunks; private BA2DX10File _state; + private uint _headerSize = 0; public uint FileHash => _state.NameHash; public uint DirHash => _state.DirHash; @@ -43,20 +45,36 @@ public async ValueTask WriteData(BinaryWriter wtr, CancellationToken token) foreach (var chunk in _chunks) await chunk.WriteData(wtr, token); } + public uint GetHeaderSize(BA2DX10File state) + { + if (_headerSize > 0) + return _headerSize; + + uint size = 0; + size += (uint)Marshal.SizeOf(DirectXTexUtility.DDSHeader.DDSMagic); + size += (uint)Marshal.SizeOf(); + var metadata = DirectXTexUtility.GenerateMetadata(state.Width, state.Height, state.NumMips, (DirectXTexUtility.DXGIFormat)state.PixelFormat, state.IsCubeMap == 1); + var pixelFormat = DirectXTexUtility.GetPixelFormat(metadata); + var hasDx10Header = DirectXTexUtility.HasDx10Header(pixelFormat); + if (hasDx10Header) + size += (uint)Marshal.SizeOf(); + + return _headerSize = size; + } - public static async Task Create(BA2DX10File state, Stream src, DiskSlabAllocator slab, + public static async Task Create(BA2DX10File state, Stream src, DiskSlabAllocator slab, bool useLz4Compression, CancellationToken token) { var builder = new DX10FileEntryBuilder {_state = state}; - var headerSize = DDS.HeaderSizeForFormat((DXGI_FORMAT) state.PixelFormat) + 4; + var headerSize = builder.GetHeaderSize(state); new BinaryReader(src).ReadBytes((int) headerSize); // This can't be parallel because it all runs off the same base IO stream. builder._chunks = new List(); foreach (var chunk in state.Chunks) - builder._chunks.Add(await ChunkBuilder.Create(state, chunk, src, slab, token)); + builder._chunks.Add(await ChunkBuilder.Create(state, chunk, src, slab, useLz4Compression, token)); return builder; } diff --git a/Wabbajack.Compression.BSA/FO4Archive/FileEntry.cs b/Wabbajack.Compression.BSA/BA2Archive/FileEntry.cs similarity index 98% rename from Wabbajack.Compression.BSA/FO4Archive/FileEntry.cs rename to Wabbajack.Compression.BSA/BA2Archive/FileEntry.cs index 8b3961b35..7d0edc289 100644 --- a/Wabbajack.Compression.BSA/FO4Archive/FileEntry.cs +++ b/Wabbajack.Compression.BSA/BA2Archive/FileEntry.cs @@ -9,7 +9,7 @@ using Wabbajack.DTOs.Streams; using Wabbajack.Paths; -namespace Wabbajack.Compression.BSA.FO4Archive; +namespace Wabbajack.Compression.BSA.BA2Archive; public class FileEntry : IBA2FileEntry { diff --git a/Wabbajack.Compression.BSA/FO4Archive/FileEntryBuilder.cs b/Wabbajack.Compression.BSA/BA2Archive/FileEntryBuilder.cs similarity index 97% rename from Wabbajack.Compression.BSA/FO4Archive/FileEntryBuilder.cs rename to Wabbajack.Compression.BSA/BA2Archive/FileEntryBuilder.cs index c620c6243..3d11a0d4c 100644 --- a/Wabbajack.Compression.BSA/FO4Archive/FileEntryBuilder.cs +++ b/Wabbajack.Compression.BSA/BA2Archive/FileEntryBuilder.cs @@ -6,7 +6,7 @@ using Wabbajack.Common; using Wabbajack.DTOs.BSA.FileStates; -namespace Wabbajack.Compression.BSA.FO4Archive; +namespace Wabbajack.Compression.BSA.BA2Archive; public class FileEntryBuilder : IFileBuilder { diff --git a/Wabbajack.Compression.BSA/FO4Archive/IBA2FileEntry.cs b/Wabbajack.Compression.BSA/BA2Archive/IBA2FileEntry.cs similarity index 72% rename from Wabbajack.Compression.BSA/FO4Archive/IBA2FileEntry.cs rename to Wabbajack.Compression.BSA/BA2Archive/IBA2FileEntry.cs index a89e965ef..d9cd85216 100644 --- a/Wabbajack.Compression.BSA/FO4Archive/IBA2FileEntry.cs +++ b/Wabbajack.Compression.BSA/BA2Archive/IBA2FileEntry.cs @@ -1,6 +1,6 @@ using Wabbajack.Compression.BSA.Interfaces; -namespace Wabbajack.Compression.BSA.FO4Archive; +namespace Wabbajack.Compression.BSA.BA2Archive; internal interface IBA2FileEntry : IFile { diff --git a/Wabbajack.Compression.BSA/FO4Archive/IFileBuilder.cs b/Wabbajack.Compression.BSA/BA2Archive/IFileBuilder.cs similarity index 87% rename from Wabbajack.Compression.BSA/FO4Archive/IFileBuilder.cs rename to Wabbajack.Compression.BSA/BA2Archive/IFileBuilder.cs index d9f5ffda8..cf78a00fd 100644 --- a/Wabbajack.Compression.BSA/FO4Archive/IFileBuilder.cs +++ b/Wabbajack.Compression.BSA/BA2Archive/IFileBuilder.cs @@ -2,7 +2,7 @@ using System.Threading; using System.Threading.Tasks; -namespace Wabbajack.Compression.BSA.FO4Archive; +namespace Wabbajack.Compression.BSA.BA2Archive; internal interface IFileBuilder { diff --git a/Wabbajack.Compression.BSA/FO4Archive/Reader.cs b/Wabbajack.Compression.BSA/BA2Archive/Reader.cs similarity index 81% rename from Wabbajack.Compression.BSA/FO4Archive/Reader.cs rename to Wabbajack.Compression.BSA/BA2Archive/Reader.cs index ecbf4dae4..686812c5a 100644 --- a/Wabbajack.Compression.BSA/FO4Archive/Reader.cs +++ b/Wabbajack.Compression.BSA/BA2Archive/Reader.cs @@ -7,7 +7,7 @@ using Wabbajack.DTOs.BSA.ArchiveStates; using Wabbajack.DTOs.Streams; -namespace Wabbajack.Compression.BSA.FO4Archive; +namespace Wabbajack.Compression.BSA.BA2Archive; public class Reader : IReader { @@ -15,9 +15,17 @@ public class Reader : IReader internal string _headerMagic; internal ulong _nameTableOffset; internal uint _numFiles; + internal uint _unknown1; + internal uint _unknown2; + internal uint _compression; internal BinaryReader _rdr; public IStreamFactory _streamFactory; internal BA2EntryType _type; + + /// + /// Fallout 4 - Version 1, 7 or 8 + /// Starfield - Version 2 or 3 + /// internal uint _version; private Reader(Stream stream) @@ -37,7 +45,10 @@ private Reader(Stream stream) Version = _version, HeaderMagic = _headerMagic, Type = _type, - HasNameTable = HasNameTable + HasNameTable = HasNameTable, + Unknown1 = _unknown1, + Unknown2 = _unknown2, + Compression = _compression, }; @@ -67,6 +78,10 @@ private Task LoadHeaders() _numFiles = _rdr.ReadUInt32(); _nameTableOffset = _rdr.ReadUInt64(); + _unknown1 = (_version == 2 || _version == 3) ? _rdr.ReadUInt32() : 0; + _unknown2 = (_version == 2 || _version == 3) ? _rdr.ReadUInt32() : 0; + _compression = (_version == 3) ? _rdr.ReadUInt32() : 0; + var files = new List(); for (var idx = 0; idx < _numFiles; idx += 1) switch (_type) @@ -94,4 +109,4 @@ private Task LoadHeaders() return Task.CompletedTask; } -} \ No newline at end of file +} diff --git a/Wabbajack.Compression.BSA/FO4Archive/TextureChunk.cs b/Wabbajack.Compression.BSA/BA2Archive/TextureChunk.cs similarity index 91% rename from Wabbajack.Compression.BSA/FO4Archive/TextureChunk.cs rename to Wabbajack.Compression.BSA/BA2Archive/TextureChunk.cs index 5003348cc..6028cccac 100644 --- a/Wabbajack.Compression.BSA/FO4Archive/TextureChunk.cs +++ b/Wabbajack.Compression.BSA/BA2Archive/TextureChunk.cs @@ -1,6 +1,6 @@ using System.IO; -namespace Wabbajack.Compression.BSA.FO4Archive; +namespace Wabbajack.Compression.BSA.BA2Archive; public class TextureChunk { diff --git a/Wabbajack.Compression.BSA/BSADispatch.cs b/Wabbajack.Compression.BSA/BSADispatch.cs index 7e989440d..8cff05ac2 100644 --- a/Wabbajack.Compression.BSA/BSADispatch.cs +++ b/Wabbajack.Compression.BSA/BSADispatch.cs @@ -22,7 +22,7 @@ public static async ValueTask Open(AbsolutePath filename) { FileType.TES3 => await Reader.Load(new NativeFileStreamFactory(filename)), FileType.BSA => await TES5Archive.Reader.Load(new NativeFileStreamFactory(filename)), - FileType.BA2 => await FO4Archive.Reader.Load(new NativeFileStreamFactory(filename)), + FileType.BA2 => await BA2Archive.Reader.Load(new NativeFileStreamFactory(filename)), _ => throw new InvalidDataException("Filename is not a .bsa or .ba2") }; } @@ -34,7 +34,7 @@ public static async ValueTask Open(IStreamFactory factory) { FileType.TES3 => await Reader.Load(factory), FileType.BSA => await TES5Archive.Reader.Load(factory), - FileType.BA2 => await FO4Archive.Reader.Load(factory), + FileType.BA2 => await BA2Archive.Reader.Load(factory), _ => throw new InvalidDataException("Filename is not a .bsa or .ba2") }; } @@ -46,7 +46,7 @@ public static async ValueTask Open(IStreamFactory factory, FileType sig { FileType.TES3 => await Reader.Load(factory), FileType.BSA => await TES5Archive.Reader.Load(factory), - FileType.BA2 => await FO4Archive.Reader.Load(factory), + FileType.BA2 => await BA2Archive.Reader.Load(factory), _ => throw new InvalidDataException("Filename is not a .bsa or .ba2") }; } @@ -57,7 +57,7 @@ public static IBuilder CreateBuilder(IArchive oldState, TemporaryFileManager man { TES3State tes3 => new Builder(tes3), BSAState bsa => TES5Archive.Builder.Create(bsa, manager), - BA2State ba2 => FO4Archive.Builder.Create(ba2, manager), + BA2State ba2 => BA2Archive.Builder.Create(ba2, manager), _ => throw new NotImplementedException() }; } diff --git a/Wabbajack.Compression.BSA/DDS.cs b/Wabbajack.Compression.BSA/DDS.cs deleted file mode 100644 index 8a7a43341..000000000 --- a/Wabbajack.Compression.BSA/DDS.cs +++ /dev/null @@ -1,215 +0,0 @@ -using System; -using System.IO; -using System.Runtime.InteropServices; -using Wabbajack.DTOs.Texture; - -namespace Compression.BSA; -/* - * Copied from https://raw.githubusercontent.com/AlexxEG/BSA_Browser/master/Sharp.BSA.BA2/BA2Util/DDS.cs - * which is also GPL3 code. Modified slightly for Wabbajack - * - */ - -/* - * Copied from dds.h. Includes (almost) only stuff I need in this project. - * - * Link: https://github.com/digitalutopia1/BA2Lib/blob/master/BA2Lib/dds.h - * - */ - -public class DDS -{ - public const int DDS_MAGIC = 0x20534444; // "DDS " - - public const int DDS_FOURCC = 0x00000004; // DDPF_FOURCC - public const int DDS_RGB = 0x00000040; // DDPF_RGB - public const int DDS_RGBA = 0x00000041; // DDPF_RGB | DDPF_ALPHAPIXELS - - public const int - DDS_HEADER_FLAGS_TEXTURE = 0x00001007; // DDSD_CAPS | DDSD_HEIGHT | DDSD_WIDTH | DDSD_PIXELFORMAT - - public const int DDS_HEADER_FLAGS_MIPMAP = 0x00020000; // DDSD_MIPMAPCOUNT - public const int DDS_HEADER_FLAGS_LINEARSIZE = 0x00080000; // DDSD_LINEARSIZE - - public const int DDS_SURFACE_FLAGS_TEXTURE = 0x00001000; // DDSCAPS_TEXTURE - public const int DDS_SURFACE_FLAGS_MIPMAP = 0x00400008; // DDSCAPS_COMPLEX | DDSCAPS_MIPMAP - - public const int DDS_ALPHA_MODE_UNKNOWN = 0x0; - - public static uint HeaderSizeForFormat(DXGI_FORMAT fmt) - { - switch (fmt) - { - case DXGI_FORMAT.BC1_UNORM_SRGB: - case DXGI_FORMAT.BC3_UNORM_SRGB: - case DXGI_FORMAT.BC4_UNORM: - case DXGI_FORMAT.BC5_SNORM: - case DXGI_FORMAT.BC6H_UF16: - case DXGI_FORMAT.BC7_UNORM: - case DXGI_FORMAT.BC7_UNORM_SRGB: - return DDS_HEADER_DXT10.Size + DDS_HEADER.Size; - default: - return DDS_HEADER.Size; - } - } - - public static uint MAKEFOURCC(char ch0, char ch1, char ch2, char ch3) - { - // This is alien to me... - return (byte) ch0 | ((uint) (byte) ch1 << 8) | ((uint) (byte) ch2 << 16) | ((uint) (byte) ch3 << 24); - } -} - -public enum DXT10_RESOURCE_DIMENSION -{ - DIMENSION_TEXTURE1D = 2, - DIMENSION_TEXTURE2D = 3, - DIMENSION_TEXTURE3D = 4 -} - -[Flags] -public enum DDSCAPS2 : uint -{ - CUBEMAP = 0x200, - CUBEMAP_POSITIVEX = 0x400, - CUBEMAP_NEGATIVEX = 0x800, - CUBEMAP_POSITIVEY = 0x1000, - CUBEMAP_NEGATIVEY = 0x2000, - CUBEMAP_POSITIVEZ = 0x4000, - CUBEMAP_NEGATIVEZ = 0x8000, - CUBEMAP_ALLFACES = 0xFC00 -} - -[StructLayout(LayoutKind.Sequential, Pack = 1)] -public struct DDS_HEADER -{ - public uint dwSize; - public uint dwHeaderFlags; - public uint dwHeight; - public uint dwWidth; - public uint dwPitchOrLinearSize; - public uint dwDepth; // only if DDS_HEADER_FLAGS_VOLUME is set in dwHeaderFlags - public uint dwMipMapCount; - public uint dwReserved1; // [11] - public DDS_PIXELFORMAT PixelFormat; // ddspf - public uint dwSurfaceFlags; - public uint dwCubemapFlags; - public uint dwReserved2; // [3] - - public uint GetSize() - { - // 9 uint + DDS_PIXELFORMAT uints + 2 uint arrays with 14 uints total - // each uint 4 bytes each - return 9 * 4 + PixelFormat.GetSize() + 14 * 4; - } - - - public void Write(BinaryWriter bw) - { - bw.Write(dwSize); - bw.Write(dwHeaderFlags); - bw.Write(dwHeight); - bw.Write(dwWidth); - bw.Write(dwPitchOrLinearSize); - bw.Write(dwDepth); - bw.Write(dwMipMapCount); - - // Just write it multiple times, since it's never assigned a value anyway - for (var i = 0; i < 11; i++) - bw.Write(dwReserved1); - - // DDS_PIXELFORMAT - bw.Write(PixelFormat.dwSize); - bw.Write(PixelFormat.dwFlags); - bw.Write(PixelFormat.dwFourCC); - bw.Write(PixelFormat.dwRGBBitCount); - bw.Write(PixelFormat.dwRBitMask); - bw.Write(PixelFormat.dwGBitMask); - bw.Write(PixelFormat.dwBBitMask); - bw.Write(PixelFormat.dwABitMask); - - bw.Write(dwSurfaceFlags); - bw.Write(dwCubemapFlags); - - // Just write it multiple times, since it's never assigned a value anyway - for (var i = 0; i < 3; i++) - bw.Write(dwReserved2); - } - - public static uint Size - { - get - { - unsafe - { - return (uint) (sizeof(DDS_HEADER) + sizeof(int) * 10 + sizeof(int) * 2); - } - - ; - } - } -} - -[StructLayout(LayoutKind.Sequential, Pack = 1)] -public struct DDS_HEADER_DXT10 -{ - public uint dxgiFormat; - public uint resourceDimension; - public uint miscFlag; - public uint arraySize; - public uint miscFlags2; - - public void Write(BinaryWriter bw) - { - bw.Write(dxgiFormat); - bw.Write(resourceDimension); - bw.Write(miscFlag); - bw.Write(arraySize); - bw.Write(miscFlags2); - } - - public static uint Size - { - get - { - unsafe - { - return (uint) sizeof(DDS_HEADER_DXT10); - } - - ; - } - } -} - -[StructLayout(LayoutKind.Sequential, Pack = 1)] -public struct DDS_PIXELFORMAT -{ - public uint dwSize; - public uint dwFlags; - public uint dwFourCC; - public uint dwRGBBitCount; - public uint dwRBitMask; - public uint dwGBitMask; - public uint dwBBitMask; - public uint dwABitMask; - - public DDS_PIXELFORMAT(uint size, uint flags, uint fourCC, uint rgbBitCount, uint rBitMask, uint gBitMask, - uint bBitMask, uint aBitMask) - { - dwSize = size; - dwFlags = flags; - dwFourCC = fourCC; - dwRGBBitCount = rgbBitCount; - dwRBitMask = rBitMask; - dwGBitMask = gBitMask; - dwBBitMask = bBitMask; - dwABitMask = aBitMask; - } - - public uint GetSize() - { - // 8 uints, each 4 bytes each - return 8 * 4; - } -} \ No newline at end of file diff --git a/Wabbajack.Compression.BSA/DirectXTexUtil.cs b/Wabbajack.Compression.BSA/DirectXTexUtil.cs new file mode 100644 index 000000000..b1c302fbf --- /dev/null +++ b/Wabbajack.Compression.BSA/DirectXTexUtil.cs @@ -0,0 +1,1116 @@ +// ------------------------------------------------------------------------ +// DirectXTex Utility - A simple class for generating DDS Headers +// Copyright(c) 2018 Philip/Scobalula +// +// Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +// in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +// furnished to do so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. +// ------------------------------------------------------------------------ +// Author: Philip/Scobalula +// Description: DirectXTex DDS Header Utilities +// Source: https://gist.github.com/Scobalula/d9474f3fcf3d5a2ca596fceb64e16c98 + +using System; +using System.Collections.Generic; +using System.Linq; +using System.Runtime.InteropServices; +using System.Text; +using System.Threading.Tasks; +using System.IO; + +namespace DirectXTex +{ + public class DirectXTexUtility + { + #region Enumerators + /// + /// DDS Formats + /// + public enum DXGIFormat : uint + { + UNKNOWN = 0, + R32G32B32A32TYPELESS = 1, + R32G32B32A32FLOAT = 2, + R32G32B32A32UINT = 3, + R32G32B32A32SINT = 4, + R32G32B32TYPELESS = 5, + R32G32B32FLOAT = 6, + R32G32B32UINT = 7, + R32G32B32SINT = 8, + R16G16B16A16TYPELESS = 9, + R16G16B16A16FLOAT = 10, + R16G16B16A16UNORM = 11, + R16G16B16A16UINT = 12, + R16G16B16A16SNORM = 13, + R16G16B16A16SINT = 14, + R32G32TYPELESS = 15, + R32G32FLOAT = 16, + R32G32UINT = 17, + R32G32SINT = 18, + R32G8X24TYPELESS = 19, + D32FLOATS8X24UINT = 20, + R32FLOATX8X24TYPELESS = 21, + X32TYPELESSG8X24UINT = 22, + R10G10B10A2TYPELESS = 23, + R10G10B10A2UNORM = 24, + R10G10B10A2UINT = 25, + R11G11B10FLOAT = 26, + R8G8B8A8TYPELESS = 27, + R8G8B8A8UNORM = 28, + R8G8B8A8UNORMSRGB = 29, + R8G8B8A8UINT = 30, + R8G8B8A8SNORM = 31, + R8G8B8A8SINT = 32, + R16G16TYPELESS = 33, + R16G16FLOAT = 34, + R16G16UNORM = 35, + R16G16UINT = 36, + R16G16SNORM = 37, + R16G16SINT = 38, + R32TYPELESS = 39, + D32FLOAT = 40, + R32FLOAT = 41, + R32UINT = 42, + R32SINT = 43, + R24G8TYPELESS = 44, + D24UNORMS8UINT = 45, + R24UNORMX8TYPELESS = 46, + X24TYPELESSG8UINT = 47, + R8G8TYPELESS = 48, + R8G8UNORM = 49, + R8G8UINT = 50, + R8G8SNORM = 51, + R8G8SINT = 52, + R16TYPELESS = 53, + R16FLOAT = 54, + D16UNORM = 55, + R16UNORM = 56, + R16UINT = 57, + R16SNORM = 58, + R16SINT = 59, + R8TYPELESS = 60, + R8UNORM = 61, + R8UINT = 62, + R8SNORM = 63, + R8SINT = 64, + A8UNORM = 65, + R1UNORM = 66, + R9G9B9E5SHAREDEXP = 67, + R8G8B8G8UNORM = 68, + G8R8G8B8UNORM = 69, + BC1TYPELESS = 70, + BC1UNORM = 71, + BC1UNORMSRGB = 72, + BC2TYPELESS = 73, + BC2UNORM = 74, + BC2UNORMSRGB = 75, + BC3TYPELESS = 76, + BC3UNORM = 77, + BC3UNORMSRGB = 78, + BC4TYPELESS = 79, + BC4UNORM = 80, + BC4SNORM = 81, + BC5TYPELESS = 82, + BC5UNORM = 83, + BC5SNORM = 84, + B5G6R5UNORM = 85, + B5G5R5A1UNORM = 86, + B8G8R8A8UNORM = 87, + B8G8R8X8UNORM = 88, + R10G10B10XRBIASA2UNORM = 89, + B8G8R8A8TYPELESS = 90, + B8G8R8A8UNORMSRGB = 91, + B8G8R8X8TYPELESS = 92, + B8G8R8X8UNORMSRGB = 93, + BC6HTYPELESS = 94, + BC6HUF16 = 95, + BC6HSF16 = 96, + BC7TYPELESS = 97, + BC7UNORM = 98, + BC7UNORMSRGB = 99, + AYUV = 100, + Y410 = 101, + Y416 = 102, + NV12 = 103, + P010 = 104, + P016 = 105, + OPAQUE420 = 106, + YUY2 = 107, + Y210 = 108, + Y216 = 109, + NV11 = 110, + AI44 = 111, + IA44 = 112, + P8 = 113, + A8P8 = 114, + B4G4R4A4UNORM = 115, + FORCEUINT = 0xffffffff + } + + /// + /// DDS Flags + /// + [Flags] + public enum DDSFlags + { + NONE = 0x0, + LEGACYDWORD = 0x1, + NOLEGACYEXPANSION = 0x2, + NOR10B10G10A2FIXUP = 0x4, + FORCERGB = 0x8, + NO16BPP = 0x10, + EXPANDLUMINANCE = 0x20, + BADDXTNTAILS = 0x40, + FORCEDX10EXT = 0x10000, + FORCEDX10EXTMISC2 = 0x20000, + } + + /// + /// Texture Dimension + /// + public enum TexDimension + { + TEXTURE1D = 2, + TEXTURE2D = 3, + TEXTURE3D = 4, + } + + /// + /// Misc. Texture Flags + /// + public enum TexMiscFlags : uint + { + TEXTURECUBE = 0x4, + }; + + /// + /// Misc. Texture Flags + /// + public enum TexMiscFlags2 : uint + { + TEXMISC2ALPHAMODEMASK = 0x7, + }; + + /// + /// Texture Alpha Modes + /// + public enum TexAlphaMode + { + UNKNOWN = 0, + STRAIGHT = 1, + PREMULTIPLIED = 2, + OPAQUE = 3, + CUSTOM = 4, + }; + + /// + /// CP Flags + /// + [Flags] + public enum CPFLAGS + { + NONE = 0x0, // Normal operation + LEGACYDWORD = 0x1, // Assume pitch is DWORD aligned instead of BYTE aligned + PARAGRAPH = 0x2, // Assume pitch is 16-byte aligned instead of BYTE aligned + YMM = 0x4, // Assume pitch is 32-byte aligned instead of BYTE aligned + ZMM = 0x8, // Assume pitch is 64-byte aligned instead of BYTE aligned + PAGE4K = 0x200, // Assume pitch is 4096-byte aligned instead of BYTE aligned + BADDXTNTAILS = 0x1000, // BC formats with malformed mipchain blocks smaller than 4x4 + BPP24 = 0x10000, // Override with a legacy 24 bits-per-pixel format size + BPP16 = 0x20000, // Override with a legacy 16 bits-per-pixel format size + BPP8 = 0x40000, // Override with a legacy 8 bits-per-pixel format size + }; + #endregion + + #region Structs/Classes + /// + /// Common Pixel Formats + /// + public class PixelFormats + { + /// + /// DDS Pixel Format Size + /// + public static readonly uint Size = (uint)Marshal.SizeOf(); + + #region PixelFormatsConstants + public const uint DDSFOURCC = 0x00000004; // DDPFFOURCC + public const uint DDSRGB = 0x00000040; // DDPFRGB + public const uint DDSRGBA = 0x00000041; // DDPFRGB | DDPFALPHAPIXELS + public const uint DDSLUMINANCE = 0x00020000; // DDPFLUMINANCE + public const uint DDSLUMINANCEA = 0x00020001; // DDPFLUMINANCE | DDPFALPHAPIXELS + public const uint DDSALPHAPIXELS = 0x00000001; // DDPFALPHAPIXELS + public const uint DDSALPHA = 0x00000002; // DDPFALPHA + public const uint DDSPAL8 = 0x00000020; // DDPFPALETTEINDEXED8 + public const uint DDSPAL8A = 0x00000021; // DDPFPALETTEINDEXED8 | DDPFALPHAPIXELS + public const uint DDSBUMPDUDV = 0x00080000; // DDPFBUMPDUDV + #endregion + + #region DDSPixelFormats + public static DDSHeader.DDSPixelFormat DXT1 = + new DDSHeader.DDSPixelFormat(Size, DDSFOURCC, MakePixelFormatFourCC('D', 'X', 'T', '1'), 0, 0, 0, 0, 0); + + public static DDSHeader.DDSPixelFormat DXT2 = + new DDSHeader.DDSPixelFormat(Size, DDSFOURCC, MakePixelFormatFourCC('D', 'X', 'T', '2'), 0, 0, 0, 0, 0); + + public static DDSHeader.DDSPixelFormat DXT3 = + new DDSHeader.DDSPixelFormat(Size, DDSFOURCC, MakePixelFormatFourCC('D', 'X', 'T', '3'), 0, 0, 0, 0, 0); + + public static DDSHeader.DDSPixelFormat DXT4 = + new DDSHeader.DDSPixelFormat(Size, DDSFOURCC, MakePixelFormatFourCC('D', 'X', 'T', '4'), 0, 0, 0, 0, 0); + + public static DDSHeader.DDSPixelFormat DXT5 = + new DDSHeader.DDSPixelFormat(Size, DDSFOURCC, MakePixelFormatFourCC('D', 'X', 'T', '5'), 0, 0, 0, 0, 0); + + public static DDSHeader.DDSPixelFormat BC4UNORM = + new DDSHeader.DDSPixelFormat(Size, DDSFOURCC, MakePixelFormatFourCC('B', 'C', '4', 'U'), 0, 0, 0, 0, 0); + + public static DDSHeader.DDSPixelFormat BC4SNORM = + new DDSHeader.DDSPixelFormat(Size, DDSFOURCC, MakePixelFormatFourCC('B', 'C', '4', 'S'), 0, 0, 0, 0, 0); + + public static DDSHeader.DDSPixelFormat BC5UNORM = + new DDSHeader.DDSPixelFormat(Size, DDSFOURCC, MakePixelFormatFourCC('B', 'C', '5', 'U'), 0, 0, 0, 0, 0); + + public static DDSHeader.DDSPixelFormat BC5SNORM = + new DDSHeader.DDSPixelFormat(Size, DDSFOURCC, MakePixelFormatFourCC('B', 'C', '5', 'S'), 0, 0, 0, 0, 0); + + public static DDSHeader.DDSPixelFormat R8G8B8G8 = + new DDSHeader.DDSPixelFormat(Size, DDSFOURCC, MakePixelFormatFourCC('R', 'G', 'B', 'G'), 0, 0, 0, 0, 0); + + public static DDSHeader.DDSPixelFormat G8R8G8B8 = + new DDSHeader.DDSPixelFormat(Size, DDSFOURCC, MakePixelFormatFourCC('G', 'R', 'G', 'B'), 0, 0, 0, 0, 0); + + public static DDSHeader.DDSPixelFormat YUY2 = + new DDSHeader.DDSPixelFormat(Size, DDSFOURCC, MakePixelFormatFourCC('Y', 'U', 'Y', '2'), 0, 0, 0, 0, 0); + + public static DDSHeader.DDSPixelFormat A8R8G8B8 = + new DDSHeader.DDSPixelFormat(Size, DDSRGBA, 0, 32, 0x00ff0000, 0x0000ff00, 0x000000ff, 0xff000000); + + public static DDSHeader.DDSPixelFormat X8R8G8B8 = + new DDSHeader.DDSPixelFormat(Size, DDSRGB, 0, 32, 0x00ff0000, 0x0000ff00, 0x000000ff, 0x00000000); + + public static DDSHeader.DDSPixelFormat A8B8G8R8 = + new DDSHeader.DDSPixelFormat(Size, DDSRGBA, 0, 32, 0x000000ff, 0x0000ff00, 0x00ff0000, 0xff000000); + + public static DDSHeader.DDSPixelFormat X8B8G8R8 = + new DDSHeader.DDSPixelFormat(Size, DDSRGB, 0, 32, 0x000000ff, 0x0000ff00, 0x00ff0000, 0x00000000); + + public static DDSHeader.DDSPixelFormat G16R16 = + new DDSHeader.DDSPixelFormat(Size, DDSRGB, 0, 32, 0x0000ffff, 0xffff0000, 0x00000000, 0x00000000); + + public static DDSHeader.DDSPixelFormat R5G6B5 = + new DDSHeader.DDSPixelFormat(Size, DDSRGB, 0, 16, 0x0000f800, 0x000007e0, 0x0000001f, 0x00000000); + + public static DDSHeader.DDSPixelFormat A1R5G5B5 = + new DDSHeader.DDSPixelFormat(Size, DDSRGBA, 0, 16, 0x00007c00, 0x000003e0, 0x0000001f, 0x00008000); + + public static DDSHeader.DDSPixelFormat A4R4G4B4 = + new DDSHeader.DDSPixelFormat(Size, DDSRGBA, 0, 16, 0x00000f00, 0x000000f0, 0x0000000f, 0x0000f000); + + public static DDSHeader.DDSPixelFormat R8G8B8 = + new DDSHeader.DDSPixelFormat(Size, DDSRGB, 0, 24, 0x00ff0000, 0x0000ff00, 0x000000ff, 0x00000000); + + public static DDSHeader.DDSPixelFormat L8 = + new DDSHeader.DDSPixelFormat(Size, DDSLUMINANCE, 0, 8, 0xff, 0x00, 0x00, 0x00); + + public static DDSHeader.DDSPixelFormat L16 = + new DDSHeader.DDSPixelFormat(Size, DDSLUMINANCE, 0, 16, 0xffff, 0x0000, 0x0000, 0x0000); + + public static DDSHeader.DDSPixelFormat A8L8 = + new DDSHeader.DDSPixelFormat(Size, DDSLUMINANCEA, 0, 16, 0x00ff, 0x0000, 0x0000, 0xff00); + + public static DDSHeader.DDSPixelFormat A8L8ALT = + new DDSHeader.DDSPixelFormat(Size, DDSLUMINANCEA, 0, 8, 0x00ff, 0x0000, 0x0000, 0xff00); + + public static DDSHeader.DDSPixelFormat A8 = + new DDSHeader.DDSPixelFormat(Size, DDSALPHA, 0, 8, 0x00, 0x00, 0x00, 0xff); + + public static DDSHeader.DDSPixelFormat V8U8 = + new DDSHeader.DDSPixelFormat(Size, DDSBUMPDUDV, 0, 16, 0x00ff, 0xff00, 0x0000, 0x0000); + + public static DDSHeader.DDSPixelFormat Q8W8V8U8 = + new DDSHeader.DDSPixelFormat(Size, DDSBUMPDUDV, 0, 32, 0x000000ff, 0x0000ff00, 0x00ff0000, 0xff000000); + + public static DDSHeader.DDSPixelFormat V16U16 = + new DDSHeader.DDSPixelFormat(Size, DDSBUMPDUDV, 0, 32, 0x0000ffff, 0xffff0000, 0x00000000, 0x00000000); + + public static DDSHeader.DDSPixelFormat DX10 = + new DDSHeader.DDSPixelFormat(Size, DDSFOURCC, MakePixelFormatFourCC('D', 'X', '1', '0'), 0, 0, 0, 0, 0); + #endregion + } + + /// + /// DDS Header + /// + public struct DDSHeader + { + /// + /// DDS Header Flags + /// + [Flags] + public enum HeaderFlags : uint + { + TEXTURE = 0x00001007, // DDSDCAPS | DDSDHEIGHT | DDSDWIDTH | DDSDPIXELFORMAT + MIPMAP = 0x00020000, // DDSDMIPMAPCOUNT + VOLUME = 0x00800000, // DDSDDEPTH + PITCH = 0x00000008, // DDSDPITCH + LINEARSIZE = 0x00080000, // DDSDLINEARSIZE + } + + /// + /// DDS Surface Flags + /// + public enum SurfaceFlags : uint + { + TEXTURE = 0x00001000, // DDSCAPSTEXTURE + MIPMAP = 0x00400008, // DDSCAPSCOMPLEX | DDSCAPSMIPMAP + CUBEMAP = 0x00000008, // DDSCAPSCOMPLEX + } + + /// + /// DDS Magic/Four CC + /// + public const uint DDSMagic = 0x20534444; + + /// + /// DDS Pixel Format + /// + public struct DDSPixelFormat + { + public uint Size; + public uint Flags; + public uint FourCC; + public uint RGBBitCount; + public uint RBitMask; + public uint GBitMask; + public uint BBitMask; + public uint ABitMask; + + /// + /// Creates a new DDS Pixel Format + /// + public DDSPixelFormat(uint size, uint flags, uint fourCC, uint rgbBitCount, uint rBitMask, uint gBitMask, uint bBitMask, uint aBitMask) + { + Size = size; + Flags = flags; + FourCC = fourCC; + RGBBitCount = rgbBitCount; + RBitMask = rBitMask; + GBitMask = gBitMask; + BBitMask = bBitMask; + ABitMask = aBitMask; + } + + } + + public uint Size; + public HeaderFlags Flags; + public uint Height; + public uint Width; + public uint PitchOrLinearSize; + public uint Depth; // only if DDSHEADERFLAGSVOLUME is set in flags + public uint MipMapCount; + [MarshalAs(UnmanagedType.ByValArray, SizeConst = 11)] + public uint[] Reserved1; + public DDSPixelFormat PixelFormat; + public uint Caps; + public uint Caps2; + public uint Caps3; + public uint Caps4; + public uint Reserved2; + } + + /// + /// DDS DX10 Header + /// + public struct DX10Header + { + public DXGIFormat Format; + public TexDimension ResourceDimension; + public TexMiscFlags MiscFlag; // see D3D11RESOURCEMISCFLAG + public uint ArraySize; + public uint MiscFlags2; // see DDSMISCFLAGS2 + } + + /// + /// Texture Metadata + /// + public struct TexMetadata + { + #region Properties + public long Width; + public long Height; // Should be 1 for 1D textures + public long Depth; // Should be 1 for 1D or 2D textures + public long ArraySize; // For cubemap, this is a multiple of 6 + public long MipLevels; + public TexMiscFlags MiscFlags; + public TexMiscFlags2 MiscFlags2; + public DXGIFormat Format; + public TexDimension Dimension; + #endregion + + /// + /// Creates a new Texture Metadata Structe + /// + public TexMetadata(long width, long height, long depth, long arraySize, long mipLevels, TexMiscFlags flags, TexMiscFlags2 flags2, DXGIFormat format, TexDimension dimension) + { + Width = width; + Height = height; + Depth = depth; + ArraySize = arraySize; + MipLevels = mipLevels; + MiscFlags = flags; + MiscFlags2 = flags2; + Format = format; + Dimension = dimension; + } + + /// + /// Checks Alpha Mode + /// + public bool IsPMAlpha() + { + return (TexAlphaMode)(MiscFlags2 & TexMiscFlags2.TEXMISC2ALPHAMODEMASK) == TexAlphaMode.PREMULTIPLIED; + } + + public bool IsCubeMap() + { + return (MiscFlags & TexMiscFlags.TEXTURECUBE) == TexMiscFlags.TEXTURECUBE; + } + } + #endregion + + #region HelperMethods + /// + /// Clamps Value to a range. + /// + /// Value to Clamp + /// Max value + /// Min value + /// Clamped Value + private static T Clamp(T value, T max, T min) where T : IComparable + { + return value.CompareTo(min) < 0 ? min : value.CompareTo(max) > 0 ? max : value; + } + + /// + /// Converts a Struct to a Byte array + /// + private static byte[] StructToBytes(T value) + { + // Size of Struct + int length = Marshal.SizeOf(); + // Destination + byte[] destination = new byte[length]; + // Get Pointer + IntPtr pointer = Marshal.AllocHGlobal(length); + // Convert it + Marshal.StructureToPtr(value, pointer, false); + Marshal.Copy(pointer, destination, 0, length); + Marshal.FreeHGlobal(pointer); + // Done + return destination; + } + + /// + /// Generates a FourCC Integer from Pixel Format Characters + /// + private static uint MakePixelFormatFourCC(char char1, char char2, char char3, char char4) + { + return Convert.ToByte(char1) | (uint)Convert.ToByte(char2) << 8 | (uint)Convert.ToByte(char3) << 16 | (uint)Convert.ToByte(char4) << 24; + } + + /// + /// Gets the Bits Per Pixel for the given format + /// + private static ulong BitsPerPixel(DXGIFormat format) + { + switch (format) + { + case DXGIFormat.R32G32B32A32TYPELESS: + case DXGIFormat.R32G32B32A32FLOAT: + case DXGIFormat.R32G32B32A32UINT: + case DXGIFormat.R32G32B32A32SINT: + return 128; + case DXGIFormat.R32G32B32TYPELESS: + case DXGIFormat.R32G32B32FLOAT: + case DXGIFormat.R32G32B32UINT: + case DXGIFormat.R32G32B32SINT: + return 96; + case DXGIFormat.R16G16B16A16TYPELESS: + case DXGIFormat.R16G16B16A16FLOAT: + case DXGIFormat.R16G16B16A16UNORM: + case DXGIFormat.R16G16B16A16UINT: + case DXGIFormat.R16G16B16A16SNORM: + case DXGIFormat.R16G16B16A16SINT: + case DXGIFormat.R32G32TYPELESS: + case DXGIFormat.R32G32FLOAT: + case DXGIFormat.R32G32UINT: + case DXGIFormat.R32G32SINT: + case DXGIFormat.R32G8X24TYPELESS: + case DXGIFormat.D32FLOATS8X24UINT: + case DXGIFormat.R32FLOATX8X24TYPELESS: + case DXGIFormat.X32TYPELESSG8X24UINT: + case DXGIFormat.Y416: + case DXGIFormat.Y210: + case DXGIFormat.Y216: + return 64; + case DXGIFormat.R10G10B10A2TYPELESS: + case DXGIFormat.R10G10B10A2UNORM: + case DXGIFormat.R10G10B10A2UINT: + case DXGIFormat.R11G11B10FLOAT: + case DXGIFormat.R8G8B8A8TYPELESS: + case DXGIFormat.R8G8B8A8UNORM: + case DXGIFormat.R8G8B8A8UNORMSRGB: + case DXGIFormat.R8G8B8A8UINT: + case DXGIFormat.R8G8B8A8SNORM: + case DXGIFormat.R8G8B8A8SINT: + case DXGIFormat.R16G16TYPELESS: + case DXGIFormat.R16G16FLOAT: + case DXGIFormat.R16G16UNORM: + case DXGIFormat.R16G16UINT: + case DXGIFormat.R16G16SNORM: + case DXGIFormat.R16G16SINT: + case DXGIFormat.R32TYPELESS: + case DXGIFormat.D32FLOAT: + case DXGIFormat.R32FLOAT: + case DXGIFormat.R32UINT: + case DXGIFormat.R32SINT: + case DXGIFormat.R24G8TYPELESS: + case DXGIFormat.D24UNORMS8UINT: + case DXGIFormat.R24UNORMX8TYPELESS: + case DXGIFormat.X24TYPELESSG8UINT: + case DXGIFormat.R9G9B9E5SHAREDEXP: + case DXGIFormat.R8G8B8G8UNORM: + case DXGIFormat.G8R8G8B8UNORM: + case DXGIFormat.B8G8R8A8UNORM: + case DXGIFormat.B8G8R8X8UNORM: + case DXGIFormat.R10G10B10XRBIASA2UNORM: + case DXGIFormat.B8G8R8A8TYPELESS: + case DXGIFormat.B8G8R8A8UNORMSRGB: + case DXGIFormat.B8G8R8X8TYPELESS: + case DXGIFormat.B8G8R8X8UNORMSRGB: + case DXGIFormat.AYUV: + case DXGIFormat.Y410: + case DXGIFormat.YUY2: + return 32; + case DXGIFormat.P010: + case DXGIFormat.P016: + return 24; + case DXGIFormat.R8G8TYPELESS: + case DXGIFormat.R8G8UNORM: + case DXGIFormat.R8G8UINT: + case DXGIFormat.R8G8SNORM: + case DXGIFormat.R8G8SINT: + case DXGIFormat.R16TYPELESS: + case DXGIFormat.R16FLOAT: + case DXGIFormat.D16UNORM: + case DXGIFormat.R16UNORM: + case DXGIFormat.R16UINT: + case DXGIFormat.R16SNORM: + case DXGIFormat.R16SINT: + case DXGIFormat.B5G6R5UNORM: + case DXGIFormat.B5G5R5A1UNORM: + case DXGIFormat.A8P8: + case DXGIFormat.B4G4R4A4UNORM: + return 16; + case DXGIFormat.NV12: + case DXGIFormat.OPAQUE420: + case DXGIFormat.NV11: + return 12; + case DXGIFormat.R8TYPELESS: + case DXGIFormat.R8UNORM: + case DXGIFormat.R8UINT: + case DXGIFormat.R8SNORM: + case DXGIFormat.R8SINT: + case DXGIFormat.A8UNORM: + case DXGIFormat.BC2TYPELESS: + case DXGIFormat.BC2UNORM: + case DXGIFormat.BC2UNORMSRGB: + case DXGIFormat.BC3TYPELESS: + case DXGIFormat.BC3UNORM: + case DXGIFormat.BC3UNORMSRGB: + case DXGIFormat.BC5TYPELESS: + case DXGIFormat.BC5UNORM: + case DXGIFormat.BC5SNORM: + case DXGIFormat.BC6HTYPELESS: + case DXGIFormat.BC6HUF16: + case DXGIFormat.BC6HSF16: + case DXGIFormat.BC7TYPELESS: + case DXGIFormat.BC7UNORM: + case DXGIFormat.BC7UNORMSRGB: + case DXGIFormat.AI44: + case DXGIFormat.IA44: + case DXGIFormat.P8: + return 8; + case DXGIFormat.R1UNORM: + return 1; + case DXGIFormat.BC1TYPELESS: + case DXGIFormat.BC1UNORM: + case DXGIFormat.BC1UNORMSRGB: + case DXGIFormat.BC4TYPELESS: + case DXGIFormat.BC4UNORM: + case DXGIFormat.BC4SNORM: + return 4; + default: + return 0; + } + } + + /// + /// Computes Row and Slice Pitch + /// + private static void ComputePitch(DXGIFormat format, uint width, uint height, out ulong rowPitch, out ulong slicePitch, CPFLAGS flags) + { + switch (format) + { + case DXGIFormat.BC1TYPELESS: + case DXGIFormat.BC1UNORM: + case DXGIFormat.BC1UNORMSRGB: + case DXGIFormat.BC4TYPELESS: + case DXGIFormat.BC4UNORM: + case DXGIFormat.BC4SNORM: + { + if (flags.HasFlag(CPFLAGS.BADDXTNTAILS)) + { + var nbw = width >> 2; + var nbh = height >> 2; + rowPitch = Clamp((ulong) nbw * 8u, ulong.MaxValue, 1u); + slicePitch = Clamp(rowPitch * nbh, ulong.MaxValue, 1u); + } + else + { + var nbw = Clamp(((ulong) width + 3u) / 4u, ulong.MaxValue, 1u); + var nbh = Clamp(((ulong) height + 3u) / 4u, ulong.MaxValue, 1u); + rowPitch = nbw * 8u; + slicePitch = rowPitch * nbh; + } + } + break; + case DXGIFormat.BC2TYPELESS: + case DXGIFormat.BC2UNORM: + case DXGIFormat.BC2UNORMSRGB: + case DXGIFormat.BC3TYPELESS: + case DXGIFormat.BC3UNORM: + case DXGIFormat.BC3UNORMSRGB: + case DXGIFormat.BC5TYPELESS: + case DXGIFormat.BC5UNORM: + case DXGIFormat.BC5SNORM: + case DXGIFormat.BC6HTYPELESS: + case DXGIFormat.BC6HUF16: + case DXGIFormat.BC6HSF16: + case DXGIFormat.BC7TYPELESS: + case DXGIFormat.BC7UNORM: + case DXGIFormat.BC7UNORMSRGB: + { + if (flags.HasFlag(CPFLAGS.BADDXTNTAILS)) + { + var nbw = width >> 2; + var nbh = height >> 2; + rowPitch = Clamp((ulong) nbw * 16u, ulong.MaxValue, 1u); + slicePitch = Clamp(rowPitch * nbh, ulong.MaxValue, 1u); + } + else + { + var nbw = Clamp((width + 3) / 4, ulong.MaxValue, 1u); + var nbh = Clamp((height + 3) / 4, ulong.MaxValue, 1u); + rowPitch = nbw * 16u; + slicePitch = rowPitch * nbh; + } + } + break; + case DXGIFormat.R8G8B8G8UNORM: + case DXGIFormat.G8R8G8B8UNORM: + case DXGIFormat.YUY2: + rowPitch = ((width + 1) >> 1) * 4; + slicePitch = rowPitch * height; + break; + case DXGIFormat.Y210: + case DXGIFormat.Y216: + rowPitch = ((width + 1) >> 1) * 8; + slicePitch = rowPitch * height; + break; + + case DXGIFormat.NV12: + case DXGIFormat.OPAQUE420: + rowPitch = ((width + 1) >> 1) * 2; + slicePitch = rowPitch * (height + ((height + 1) >> 1)); + break; + + case DXGIFormat.P010: + case DXGIFormat.P016: + rowPitch = ((width + 1) >> 1) * 4; + slicePitch = rowPitch * (height + ((height + 1) >> 1)); + break; + case DXGIFormat.NV11: + rowPitch = ((width + 3) >> 2) * 4; + slicePitch = rowPitch * height * 2; + break; + default: + { + ulong bpp; + + if (flags.HasFlag(CPFLAGS.BPP24)) + bpp = 24; + else if (flags.HasFlag(CPFLAGS.BPP16)) + bpp = 16; + else if (flags.HasFlag(CPFLAGS.BPP8)) + bpp = 8; + else + bpp = BitsPerPixel(format); + + if (flags.HasFlag(CPFLAGS.LEGACYDWORD | CPFLAGS.PARAGRAPH | CPFLAGS.YMM | CPFLAGS.ZMM | CPFLAGS.PAGE4K)) + { + if (flags.HasFlag(CPFLAGS.PAGE4K)) + { + rowPitch = (width * bpp + 32767u) / 32768u * 4096u; + slicePitch = rowPitch * height; + } + else if (flags.HasFlag(CPFLAGS.ZMM)) + { + rowPitch = (width * bpp + 511u) / 512u * 64u; + slicePitch = rowPitch * height; + } + else if (flags.HasFlag(CPFLAGS.YMM)) + { + rowPitch = (width * bpp + 255u) / 256u * 32u; + slicePitch = rowPitch * height; + } + else if (flags.HasFlag(CPFLAGS.PARAGRAPH)) + { + rowPitch = (width * bpp + 127u) / 128u * 16u; + slicePitch = rowPitch * height; + } + else // DWORD alignment + { + // Special computation for some incorrectly created DDS files based on + // legacy DirectDraw assumptions about pitch alignment + rowPitch = (width * bpp + 31u) / 32u * sizeof(uint); + slicePitch = rowPitch * height; + } + } + else + { + // Default byte alignment + rowPitch = (width * bpp + 7u) / 8u; + slicePitch = rowPitch * height; + } + } + break; + } + } + + /// + /// Checks is the given format compressed + /// + private static bool IsCompressed(DXGIFormat format) + { + switch (format) + { + case DXGIFormat.BC1TYPELESS: + case DXGIFormat.BC1UNORM: + case DXGIFormat.BC1UNORMSRGB: + case DXGIFormat.BC2TYPELESS: + case DXGIFormat.BC2UNORM: + case DXGIFormat.BC2UNORMSRGB: + case DXGIFormat.BC3TYPELESS: + case DXGIFormat.BC3UNORM: + case DXGIFormat.BC3UNORMSRGB: + case DXGIFormat.BC4TYPELESS: + case DXGIFormat.BC4UNORM: + case DXGIFormat.BC4SNORM: + case DXGIFormat.BC5TYPELESS: + case DXGIFormat.BC5UNORM: + case DXGIFormat.BC5SNORM: + case DXGIFormat.BC6HTYPELESS: + case DXGIFormat.BC6HUF16: + case DXGIFormat.BC6HSF16: + case DXGIFormat.BC7TYPELESS: + case DXGIFormat.BC7UNORM: + case DXGIFormat.BC7UNORMSRGB: + return true; + + default: + return false; + } + } + #endregion + + #region MainMethods + /// + /// Encodes the DDS Header and if DX10, the DX10 Header + /// + /// DDS Header + /// DX10 Header + /// Resulting DDS File Header in bytes + public static byte[] EncodeDDSHeader(DDSHeader header, DX10Header dx10Header) + { + // Create stream + using (var output = new BinaryWriter(new MemoryStream())) + { + // Write DDS Magic + output.Write(DDSHeader.DDSMagic); + // Write Header + output.Write(StructToBytes(header)); + // Check for DX10 Header + if (header.PixelFormat.FourCC == PixelFormats.DX10.FourCC) + // Write Header + output.Write(StructToBytes(dx10Header)); + // Done + return ((MemoryStream)(output.BaseStream)).ToArray(); + } + } + + /// + /// Generates DirectXTex Meta Data + /// + /// Image Width + /// Image Height + /// Number of Mip Maps + /// Compression Format + /// Whether or not this is a cube map + /// Resulting TexMetaData Object + public static TexMetadata GenerateMetadata(int width, int height, int mipMapLevels, DXGIFormat format, bool isCubeMap) + { + // Create Texture MetaData + return new TexMetadata( + width, + height, + 1, + isCubeMap ? 6 : 1, + mipMapLevels, + isCubeMap ? TexMiscFlags.TEXTURECUBE : 0, + 0, + format, + TexDimension.TEXTURE2D + ); + } + + private static readonly uint[] _elevenUInts = Enumerable.Repeat((uint)0, 11).ToArray(); + + /// + /// Generates a DDS Header, and if requires, a DX10 Header + /// + /// Meta Data + /// Flags + /// DDS Header Output + /// DX10 Header Output + public static void GenerateDDSHeader(TexMetadata metaData, DDSFlags flags, out DDSHeader header, out DX10Header dx10Header) + { + // Check array size + if (metaData.ArraySize > 1) + // Check if we have an array and whether we're cube maps/non-2D + if (metaData.ArraySize != 6 || metaData.Dimension != TexDimension.TEXTURE2D || !metaData.IsCubeMap()) + // Texture1D arrays, Texture2D arrays, and Cubemap arrays must be stored using 'DX10' extended header + flags |= DDSFlags.FORCEDX10EXT; + + // Check for DX10 Ext + if (flags.HasFlag(DDSFlags.FORCEDX10EXTMISC2)) + flags |= DDSFlags.FORCEDX10EXT; + + // Create DDS Header + header = new DDSHeader + { + // Set Data + Size = (uint) Marshal.SizeOf(), + Flags = DDSHeader.HeaderFlags.TEXTURE, + Height = 0, + Width = 0, + PitchOrLinearSize = 0, + Depth = 0, + MipMapCount = 0, + Reserved1 = _elevenUInts, + Caps = (uint) DDSHeader.SurfaceFlags.TEXTURE, + Caps2 = 0, + Caps3 = 0, + Caps4 = 0, + Reserved2 = 0, + PixelFormat = new DDSHeader.DDSPixelFormat(), + + }; + + // Create DX10 Header + dx10Header = new DX10Header + { + Format = 0, + ResourceDimension = (TexDimension) 0, + MiscFlag = (TexMiscFlags) 0, + ArraySize = 0, + MiscFlags2 = 0, + }; + + // Switch format + header.PixelFormat = GetPixelFormat(metaData); + // Check for mips + if (metaData.MipLevels > 0) + { + // Set flag + header.Flags |= DDSHeader.HeaderFlags.MIPMAP; + // Check size + if (metaData.MipLevels > UInt16.MaxValue) + throw new ArgumentException(String.Format("Too many mipmaps: {0}. Max: {1}", metaData.MipLevels, UInt16.MaxValue)); + // Set + header.MipMapCount = (uint)metaData.MipLevels; + // Check count + if (header.MipMapCount > 1) + header.Caps |= (uint)DDSHeader.SurfaceFlags.MIPMAP; + } + + // Switch Dimension + switch (metaData.Dimension) + { + case TexDimension.TEXTURE1D: + { + // Check size + if (metaData.Width > Int32.MaxValue) + throw new ArgumentException(String.Format("Image Width too large: {0}. Max: {1}", metaData.Width, Int32.MaxValue)); + // Set + header.Width = (uint)metaData.Width; + header.Height = header.Depth = 1; + // Check size + break; + } + case TexDimension.TEXTURE2D: + { + // Check size + if (metaData.Width > Int32.MaxValue || metaData.Height > Int32.MaxValue) + throw new ArgumentException(String.Format("Image Width and/or Height too large: {0}x{1}. Max: {2}", + metaData.Width, + metaData.Height, + Int32.MaxValue)); + // Set + header.Width = (uint)metaData.Width; + header.Height = (uint)metaData.Height; + header.Depth = 1; + // Check size + break; + } + case TexDimension.TEXTURE3D: + { + // Check size + if (metaData.Width > Int32.MaxValue || metaData.Height > Int32.MaxValue) + throw new ArgumentException(String.Format("Image Width and/or Height too large: {0}x{1}. Max: {2}", + metaData.Width, + metaData.Height, + Int32.MaxValue)); + // Check size + if (metaData.Depth > UInt16.MaxValue) + throw new ArgumentException(String.Format("Image Depth too large: {0}. Max: {1}", metaData.Depth, UInt16.MaxValue)); + // Set + header.Flags |= DDSHeader.HeaderFlags.VOLUME; + header.Caps2 |= 0x00200000; + header.Width = (uint)metaData.Width; + header.Height = (uint)metaData.Height; + header.Depth = (uint)metaData.Depth; + // Check size + break; + } + default: + throw new ArgumentException("Invalid Texture Dimension."); + + } + // Calculate the Pitch + ComputePitch(metaData.Format, (uint) metaData.Width, (uint) metaData.Height, out var rowPitch, out var slicePitch, CPFLAGS.NONE); + // Validate results + if (slicePitch > UInt32.MaxValue || rowPitch > UInt32.MaxValue) + throw new ArgumentException("Failed to calculate row and/or slice pitch, values returned were too large"); + // Check is it compressed + if (IsCompressed(metaData.Format)) + { + header.Flags |= DDSHeader.HeaderFlags.LINEARSIZE; + header.PitchOrLinearSize = (uint)slicePitch; + } + else + { + header.Flags |= DDSHeader.HeaderFlags.PITCH; + header.PitchOrLinearSize = (uint)rowPitch; + } + + // Check for do we need to create the DX10 Header + if (HasDx10Header(header.PixelFormat)) + { + // Check size + if (metaData.ArraySize > UInt16.MaxValue) + throw new ArgumentException(String.Format("Array Size too large: {0}. Max: {1}", metaData.ArraySize, UInt16.MaxValue)); + // Set Pixel format + header.PixelFormat = PixelFormats.DX10; + // Set Data + dx10Header.Format = metaData.Format; + dx10Header.ResourceDimension = metaData.Dimension; + dx10Header.MiscFlag = metaData.MiscFlags & ~TexMiscFlags.TEXTURECUBE; + dx10Header.ArraySize = (uint)metaData.ArraySize; + // Check for Cube Maps + if (metaData.MiscFlags.HasFlag(TexMiscFlags.TEXTURECUBE)) + { + // Check array size, must be a multiple of 6 for cube maps + if ((metaData.ArraySize % 6) != 0) + throw new ArgumentException("Array size must be a multiple of 6"); + // Set Flag + dx10Header.MiscFlag |= TexMiscFlags.TEXTURECUBE; + dx10Header.ArraySize /= 6; + } + // Check for mist flags + if (flags.HasFlag(DDSFlags.FORCEDX10EXTMISC2)) + // This was formerly 'reserved'. D3DX10 and D3DX11 will fail if this value is anything other than 0 + dx10Header.MiscFlags2 = (uint)metaData.MiscFlags2; + } + } + + public static bool HasDx10Header(DDSHeader.DDSPixelFormat pixelFormat) => pixelFormat.Size == 0; + + public static DDSHeader.DDSPixelFormat GetPixelFormat(TexMetadata metaData) => + metaData.Format switch + { + DXGIFormat.R8G8B8A8UNORM => PixelFormats.A8B8G8R8, + DXGIFormat.R16G16UNORM => PixelFormats.G16R16, + DXGIFormat.R8G8UNORM => PixelFormats.A8L8, + DXGIFormat.R16UNORM => PixelFormats.L16, + DXGIFormat.R8UNORM => PixelFormats.L8, + DXGIFormat.A8UNORM => PixelFormats.A8, + DXGIFormat.R8G8B8G8UNORM => PixelFormats.R8G8B8G8, + DXGIFormat.G8R8G8B8UNORM => PixelFormats.G8R8G8B8, + DXGIFormat.BC1UNORM => PixelFormats.DXT1, + DXGIFormat.BC2UNORM => metaData.IsPMAlpha() ? (PixelFormats.DXT2) : (PixelFormats.DXT3), + DXGIFormat.BC3UNORM => metaData.IsPMAlpha() ? (PixelFormats.DXT4) : (PixelFormats.DXT5), + DXGIFormat.BC4UNORM => PixelFormats.BC4UNORM, + DXGIFormat.BC4SNORM => PixelFormats.BC4SNORM, + DXGIFormat.BC5UNORM => PixelFormats.BC5UNORM, + DXGIFormat.BC5SNORM => PixelFormats.BC5SNORM, + DXGIFormat.B5G6R5UNORM => PixelFormats.R5G6B5, + DXGIFormat.B5G5R5A1UNORM => PixelFormats.A1R5G5B5, + DXGIFormat.R8G8SNORM => PixelFormats.V8U8, + DXGIFormat.R8G8B8A8SNORM => PixelFormats.Q8W8V8U8, + DXGIFormat.R16G16SNORM => PixelFormats.V16U16, + DXGIFormat.B8G8R8A8UNORM => PixelFormats.A8R8G8B8, + DXGIFormat.B8G8R8X8UNORM => PixelFormats.X8R8G8B8, + DXGIFormat.B4G4R4A4UNORM => PixelFormats.A4R4G4B4, + DXGIFormat.YUY2 => PixelFormats.YUY2, + // Legacy D3DX formats using D3DFMT enum value as FourCC + DXGIFormat.R32G32B32A32FLOAT => new DDSHeader.DDSPixelFormat { Flags = PixelFormats.DDSFOURCC, FourCC = 116 /* D3DFMTA32B32G32R32F */ }, + DXGIFormat.R16G16B16A16FLOAT => new DDSHeader.DDSPixelFormat { Flags = PixelFormats.DDSFOURCC, FourCC = 113 /* D3DFMTA16B16G16R16F */ }, + DXGIFormat.R16G16B16A16UNORM => new DDSHeader.DDSPixelFormat { Flags = PixelFormats.DDSFOURCC, FourCC = 36 /* D3DFMTA16B16G16R16 */ }, + DXGIFormat.R16G16B16A16SNORM => new DDSHeader.DDSPixelFormat { Flags = PixelFormats.DDSFOURCC, FourCC = 110 /* D3DFMTQ16W16V16U16 */ }, + DXGIFormat.R32G32FLOAT => new DDSHeader.DDSPixelFormat { Flags = PixelFormats.DDSFOURCC, FourCC = 115 /* D3DFMTG32R32F */ }, + DXGIFormat.R16G16FLOAT => new DDSHeader.DDSPixelFormat { Flags = PixelFormats.DDSFOURCC, FourCC = 112 /* D3DFMTG16R16F */ }, + DXGIFormat.R32FLOAT => new DDSHeader.DDSPixelFormat { Flags = PixelFormats.DDSFOURCC, FourCC = 114 /* D3DFMTR32F */ }, + DXGIFormat.R16FLOAT => new DDSHeader.DDSPixelFormat { Flags = PixelFormats.DDSFOURCC, FourCC = 111 /* D3DFMTR16F */ }, + _ => new DDSHeader.DDSPixelFormat() + }; + + #endregion + } +} diff --git a/Wabbajack.Compression.BSA/FO4Archive/ChunkBuilder.cs b/Wabbajack.Compression.BSA/FO4Archive/ChunkBuilder.cs deleted file mode 100644 index 12f930e3f..000000000 --- a/Wabbajack.Compression.BSA/FO4Archive/ChunkBuilder.cs +++ /dev/null @@ -1,69 +0,0 @@ -using System.IO; -using System.Threading; -using System.Threading.Tasks; -using ICSharpCode.SharpZipLib.Zip.Compression; -using ICSharpCode.SharpZipLib.Zip.Compression.Streams; -using Wabbajack.Common; -using Wabbajack.DTOs.BSA.FileStates; - -namespace Wabbajack.Compression.BSA.FO4Archive; - -public class ChunkBuilder -{ - private BA2Chunk _chunk; - private Stream _dataSlab; - private long _offsetOffset; - private uint _packSize; - - public static async Task Create(BA2DX10File state, BA2Chunk chunk, Stream src, - DiskSlabAllocator slab, CancellationToken token) - { - var builder = new ChunkBuilder {_chunk = chunk}; - - if (!chunk.Compressed) - { - builder._dataSlab = slab.Allocate(chunk.FullSz); - await src.CopyToLimitAsync(builder._dataSlab, (int) chunk.FullSz, token); - } - else - { - var deflater = new Deflater(Deflater.BEST_COMPRESSION); - await using var ms = new MemoryStream(); - await using (var ds = new DeflaterOutputStream(ms, deflater)) - { - ds.IsStreamOwner = false; - await src.CopyToLimitAsync(ds, (int) chunk.FullSz, token); - } - - builder._dataSlab = slab.Allocate(ms.Length); - ms.Position = 0; - await ms.CopyToLimitAsync(builder._dataSlab, (int) ms.Length, token); - builder._packSize = (uint) ms.Length; - } - - builder._dataSlab.Position = 0; - - return builder; - } - - public void WriteHeader(BinaryWriter bw) - { - _offsetOffset = bw.BaseStream.Position; - bw.Write((ulong) 0); - bw.Write(_packSize); - bw.Write(_chunk.FullSz); - bw.Write(_chunk.StartMip); - bw.Write(_chunk.EndMip); - bw.Write(_chunk.Align); - } - - public async ValueTask WriteData(BinaryWriter bw, CancellationToken token) - { - var pos = bw.BaseStream.Position; - bw.BaseStream.Position = _offsetOffset; - bw.Write((ulong) pos); - bw.BaseStream.Position = pos; - await _dataSlab.CopyToLimitAsync(bw.BaseStream, (int) _dataSlab.Length, token); - await _dataSlab.DisposeAsync(); - } -} \ No newline at end of file diff --git a/Wabbajack.Compression.BSA/FO4Archive/DX10Entry.cs b/Wabbajack.Compression.BSA/FO4Archive/DX10Entry.cs deleted file mode 100644 index a793f4f4e..000000000 --- a/Wabbajack.Compression.BSA/FO4Archive/DX10Entry.cs +++ /dev/null @@ -1,249 +0,0 @@ -using System; -using System.Collections.Generic; -using System.IO; -using System.Linq; -using System.Text; -using System.Threading; -using System.Threading.Tasks; -using Compression.BSA; -using ICSharpCode.SharpZipLib.Zip.Compression; -using Wabbajack.Common; -using Wabbajack.DTOs.BSA.FileStates; -using Wabbajack.DTOs.Streams; -using Wabbajack.DTOs.Texture; -using Wabbajack.Paths; - -namespace Wabbajack.Compression.BSA.FO4Archive; - -public class DX10Entry : IBA2FileEntry -{ - private readonly Reader _bsa; - private ushort _chunkHdrLen; - private List _chunks; - private uint _dirHash; - private string _extension; - private byte _format; - private ushort _height; - private int _index; - private uint _nameHash; - private byte _numChunks; - private byte _numMips; - private ushort _unk16; - private byte _unk8; - private ushort _width; - private readonly byte _isCubemap; - private readonly byte _tileMode; - - public DX10Entry(Reader ba2Reader, int idx) - { - _bsa = ba2Reader; - var _rdr = ba2Reader._rdr; - _nameHash = _rdr.ReadUInt32(); - FullPath = _nameHash.ToString("X"); - _extension = Encoding.UTF8.GetString(_rdr.ReadBytes(4)); - _dirHash = _rdr.ReadUInt32(); - _unk8 = _rdr.ReadByte(); - _numChunks = _rdr.ReadByte(); - _chunkHdrLen = _rdr.ReadUInt16(); - _height = _rdr.ReadUInt16(); - _width = _rdr.ReadUInt16(); - _numMips = _rdr.ReadByte(); - _format = _rdr.ReadByte(); - _isCubemap = _rdr.ReadByte(); - _tileMode = _rdr.ReadByte(); - _index = idx; - - _chunks = Enumerable.Range(0, _numChunks) - .Select(_ => new TextureChunk(_rdr)) - .ToList(); - } - - public uint HeaderSize => DDS.HeaderSizeForFormat((DXGI_FORMAT) _format); - - public string FullPath { get; set; } - - public RelativePath Path => FullPath.ToRelativePath(); - public uint Size => (uint) _chunks.Sum(f => f._fullSz) + HeaderSize + sizeof(uint); - - public AFile State => new BA2DX10File - { - Path = Path, - NameHash = _nameHash, - Extension = _extension, - DirHash = _dirHash, - Unk8 = _unk8, - ChunkHdrLen = _chunkHdrLen, - Height = _height, - Width = _width, - NumMips = _numMips, - PixelFormat = _format, - IsCubeMap = _isCubemap, - TileMode = _tileMode, - Index = _index, - Chunks = _chunks.Select(ch => new BA2Chunk - { - FullSz = ch._fullSz, - StartMip = ch._startMip, - EndMip = ch._endMip, - Align = ch._align, - Compressed = ch._packSz != 0 - }).ToArray() - }; - - public async ValueTask CopyDataTo(Stream output, CancellationToken token) - { - var bw = new BinaryWriter(output); - - WriteHeader(bw); - - await using var fs = await _bsa._streamFactory.GetStream(); - using var br = new BinaryReader(fs); - foreach (var chunk in _chunks) - { - var full = new byte[chunk._fullSz]; - var isCompressed = chunk._packSz != 0; - - br.BaseStream.Seek((long) chunk._offset, SeekOrigin.Begin); - - if (!isCompressed) - { - await br.BaseStream.ReadAsync(full, token); - } - else - { - var compressed = new byte[chunk._packSz]; - await br.BaseStream.ReadAsync(compressed, token); - var inflater = new Inflater(); - inflater.SetInput(compressed); - inflater.Inflate(full); - } - - await bw.BaseStream.WriteAsync(full, token); - } - } - - public async ValueTask GetStreamFactory(CancellationToken token) - { - var ms = new MemoryStream(); - await CopyDataTo(ms, token); - ms.Position = 0; - return new MemoryStreamFactory(ms, Path, _bsa._streamFactory.LastModifiedUtc); - } - - private void WriteHeader(BinaryWriter bw) - { - var ddsHeader = new DDS_HEADER(); - - ddsHeader.dwSize = ddsHeader.GetSize(); - ddsHeader.dwHeaderFlags = DDS.DDS_HEADER_FLAGS_TEXTURE | DDS.DDS_HEADER_FLAGS_LINEARSIZE | - DDS.DDS_HEADER_FLAGS_MIPMAP; - ddsHeader.dwHeight = _height; - ddsHeader.dwWidth = _width; - ddsHeader.dwMipMapCount = _numMips; - ddsHeader.PixelFormat.dwSize = ddsHeader.PixelFormat.GetSize(); - ddsHeader.dwDepth = 1; - ddsHeader.dwSurfaceFlags = DDS.DDS_SURFACE_FLAGS_TEXTURE | DDS.DDS_SURFACE_FLAGS_MIPMAP; - ddsHeader.dwCubemapFlags = _isCubemap == 1 ? (uint)(DDSCAPS2.CUBEMAP - | DDSCAPS2.CUBEMAP_NEGATIVEX | DDSCAPS2.CUBEMAP_POSITIVEX - | DDSCAPS2.CUBEMAP_NEGATIVEY | DDSCAPS2.CUBEMAP_POSITIVEY - | DDSCAPS2.CUBEMAP_NEGATIVEZ | DDSCAPS2.CUBEMAP_POSITIVEZ - | DDSCAPS2.CUBEMAP_ALLFACES) : 0u; - - - switch ((DXGI_FORMAT) _format) - { - case DXGI_FORMAT.BC1_UNORM: - ddsHeader.PixelFormat.dwFlags = DDS.DDS_FOURCC; - ddsHeader.PixelFormat.dwFourCC = DDS.MAKEFOURCC('D', 'X', 'T', '1'); - ddsHeader.dwPitchOrLinearSize = (uint) (_width * _height / 2); // 4bpp - break; - case DXGI_FORMAT.BC2_UNORM: - ddsHeader.PixelFormat.dwFlags = DDS.DDS_FOURCC; - ddsHeader.PixelFormat.dwFourCC = DDS.MAKEFOURCC('D', 'X', 'T', '3'); - ddsHeader.dwPitchOrLinearSize = (uint) (_width * _height); // 8bpp - break; - case DXGI_FORMAT.BC3_UNORM: - ddsHeader.PixelFormat.dwFlags = DDS.DDS_FOURCC; - ddsHeader.PixelFormat.dwFourCC = DDS.MAKEFOURCC('D', 'X', 'T', '5'); - ddsHeader.dwPitchOrLinearSize = (uint) (_width * _height); // 8bpp - break; - case DXGI_FORMAT.BC5_UNORM: - ddsHeader.PixelFormat.dwFlags = DDS.DDS_FOURCC; - if (_bsa.UseATIFourCC) - ddsHeader.PixelFormat.dwFourCC = - DDS.MAKEFOURCC('A', 'T', 'I', - '2'); // this is more correct but the only thing I have found that supports it is the nvidia photoshop plugin - else - ddsHeader.PixelFormat.dwFourCC = DDS.MAKEFOURCC('B', 'C', '5', 'U'); - ddsHeader.dwPitchOrLinearSize = (uint) (_width * _height); // 8bpp - break; - case DXGI_FORMAT.BC1_UNORM_SRGB: - ddsHeader.PixelFormat.dwFlags = DDS.DDS_FOURCC; - ddsHeader.PixelFormat.dwFourCC = DDS.MAKEFOURCC('D', 'X', '1', '0'); - ddsHeader.dwPitchOrLinearSize = (uint) (_width * _height / 2); // 4bpp - break; - case DXGI_FORMAT.BC3_UNORM_SRGB: - case DXGI_FORMAT.BC6H_UF16: - case DXGI_FORMAT.BC4_UNORM: - case DXGI_FORMAT.BC5_SNORM: - case DXGI_FORMAT.BC7_UNORM: - case DXGI_FORMAT.BC7_UNORM_SRGB: - ddsHeader.PixelFormat.dwFlags = DDS.DDS_FOURCC; - ddsHeader.PixelFormat.dwFourCC = DDS.MAKEFOURCC('D', 'X', '1', '0'); - ddsHeader.dwPitchOrLinearSize = (uint) (_width * _height); // 8bpp - break; - case DXGI_FORMAT.R8G8B8A8_UNORM: - case DXGI_FORMAT.R8G8B8A8_UNORM_SRGB: - ddsHeader.PixelFormat.dwFlags = DDS.DDS_RGBA; - ddsHeader.PixelFormat.dwRGBBitCount = 32; - ddsHeader.PixelFormat.dwRBitMask = 0x000000FF; - ddsHeader.PixelFormat.dwGBitMask = 0x0000FF00; - ddsHeader.PixelFormat.dwBBitMask = 0x00FF0000; - ddsHeader.PixelFormat.dwABitMask = 0xFF000000; - ddsHeader.dwPitchOrLinearSize = (uint) (_width * _height * 4); // 32bpp - break; - case DXGI_FORMAT.B8G8R8A8_UNORM: - case DXGI_FORMAT.B8G8R8X8_UNORM: - ddsHeader.PixelFormat.dwFlags = DDS.DDS_RGBA; - ddsHeader.PixelFormat.dwRGBBitCount = 32; - ddsHeader.PixelFormat.dwRBitMask = 0x00FF0000; - ddsHeader.PixelFormat.dwGBitMask = 0x0000FF00; - ddsHeader.PixelFormat.dwBBitMask = 0x000000FF; - ddsHeader.PixelFormat.dwABitMask = 0xFF000000; - ddsHeader.dwPitchOrLinearSize = (uint) (_width * _height * 4); // 32bpp - break; - case DXGI_FORMAT.R8_UNORM: - ddsHeader.PixelFormat.dwFlags = DDS.DDS_RGB; - ddsHeader.PixelFormat.dwRGBBitCount = 8; - ddsHeader.PixelFormat.dwRBitMask = 0xFF; - ddsHeader.dwPitchOrLinearSize = (uint) (_width * _height); // 8bpp - break; - default: - throw new Exception("Unsupported DDS header format. File: " + FullPath); - } - - bw.Write((uint) DDS.DDS_MAGIC); - ddsHeader.Write(bw); - - switch ((DXGI_FORMAT) _format) - { - case DXGI_FORMAT.BC1_UNORM_SRGB: - case DXGI_FORMAT.BC3_UNORM_SRGB: - case DXGI_FORMAT.BC4_UNORM: - case DXGI_FORMAT.BC5_SNORM: - case DXGI_FORMAT.BC6H_UF16: - case DXGI_FORMAT.BC7_UNORM: - case DXGI_FORMAT.BC7_UNORM_SRGB: - var dxt10 = new DDS_HEADER_DXT10 - { - dxgiFormat = _format, - resourceDimension = (uint) DXT10_RESOURCE_DIMENSION.DIMENSION_TEXTURE2D, - miscFlag = 0, - arraySize = 1, - miscFlags2 = DDS.DDS_ALPHA_MODE_UNKNOWN - }; - dxt10.Write(bw); - break; - } - } -} \ No newline at end of file diff --git a/Wabbajack.DTOs/ModList/BSA/ArchiveStates/BA2State.cs b/Wabbajack.DTOs/ModList/BSA/ArchiveStates/BA2State.cs index dd94cadc3..165b07f4b 100644 --- a/Wabbajack.DTOs/ModList/BSA/ArchiveStates/BA2State.cs +++ b/Wabbajack.DTOs/ModList/BSA/ArchiveStates/BA2State.cs @@ -17,4 +17,7 @@ public class BA2State : IArchive public BA2EntryType Type { get; set; } public string HeaderMagic { get; set; } public uint Version { get; set; } + public uint Unknown1 { get; set; } + public uint Unknown2 { get; set; } + public uint Compression { get; set; } } \ No newline at end of file