mirror of
https://github.com/crskycode/GARbro.git
synced 2025-01-03 00:34:13 +08:00
512 lines
20 KiB
C#
512 lines
20 KiB
C#
|
//! \file ImagePT1.cs
|
||
|
//! \date Wed Apr 15 15:17:24 2015
|
||
|
//! \brief Black Package image format implementation.
|
||
|
//
|
||
|
// Copyright (C) 2015 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.Text;
|
||
|
using System.Windows;
|
||
|
using System.Windows.Media;
|
||
|
using System.Windows.Media.Imaging;
|
||
|
using GameRes.Utility;
|
||
|
|
||
|
namespace GameRes.Formats.BlackPackage
|
||
|
{
|
||
|
internal class Pt1MetaData : ImageMetaData
|
||
|
{
|
||
|
public uint PackedSize;
|
||
|
public uint UnpackedSize;
|
||
|
}
|
||
|
|
||
|
[Export(typeof(ImageFormat))]
|
||
|
public class Pt1Format : ImageFormat
|
||
|
{
|
||
|
public override string Tag { get { return "PT1"; } }
|
||
|
public override string Description { get { return "Black Package RGB image format"; } }
|
||
|
public override uint Signature { get { return 2u; } }
|
||
|
|
||
|
public override void Write (Stream file, ImageData image)
|
||
|
{
|
||
|
throw new NotImplementedException ("Pt1Format.Write not implemented");
|
||
|
}
|
||
|
|
||
|
public override ImageMetaData ReadMetaData (Stream stream)
|
||
|
{
|
||
|
using (var input = new ArcView.Reader (stream))
|
||
|
{
|
||
|
stream.Seek (0x10, SeekOrigin.Current);
|
||
|
uint m_width = input.ReadUInt32();
|
||
|
uint m_height = input.ReadUInt32();
|
||
|
uint comp_size = input.ReadUInt32();
|
||
|
uint uncomp_size = input.ReadUInt32();
|
||
|
if (uncomp_size != m_width*m_height*3u)
|
||
|
return null;
|
||
|
return new Pt1MetaData {
|
||
|
Width = m_width,
|
||
|
Height = m_height,
|
||
|
BPP = 24,
|
||
|
PackedSize = comp_size,
|
||
|
UnpackedSize = uncomp_size
|
||
|
};
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public override ImageData Read (Stream stream, ImageMetaData info)
|
||
|
{
|
||
|
var meta = info as Pt1MetaData;
|
||
|
if (null == meta)
|
||
|
throw new ArgumentException ("Pt1Format.Read should be supplied with Pt1MetaData", "info");
|
||
|
|
||
|
stream.Position = 0x20;
|
||
|
var reader = new Reader (stream, meta);
|
||
|
// try
|
||
|
{
|
||
|
reader.UnpackV2();
|
||
|
}
|
||
|
// catch { }
|
||
|
byte[] pixels = reader.Data;
|
||
|
var bitmap = BitmapSource.Create ((int)meta.Width, (int)meta.Height, 96, 96,
|
||
|
PixelFormats.Bgr24, null, pixels, (int)meta.Width*3);
|
||
|
bitmap.Freeze();
|
||
|
return new ImageData (bitmap, meta);
|
||
|
}
|
||
|
|
||
|
internal class Reader
|
||
|
{
|
||
|
byte[] m_input;
|
||
|
byte[] m_output;
|
||
|
int m_width;
|
||
|
int m_height;
|
||
|
int m_left = 0;
|
||
|
int m_stride = 0;
|
||
|
|
||
|
public byte[] Data { get { return m_output; } }
|
||
|
|
||
|
public Reader (Stream input, Pt1MetaData info)
|
||
|
{
|
||
|
m_input = new byte[info.PackedSize+8];
|
||
|
if ((int)info.PackedSize != input.Read (m_input, 0, (int)info.PackedSize))
|
||
|
throw new InvalidFormatException ("Unexpected end of file");
|
||
|
m_width = (int)info.Width;
|
||
|
m_height = (int)info.Height;
|
||
|
m_output = new byte[info.UnpackedSize];
|
||
|
m_stride = m_width*3;
|
||
|
}
|
||
|
|
||
|
uint edx;
|
||
|
byte ch;
|
||
|
int src;
|
||
|
|
||
|
void ReadNext ()
|
||
|
{
|
||
|
byte cl = (byte)(32 - ch);
|
||
|
edx &= 0xFFFFFFFFu >> cl;
|
||
|
edx += LittleEndian.ToUInt32 (m_input, src) << ch;
|
||
|
src += cl >> 3;
|
||
|
ch += (byte)(cl & 0xf8);
|
||
|
}
|
||
|
|
||
|
public void UnpackV2 ()
|
||
|
{
|
||
|
src = 0;
|
||
|
int dst = 0;
|
||
|
Buffer.BlockCopy (m_input, src, m_output, dst, 3);
|
||
|
src += 3;
|
||
|
dst += 3;
|
||
|
edx = LittleEndian.ToUInt32 (m_input, src);
|
||
|
src += 3;
|
||
|
ch = 0x18;
|
||
|
uint _CF;
|
||
|
uint ebx;
|
||
|
sbyte ah;
|
||
|
byte al;
|
||
|
|
||
|
// [ebp+var_8] = i
|
||
|
for (int i = 1; i < m_width; ++i)
|
||
|
{
|
||
|
ReadNext();
|
||
|
|
||
|
_CF = edx & 1;
|
||
|
edx >>= 1;
|
||
|
|
||
|
if (0 != _CF)
|
||
|
{
|
||
|
--ch;
|
||
|
Buffer.BlockCopy (m_output, dst-3, m_output, dst, 3);
|
||
|
dst += 3;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ch -= 2;
|
||
|
_CF = edx & 1;
|
||
|
edx >>= 1;
|
||
|
if (0 != _CF)
|
||
|
{
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst-3]);
|
||
|
m_output[dst++] = al;
|
||
|
|
||
|
ReadNext();
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst-3]);
|
||
|
m_output[dst++] = al;
|
||
|
|
||
|
ReadNext();
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst-3]);
|
||
|
m_output[dst++] = al;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ReadNext();
|
||
|
LittleEndian.Pack ((ushort)edx, m_output, dst);
|
||
|
edx >>= 16;
|
||
|
m_output[dst+2] = (byte)edx;
|
||
|
dst += 3;
|
||
|
edx >>= 8;
|
||
|
ch -= 24;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
for (int i = 1; i < m_height; ++i)
|
||
|
{
|
||
|
dst += m_left; // XXX add edi, [ebp+arg_8]
|
||
|
ReadNext();
|
||
|
_CF = edx & 1;
|
||
|
edx >>= 1;
|
||
|
if (0 != _CF)
|
||
|
{
|
||
|
--ch;
|
||
|
Buffer.BlockCopy (m_output, dst-m_stride, m_output, dst, 3);
|
||
|
dst += 3;
|
||
|
}
|
||
|
else // loc_42207F
|
||
|
{
|
||
|
ch -= 2;
|
||
|
_CF = edx & 1;
|
||
|
edx >>= 1;
|
||
|
if (0 != _CF)
|
||
|
{
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst-m_stride]);
|
||
|
m_output[dst++] = al;
|
||
|
|
||
|
ReadNext();
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst-m_stride]);
|
||
|
m_output[dst++] = al;
|
||
|
|
||
|
ReadNext();
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst-m_stride]);
|
||
|
m_output[dst++] = al;
|
||
|
}
|
||
|
else // loc_4220FC
|
||
|
{
|
||
|
ReadNext();
|
||
|
LittleEndian.Pack ((ushort)edx, m_output, dst);
|
||
|
edx >>= 16;
|
||
|
m_output[dst+2] = (byte)edx;
|
||
|
dst += 3;
|
||
|
edx >>= 8;
|
||
|
ch -= 24;
|
||
|
}
|
||
|
}
|
||
|
for (int j = 1; j < m_width; ++j)
|
||
|
{
|
||
|
ReadNext();
|
||
|
_CF = edx & 1;
|
||
|
edx >>= 1;
|
||
|
if (0 != _CF)
|
||
|
{
|
||
|
--ch;
|
||
|
ebx = (uint)(dst - m_stride);
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(m_output[dst-3] - m_output[ebx-3] + m_output[ebx] + ah);
|
||
|
m_output[dst++] = al;
|
||
|
|
||
|
ReadNext();
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(m_output[dst-3] - m_output[ebx-2] + m_output[ebx+1] + ah);
|
||
|
m_output[dst++] = al;
|
||
|
|
||
|
ReadNext();
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(m_output[dst-3] - m_output[ebx-1] + m_output[ebx+2] + ah);
|
||
|
m_output[dst++] = al;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
_CF = edx & 1;
|
||
|
edx >>= 1;
|
||
|
if (0 != _CF)
|
||
|
{
|
||
|
ch -= 2;
|
||
|
ebx = (uint)(dst - m_stride);
|
||
|
al = (byte)(m_output[dst-3] - m_output[ebx-3] + m_output[ebx]);
|
||
|
m_output[dst++] = al;
|
||
|
al = (byte)(m_output[dst-3] - m_output[ebx-2] + m_output[ebx+1]);
|
||
|
m_output[dst++] = al;
|
||
|
al = (byte)(m_output[dst-3] - m_output[ebx-1] + m_output[ebx+2]);
|
||
|
m_output[dst++] = al;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ebx = edx & 3;
|
||
|
if (3 == ebx)
|
||
|
{
|
||
|
edx >>= 2;
|
||
|
ch -= 4;
|
||
|
Buffer.BlockCopy (m_output, dst-3, m_output, dst, 3);
|
||
|
dst += 3;
|
||
|
}
|
||
|
else if (2 == ebx)
|
||
|
{
|
||
|
edx >>= 2;
|
||
|
ch -= 4;
|
||
|
|
||
|
ReadNext();
|
||
|
LittleEndian.Pack ((ushort)edx, m_output, dst);
|
||
|
edx >>= 16;
|
||
|
m_output[dst+2] = (byte)edx;
|
||
|
dst += 3;
|
||
|
edx >>= 8;
|
||
|
ch -= 24;
|
||
|
}
|
||
|
else if (1 == ebx)
|
||
|
{
|
||
|
edx >>= 2;
|
||
|
ch -= 4;
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst-3]);
|
||
|
m_output[dst++] = al;
|
||
|
|
||
|
ReadNext();
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst-3]);
|
||
|
m_output[dst++] = al;
|
||
|
|
||
|
ReadNext();
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst-3]);
|
||
|
m_output[dst++] = al;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ebx = edx & 0xf;
|
||
|
edx >>= 4;
|
||
|
ch -= 6;
|
||
|
if (0 == ebx)
|
||
|
{
|
||
|
Buffer.BlockCopy (m_output, dst - m_stride - 3, m_output, dst, 3);
|
||
|
dst += 3;
|
||
|
}
|
||
|
else if (8 == ebx)
|
||
|
{
|
||
|
Buffer.BlockCopy (m_output, dst - m_stride, m_output, dst, 3);
|
||
|
dst += 3;
|
||
|
}
|
||
|
else if (4 == ebx)
|
||
|
{
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst - m_stride - 3]);
|
||
|
m_output[dst++] = al;
|
||
|
|
||
|
ReadNext();
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst - m_stride - 3]);
|
||
|
m_output[dst++] = al;
|
||
|
|
||
|
ReadNext();
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst - m_stride - 3]);
|
||
|
m_output[dst++] = al;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst - m_stride]);
|
||
|
m_output[dst++] = al;
|
||
|
|
||
|
ReadNext();
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst - m_stride]);
|
||
|
m_output[dst++] = al;
|
||
|
|
||
|
ReadNext();
|
||
|
ah = sub_4225EA();
|
||
|
al = (byte)(ah + m_output[dst - m_stride]);
|
||
|
m_output[dst++] = al;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
sbyte sub_4225EA ()
|
||
|
{
|
||
|
uint _CF = edx & 1;
|
||
|
edx >>= 1;
|
||
|
if (0 != _CF)
|
||
|
{
|
||
|
--ch;
|
||
|
return 0;
|
||
|
}
|
||
|
uint bits = edx & 3;
|
||
|
if (2 == bits)
|
||
|
{
|
||
|
edx >>= 2;
|
||
|
ch -= 3;
|
||
|
return -1;
|
||
|
}
|
||
|
if (1 == bits)
|
||
|
{
|
||
|
edx >>= 2;
|
||
|
ch -= 3;
|
||
|
return 1;
|
||
|
}
|
||
|
switch (edx & 7)
|
||
|
{
|
||
|
case 7:
|
||
|
edx >>= 3;
|
||
|
ch -= 4;
|
||
|
return -2;
|
||
|
case 3:
|
||
|
edx >>= 3;
|
||
|
ch -= 4;
|
||
|
return 2;
|
||
|
case 4:
|
||
|
edx >>= 3;
|
||
|
ch -= 4;
|
||
|
return -3;
|
||
|
default:
|
||
|
switch (edx & 0x3f)
|
||
|
{
|
||
|
case 0x38:
|
||
|
edx >>= 6;
|
||
|
ch -= 7;
|
||
|
return 3;
|
||
|
case 0x18:
|
||
|
edx >>= 6;
|
||
|
ch -= 7;
|
||
|
return -4;
|
||
|
case 0x28:
|
||
|
edx >>= 6;
|
||
|
ch -= 7;
|
||
|
return 4;
|
||
|
case 0x08:
|
||
|
edx >>= 6;
|
||
|
ch -= 7;
|
||
|
return -5;
|
||
|
case 0x30:
|
||
|
edx >>= 6;
|
||
|
ch -= 7;
|
||
|
return 5;
|
||
|
case 0x10:
|
||
|
edx >>= 6;
|
||
|
ch -= 7;
|
||
|
return -6;
|
||
|
case 0x20:
|
||
|
edx >>= 6;
|
||
|
ch -= 7;
|
||
|
return 6;
|
||
|
default:
|
||
|
switch (edx & 0xff)
|
||
|
{
|
||
|
case 0xc0:
|
||
|
edx >>= 8;
|
||
|
ch -= 9;
|
||
|
return -7;
|
||
|
case 0x40:
|
||
|
edx >>= 8;
|
||
|
ch -= 9;
|
||
|
return 7;
|
||
|
case 0x80:
|
||
|
edx >>= 8;
|
||
|
ch -= 9;
|
||
|
return -8;
|
||
|
default:
|
||
|
switch (edx & 0x3ff)
|
||
|
{
|
||
|
case 0x300:
|
||
|
edx >>= 10;
|
||
|
ch -= 11;
|
||
|
return 8;
|
||
|
case 0x100:
|
||
|
edx >>= 10;
|
||
|
ch -= 11;
|
||
|
return -9;
|
||
|
case 0x200:
|
||
|
edx >>= 10;
|
||
|
ch -= 11;
|
||
|
return 9;
|
||
|
default:
|
||
|
switch (edx & 0xfff)
|
||
|
{
|
||
|
case 0xc00:
|
||
|
edx >>= 12;
|
||
|
ch -= 13;
|
||
|
return -10;
|
||
|
case 0x400:
|
||
|
edx >>= 12;
|
||
|
ch -= 13;
|
||
|
return 10;
|
||
|
case 0x800:
|
||
|
edx >>= 12;
|
||
|
ch -= 13;
|
||
|
return -11;
|
||
|
default:
|
||
|
switch (edx & 0x3fff)
|
||
|
{
|
||
|
case 0x3000:
|
||
|
edx >>= 14;
|
||
|
ch -= 15;
|
||
|
return 0x0b;
|
||
|
case 0x1000:
|
||
|
edx >>= 14;
|
||
|
ch -= 15;
|
||
|
return -12;
|
||
|
case 0x2000:
|
||
|
edx >>= 14;
|
||
|
ch -= 15;
|
||
|
return 0x0c;
|
||
|
default:
|
||
|
edx >>= 14;
|
||
|
ch -= 15;
|
||
|
return -13;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|