mirror of
https://github.com/crskycode/GARbro.git
synced 2024-11-23 13:45:34 +08:00
(Legacy): 'Types' archives and TPGF images.
This commit is contained in:
parent
6fa882238b
commit
a698488371
80
Legacy/Types/ArcARC.cs
Normal file
80
Legacy/Types/ArcARC.cs
Normal file
@ -0,0 +1,80 @@
|
||||
//! \file ArcARC.cs
|
||||
//! \date 2018 Jun 08
|
||||
//! \brief Types resource archive.
|
||||
//
|
||||
// Copyright (C) 2018 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.Collections.Generic;
|
||||
using System.ComponentModel.Composition;
|
||||
using System.IO;
|
||||
|
||||
namespace GameRes.Formats.Types
|
||||
{
|
||||
[Export(typeof(ArchiveFormat))]
|
||||
public class ArcOpener : ArchiveFormat
|
||||
{
|
||||
public override string Tag { get { return "ARC/TYPES"; } }
|
||||
public override string Description { get { return "Types resource archive"; } }
|
||||
public override uint Signature { get { return 0; } }
|
||||
public override bool IsHierarchic { get { return false; } }
|
||||
public override bool CanWrite { get { return false; } }
|
||||
|
||||
public override ArcFile TryOpen (ArcView file)
|
||||
{
|
||||
if (!file.Name.HasExtension (".arc") || file.MaxOffset > uint.MaxValue)
|
||||
return null;
|
||||
var dir = new List<Entry>();
|
||||
using (var input = file.CreateStream())
|
||||
{
|
||||
var header = new byte[8];
|
||||
while (input.PeekByte() != -1)
|
||||
{
|
||||
uint size = input.ReadUInt32();
|
||||
if (0 == size)
|
||||
break;
|
||||
uint idx = input.ReadUInt32();
|
||||
int name_length = input.ReadUInt16();
|
||||
if (0 == name_length || name_length > 0x100)
|
||||
return null;
|
||||
var name = input.ReadCString (name_length);
|
||||
var entry = new Entry {
|
||||
Name = name,
|
||||
Offset = input.Position,
|
||||
Size = size,
|
||||
};
|
||||
if (!entry.CheckPlacement (file.MaxOffset))
|
||||
return null;
|
||||
input.Read (header, 0, header.Length);
|
||||
if (header.AsciiEqual (0, "RIFF"))
|
||||
entry.Type = "audio";
|
||||
else if (header.AsciiEqual (4, "TPGF"))
|
||||
entry.Type = "image";
|
||||
dir.Add (entry);
|
||||
input.Position = entry.Offset + entry.Size;
|
||||
}
|
||||
}
|
||||
if (0 == dir.Count)
|
||||
return null;
|
||||
return new ArcFile (file, this, dir);
|
||||
}
|
||||
}
|
||||
}
|
278
Legacy/Types/ImageTPGF.cs
Normal file
278
Legacy/Types/ImageTPGF.cs
Normal file
@ -0,0 +1,278 @@
|
||||
//! \file ImageTPGF.cs
|
||||
//! \date 2018 Jun 09
|
||||
//! \brief Types image format.
|
||||
//
|
||||
// Copyright (C) 2018 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 System.Windows.Media.Imaging;
|
||||
using GameRes.Utility;
|
||||
|
||||
namespace GameRes.Formats.Types
|
||||
{
|
||||
[Export(typeof(ImageFormat))]
|
||||
public class TpgFormat : ImageFormat
|
||||
{
|
||||
public override string Tag { get { return "TPGF"; } }
|
||||
public override string Description { get { return "Types image format"; } }
|
||||
public override uint Signature { get { return 0; } }
|
||||
|
||||
public override ImageMetaData ReadMetaData (IBinaryStream file)
|
||||
{
|
||||
var header = file.ReadHeader (13);
|
||||
if (!header.AsciiEqual (4, "TPGF"))
|
||||
return null;
|
||||
int bpp = header[12];
|
||||
if (bpp != 8 && bpp != 24)
|
||||
return null;
|
||||
return new ImageMetaData {
|
||||
Width = BigEndian.ToUInt16 (header, 8),
|
||||
Height = BigEndian.ToUInt16 (header, 10),
|
||||
BPP = bpp,
|
||||
};
|
||||
}
|
||||
|
||||
public override ImageData Read (IBinaryStream file, ImageMetaData info)
|
||||
{
|
||||
using (var tpgf = new TpgfReader (file, info))
|
||||
{
|
||||
var pixels = tpgf.Unpack();
|
||||
return ImageData.Create (info, tpgf.Format, tpgf.Palette, pixels);
|
||||
}
|
||||
}
|
||||
|
||||
public override void Write (Stream file, ImageData image)
|
||||
{
|
||||
throw new System.NotImplementedException ("TpgFormat.Write not implemented");
|
||||
}
|
||||
}
|
||||
|
||||
internal class TpgfReader : IDisposable
|
||||
{
|
||||
BitStreamEx m_input;
|
||||
byte[] m_output;
|
||||
ImageMetaData m_info;
|
||||
int m_stride;
|
||||
|
||||
public byte[] Data { get { return m_output; } }
|
||||
public PixelFormat Format { get; private set; }
|
||||
public BitmapPalette Palette { get; private set; }
|
||||
|
||||
public TpgfReader (IBinaryStream input, ImageMetaData info)
|
||||
{
|
||||
if (8 == info.BPP)
|
||||
throw new NotImplementedException();
|
||||
m_input = new BitStreamEx (input.AsStream, true);
|
||||
m_info = info;
|
||||
m_stride = (int)m_info.Height * (info.BPP + 7) / 8;
|
||||
m_output = new byte[m_stride * (int)m_info.Width];
|
||||
Format = PixelFormats.Bgr24;
|
||||
}
|
||||
|
||||
public byte[] Unpack ()
|
||||
{
|
||||
m_input.Reset();
|
||||
m_input.Input.Position = 13;
|
||||
var scanline = new byte[m_info.Width];
|
||||
int dst_line = 0;
|
||||
for (uint y = 0; y < m_info.Height; ++y)
|
||||
{
|
||||
for (int i = 0; i < 3; ++i)
|
||||
{
|
||||
ReadScanLine (scanline);
|
||||
TransformLine (scanline);
|
||||
int dst = dst_line + i;
|
||||
for (uint x = 0; x < m_info.Width; ++x)
|
||||
{
|
||||
m_output[dst] = scanline[x];
|
||||
dst += 3;
|
||||
}
|
||||
}
|
||||
dst_line += m_stride;
|
||||
}
|
||||
return m_output;
|
||||
}
|
||||
|
||||
void ReadScanLine (byte[] line)
|
||||
{
|
||||
int dst = 0;
|
||||
while (dst < line.Length)
|
||||
{
|
||||
int ctl = m_input.GetBits (3);
|
||||
int count = ReadCount() + 1;
|
||||
if (ctl != 0)
|
||||
{
|
||||
m_input.ReadEncodedBits (line, dst, count, ctl + 1);
|
||||
}
|
||||
else
|
||||
{
|
||||
for (int i = 0; i < count; ++i)
|
||||
line[dst+i] = 0;
|
||||
}
|
||||
dst += count;
|
||||
}
|
||||
}
|
||||
|
||||
int ReadCount ()
|
||||
{
|
||||
int i = 1;
|
||||
while (m_input.GetNextBit() == 0)
|
||||
++i;
|
||||
return m_input.GetBits (i) + (1 << i) - 2;
|
||||
}
|
||||
|
||||
void TransformLine (byte[] line)
|
||||
{
|
||||
var tmp = line.Clone() as byte[];
|
||||
for (int i = 1; i < line.Length; ++i)
|
||||
{
|
||||
byte a = tmp[i];
|
||||
byte b = line[i-1];
|
||||
line[i] = TransformMap[a, b];
|
||||
}
|
||||
}
|
||||
|
||||
static readonly byte[,] TransformMap = InitTransformMap();
|
||||
|
||||
static byte[,] InitTransformMap ()
|
||||
{
|
||||
var table = new byte[256,256];
|
||||
for (int i = 0; i < 256; ++i)
|
||||
for (int j = 0; j < 256; ++j)
|
||||
{
|
||||
int v;
|
||||
if (j >= 128)
|
||||
v = -1 - j;
|
||||
else
|
||||
v = j;
|
||||
if (2 * v < i)
|
||||
v = i;
|
||||
else if ((i & 1) != 0)
|
||||
v += (i + 1) >> 1;
|
||||
else
|
||||
v -= i >> 1;
|
||||
|
||||
if (j >= 128)
|
||||
table[i,j] = (byte)(-1 - v);
|
||||
else
|
||||
table[i,j] = (byte)v;
|
||||
}
|
||||
return table;
|
||||
}
|
||||
|
||||
bool m_disposed = false;
|
||||
public void Dispose ()
|
||||
{
|
||||
if (!m_disposed)
|
||||
{
|
||||
m_input.Dispose();
|
||||
m_disposed = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
internal class BitStreamEx : BitStream, IBitStream
|
||||
{
|
||||
public BitStreamEx (Stream file, bool leave_open = false) : base (file, leave_open)
|
||||
{
|
||||
}
|
||||
|
||||
public int GetBits (int count)
|
||||
{
|
||||
int mask = (1 << count) - 1;
|
||||
int v = 0;
|
||||
for (;;)
|
||||
{
|
||||
if (0 == m_cached_bits)
|
||||
{
|
||||
m_bits = m_input.ReadByte();
|
||||
if (-1 == m_bits)
|
||||
return -1;
|
||||
m_cached_bits = 8;
|
||||
}
|
||||
if (m_cached_bits >= count)
|
||||
break;
|
||||
count -= m_cached_bits;
|
||||
v |= m_bits << count;
|
||||
m_cached_bits = 0;
|
||||
}
|
||||
m_cached_bits -= count;
|
||||
return (m_bits >> m_cached_bits | v) & mask;
|
||||
}
|
||||
|
||||
public int GetNextBit ()
|
||||
{
|
||||
return GetBits (1);
|
||||
}
|
||||
|
||||
byte[] m_bit_buffer = new byte[1024];
|
||||
|
||||
public void ReadEncodedBits (byte[] buffer, int dst, int count, int ctl)
|
||||
{
|
||||
int mask = (1 << ctl) - 1;
|
||||
var cur_pos = m_input.Position;
|
||||
int byte_count = 0;
|
||||
m_input.Read (m_bit_buffer, 0, (count * ctl + 7) / 8 + 1);
|
||||
for (int i = 0; i < count; ++i)
|
||||
{
|
||||
int v = 0;
|
||||
int bit_count = ctl;
|
||||
for (;;)
|
||||
{
|
||||
if (0 == m_cached_bits)
|
||||
{
|
||||
m_bits = m_bit_buffer[byte_count++];
|
||||
m_cached_bits = 8;
|
||||
}
|
||||
if (m_cached_bits >= bit_count)
|
||||
break;
|
||||
bit_count -= m_cached_bits;
|
||||
v |= BitMap1[m_bits, bit_count];
|
||||
m_cached_bits = 0;
|
||||
}
|
||||
m_cached_bits -= bit_count;
|
||||
buffer[dst+i] = (byte)((BitMap2[m_bits, m_cached_bits] | v) & mask);
|
||||
}
|
||||
m_input.Position = cur_pos + byte_count;
|
||||
}
|
||||
|
||||
static readonly byte[,] BitMap1;
|
||||
static readonly byte[,] BitMap2;
|
||||
|
||||
static BitStreamEx ()
|
||||
{
|
||||
BitMap1 = new byte[256,8];
|
||||
BitMap2 = new byte[256,8];
|
||||
for (int i = 0; i < 256; ++i)
|
||||
{
|
||||
for (int j = 0; j < 8; ++j)
|
||||
{
|
||||
BitMap1[i, j] = (byte)(i << j);
|
||||
BitMap2[i, j] = (byte)(i >> j);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user