implemented CPK archives, XTX images and HCA audio.

This commit is contained in:
morkt 2016-03-03 08:55:23 +04:00
parent b41f93cd0e
commit ed365530a6
6 changed files with 1897 additions and 1 deletions

View File

@ -66,6 +66,10 @@
<Compile Include="Abel\ImageGPS.cs" />
<Compile Include="Actgs\ArcDAT.cs" />
<Compile Include="ArcCG.cs" />
<Compile Include="Cri\ArcCPK.cs" />
<Compile Include="Cri\AudioHCA.cs" />
<Compile Include="Cri\BigEndianReader.cs" />
<Compile Include="Cri\ImageXTX.cs" />
<Compile Include="Entis\ErisaMatrix.cs" />
<Compile Include="ImageLZ.cs" />
<Compile Include="NitroPlus\ArcNPK.cs" />

416
ArcFormats/Cri/ArcCPK.cs Normal file
View File

@ -0,0 +1,416 @@
//! \file ArcCPK.cs
//! \date Mon Feb 29 12:39:36 2016
//! \brief CRI Middleware resource archive.
//
// Copyright (C) 2016 by morkt
//
// 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.
//
using System;
using System.Collections.Generic;
using System.ComponentModel.Composition;
using System.IO;
using System.Linq;
using System.Text;
using GameRes.Utility;
namespace GameRes.Formats.Cri
{
using TableRow = Dictionary<string, object>;
internal class CpkEntry : PackedEntry
{
public int Id;
}
[Export(typeof(ArchiveFormat))]
public class CpkOpener : ArchiveFormat
{
public override string Tag { get { return "CPK"; } }
public override string Description { get { return "CRI Middleware resource archive"; } }
public override uint Signature { get { return 0x204B5043; } } // 'CPK '
public override bool IsHierarchic { get { return false; } }
public override bool CanCreate { get { return false; } }
public override ArcFile TryOpen (ArcView file)
{
var reader = new IndexReader (file);
var dir = reader.ReadIndex();
if (null == dir || !dir.Any())
return null;
if (!reader.HasNames)
DetectFileTypes (file, dir);
return new ArcFile (file, this, dir);
}
public override Stream OpenEntry (ArcFile arc, Entry entry)
{
if (entry.Size < 0x10 || !arc.File.View.AsciiEqual (entry.Offset, "CRILAYLA"))
return base.OpenEntry (arc, entry);
var unpacked_size = arc.File.View.ReadInt32 (entry.Offset+8);
var packed_size = arc.File.View.ReadUInt32 (entry.Offset+12);
if (unpacked_size < 0 || packed_size > entry.Size - 0x10)
return base.OpenEntry (arc, entry);
uint prefix_size = entry.Size - (0x10+packed_size);
var output = new byte[unpacked_size+prefix_size];
var packed = arc.File.View.ReadBytes (entry.Offset+0x10, packed_size);
Array.Reverse (packed);
using (var mem = new MemoryStream (packed))
using (var input = new MsbBitStream (mem))
{
byte[] sizes = { 2, 3, 5, 8 };
int dst = (int)prefix_size;
while (dst < output.Length)
{
if (0 == input.GetNextBit())
{
output[dst++] = (byte)input.GetBits (8);
continue;
}
int count = 3;
int offset = input.GetBits (13) + 3;
int rank = 0;
int bits, step;
do
{
bits = sizes[rank];
step = input.GetBits (bits);
count += step;
if (rank < 3)
rank++;
}
while (((1 << bits) - 1) == step);
Binary.CopyOverlapped (output, dst-offset, dst, count);
dst += count;
}
}
Array.Reverse (output, (int)prefix_size, unpacked_size);
arc.File.View.Read (entry.Offset+0x10+packed_size, output, 0, prefix_size);
return new MemoryStream (output);
}
void DetectFileTypes (ArcView file, List<Entry> dir)
{
foreach (var entry in dir)
{
var offset = entry.Offset;
var signature = file.View.ReadUInt32 (offset);
if (entry.Size > 0x10 && 0x4C495243 == signature) // 'CRIL'
{
uint packed_size = file.View.ReadUInt32 (offset+12);
if (packed_size < entry.Size - 0x10)
{
signature = file.View.ReadUInt32 (offset+0x10+packed_size);
if (0x10 == signature)
signature = file.View.ReadUInt32 (offset+0x10+packed_size+signature);
}
}
var res = AutoEntry.DetectFileType (signature);
if (null != res)
{
entry.Type = res.Type;
entry.Name = Path.ChangeExtension (entry.Name, res.Extensions.FirstOrDefault());
}
}
}
}
internal class IndexReader
{
ArcView m_file;
Deserializer m_des = new Deserializer();
long m_content_offset;
Dictionary<int, Entry> m_dir = new Dictionary<int, Entry>();
public bool HasNames { get; private set; }
public IndexReader (ArcView file)
{
m_file = file;
}
public List<Entry> ReadIndex ()
{
var chunk = ReadUTFChunk (4);
var header = m_des.DeserializeUTFChunk (chunk).First();
m_content_offset = (long)header["ContentOffset"];
HasNames = header.ContainsKey ("TocOffset");
if (HasNames)
{
ReadToc ((long)header["TocOffset"]);
}
if (header.ContainsKey ("ItocOffset"))
{
var align = (uint)(int)header["Align"];
ReadItoc ((long)header["ItocOffset"], align);
}
return m_dir.Values.ToList();
}
void ReadToc (long toc_offset)
{
var base_offset = Math.Min (m_content_offset, toc_offset);
if (!m_file.View.AsciiEqual (toc_offset, "TOC "))
throw new InvalidFormatException();
var chunk = ReadUTFChunk (toc_offset+4);
var table = m_des.DeserializeUTFChunk (chunk);
foreach (var row in table)
{
var entry = new CpkEntry
{
Id = (int)row["ID"],
Offset = (long)row["FileOffset"] + base_offset,
Size = (uint)(int)row["FileSize"],
};
if (row.ContainsKey ("ExtractSize"))
entry.UnpackedSize = (uint)(int)row["ExtractSize"];
else
entry.UnpackedSize = entry.Size;
entry.IsPacked = entry.Size != entry.UnpackedSize;
var name = (string)row["FileName"];
if (row.ContainsKey ("DirName"))
name = Path.Combine ((string)row["DirName"], name);
entry.Name = name;
entry.Type = FormatCatalog.Instance.GetTypeFromName (name);
m_dir[entry.Id] = entry;
}
}
void ReadItoc (long toc_offset, uint align)
{
if (!m_file.View.AsciiEqual (toc_offset, "ITOC"))
throw new InvalidFormatException();
var chunk = ReadUTFChunk (toc_offset+4);
var itoc = m_des.DeserializeUTFChunk (chunk).FirstOrDefault();
if (null == itoc || !itoc.ContainsKey ("DataL"))
return;
var dataL = m_des.DeserializeUTFChunk ((byte[])itoc["DataL"]);
var dataH = m_des.DeserializeUTFChunk ((byte[])itoc["DataH"]);
foreach (var row in dataL.Concat (dataH))
{
int id = (int)row["ID"];
var entry = GetEntryById (id);
entry.Size = (uint)(int)row["FileSize"];
if (row.ContainsKey ("ExtractSize"))
entry.UnpackedSize = (uint)(int)row["ExtractSize"];
else
entry.UnpackedSize = entry.Size;
entry.IsPacked = entry.Size != entry.UnpackedSize;
}
long current_offset = m_content_offset;
foreach (var id in m_dir.Keys.OrderBy (x => x))
{
var entry = m_dir[id];
entry.Offset = current_offset;
current_offset += entry.Size;
if (align != 0)
{
var tail = entry.Size % align;
if (tail > 0)
current_offset += align - tail;
}
if (string.IsNullOrEmpty (entry.Name))
entry.Name = id.ToString ("D5");
}
}
CpkEntry GetEntryById (int id)
{
Entry entry;
if (!m_dir.TryGetValue (id, out entry))
{
entry = new CpkEntry { Id = id };
m_dir[id] = entry;
}
return entry as CpkEntry;
}
byte[] ReadUTFChunk (long offset)
{
long chunk_size = m_file.View.ReadInt64 (offset+4);
if (chunk_size < 0 || chunk_size > int.MaxValue)
throw new FileSizeException();
var chunk = m_file.View.ReadBytes (offset+12, (uint)chunk_size);
if (chunk.Length < chunk_size)
throw new EndOfStreamException ("Unexpected end of file");
if (!Binary.AsciiEqual (chunk, 0, "@UTF"))
DecryptUTFChunk (chunk);
return chunk;
}
internal static void DecryptUTFChunk (byte[] chunk)
{
int key = 0x655F;
for (int i = 0; i < chunk.Length; i++)
{
chunk[i] ^= (byte)key;
key *= 0x4115;
}
}
}
internal class Deserializer
{
byte[] m_chunk;
public List<TableRow> DeserializeUTFChunk (byte[] chunk)
{
m_chunk = chunk;
if (!Binary.AsciiEqual (m_chunk, 0, "@UTF"))
throw new InvalidFormatException();
var chunk_length = BigEndian.ToInt32 (m_chunk, 4);
using (var mem = new MemoryStream (m_chunk, 8, chunk_length))
using (var input = new BigEndianReader (mem))
{
int rows_offset = input.ReadInt32();
int strings_offset = input.ReadInt32() + 8;
int data_offset = input.ReadInt32() + 8;
input.Skip (4);
int column_count = input.ReadInt16();
int row_length = input.ReadInt16();
int row_count = input.ReadInt32();
var columns = new List<Column> (column_count);
for (int i = 0; i < column_count; ++i)
{
byte flags = input.ReadByte();
if (0 == flags)
{
input.Skip (3);
flags = input.ReadByte();
}
int name_offset = strings_offset + input.ReadInt32();
var column = new Column
{
Flags = (TableFlags)flags,
Name = ReadString (name_offset),
};
columns.Add (column);
}
var table = new List<TableRow> (row_count);
int next_offset = rows_offset;
for (int i = 0; i < row_count; ++i)
{
input.Position = next_offset;
next_offset += row_length;
var row = new TableRow (column_count);
table.Add (row);
foreach (var column in columns)
{
var storage = column.Flags & TableFlags.StorageMask;
if (TableFlags.StorageNone == storage
|| TableFlags.StorageZero == storage
|| TableFlags.StorageConstant == storage)
continue;
switch (column.Flags & TableFlags.TypeMask)
{
case TableFlags.TypeByte:
row[column.Name] = (int)input.ReadByte();
break;
case TableFlags.TypeSByte:
row[column.Name] = (int)input.ReadSByte();
break;
case TableFlags.TypeUInt16:
row[column.Name] = (int)input.ReadUInt16();
break;
case TableFlags.TypeInt16:
row[column.Name] = (int)input.ReadInt16();
break;
case TableFlags.TypeUInt32:
case TableFlags.TypeInt32:
row[column.Name] = input.ReadInt32();
break;
case TableFlags.TypeUInt64:
case TableFlags.TypeInt64:
row[column.Name] = input.ReadInt64();
break;
case TableFlags.TypeFloat32:
row[column.Name] = input.ReadSingle();
break;
case TableFlags.TypeString:
{
int offset = strings_offset + input.ReadInt32();
row[column.Name] = ReadString (offset);
break;
}
case TableFlags.TypeData:
{
int offset = data_offset + input.ReadInt32();
int length = input.ReadInt32();
row[column.Name] = m_chunk.Skip (offset).Take (length).ToArray();
break;
}
default:
throw new NotSupportedException();
}
}
}
return table;
}
}
string ReadString (int offset)
{
return Binary.GetCString (m_chunk, offset, 0xFF, Encoding.UTF8);
}
}
internal class Column
{
public TableFlags Flags;
public string Name;
}
[Flags]
internal enum TableFlags : byte
{
StorageMask = 0xF0,
StorageNone = 0x00,
StorageZero = 0x10,
StorageConstant = 0x30,
TypeMask = 0x0F,
TypeByte = 0x00,
TypeSByte = 0x01,
TypeUInt16 = 0x02,
TypeInt16 = 0x03,
TypeUInt32 = 0x04,
TypeInt32 = 0x05,
TypeUInt64 = 0x06,
TypeInt64 = 0x07,
TypeFloat32 = 0x08,
TypeFloat64 = 0x09,
TypeString = 0x0A,
TypeData = 0x0B,
}
}

