mirror of
https://github.com/KillzXGaming/Switch-Toolbox
synced 2024-11-10 07:04:36 +00:00
Full support for Oodle compression (#440)
* Full support for Oodle descompression * ups * Save/Compress support * throw exception if oo2core_6_win64 dll` dont exists * Show a MessageBox when the dll is not present
This commit is contained in:
parent
7d1da60731
commit
55b9969a8e
4 changed files with 279 additions and 3 deletions
|
@ -493,6 +493,11 @@ namespace FirstPlugin
|
|||
|
||||
public void Save(System.IO.Stream stream)
|
||||
{
|
||||
if (version == 0x1000 && !File.Exists($"{Runtime.ExecutableDir}\\oo2core_6_win64.dll"))
|
||||
{
|
||||
MessageBox.Show("It is necessary to have 'oo2core_6_win64.dll' in the executable folder.");
|
||||
return;
|
||||
}
|
||||
Write(new FileWriter(stream));
|
||||
}
|
||||
|
||||
|
@ -544,6 +549,13 @@ namespace FirstPlugin
|
|||
GFPAKHashCache.EnsureHashCache();
|
||||
|
||||
version = reader.ReadInt32();
|
||||
|
||||
if (version == 0x1000 && !File.Exists($"{Runtime.ExecutableDir}\\oo2core_6_win64.dll"))
|
||||
{
|
||||
MessageBox.Show("It is necessary to have 'oo2core_6_win64.dll' in the executable folder.");
|
||||
return;
|
||||
}
|
||||
|
||||
uint padding = reader.ReadUInt32();
|
||||
uint FileCount = reader.ReadUInt32();
|
||||
FolderCount = reader.ReadInt32();
|
||||
|
@ -582,7 +594,7 @@ namespace FirstPlugin
|
|||
|
||||
fileEntry.Read(reader);
|
||||
string Extension = FindMatch(fileEntry.FileData);
|
||||
if (Extension.EndsWith("gfbanmcfg"))
|
||||
if (Extension.EndsWith("gfbanmcfg") && version != 0x1000)
|
||||
{
|
||||
GFBANMCFG cfg = new GFBANMCFG();
|
||||
cfg.Load(new MemoryStream(fileEntry.FileData));
|
||||
|
@ -847,6 +859,11 @@ namespace FirstPlugin
|
|||
}
|
||||
else if (Type == CompressionType.None)
|
||||
FileData = reader.ReadBytes((int)DecompressedFileSize);
|
||||
else if (Type == CompressionType.Oodle)
|
||||
{
|
||||
FileData = reader.ReadBytes((int)CompressedFileSize);
|
||||
FileData = STLibraryCompression.Type_Oodle.Decompress(FileData, (int)DecompressedFileSize);
|
||||
}
|
||||
else
|
||||
FileData = reader.ReadBytes((int)CompressedFileSize);
|
||||
}
|
||||
|
@ -882,7 +899,8 @@ namespace FirstPlugin
|
|||
else if (Type == CompressionType.Zlib)
|
||||
return STLibraryCompression.ZLIB.Compress(data);
|
||||
else if (Type == CompressionType.Oodle)
|
||||
throw new Exception("Oodle compression type not supported yet for saving!");
|
||||
return STLibraryCompression.Type_Oodle.Compress(data, Toolbox.Library.Compression.Oodle.OodleLZ_Compressor.OodleLZ_Compressor_Kraken,
|
||||
Toolbox.Library.Compression.Oodle.OodleLZ_CompressionLevel.OodleLZ_CompressionLevel_Optimal2);
|
||||
else
|
||||
return data;
|
||||
}
|
||||
|
|
245
Switch_Toolbox_Library/Compression/Oodle.cs
Normal file
245
Switch_Toolbox_Library/Compression/Oodle.cs
Normal file
|
@ -0,0 +1,245 @@
|
|||
using System;
|
||||
using System.Runtime.InteropServices;
|
||||
|
||||
namespace Toolbox.Library.Compression
|
||||
{
|
||||
// Code from https://github.com/JKAnderson/SoulsFormats/blob/master/SoulsFormats/Util/Oodle26.cs
|
||||
public static class Oodle
|
||||
{
|
||||
public static byte[] Compress(byte[] source, OodleLZ_Compressor compressor, OodleLZ_CompressionLevel level)
|
||||
{
|
||||
IntPtr pOptions = OodleLZ_CompressOptions_GetDefault(compressor, level);
|
||||
OodleLZ_CompressOptions options = Marshal.PtrToStructure<OodleLZ_CompressOptions>(pOptions);
|
||||
// Required for the game to not crash
|
||||
options.seekChunkReset = true;
|
||||
// This is already the default but I am including it for authenticity to game code
|
||||
options.seekChunkLen = 0x40000;
|
||||
pOptions = Marshal.AllocHGlobal(Marshal.SizeOf<OodleLZ_CompressOptions>());
|
||||
|
||||
try
|
||||
{
|
||||
Marshal.StructureToPtr(options, pOptions, false);
|
||||
long compressedBufferSizeNeeded = OodleLZ_GetCompressedBufferSizeNeeded(source.LongLength);
|
||||
byte[] compBuf = new byte[compressedBufferSizeNeeded];
|
||||
long compLen = OodleLZ_Compress(compressor, source, source.LongLength, compBuf, level, pOptions, IntPtr.Zero, IntPtr.Zero, IntPtr.Zero, 0);
|
||||
Array.Resize(ref compBuf, (int)compLen);
|
||||
return compBuf;
|
||||
}
|
||||
finally
|
||||
{
|
||||
Marshal.FreeHGlobal(pOptions);
|
||||
}
|
||||
}
|
||||
|
||||
public static byte[] Decompress(byte[] source, long uncompressedSize)
|
||||
{
|
||||
long decodeBufferSize = OodleLZ_GetDecodeBufferSize(uncompressedSize, true);
|
||||
byte[] rawBuf = new byte[decodeBufferSize];
|
||||
long rawLen = OodleLZ_Decompress(source, source.LongLength, rawBuf, uncompressedSize);
|
||||
Array.Resize(ref rawBuf, (int)rawLen);
|
||||
return rawBuf;
|
||||
}
|
||||
|
||||
|
||||
/// <param name="compressor"></param>
|
||||
/// <param name="rawBuf"></param>
|
||||
/// <param name="rawLen"></param>
|
||||
/// <param name="compBuf"></param>
|
||||
/// <param name="level"></param>
|
||||
/// <param name="pOptions">= NULL</param>
|
||||
/// <param name="dictionaryBase">= NULL</param>
|
||||
/// <param name="lrm">= NULL</param>
|
||||
/// <param name="scratchMem">= NULL</param>
|
||||
/// <param name="scratchSize">= 0</param>
|
||||
[DllImport("oo2core_6_win64.dll", CallingConvention = CallingConvention.StdCall)]
|
||||
private static extern long OodleLZ_Compress(
|
||||
OodleLZ_Compressor compressor,
|
||||
[MarshalAs(UnmanagedType.LPArray)]
|
||||
byte[] rawBuf,
|
||||
long rawLen,
|
||||
[MarshalAs(UnmanagedType.LPArray)]
|
||||
byte[] compBuf,
|
||||
OodleLZ_CompressionLevel level,
|
||||
IntPtr pOptions,
|
||||
IntPtr dictionaryBase,
|
||||
IntPtr lrm,
|
||||
IntPtr scratchMem,
|
||||
long scratchSize);
|
||||
|
||||
private static long OodleLZ_Compress(OodleLZ_Compressor compressor, byte[] rawBuf, long rawLen, byte[] compBuf, OodleLZ_CompressionLevel level)
|
||||
=> OodleLZ_Compress(compressor, rawBuf, rawLen, compBuf, level,
|
||||
IntPtr.Zero, IntPtr.Zero, IntPtr.Zero, IntPtr.Zero, 0);
|
||||
|
||||
|
||||
/// <param name="compressor">= OodleLZ_Compressor_Invalid</param>
|
||||
/// <param name="lzLevel">= OodleLZ_CompressionLevel_Normal</param>
|
||||
[DllImport("oo2core_6_win64.dll", CallingConvention = CallingConvention.StdCall)]
|
||||
private static extern IntPtr OodleLZ_CompressOptions_GetDefault(
|
||||
OodleLZ_Compressor compressor,
|
||||
OodleLZ_CompressionLevel lzLevel);
|
||||
|
||||
private static IntPtr OodleLZ_CompressOptions_GetDefault()
|
||||
=> OodleLZ_CompressOptions_GetDefault(OodleLZ_Compressor.OodleLZ_Compressor_Invalid, OodleLZ_CompressionLevel.OodleLZ_CompressionLevel_Normal);
|
||||
|
||||
|
||||
/// <param name="compBuf"></param>
|
||||
/// <param name="compBufSize"></param>
|
||||
/// <param name="rawBuf"></param>
|
||||
/// <param name="rawLen"></param>
|
||||
/// <param name="fuzzSafe">= OodleLZ_FuzzSafe_Yes</param>
|
||||
/// <param name="checkCRC">= OodleLZ_CheckCRC_No</param>
|
||||
/// <param name="verbosity">= OodleLZ_Verbosity_None</param>
|
||||
/// <param name="decBufBase">= NULL</param>
|
||||
/// <param name="decBufSize">= 0</param>
|
||||
/// <param name="fpCallback">= NULL</param>
|
||||
/// <param name="callbackUserData">= NULL</param>
|
||||
/// <param name="decoderMemory">= NULL</param>
|
||||
/// <param name="decoderMemorySize">= 0</param>
|
||||
/// <param name="threadPhase">= OodleLZ_Decode_Unthreaded</param>
|
||||
[DllImport("oo2core_6_win64.dll", CallingConvention = CallingConvention.StdCall)]
|
||||
private static extern long OodleLZ_Decompress(
|
||||
[MarshalAs(UnmanagedType.LPArray)]
|
||||
byte[] compBuf,
|
||||
long compBufSize,
|
||||
[MarshalAs(UnmanagedType.LPArray)]
|
||||
byte[] rawBuf,
|
||||
long rawLen,
|
||||
OodleLZ_FuzzSafe fuzzSafe,
|
||||
OodleLZ_CheckCRC checkCRC,
|
||||
OodleLZ_Verbosity verbosity,
|
||||
IntPtr decBufBase,
|
||||
long decBufSize,
|
||||
IntPtr fpCallback,
|
||||
IntPtr callbackUserData,
|
||||
IntPtr decoderMemory,
|
||||
long decoderMemorySize,
|
||||
OodleLZ_Decode_ThreadPhase threadPhase);
|
||||
|
||||
private static long OodleLZ_Decompress(byte[] compBuf, long compBufSize, byte[] rawBuf, long rawLen)
|
||||
=> OodleLZ_Decompress(compBuf, compBufSize, rawBuf, rawLen,
|
||||
OodleLZ_FuzzSafe.OodleLZ_FuzzSafe_Yes, OodleLZ_CheckCRC.OodleLZ_CheckCRC_No, OodleLZ_Verbosity.OodleLZ_Verbosity_None,
|
||||
IntPtr.Zero, 0, IntPtr.Zero, IntPtr.Zero, IntPtr.Zero, 0, OodleLZ_Decode_ThreadPhase.OodleLZ_Decode_Unthreaded);
|
||||
|
||||
|
||||
[DllImport("oo2core_6_win64.dll", CallingConvention = CallingConvention.StdCall)]
|
||||
private static extern long OodleLZ_GetCompressedBufferSizeNeeded(
|
||||
long rawSize);
|
||||
|
||||
|
||||
[DllImport("oo2core_6_win64.dll", CallingConvention = CallingConvention.StdCall)]
|
||||
private static extern long OodleLZ_GetDecodeBufferSize(
|
||||
long rawSize,
|
||||
[MarshalAs(UnmanagedType.Bool)]
|
||||
bool corruptionPossible);
|
||||
|
||||
|
||||
[StructLayout(LayoutKind.Sequential)]
|
||||
private struct OodleLZ_CompressOptions
|
||||
{
|
||||
public uint verbosity;
|
||||
public int minMatchLen;
|
||||
[MarshalAs(UnmanagedType.Bool)]
|
||||
public bool seekChunkReset;
|
||||
public int seekChunkLen;
|
||||
public OodleLZ_Profile profile;
|
||||
public int dictionarySize;
|
||||
public int spaceSpeedTradeoffBytes;
|
||||
public int maxHuffmansPerChunk;
|
||||
[MarshalAs(UnmanagedType.Bool)]
|
||||
public bool sendQuantumCRCs;
|
||||
public int maxLocalDictionarySize;
|
||||
public int makeLongRangeMatcher;
|
||||
public int matchTableSizeLog2;
|
||||
}
|
||||
|
||||
|
||||
private enum OodleLZ_CheckCRC : int
|
||||
{
|
||||
OodleLZ_CheckCRC_No = 0,
|
||||
OodleLZ_CheckCRC_Yes = 1,
|
||||
OodleLZ_CheckCRC_Force32 = 0x40000000
|
||||
}
|
||||
|
||||
public enum OodleLZ_CompressionLevel : int
|
||||
{
|
||||
OodleLZ_CompressionLevel_None = 0,
|
||||
OodleLZ_CompressionLevel_SuperFast = 1,
|
||||
OodleLZ_CompressionLevel_VeryFast = 2,
|
||||
OodleLZ_CompressionLevel_Fast = 3,
|
||||
OodleLZ_CompressionLevel_Normal = 4,
|
||||
|
||||
OodleLZ_CompressionLevel_Optimal1 = 5,
|
||||
OodleLZ_CompressionLevel_Optimal2 = 6,
|
||||
OodleLZ_CompressionLevel_Optimal3 = 7,
|
||||
OodleLZ_CompressionLevel_Optimal4 = 8,
|
||||
OodleLZ_CompressionLevel_Optimal5 = 9,
|
||||
|
||||
OodleLZ_CompressionLevel_HyperFast1 = -1,
|
||||
OodleLZ_CompressionLevel_HyperFast2 = -2,
|
||||
OodleLZ_CompressionLevel_HyperFast3 = -3,
|
||||
OodleLZ_CompressionLevel_HyperFast4 = -4,
|
||||
|
||||
OodleLZ_CompressionLevel_HyperFast = OodleLZ_CompressionLevel_HyperFast1,
|
||||
OodleLZ_CompressionLevel_Optimal = OodleLZ_CompressionLevel_Optimal2,
|
||||
OodleLZ_CompressionLevel_Max = OodleLZ_CompressionLevel_Optimal5,
|
||||
OodleLZ_CompressionLevel_Min = OodleLZ_CompressionLevel_HyperFast4,
|
||||
|
||||
OodleLZ_CompressionLevel_Force32 = 0x40000000,
|
||||
OodleLZ_CompressionLevel_Invalid = OodleLZ_CompressionLevel_Force32
|
||||
}
|
||||
|
||||
public enum OodleLZ_Compressor : int
|
||||
{
|
||||
OodleLZ_Compressor_Invalid = -1,
|
||||
OodleLZ_Compressor_None = 3,
|
||||
|
||||
OodleLZ_Compressor_Kraken = 8,
|
||||
OodleLZ_Compressor_Leviathan = 13,
|
||||
OodleLZ_Compressor_Mermaid = 9,
|
||||
OodleLZ_Compressor_Selkie = 11,
|
||||
OodleLZ_Compressor_Hydra = 12,
|
||||
|
||||
OodleLZ_Compressor_BitKnit = 10,
|
||||
OodleLZ_Compressor_LZB16 = 4,
|
||||
OodleLZ_Compressor_LZNA = 7,
|
||||
OodleLZ_Compressor_LZH = 0,
|
||||
OodleLZ_Compressor_LZHLW = 1,
|
||||
OodleLZ_Compressor_LZNIB = 2,
|
||||
OodleLZ_Compressor_LZBLW = 5,
|
||||
OodleLZ_Compressor_LZA = 6,
|
||||
|
||||
OodleLZ_Compressor_Count = 14,
|
||||
OodleLZ_Compressor_Force32 = 0x40000000
|
||||
}
|
||||
|
||||
private enum OodleLZ_Decode_ThreadPhase : int
|
||||
{
|
||||
OodleLZ_Decode_ThreadPhase1 = 1,
|
||||
OodleLZ_Decode_ThreadPhase2 = 2,
|
||||
OodleLZ_Decode_ThreadPhaseAll = 3,
|
||||
OodleLZ_Decode_Unthreaded = OodleLZ_Decode_ThreadPhaseAll
|
||||
}
|
||||
|
||||
private enum OodleLZ_FuzzSafe : int
|
||||
{
|
||||
OodleLZ_FuzzSafe_No = 0,
|
||||
OodleLZ_FuzzSafe_Yes = 1
|
||||
}
|
||||
|
||||
private enum OodleLZ_Profile : int
|
||||
{
|
||||
OodleLZ_Profile_Main = 0,
|
||||
OodleLZ_Profile_Reduced = 1,
|
||||
OodleLZ_Profile_Force32 = 0x40000000
|
||||
}
|
||||
|
||||
private enum OodleLZ_Verbosity : int
|
||||
{
|
||||
OodleLZ_Verbosity_None = 0,
|
||||
OodleLZ_Verbosity_Minimal = 1,
|
||||
OodleLZ_Verbosity_Some = 2,
|
||||
OodleLZ_Verbosity_Lots = 3,
|
||||
OodleLZ_Verbosity_Force32 = 0x40000000
|
||||
}
|
||||
}
|
||||
}
|
|
@ -712,6 +712,18 @@ namespace Toolbox.Library.IO
|
|||
return LZ4.LZ4Codec.Encode(data, inputOffset, data.Length);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public class Type_Oodle
|
||||
{
|
||||
public static byte[] Decompress(byte[] data, int decompressedSize)
|
||||
{
|
||||
return Toolbox.Library.Compression.Oodle.Decompress(data, decompressedSize);
|
||||
}
|
||||
|
||||
public static byte[] Compress(byte[] source, Compression.Oodle.OodleLZ_Compressor compressor, Compression.Oodle.OodleLZ_CompressionLevel level)
|
||||
{
|
||||
return Toolbox.Library.Compression.Oodle.Compress(source, compressor, level);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -248,6 +248,7 @@
|
|||
<Compile Include="Compression\7ZIP\RangeCoder\RangeCoderBit.cs" />
|
||||
<Compile Include="Compression\7ZIP\RangeCoder\RangeCoderBitTree.cs" />
|
||||
<Compile Include="Compression\MarioTennisCmp32.cs" />
|
||||
<Compile Include="Compression\Oodle.cs" />
|
||||
<Compile Include="Compression\STLibraryCompression.cs" />
|
||||
<Compile Include="Compression\Formats\Yay0.cs" />
|
||||
<Compile Include="Compression\Formats\Yaz0.cs" />
|
||||
|
|
Loading…
Reference in a new issue