mirror of
https://github.com/crskycode/GARbro.git
synced 2025-01-12 21:03:55 +08:00
292 lines
10 KiB
C#
292 lines
10 KiB
C#
|
//! \file ArcA5R.cs
|
||
|
//! \date 2018 Mar 16
|
||
|
//! \brief Pinky Soft 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;
|
||
|
using System.Collections.Generic;
|
||
|
using System.ComponentModel.Composition;
|
||
|
using System.IO;
|
||
|
using System.Linq;
|
||
|
using GameRes.Compression;
|
||
|
|
||
|
namespace GameRes.Formats.Pinky
|
||
|
{
|
||
|
internal class A5Segment
|
||
|
{
|
||
|
public uint Offset;
|
||
|
public uint Size;
|
||
|
public uint UnpackedSize;
|
||
|
public byte Type;
|
||
|
public byte Compression;
|
||
|
|
||
|
public bool IsCompressed { get { return 3 == Compression; } }
|
||
|
}
|
||
|
|
||
|
internal class A5rEntry : PackedEntry
|
||
|
{
|
||
|
public IEnumerable<A5Segment> Segments;
|
||
|
}
|
||
|
|
||
|
[Export(typeof(ArchiveFormat))]
|
||
|
public class A5rOpener : ArchiveFormat
|
||
|
{
|
||
|
public override string Tag { get { return "A5R"; } }
|
||
|
public override string Description { get { return "Pinky Soft resource archive"; } }
|
||
|
public override uint Signature { get { return 0x53524350; } } // 'PCRS'
|
||
|
public override bool IsHierarchic { get { return false; } }
|
||
|
public override bool CanWrite { get { return false; } }
|
||
|
|
||
|
public A5rOpener ()
|
||
|
{
|
||
|
Signatures = new uint[] { 0x53524350, 0x42494C50 }; // 'PLIB'
|
||
|
Extensions = new string[] { "a5r", "a5e" };
|
||
|
}
|
||
|
|
||
|
public override ArcFile TryOpen (ArcView file)
|
||
|
{
|
||
|
uint id = file.View.ReadUInt32 (0);
|
||
|
if (file.View.ReadUInt32 (4) != ~id)
|
||
|
return null;
|
||
|
int count = file.View.ReadInt32 (0x30);
|
||
|
if (!IsSaneCount (count))
|
||
|
return null;
|
||
|
uint index_offset = file.View.ReadUInt32 (0x34);
|
||
|
if (index_offset >= file.MaxOffset)
|
||
|
return null;
|
||
|
|
||
|
var base_name = Path.GetFileNameWithoutExtension (file.Name);
|
||
|
uint next_offset = file.View.ReadUInt32 (index_offset);
|
||
|
var segments = new A5Segment[count];
|
||
|
for (int i = 0; i < count; ++i)
|
||
|
{
|
||
|
var segment = new A5Segment {
|
||
|
Offset = next_offset,
|
||
|
UnpackedSize = file.View.ReadUInt32 (index_offset+4),
|
||
|
Type = file.View.ReadByte (index_offset+8),
|
||
|
Compression = file.View.ReadByte (index_offset+9),
|
||
|
};
|
||
|
next_offset = file.View.ReadUInt32 (index_offset+0xA);
|
||
|
if (next_offset > file.MaxOffset || next_offset < segment.Offset)
|
||
|
return null;
|
||
|
segment.Size = (uint)(next_offset - segment.Offset);
|
||
|
segments[i] = segment;
|
||
|
index_offset += 0xA;
|
||
|
}
|
||
|
var dir = new List<Entry> (count);
|
||
|
var riff_buffer = new byte[8];
|
||
|
for (int i = 0; i < count; )
|
||
|
{
|
||
|
A5rEntry entry;
|
||
|
var segment = segments[i];
|
||
|
var name = string.Format ("{0}#{1:D5}", base_name, i);
|
||
|
if (0x3C == segment.Type)
|
||
|
{
|
||
|
Stream input = file.CreateStream (segment.Offset, segment.Size);
|
||
|
if (3 == segment.Compression)
|
||
|
input = new ZLibStream (input, CompressionMode.Decompress);
|
||
|
using (input)
|
||
|
{
|
||
|
if (8 == input.Read (riff_buffer, 0, 8) && riff_buffer.AsciiEqual ("RIFF"))
|
||
|
{
|
||
|
uint riff_size = riff_buffer.ToUInt32 (4);
|
||
|
entry = new A5rEntry {
|
||
|
Name = name + ".wav",
|
||
|
Type = "audio",
|
||
|
Offset = segment.Offset,
|
||
|
Size = 0,
|
||
|
UnpackedSize = 0,
|
||
|
};
|
||
|
var segment_list = new List<A5Segment>();
|
||
|
for (;;)
|
||
|
{
|
||
|
entry.Size += segment.Size;
|
||
|
entry.UnpackedSize += segment.UnpackedSize;
|
||
|
entry.IsPacked |= segment.Compression == 3;
|
||
|
segment_list.Add (segment);
|
||
|
++i;
|
||
|
if (i >= count || entry.UnpackedSize >= riff_size)
|
||
|
break;
|
||
|
segment = segments[i];
|
||
|
if (segment.Type != 0x3C)
|
||
|
break;
|
||
|
}
|
||
|
entry.Segments = segment_list;
|
||
|
dir.Add (entry);
|
||
|
continue;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if (0x3E == segment.Type)
|
||
|
name += ".bmp";
|
||
|
entry = new A5rEntry {
|
||
|
Name = name,
|
||
|
Type = 0x3E == segment.Type ? "image" : "",
|
||
|
Offset = segment.Offset,
|
||
|
Size = segment.Size,
|
||
|
UnpackedSize = segment.UnpackedSize,
|
||
|
IsPacked = segment.Compression == 3,
|
||
|
Segments = new A5Segment[1] { segment },
|
||
|
};
|
||
|
dir.Add (entry);
|
||
|
++i;
|
||
|
}
|
||
|
return new ArcFile (file, this, dir);
|
||
|
}
|
||
|
|
||
|
public override Stream OpenEntry (ArcFile arc, Entry entry)
|
||
|
{
|
||
|
var a5ent = (A5rEntry)entry;
|
||
|
Stream input;
|
||
|
if (a5ent.Segments.Count() == 1)
|
||
|
{
|
||
|
input = arc.File.CreateStream (entry.Offset, entry.Size);
|
||
|
if (a5ent.IsPacked)
|
||
|
input = new ZLibStream (input, CompressionMode.Decompress);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
input = new A5rStream (arc.File, a5ent);
|
||
|
}
|
||
|
return input;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal class A5rStream : Stream
|
||
|
{
|
||
|
ArcView m_file;
|
||
|
A5rEntry m_entry;
|
||
|
IEnumerator<A5Segment> m_segment;
|
||
|
Stream m_stream;
|
||
|
long m_offset = 0;
|
||
|
bool m_eof = false;
|
||
|
|
||
|
public override bool CanRead { get { return !disposed; } }
|
||
|
public override bool CanSeek { get { return false; } }
|
||
|
public override bool CanWrite { get { return false; } }
|
||
|
public override long Length { get { return m_entry.UnpackedSize; } }
|
||
|
public override long Position
|
||
|
{
|
||
|
get { return m_offset; }
|
||
|
set { throw new NotSupportedException ("A5rStream.Position not supported."); }
|
||
|
}
|
||
|
|
||
|
public A5rStream (ArcView file, A5rEntry entry)
|
||
|
{
|
||
|
m_file = file;
|
||
|
m_entry = entry;
|
||
|
m_segment = entry.Segments.GetEnumerator();
|
||
|
NextSegment();
|
||
|
}
|
||
|
|
||
|
private void NextSegment ()
|
||
|
{
|
||
|
if (!m_segment.MoveNext())
|
||
|
{
|
||
|
m_eof = true;
|
||
|
return;
|
||
|
}
|
||
|
if (null != m_stream)
|
||
|
m_stream.Dispose();
|
||
|
var segment = m_segment.Current;
|
||
|
m_stream = m_file.CreateStream (segment.Offset, segment.Size);
|
||
|
if (segment.IsCompressed)
|
||
|
m_stream = new ZLibStream (m_stream, CompressionMode.Decompress);
|
||
|
}
|
||
|
|
||
|
public override int Read (byte[] buffer, int offset, int count)
|
||
|
{
|
||
|
int total = 0;
|
||
|
while (!m_eof && count > 0)
|
||
|
{
|
||
|
int read = m_stream.Read (buffer, offset, count);
|
||
|
if (0 != read)
|
||
|
{
|
||
|
m_offset += read;
|
||
|
total += read;
|
||
|
offset += read;
|
||
|
count -= read;
|
||
|
}
|
||
|
if (0 != count)
|
||
|
NextSegment();
|
||
|
}
|
||
|
return total;
|
||
|
}
|
||
|
|
||
|
public override int ReadByte ()
|
||
|
{
|
||
|
int b = -1;
|
||
|
while (!m_eof)
|
||
|
{
|
||
|
b = m_stream.ReadByte();
|
||
|
if (-1 != b)
|
||
|
break;
|
||
|
NextSegment();
|
||
|
}
|
||
|
return b;
|
||
|
}
|
||
|
|
||
|
public override void Flush ()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
public override long Seek (long offset, SeekOrigin origin)
|
||
|
{
|
||
|
throw new NotSupportedException ("A5rStream.Seek method is not supported");
|
||
|
}
|
||
|
|
||
|
public override void SetLength (long length)
|
||
|
{
|
||
|
throw new NotSupportedException ("A5rStream.SetLength method is not supported");
|
||
|
}
|
||
|
|
||
|
public override void Write (byte[] buffer, int offset, int count)
|
||
|
{
|
||
|
throw new NotSupportedException ("A5rStream.Write method is not supported");
|
||
|
}
|
||
|
|
||
|
public override void WriteByte (byte value)
|
||
|
{
|
||
|
throw new NotSupportedException("A5rStream.WriteByte method is not supported");
|
||
|
}
|
||
|
|
||
|
#region IDisposable Members
|
||
|
bool disposed = false;
|
||
|
protected override void Dispose (bool disposing)
|
||
|
{
|
||
|
if (!disposed)
|
||
|
{
|
||
|
if (disposing)
|
||
|
{
|
||
|
if (null != m_stream)
|
||
|
m_stream.Dispose();
|
||
|
m_segment.Dispose();
|
||
|
}
|
||
|
disposed = true;
|
||
|
base.Dispose (disposing);
|
||
|
}
|
||
|
}
|
||
|
#endregion
|
||
|
}
|
||
|
}
|