1011
ArcFormats/Cri/AudioHCA.cs Normal file

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,125 @@
//! \file BigEndianReader.cs
//! \date Wed Mar 02 23:27:29 2016
//! \brief Wrapper around BinaryReader that reads data in a big endian order.
//
// Copyright (C) 2016 by morkt
//
// 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.
//
using System;
using System.IO;
using System.Text;
using GameRes.Utility;
namespace GameRes.Formats.Cri
{
public class BigEndianReader : IDisposable
{
BinaryReader m_input;
byte[] m_buffer = new byte[8];
public long Position
{
get { return m_input.BaseStream.Position; }
set { m_input.BaseStream.Position = value; }
}
public BigEndianReader(Stream input)
{
m_input = new BinaryReader (input, Encoding.UTF8, false);
}
public BigEndianReader (Stream input, Encoding enc, bool leave_open = false)
{
m_input = new BinaryReader (input, enc, leave_open);
}
public int Read (byte[] buffer, int index, int count)
{
return m_input.Read (buffer, index, count);
}
public void Skip (int amount)
{
m_input.BaseStream.Seek (amount, SeekOrigin.Current);
}
public byte ReadByte ()
{
return m_input.ReadByte();
}
public sbyte ReadSByte ()
{
return m_input.ReadSByte();
}
public short ReadInt16 ()
{
return Binary.BigEndian (m_input.ReadInt16());
}
public ushort ReadUInt16 ()
{
return Binary.BigEndian (m_input.ReadUInt16());
}
public int ReadInt32 ()
{
return Binary.BigEndian (m_input.ReadInt32());
}
public uint ReadUInt32 ()
{
return Binary.BigEndian (m_input.ReadUInt32());
}
public long ReadInt64 ()
{
return Binary.BigEndian (m_input.ReadInt64());
}
public ulong ReadUInt64 ()
{
return Binary.BigEndian (m_input.ReadUInt64());
}
public float ReadSingle ()
{
if (4 != m_input.Read (m_buffer, 0, 4))
throw new EndOfStreamException();
if (BitConverter.IsLittleEndian)
Array.Reverse (m_buffer, 0, 4);
return BitConverter.ToSingle (m_buffer, 0);
}
#region IDisposable Members
bool _disposed = false;
public void Dispose ()
{
if (!_disposed)
{
m_input.Dispose();
_disposed = true;
}
}
#endregion
}
}

329
ArcFormats/Cri/ImageXTX.cs Normal file
View File

@ -0,0 +1,329 @@
//! \file ImageXTX.cs
//! \date Mon Feb 29 19:14:55 2016
//! \brief Xbox 360 texture.
//
// Copyright (C) 2016 by morkt
//
// 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.
//
using System;
using System.ComponentModel.Composition;
using System.IO;
using System.Windows.Media;
using GameRes.Utility;
namespace GameRes.Formats.Cri
{
internal class XtxMetaData : ImageMetaData
{
public byte Format;
public uint DataOffset;
public int AlignedWidth;
public int AlignedHeight;
}
[Export(typeof(ImageFormat))]
public class XtxFormat : ImageFormat
{
public override string Tag { get { return "XTX"; } }
public override string Description { get { return "Xbox 360 texture format"; } }
public override uint Signature { get { return 0x00787478; } } // 'xtx'
public XtxFormat ()
{
Signatures = new uint[] { 0x00787478, 0 };
}
public override ImageMetaData ReadMetaData (Stream stream)
{
var header = new byte[0x20];
if (0x20 != stream.Read (header, 0, 0x20))
return null;
if (!Binary.AsciiEqual (header, 0, "xtx\0"))
{
var header_size = LittleEndian.ToUInt32 (header, 0);
if (header_size >= stream.Length)
return null;
stream.Position = header_size;
if (0x20 != stream.Read (header, 0, 0x20))
return null;
if (!Binary.AsciiEqual (header, 0, "xtx\0"))
return null;
}
if (header[4] > 2)
return null;
int aligned_width = BigEndian.ToInt32 (header, 8);
int aligned_height = BigEndian.ToInt32 (header, 0xC);
if (aligned_width <= 0 || aligned_height <= 0)
return null;
return new XtxMetaData
{
Width = BigEndian.ToUInt32 (header, 0x10),
Height = BigEndian.ToUInt32 (header, 0x14),
OffsetX = BigEndian.ToInt32 (header, 0x18),
OffsetY = BigEndian.ToInt32 (header, 0x1C),
BPP = 32,
Format = header[4],
AlignedWidth = aligned_width,
AlignedHeight = aligned_height,
DataOffset = (uint)stream.Position,
};
}
public override ImageData Read (Stream stream, ImageMetaData info)
{
var reader = new XtxReader (stream, (XtxMetaData)info);
var pixels = reader.Unpack();
return ImageData.Create (info, reader.Format, null, pixels);
}
public override void Write (Stream file, ImageData image)
{
throw new System.NotImplementedException ("XtxFormat.Write not implemented");
}
}
internal sealed class XtxReader
{
Stream m_input;
int m_width;
int m_height;
XtxMetaData m_info;
byte[] m_output;
int m_output_stride;
public byte[] Data { get { return m_output; } }
public PixelFormat Format { get; private set; }
public XtxReader (Stream input, XtxMetaData info)
{
m_input = input;
m_info = info;
m_input.Position = info.DataOffset;
m_width = (int)m_info.Width;
m_output_stride = m_width * 4;
m_height = (int)m_info.Height;
m_output = new byte[m_output_stride*m_height];
}
public byte[] Unpack ()
{
Format = PixelFormats.Bgra32;
if (0 == m_info.Format)
ReadTex0();
else if (1 == m_info.Format)
ReadTex1();
else
ReadTex2();
return m_output;
}
void ReadTex0 ()
{
int total = m_info.AlignedWidth * m_info.AlignedHeight;
var texture = new byte[total*4];
m_input.Read (texture, 0, texture.Length);
int src = 0;
for (int i = 0; i < total; ++i)
{
int y = GetY (i, m_info.AlignedWidth, 4);
int x = GetX (i, m_info.AlignedWidth, 4);
if (y < m_height && x < m_width)
{
int dst = m_output_stride * y + x * 4;
m_output[dst] = texture[src+3];
m_output[dst+1] = texture[src+2];
m_output[dst+2] = texture[src+1];
m_output[dst+3] = texture[src];
}
src += 4;
}
}
void ReadTex1 ()
{
int total = m_info.AlignedWidth * m_info.AlignedHeight;
var texture = new byte[total*2];
var packed = new byte[total*2];
m_input.Read (texture, 0, texture.Length);
int stride = m_info.AlignedWidth;
int src = 0;
for (int i = 0; i < total; ++i)
{
int y = GetY (i, m_info.AlignedWidth, 2);
int x = GetX (i, m_info.AlignedWidth, 2);
int dst = (x + y * stride) * 2;
packed[dst] = texture[src+1];
packed[dst+1] = texture[src];
src += 2;
}
Format = PixelFormats.Bgr565;
m_output = packed;
throw new NotImplementedException ("XTX textures format 1 not implemented");
}
void ReadTex2 ()
{
int tex_width = m_info.AlignedWidth >> 2;
int total = tex_width * (m_info.AlignedHeight >> 2);
var texture = new byte[m_info.AlignedWidth * m_info.AlignedHeight];
var packed = new byte[m_info.AlignedWidth * m_info.AlignedHeight];
m_input.Read (texture, 0, texture.Length);
int src = 0;
for (int i = 0; i < total; ++i)
{
int y = GetY (i, tex_width, 0x10);
int x = GetX (i, tex_width, 0x10);
int dst = (x + y * tex_width) * 16;
for (int j = 0; j < 8; ++j)
{
packed[dst++] = texture[src+1];
packed[dst++] = texture[src];
src += 2;
}
}
UnpackDXT5 (packed);
}
static int GetY (int i, int width, byte level)
{
int v1 = (level >> 2) + (level >> 1 >> (level >> 2));
int v2 = i << v1;
int v3 = (v2 & 0x3F) + ((v2 >> 2) & 0x1C0) + ((v2 >> 3) & 0x1FFFFE00);
return ((v3 >> 4) & 1)
+ ((((v3 & ((level << 6) - 1) & -0x20)
+ ((((v2 & 0x3F) + ((v2 >> 2) & 0xC0)) & 0xF) << 1)) >> (v1 + 3)) & -2)
+ ((((v2 >> 10) & 2) + ((v3 >> (v1 + 6)) & 1)
+ (((v3 >> (v1 + 7)) / ((width + 31) >> 5)) << 2)) << 3);
}
static int GetX (int i, int width, byte level)
{
int v1 = (level >> 2) + (level >> 1 >> (level >> 2));
int v2 = i << v1;
int v3 = (v2 & 0x3F) + ((v2 >> 2) & 0x1C0) + ((v2 >> 3) & 0x1FFFFE00);
return ((((level << 3) - 1) & ((v3 >> 1) ^ (v3 ^ (v3 >> 1)) & 0xF)) >> v1)
+ ((((((v2 >> 6) & 0xFF) + ((v3 >> (v1 + 5)) & 0xFE)) & 3)
+ (((v3 >> (v1 + 7)) % (((width + 31)) >> 5)) << 2)) << 3);
}
void UnpackDXT5 (byte[] input)
{
int src = 0;
for (int y = 0; y < m_info.AlignedHeight; y += 4)
for (int x = 0; x < m_info.AlignedWidth; x += 4)
{
DecompressDXT5Block (input, src, y, x);
src += 16;
}
}
byte[] m_dxt5_alpha = new byte[16];
void DecompressDXT5Block (byte[] input, int src, int block_y, int block_x)
{
byte alpha0 = input[src];
byte alpha1 = input[src+1];
DecompressDXT5Alpha (input, src+2, m_dxt5_alpha);
ushort color0 = LittleEndian.ToUInt16 (input, src+8);
ushort color1 = LittleEndian.ToUInt16 (input, src+10);
int t = (color0 >> 11) * 255 + 16;
byte r0 = (byte)((t / 32 + t) / 32);
t = ((color0 & 0x07E0) >> 5) * 255 + 32;
byte g0 = (byte)((t / 64 + t) / 64);
t = (color0 & 0x001F) * 255 + 16;
byte b0 = (byte)((t / 32 + t) / 32);
t = (color1 >> 11) * 255 + 16;
byte r1 = (byte)((t / 32 + t) / 32);
t = ((color1 & 0x07E0) >> 5) * 255 + 32;
byte g1 = (byte)((t / 64 + t) / 64);
t = (color1 & 0x001F) * 255 + 16;
byte b1 = (byte)((t / 32 + t) / 32);
uint code = LittleEndian.ToUInt32 (input, src+12);
for (int y = 0; y < 4 && (block_y + y) < m_height; ++y)
for (int x = 0; x < 4 && (block_x + x) < m_width; ++x)
{
int alpha_code = m_dxt5_alpha[4 * y + x];
byte alpha;
if (0 == alpha_code)
alpha = alpha0;
else if (1 == alpha_code)
alpha = alpha1;
else if (alpha0 > alpha1)
alpha = (byte)(((8 - alpha_code) * alpha0 + (alpha_code - 1) * alpha1) / 7);
else if (6 == alpha_code)
alpha = 0;
else if (7 == alpha_code)
alpha = 0xFF;
else
alpha = (byte)(((6 - alpha_code) * alpha0 + (alpha_code - 1) * alpha1) / 5);
int dst = m_output_stride * (block_y + y) + (block_x + x) * 4;
switch (code & 3)
{
case 0:
PutPixel (dst, r0, g0, b0, alpha);
break;
case 1:
PutPixel (dst, r1, g1, b1, alpha);
break;
case 2:
PutPixel (dst, (byte)((2 * r0 + r1) / 3), (byte)((2 * g0 + g1) / 3), (byte)((2 * b0 + b1) / 3), alpha);
break;
case 3:
PutPixel (dst, (byte)((r0 + 2 * r1) / 3), (byte)((g0 + 2 * g1) / 3), (byte)((b0 + 2 * b1) / 3), alpha);
break;
}
code >>= 2;
}
}
static void DecompressDXT5Alpha (byte[] input, int src, byte[] output)
{
int dst = 0;
for (int j = 0; j < 2; ++j)
{
int block = input[src++];
block |= input[src++] << 8;
block |= input[src++] << 16;
for (int i = 0; i < 8; ++i)
{
output[dst++] = (byte)(block & 7);
block >>= 3;
}
}
}
void PutPixel (int dst, byte r, byte g, byte b, byte a)
{
m_output[dst] = b;
m_output[dst+1] = g;
m_output[dst+2] = r;
m_output[dst+3] = a;
}
}
}

View File

@ -197,6 +197,7 @@ Hime to Majin to Koi Suru Tamashii<br/>
Imouto Style<br/>
Inaho no Mirai</br>
Mayoeru Futari to Sekai no Subete<br/>
Mahoutsukai no Yoru<br/>
Natsupochi<br/>
Nidaime wa ☆ Mahou Shoujo<br/>
Nuki Doki!<br/>
@ -338,7 +339,10 @@ Tsurugi Otome Noah<br/>
<tr><td>*.cpb</td><td><tt>CPB\x1a</tt><br><tt>TYP1</tt></td><td>No</td></tr>
<tr class="odd"><td>*.mfg<br/>*.mfm<br/>*.mfs</td><td><tt>ALPF</tt></td><td>No</td><td rowspan="2">Silky's</td><td rowspan="2">Jokei Kazoku</td></tr>
<tr class="odd"><td>*</td><td><tt>MFG_</tt><br/><tt>MFGA</tt><br/><tt>MFGC</tt></td><td>No</td></tr>
<tr><td>*.pmp<br/>*.pmw</td><td>-</td><td>Yes</td><td>ScenePlayer</td><td>Nyuujoku Hitozuma Jogakuen</td></tr>
<tr><td>*.pmp<br/>*.pmw</td><td>-</td><td>Yes</td><td>ScenePlayer</td><td>
Eraburu ~Erabu + Love x Double de~<br/>
Nyuujoku Hitozuma Jogakuen<br/>
</td></tr>
<tr class="odd"><td>*.dat</td><td><tt>GAMEDAT PACK</tt><br/><tt>GAMEDAT PAC2</tt></td><td>No</td><td rowspan="2"> bootUP!<br/>Pajamas Soft<br/>Aries</td><td rowspan="2">
Aneimo 2 ~Second Stage~<br/>
Momichupa Teacher!<br/>
@ -386,6 +390,7 @@ Yatohime Zankikou<br/>
<tr class="odd"><td>*.eri<br/>*.mio</td><td><tt>Entis\x1a</tt></td><td>No</td></tr>
<tr><td>*.saf</td><td>-</td><td>No</td><td>Lune</td><td>Rinkan Gakuen</td></tr>
<tr class="odd"><td>*.arc+*.ari</td><td><tt>WFL1</tt></td><td>No</td><td rowspan="2">KaGuYa</td><td rowspan="2">
Dungeon Crusaderz ~Tales of Demon Eater~<br/>
Dungeon Crusaderz 2 ~Eigou no Rakudo~<br/>
Onna Kyoushi<br/>
Magical Witch Academy<br/>
@ -671,6 +676,7 @@ MILK Junkies<br/>
<tr class="odd"><td>*.zbm</td><td><tt>amp_</tt></td><td>No</td></tr>
<tr><td>*.vfs</td><td><tt>VF</tt></td><td>No</td><td rowspan="4">Aoi</td><td rowspan="4">
Dancing Crazies<br/>
Level Justice<br/>
</td></tr>
<tr><td>*.iph</td><td><tt>RIFF....IPH fmt</tt></td><td>No</td></tr>
<tr><td>*.aog</td><td><tt>AoiOgg</tt></td><td>No</td></tr>
@ -711,6 +717,11 @@ Tokyo Necro<br/>
Thanatos no Koi ~In Ane Otouto Soukan~<br/>
</td></tr>
<tr class="odd"><td>*.gps</td><td><tt>GPS</tt></td><td>No</td></tr>
<tr><td>*.cpk</td><td><tt>CPK</tt></td><td>No</td><td rowspan="3">CRI</td><td rowspan="3">
Iwaihime<br/>
</td></tr>
<tr><td>*.xtx</td><td><tt>xtx</tt></td><td>No</td></tr>
<tr><td>*.hca</td><td><tt>HCA</tt></td><td>No</td></tr>
</table>
<p><a name="note-1" class="footnote">1</a> Non-encrypted only</p>
</body>