2018-02-24 21:59:00 +08:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using TC = System.ComponentModel.TypeConverterAttribute;
|
|
|
|
|
using EXP = System.ComponentModel.ExpandableObjectConverter;
|
2020-02-08 03:24:04 +08:00
|
|
|
|
using System.Xml;
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
|
|
|
|
namespace CodeWalker.GameFiles
|
|
|
|
|
{
|
|
|
|
|
[TC(typeof(EXP))]public class AwcFile : PackedFile
|
|
|
|
|
{
|
|
|
|
|
public string Name { get; set; }
|
|
|
|
|
public RpfFileEntry FileEntry { get; set; }
|
|
|
|
|
public byte[] Data { get; set; }
|
|
|
|
|
|
|
|
|
|
public string ErrorMessage { get; set; }
|
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
public uint Magic { get; set; } = 0x54414441;
|
2018-02-24 21:59:00 +08:00
|
|
|
|
public ushort Version { get; set; }
|
2020-02-08 03:24:04 +08:00
|
|
|
|
public ushort Flags { get; set; } = 0xFF00;
|
2018-02-24 21:59:00 +08:00
|
|
|
|
public int StreamCount { get; set; }
|
|
|
|
|
public int InfoOffset { get; set; }
|
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
public bool UnkUshortsFlag { get { return ((Flags & 1) == 1); } set { Flags = (ushort)((Flags & 0xFFFE) + (value ? 1 : 0)); } }
|
|
|
|
|
public bool SingleChannelEncryptFlag { get { return ((Flags & 2) == 2); } set { Flags = (ushort)((Flags & 0xFFFD) + (value ? 2 : 0)); } }
|
|
|
|
|
public bool MultiChannelFlag { get { return ((Flags & 4) == 4); } set { Flags = (ushort)((Flags & 0xFFFB) + (value ? 4 : 0)); } }
|
|
|
|
|
public bool MultiChannelEncryptFlag { get { return ((Flags & 8) == 8); } set { Flags = (ushort)((Flags & 0xFFF7) + (value ? 8 : 0)); } }
|
2020-02-07 01:02:55 +08:00
|
|
|
|
|
2020-02-06 02:33:12 +08:00
|
|
|
|
public ushort[] UnkUshorts { get; set; } //offsets of some sort?
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
|
|
|
|
|
public bool WholeFileEncrypted { get; set; }
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
|
|
|
|
public AwcStreamInfo[] StreamInfos { get; set; }
|
|
|
|
|
public uint[] AudioIds { get; set; }
|
|
|
|
|
public AwcAudio[] Audios { get; set; }
|
|
|
|
|
|
|
|
|
|
static public void Decrypt_RSXXTEA(byte[] data, uint[] key)
|
|
|
|
|
{
|
|
|
|
|
// Rockstar's modified version of XXTEA
|
|
|
|
|
uint[] blocks = new uint[data.Length / 4];
|
|
|
|
|
Buffer.BlockCopy(data, 0, blocks, 0, data.Length);
|
|
|
|
|
|
|
|
|
|
int block_count = blocks.Length;
|
|
|
|
|
uint a, b = blocks[0], i;
|
|
|
|
|
|
|
|
|
|
i = (uint)(0x9E3779B9 * (6 + 52 / block_count));
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
for (int block_index = block_count - 1; block_index >= 0; --block_index)
|
|
|
|
|
{
|
|
|
|
|
a = blocks[(block_index > 0 ? block_index : block_count) - 1];
|
|
|
|
|
b = blocks[block_index] -= (a >> 5 ^ b << 2) + (b >> 3 ^ a << 4) ^ (i ^ b) + (key[block_index & 3 ^ (i >> 2 & 3)] ^ a ^ 0x7B3A207F);
|
|
|
|
|
}
|
|
|
|
|
i -= 0x9E3779B9;
|
|
|
|
|
} while (i != 0);
|
|
|
|
|
|
|
|
|
|
Buffer.BlockCopy(blocks, 0, data, 0, data.Length);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Load(byte[] data, RpfFileEntry entry)
|
|
|
|
|
{
|
|
|
|
|
Name = entry.Name;
|
|
|
|
|
FileEntry = entry;
|
|
|
|
|
Data = data;
|
|
|
|
|
|
|
|
|
|
if ((data == null) || (data.Length < 8))
|
|
|
|
|
{
|
|
|
|
|
ErrorMessage = "Data null or too short!";
|
|
|
|
|
return; //nothing to do, not enough data...
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Endianess endianess = Endianess.LittleEndian;
|
|
|
|
|
|
|
|
|
|
Magic = BitConverter.ToUInt32(data, 0);
|
|
|
|
|
if (Magic != 0x54414441 && Magic != 0x41444154)
|
|
|
|
|
{
|
|
|
|
|
if (data.Length % 4 == 0)
|
|
|
|
|
{
|
|
|
|
|
Decrypt_RSXXTEA(data, GTA5Keys.PC_AWC_KEY);
|
|
|
|
|
Magic = BitConverter.ToUInt32(data, 0);
|
2020-02-07 01:02:55 +08:00
|
|
|
|
WholeFileEncrypted = true;
|
2020-02-06 02:33:12 +08:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2018-02-24 21:59:00 +08:00
|
|
|
|
ErrorMessage = "Corrupted data!";
|
2020-02-06 02:33:12 +08:00
|
|
|
|
}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (Magic)
|
|
|
|
|
{
|
|
|
|
|
default:
|
|
|
|
|
ErrorMessage = "Unexpected Magic 0x" + Magic.ToString("X");
|
|
|
|
|
return;
|
|
|
|
|
case 0x54414441:
|
|
|
|
|
endianess = Endianess.LittleEndian;
|
|
|
|
|
break;
|
|
|
|
|
case 0x41444154:
|
|
|
|
|
endianess = Endianess.BigEndian;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
using (MemoryStream ms = new MemoryStream(data))
|
|
|
|
|
{
|
|
|
|
|
DataReader r = new DataReader(ms, endianess);
|
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
Read(r);
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
private void Read(DataReader r)
|
|
|
|
|
{
|
|
|
|
|
Magic = r.ReadUInt32();
|
|
|
|
|
Version = r.ReadUInt16();
|
|
|
|
|
Flags = r.ReadUInt16();
|
|
|
|
|
StreamCount = r.ReadInt32();
|
|
|
|
|
InfoOffset = r.ReadInt32();
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
if ((Flags >> 8) != 0xFF)
|
|
|
|
|
{
|
|
|
|
|
ErrorMessage = "Flags 0 not supported!";
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (UnkUshortsFlag)
|
|
|
|
|
{
|
|
|
|
|
UnkUshorts = new ushort[StreamCount]; //offsets of some sort?
|
|
|
|
|
for (int i = 0; i < StreamCount; i++)
|
2020-02-06 02:33:12 +08:00
|
|
|
|
{
|
2020-02-08 03:24:04 +08:00
|
|
|
|
UnkUshorts[i] = r.ReadUInt16();
|
2020-02-06 02:33:12 +08:00
|
|
|
|
}
|
2020-02-08 03:24:04 +08:00
|
|
|
|
}
|
2020-02-06 02:33:12 +08:00
|
|
|
|
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
var infoStart = 16 + (UnkUshortsFlag ? (StreamCount * 2) : 0);
|
|
|
|
|
if (r.Position != infoStart)
|
|
|
|
|
{ }//no hit
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-06 02:33:12 +08:00
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
List<AwcStreamInfo> infos = new List<AwcStreamInfo>();
|
|
|
|
|
Dictionary<uint, AwcStreamInfo> infoDict = new Dictionary<uint, AwcStreamInfo>();
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
for (int i = 0; i < StreamCount; i++)
|
|
|
|
|
{
|
|
|
|
|
var info = new AwcStreamInfo(r);
|
|
|
|
|
infos.Add(info);
|
|
|
|
|
infoDict[info.Id] = info;
|
|
|
|
|
}
|
|
|
|
|
for (int i = 0; i < StreamCount; i++)
|
|
|
|
|
{
|
|
|
|
|
var info = infos[i];
|
|
|
|
|
for (int j = 0; j < info.ChunkCount; j++)
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-08 03:24:04 +08:00
|
|
|
|
var chunk = new AwcChunkInfo(r);
|
|
|
|
|
info.ChunkList.Add(chunk);
|
|
|
|
|
info.ChunkDict[chunk.Tag] = chunk;
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
2020-02-08 03:24:04 +08:00
|
|
|
|
}
|
|
|
|
|
StreamInfos = infos.ToArray();
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
var infoOffset = infoStart + StreamCount * 4;
|
|
|
|
|
foreach (var info in infos) infoOffset += (int)info.ChunkCount * 8;
|
|
|
|
|
if (infoOffset != InfoOffset)
|
|
|
|
|
{ }//no hit
|
|
|
|
|
if (r.Position != InfoOffset)
|
|
|
|
|
{ }//no hit
|
2020-02-07 01:02:55 +08:00
|
|
|
|
|
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
ReadAudios(r);
|
|
|
|
|
|
2020-02-06 02:33:12 +08:00
|
|
|
|
}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-06 02:33:12 +08:00
|
|
|
|
private void ReadAudios(DataReader r)
|
|
|
|
|
{
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-06 02:33:12 +08:00
|
|
|
|
List<uint> audioIds = new List<uint>();
|
|
|
|
|
List<AwcAudio> audios = new List<AwcAudio>();
|
2020-02-07 01:02:55 +08:00
|
|
|
|
AwcAudio multiaudio = null;
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-06 02:33:12 +08:00
|
|
|
|
for (int i = 0; i < StreamCount; i++)
|
|
|
|
|
{
|
|
|
|
|
var info = StreamInfos[i];
|
|
|
|
|
var audio = new AwcAudio();
|
|
|
|
|
audio.StreamInfo = info;
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-06 02:33:12 +08:00
|
|
|
|
for (int j = 0; j < info.ChunkList.Count; j++)
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-06 02:33:12 +08:00
|
|
|
|
var chunk = info.ChunkList[j];
|
|
|
|
|
r.Position = chunk.Offset;
|
2020-02-08 03:24:04 +08:00
|
|
|
|
|
|
|
|
|
if ((chunk.Offset % 2) != 0)
|
|
|
|
|
{ }
|
|
|
|
|
|
2020-02-06 02:33:12 +08:00
|
|
|
|
switch (chunk.Tag)
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-06 02:33:12 +08:00
|
|
|
|
case 85: //data
|
|
|
|
|
audio.DataInfo = chunk;
|
|
|
|
|
audio.Data = r.ReadBytes(chunk.Size);
|
|
|
|
|
break;
|
|
|
|
|
case 250://format
|
|
|
|
|
audio.Format = new AwcFormatChunk(r, chunk);
|
|
|
|
|
break;
|
|
|
|
|
case 92: //animation - YCD resource chunk
|
|
|
|
|
audio.ClipDict = new AwcAudioAnimClipDict(r, chunk);
|
|
|
|
|
break;
|
2020-02-08 03:24:04 +08:00
|
|
|
|
case 43: //0x2B
|
|
|
|
|
audio.UnkData2B = new AwcAudioUnk2B(r, chunk);
|
2020-02-06 02:33:12 +08:00
|
|
|
|
break;
|
2020-02-08 03:24:04 +08:00
|
|
|
|
case 54: //0x36 - peak
|
|
|
|
|
audio.PeakData = new AwcAudioPeak(r, chunk);
|
2020-02-06 02:33:12 +08:00
|
|
|
|
break;
|
2020-02-08 03:24:04 +08:00
|
|
|
|
case 90: //0x5A
|
|
|
|
|
audio.UnkData5A = new AwcAudioUnk5A(r, chunk);
|
2020-02-06 02:33:12 +08:00
|
|
|
|
break;
|
2020-02-07 01:02:55 +08:00
|
|
|
|
case 104://MIDI
|
|
|
|
|
audio.MIDIData = new AwcAudioMIDI(r, chunk);
|
2020-02-06 02:33:12 +08:00
|
|
|
|
break;
|
2020-02-08 03:24:04 +08:00
|
|
|
|
case 217://0xD9 - length 200+
|
|
|
|
|
audio.UnkDataD9 = new AwcAudioUnkD9(r, chunk);
|
2020-02-06 02:33:12 +08:00
|
|
|
|
break;
|
|
|
|
|
case 72: //multi channel info
|
2020-02-07 01:02:55 +08:00
|
|
|
|
audio.MultiChannelInfo = new AwcChannelBlockInfo(r);
|
2020-02-06 02:33:12 +08:00
|
|
|
|
break;
|
2020-02-08 03:24:04 +08:00
|
|
|
|
case 163://multi sample offsets
|
2020-02-07 01:02:55 +08:00
|
|
|
|
audio.MultiChannelSampleOffsets = new AwcChannelOffsetsInfo(r, chunk);
|
2020-02-06 02:33:12 +08:00
|
|
|
|
break;
|
2020-02-08 03:24:04 +08:00
|
|
|
|
case 189://multi 0xBD - markers
|
|
|
|
|
audio.MarkersData = new AwcAudioMarkers(r, chunk);
|
2020-02-06 02:33:12 +08:00
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
2020-02-06 02:33:12 +08:00
|
|
|
|
if ((r.Position - chunk.Offset) != chunk.Size)
|
|
|
|
|
{ }//make sure everything was read!
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
//decrypt data where necessary
|
|
|
|
|
if ((audio.Data != null) && (!WholeFileEncrypted))
|
|
|
|
|
{
|
|
|
|
|
if (MultiChannelFlag)
|
|
|
|
|
{
|
|
|
|
|
var ocount = (int)(audio.MultiChannelSampleOffsets?.SampleOffsets?.Length ?? 0);
|
|
|
|
|
var ccount = (int)(audio.MultiChannelInfo?.ChannelCount ?? 0);
|
|
|
|
|
var bcount = (int)(audio.MultiChannelInfo?.BlockCount ?? 0);
|
|
|
|
|
var bsize = (int)(audio.MultiChannelInfo?.BlockSize ?? 0);
|
|
|
|
|
var blist = new List<AwcChannelDataBlock>();
|
|
|
|
|
for (int b = 0; b < bcount; b++)
|
|
|
|
|
{
|
|
|
|
|
int srcoff = b * bsize;
|
|
|
|
|
int mcsoff = (b < ocount) ? (int)audio.MultiChannelSampleOffsets.SampleOffsets[b] : 0;
|
|
|
|
|
int blen = Math.Max(Math.Min(bsize, audio.Data.Length - srcoff), 0);
|
|
|
|
|
var bdat = new byte[blen];
|
|
|
|
|
Buffer.BlockCopy(audio.Data, srcoff, bdat, 0, blen);
|
|
|
|
|
if (MultiChannelEncryptFlag)
|
|
|
|
|
{
|
|
|
|
|
AwcFile.Decrypt_RSXXTEA(bdat, GTA5Keys.PC_AWC_KEY);
|
|
|
|
|
}
|
|
|
|
|
var blk = new AwcChannelDataBlock(bdat, audio.MultiChannelInfo, r.Endianess, b, mcsoff);
|
|
|
|
|
blist.Add(blk);
|
|
|
|
|
}
|
|
|
|
|
audio.MultiChannelBlocks = blist.ToArray();
|
|
|
|
|
multiaudio = audio;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (SingleChannelEncryptFlag)
|
|
|
|
|
{
|
|
|
|
|
AwcFile.Decrypt_RSXXTEA(audio.Data, GTA5Keys.PC_AWC_KEY);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if ((audio.Data != null) && WholeFileEncrypted && MultiChannelFlag)
|
|
|
|
|
{ }//no hit
|
2020-02-06 02:33:12 +08:00
|
|
|
|
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
audios.Add(audio);
|
|
|
|
|
audioIds.Add(info.Id);
|
|
|
|
|
}
|
2020-02-06 02:33:12 +08:00
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
if (multiaudio != null)
|
|
|
|
|
{
|
|
|
|
|
var multichaninfos = multiaudio.MultiChannelInfo;
|
|
|
|
|
for (int i = 0; i < audios.Count; i++)
|
|
|
|
|
{
|
|
|
|
|
var audio = audios[i];
|
|
|
|
|
if (audio != multiaudio)
|
|
|
|
|
{
|
|
|
|
|
var id = audio.StreamInfo?.Id ?? 0;
|
|
|
|
|
var srcind = 0;
|
|
|
|
|
var chancnt = multiaudio.MultiChannelInfo?.Channels?.Length ?? 0;
|
|
|
|
|
var found = false;
|
|
|
|
|
for (int ind = 0; ind < chancnt; ind++)
|
|
|
|
|
{
|
|
|
|
|
var mchan = multiaudio.MultiChannelInfo.Channels[ind];
|
|
|
|
|
if (mchan.Id == id)
|
|
|
|
|
{
|
|
|
|
|
srcind = ind;
|
|
|
|
|
found = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!found)
|
|
|
|
|
{ }//no hit
|
|
|
|
|
|
|
|
|
|
audio.MultiChannelSource = multiaudio;
|
|
|
|
|
audio.MultiChannelFormat = (srcind < multichaninfos?.Channels?.Length) ? multichaninfos.Channels[srcind] : null;
|
|
|
|
|
audio.MultiChannelIndex = srcind;
|
|
|
|
|
}
|
2020-02-06 02:33:12 +08:00
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
Audios = audios.ToArray();
|
|
|
|
|
AudioIds = audioIds.ToArray();
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
|
|
|
|
|
public byte[] Save()
|
|
|
|
|
{
|
|
|
|
|
//TODO
|
|
|
|
|
return new byte[1];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void WriteXml(StringBuilder sb, int indent, string wavfolder)
|
|
|
|
|
{
|
|
|
|
|
//AwcXml.StringTag(sb, indent, "Name", AwcXml.XmlEscape(Name));
|
|
|
|
|
AwcXml.ValueTag(sb, indent, "Version", Version.ToString());
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (UnkUshorts != null)
|
|
|
|
|
{
|
|
|
|
|
AwcXml.WriteRawArray(sb, UnkUshorts, indent, "UnkUshorts", "", null, 32);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
public void ReadXml(XmlNode node, string wavfolder)
|
|
|
|
|
{
|
|
|
|
|
//Name = Xml.GetChildInnerText(node, "Name");
|
|
|
|
|
Version = (ushort)Xml.GetChildUIntAttribute(node, "Version");
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
var unode = node.SelectSingleNode("UnkUshorts");
|
|
|
|
|
if (unode != null)
|
|
|
|
|
{
|
|
|
|
|
UnkUshorts = Xml.GetRawUshortArray(node);
|
|
|
|
|
UnkUshortsFlag = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
public static void WriteXmlNode(AwcFile f, StringBuilder sb, int indent, string wavfolder, string name = "AudioWaveContainer")
|
|
|
|
|
{
|
|
|
|
|
if (f == null) return;
|
|
|
|
|
AwcXml.OpenTag(sb, indent, name);
|
|
|
|
|
f.WriteXml(sb, indent + 1, wavfolder);
|
|
|
|
|
AwcXml.CloseTag(sb, indent, name);
|
|
|
|
|
}
|
|
|
|
|
public static AwcFile ReadXmlNode(XmlNode node, string wavfolder)
|
|
|
|
|
{
|
|
|
|
|
if (node == null) return null;
|
|
|
|
|
var f = new AwcFile();
|
|
|
|
|
f.ReadXml(node, wavfolder);
|
|
|
|
|
return f;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public enum AwcCodecFormat
|
|
|
|
|
{
|
|
|
|
|
PCM = 0,
|
|
|
|
|
ADPCM = 4
|
|
|
|
|
}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
|
|
|
|
[TC(typeof(EXP))] public class AwcStreamInfo
|
|
|
|
|
{
|
|
|
|
|
public uint RawVal { get; set; }
|
2020-02-06 02:33:12 +08:00
|
|
|
|
public uint ChunkCount { get; set; }
|
2018-02-24 21:59:00 +08:00
|
|
|
|
public uint Id { get; set; }
|
|
|
|
|
|
2020-02-06 02:33:12 +08:00
|
|
|
|
public List<AwcChunkInfo> ChunkList { get; set; } = new List<AwcChunkInfo>();
|
|
|
|
|
public Dictionary<byte, AwcChunkInfo> ChunkDict { get; set; } = new Dictionary<byte, AwcChunkInfo>();
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
|
|
|
|
public AwcStreamInfo(DataReader r)
|
|
|
|
|
{
|
|
|
|
|
RawVal = r.ReadUInt32();
|
2020-02-06 02:33:12 +08:00
|
|
|
|
ChunkCount = (RawVal >> 29);
|
2018-02-24 21:59:00 +08:00
|
|
|
|
Id = (RawVal & 0x1FFFFFFF);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
2020-02-06 02:33:12 +08:00
|
|
|
|
return Id.ToString("X") + ": " + ChunkCount.ToString() + " chunks";
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TC(typeof(EXP))] public class AwcChunkInfo
|
|
|
|
|
{
|
|
|
|
|
public ulong RawVal { get; set; }
|
|
|
|
|
public byte Tag { get; set; }
|
|
|
|
|
public int Size { get; set; }
|
|
|
|
|
public int Offset { get; set; }
|
|
|
|
|
|
|
|
|
|
public AwcChunkInfo(DataReader r)
|
|
|
|
|
{
|
|
|
|
|
RawVal = r.ReadUInt64();
|
|
|
|
|
Tag = (byte)(RawVal >> 56);
|
|
|
|
|
Size = (int)((RawVal >> 28) & 0x0FFFFFFF);
|
|
|
|
|
Offset = (int)(RawVal & 0x0FFFFFFF);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return Tag.ToString() + ": " + Size.ToString() + ", " + Offset.ToString();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
[TC(typeof(EXP))] public class AwcChannelBlockInfo
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public uint BlockCount { get; set; }
|
|
|
|
|
public uint BlockSize { get; set; }
|
2018-02-24 21:59:00 +08:00
|
|
|
|
public uint ChannelCount { get; set; }
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public AwcChannelBlockItemInfo[] Channels { get; set; }
|
2020-02-06 02:33:12 +08:00
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public uint TotalSize { get { return BlockCount * BlockSize; } }
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public AwcChannelBlockInfo(DataReader r)
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
BlockCount = r.ReadUInt32();
|
|
|
|
|
BlockSize = r.ReadUInt32();
|
2018-02-24 21:59:00 +08:00
|
|
|
|
ChannelCount = r.ReadUInt32();
|
2020-02-06 02:33:12 +08:00
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
List<AwcChannelBlockItemInfo> channels = new List<AwcChannelBlockItemInfo>();
|
2020-02-06 02:33:12 +08:00
|
|
|
|
for (int i = 0; i < ChannelCount; i++)
|
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
var itemInfo = new AwcChannelBlockItemInfo(r);
|
2020-02-06 02:33:12 +08:00
|
|
|
|
channels.Add(itemInfo);
|
|
|
|
|
}
|
|
|
|
|
Channels = channels.ToArray();
|
|
|
|
|
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
return BlockCount.ToString() + ": " + BlockSize.ToString() + ", " + ChannelCount.ToString() + " channels";
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
[TC(typeof(EXP))] public class AwcChannelBlockItemInfo
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
|
|
|
|
public uint Id { get; set; }
|
|
|
|
|
public uint Samples { get; set; }
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public ushort Unk0 { get; set; }//flags?
|
2018-02-24 21:59:00 +08:00
|
|
|
|
public ushort SamplesPerSecond { get; set; }
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public AwcCodecFormat Codec { get; set; }
|
2018-02-24 21:59:00 +08:00
|
|
|
|
public byte RoundSize { get; set; }
|
|
|
|
|
public ushort Unk2 { get; set; }
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public AwcChannelBlockItemInfo(DataReader r)
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
|
|
|
|
Id = r.ReadUInt32();
|
|
|
|
|
Samples = r.ReadUInt32();
|
|
|
|
|
Unk0 = r.ReadUInt16();
|
|
|
|
|
SamplesPerSecond = r.ReadUInt16();
|
2020-02-07 01:02:55 +08:00
|
|
|
|
Codec = (AwcCodecFormat)r.ReadByte();
|
2018-02-24 21:59:00 +08:00
|
|
|
|
RoundSize = r.ReadByte();
|
|
|
|
|
Unk2 = r.ReadUInt16();
|
2020-02-07 01:02:55 +08:00
|
|
|
|
|
|
|
|
|
//switch (Unk0)
|
|
|
|
|
//{
|
|
|
|
|
// case 65395:
|
|
|
|
|
// case 65485:
|
|
|
|
|
// case 33:
|
|
|
|
|
// case 9:
|
|
|
|
|
// case 674:
|
|
|
|
|
// case 529:
|
|
|
|
|
// case 11:
|
|
|
|
|
// case 105:
|
|
|
|
|
// case 65396:
|
|
|
|
|
// case 437:
|
|
|
|
|
// case 477:
|
|
|
|
|
// case 65475:
|
|
|
|
|
// case 519:
|
|
|
|
|
// case 349:
|
|
|
|
|
// case 65517:
|
|
|
|
|
// case 106:
|
|
|
|
|
// case 219:
|
|
|
|
|
// case 58:
|
|
|
|
|
// case 63:
|
|
|
|
|
// case 65530:
|
|
|
|
|
// case 65511:
|
|
|
|
|
// case 191:
|
|
|
|
|
// case 65439:
|
|
|
|
|
// case 65345:
|
|
|
|
|
// case 560:
|
|
|
|
|
// case 640:
|
|
|
|
|
// case 65335:
|
|
|
|
|
// case 208:
|
|
|
|
|
// case 392:
|
|
|
|
|
// case 1968:
|
|
|
|
|
// case 2257:
|
|
|
|
|
// case 1101:
|
|
|
|
|
// case 906:
|
|
|
|
|
// case 1087:
|
|
|
|
|
// case 718:
|
|
|
|
|
// case 65362:
|
|
|
|
|
// case 259:
|
|
|
|
|
// case 234:
|
|
|
|
|
// case 299:
|
|
|
|
|
// case 307:
|
|
|
|
|
// case 75:
|
|
|
|
|
// case 67:
|
|
|
|
|
// case 4:
|
|
|
|
|
// case 290:
|
|
|
|
|
// case 587:
|
|
|
|
|
// case 65448:
|
|
|
|
|
// case 15:
|
|
|
|
|
// case 398:
|
|
|
|
|
// case 399:
|
|
|
|
|
// case 65375:
|
|
|
|
|
// case 65376:
|
|
|
|
|
// case 65336:
|
|
|
|
|
// break;
|
|
|
|
|
// default:
|
|
|
|
|
// break;//more... flags?
|
|
|
|
|
//}
|
|
|
|
|
//switch (Codec)
|
|
|
|
|
//{
|
|
|
|
|
// case AwcCodecFormat.ADPCM:
|
|
|
|
|
// break;
|
|
|
|
|
// default:
|
|
|
|
|
// break;//no hit
|
|
|
|
|
//}
|
|
|
|
|
//switch (Unk2)
|
|
|
|
|
//{
|
|
|
|
|
// case 0:
|
|
|
|
|
// break;
|
|
|
|
|
// default:
|
|
|
|
|
// break;//no hit
|
|
|
|
|
//}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
return Id.ToString() + ", " + Codec.ToString() + ": " + Samples.ToString() + " samples, " + SamplesPerSecond.ToString() + " samples/sec, size: " + RoundSize.ToString() + " unk0: " + Unk0.ToString();
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TC(typeof(EXP))] public class AwcFormatChunk
|
|
|
|
|
{
|
|
|
|
|
public uint Samples { get; set; }
|
|
|
|
|
public int LoopPoint { get; set; }
|
|
|
|
|
public ushort SamplesPerSecond { get; set; }
|
|
|
|
|
public short Headroom { get; set; }
|
|
|
|
|
public ushort LoopBegin { get; set; }
|
|
|
|
|
public ushort LoopEnd { get; set; }
|
|
|
|
|
public ushort PlayEnd { get; set; }
|
|
|
|
|
public byte PlayBegin { get; set; }
|
2020-02-06 02:33:12 +08:00
|
|
|
|
public uint UnkExtra { get; set; }
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public AwcCodecFormat Codec { get; set; }
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-06 02:33:12 +08:00
|
|
|
|
public AwcFormatChunk(DataReader r, AwcChunkInfo info)
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
|
|
|
|
Samples = r.ReadUInt32();
|
|
|
|
|
LoopPoint = r.ReadInt32();
|
|
|
|
|
SamplesPerSecond = r.ReadUInt16();
|
|
|
|
|
Headroom = r.ReadInt16();
|
|
|
|
|
LoopBegin = r.ReadUInt16();
|
|
|
|
|
LoopEnd = r.ReadUInt16();
|
|
|
|
|
PlayEnd = r.ReadUInt16();
|
|
|
|
|
PlayBegin = r.ReadByte();
|
2020-02-07 01:02:55 +08:00
|
|
|
|
Codec = (AwcCodecFormat)r.ReadByte();
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-06 02:33:12 +08:00
|
|
|
|
switch (info.Size) //Apparently sometimes this struct is longer?
|
|
|
|
|
{
|
|
|
|
|
case 20:
|
|
|
|
|
break;
|
|
|
|
|
case 24:
|
|
|
|
|
UnkExtra = r.ReadUInt32();
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;//no hit
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return Headroom.ToString() + ", " + Codec.ToString() + ": " + Samples.ToString() + " samples, " + SamplesPerSecond.ToString() + " samples/sec";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TC(typeof(EXP))] public class AwcAudio
|
|
|
|
|
{
|
|
|
|
|
public AwcStreamInfo StreamInfo { get; set; }
|
|
|
|
|
public AwcFormatChunk Format { get; set; }
|
|
|
|
|
public AwcChunkInfo DataInfo { get; set; }
|
|
|
|
|
public byte[] Data { get; set; }
|
|
|
|
|
|
|
|
|
|
public AwcAudioAnimClipDict ClipDict { get; set; }
|
2020-02-08 03:24:04 +08:00
|
|
|
|
public AwcAudioUnk2B UnkData2B { get; set; }
|
|
|
|
|
public AwcAudioPeak PeakData { get; set; }
|
|
|
|
|
public AwcAudioUnk5A UnkData5A { get; set; }
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public AwcAudioMIDI MIDIData { get; set; }
|
2020-02-08 03:24:04 +08:00
|
|
|
|
public AwcAudioUnkD9 UnkDataD9 { get; set; }
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public AwcChannelBlockInfo MultiChannelInfo { get; set; }
|
|
|
|
|
public AwcChannelOffsetsInfo MultiChannelSampleOffsets { get; set; }
|
|
|
|
|
public AwcChannelDataBlock[] MultiChannelBlocks { get; set; }
|
2020-02-08 03:24:04 +08:00
|
|
|
|
public AwcAudioMarkers MarkersData { get; set; }
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public AwcAudio MultiChannelSource { get; set; }
|
|
|
|
|
public AwcChannelBlockItemInfo MultiChannelFormat { get; set; }
|
|
|
|
|
public int MultiChannelIndex { get; set; }
|
|
|
|
|
|
2018-02-24 21:59:00 +08:00
|
|
|
|
public short Channels = 1;
|
|
|
|
|
public short BitsPerSample = 16;
|
|
|
|
|
public int SamplesPerSecond
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
return Format?.SamplesPerSecond ?? MultiChannelFormat?.SamplesPerSecond ?? 0;
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
public int SampleCount
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
return (int)(Format?.Samples ?? MultiChannelFormat?.Samples ?? 0);
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public string Name
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
return "0x" + StreamInfo?.Id.ToString("X").PadLeft(8, '0') ?? "0";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
public string Type
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
if (MIDIData != null)
|
|
|
|
|
{
|
|
|
|
|
return "MIDI";
|
|
|
|
|
}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
var fc = AwcCodecFormat.PCM;
|
|
|
|
|
var hz = 0;
|
|
|
|
|
if (Format != null)
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
fc = Format.Codec;
|
|
|
|
|
hz = Format.SamplesPerSecond;
|
|
|
|
|
}
|
|
|
|
|
if (MultiChannelFormat != null)
|
|
|
|
|
{
|
|
|
|
|
fc = MultiChannelFormat.Codec;
|
|
|
|
|
hz = MultiChannelFormat.SamplesPerSecond;
|
|
|
|
|
}
|
|
|
|
|
string codec = fc.ToString();
|
|
|
|
|
switch (fc)
|
|
|
|
|
{
|
|
|
|
|
case AwcCodecFormat.PCM:
|
|
|
|
|
case AwcCodecFormat.ADPCM:
|
2018-02-24 21:59:00 +08:00
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
codec = "Unknown";
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return codec + ((hz > 0) ? (", " + hz.ToString() + " Hz") : "");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public float Length
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
if (Format != null) return (float)Format.Samples / Format.SamplesPerSecond;
|
|
|
|
|
if (MultiChannelFormat != null) return (float)MultiChannelFormat.Samples / MultiChannelFormat.SamplesPerSecond;
|
|
|
|
|
return 0;
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
public string LengthStr
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
return TimeSpan.FromSeconds(Length).ToString("m\\:ss");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public int ByteLength
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
if (MIDIData?.Data != null)
|
|
|
|
|
{
|
|
|
|
|
return MIDIData.Data?.Length ?? 0;
|
|
|
|
|
}
|
|
|
|
|
if (Data != null)
|
|
|
|
|
{
|
|
|
|
|
return Data.Length;
|
|
|
|
|
}
|
|
|
|
|
if (MultiChannelSource != null)
|
|
|
|
|
{
|
|
|
|
|
int c = 0;
|
|
|
|
|
if (MultiChannelSource?.MultiChannelBlocks != null)
|
|
|
|
|
{
|
|
|
|
|
foreach (var blk in MultiChannelSource.MultiChannelBlocks)
|
|
|
|
|
{
|
|
|
|
|
if (MultiChannelIndex < (blk?.Channels?.Length ?? 0))
|
|
|
|
|
{
|
|
|
|
|
var chan = blk.Channels[MultiChannelIndex];
|
|
|
|
|
c += chan?.ChannelData?.Length ?? 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return c;
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-02-06 02:33:12 +08:00
|
|
|
|
public AwcAudio()
|
|
|
|
|
{
|
|
|
|
|
}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
public AwcAudio(DataReader r, AwcStreamInfo s, AwcFormatChunk f, AwcChunkInfo d)
|
|
|
|
|
{
|
|
|
|
|
StreamInfo = s;
|
|
|
|
|
Format = f;
|
|
|
|
|
DataInfo = d;
|
|
|
|
|
|
|
|
|
|
Data = r.ReadBytes(d.Size);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
var hash = "0x" + (StreamInfo?.Id.ToString("X") ?? "0").PadLeft(8, '0') + ": ";
|
|
|
|
|
if (Format != null)
|
|
|
|
|
{
|
|
|
|
|
return hash + Format?.ToString() ?? "AwcAudio";
|
|
|
|
|
}
|
|
|
|
|
if (MultiChannelFormat != null)
|
|
|
|
|
{
|
|
|
|
|
return hash + MultiChannelFormat?.ToString() ?? "AwcAudio";
|
|
|
|
|
}
|
|
|
|
|
if (MIDIData != null)
|
|
|
|
|
{
|
|
|
|
|
return hash + MIDIData.ToString();
|
|
|
|
|
}
|
|
|
|
|
return hash + "Unknown";
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2020-02-07 16:27:23 +08:00
|
|
|
|
public byte[] GetWavData()
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
if (MultiChannelFormat != null)
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-07 16:27:23 +08:00
|
|
|
|
if (Data == null)
|
2020-02-07 01:02:55 +08:00
|
|
|
|
{
|
2020-02-07 16:27:23 +08:00
|
|
|
|
var ms = new MemoryStream();
|
|
|
|
|
var bw = new BinaryWriter(ms);
|
|
|
|
|
if (MultiChannelSource?.MultiChannelBlocks != null)
|
2020-02-07 01:02:55 +08:00
|
|
|
|
{
|
2020-02-07 16:27:23 +08:00
|
|
|
|
foreach (var blk in MultiChannelSource.MultiChannelBlocks)
|
2020-02-07 01:02:55 +08:00
|
|
|
|
{
|
2020-02-07 16:27:23 +08:00
|
|
|
|
if (MultiChannelIndex < (blk?.Channels?.Length ?? 0))
|
|
|
|
|
{
|
|
|
|
|
var chan = blk.Channels[MultiChannelIndex];
|
|
|
|
|
var cdata = chan.ChannelData;
|
|
|
|
|
bw.Write(cdata);
|
|
|
|
|
}
|
2020-02-07 01:02:55 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2020-02-07 16:27:23 +08:00
|
|
|
|
bw.Flush();
|
|
|
|
|
ms.Position = 0;
|
|
|
|
|
Data = new byte[ms.Length];
|
|
|
|
|
ms.Read(Data, 0, (int)ms.Length);
|
2020-02-07 01:02:55 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
2020-02-07 16:27:23 +08:00
|
|
|
|
return Data;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Stream GetWavStream()
|
|
|
|
|
{
|
|
|
|
|
byte[] dataPCM = GetWavData();
|
|
|
|
|
var bitsPerSample = BitsPerSample;
|
|
|
|
|
var channels = Channels;
|
|
|
|
|
var codec = MultiChannelFormat?.Codec ?? Format?.Codec ?? AwcCodecFormat.PCM;
|
|
|
|
|
|
|
|
|
|
if (codec == AwcCodecFormat.ADPCM)//just convert ADPCM to PCM for compatibility reasons
|
2020-02-07 01:02:55 +08:00
|
|
|
|
{
|
2020-02-07 16:27:23 +08:00
|
|
|
|
dataPCM = ADPCMCodec.DecodeADPCM(dataPCM, SampleCount);
|
|
|
|
|
bitsPerSample = 16;
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-02-07 16:27:23 +08:00
|
|
|
|
short formatcodec = 1; // 1 = WAVE_FORMAT_PCM
|
2020-02-07 01:02:55 +08:00
|
|
|
|
int byteRate = SamplesPerSecond * channels * bitsPerSample / 8;
|
|
|
|
|
short blockAlign = (short)(channels * bitsPerSample / 8);
|
2020-02-07 16:27:23 +08:00
|
|
|
|
short samplesPerBlock = 0;
|
|
|
|
|
bool addextrafmt = false;
|
|
|
|
|
|
|
|
|
|
//if (codec == AwcCodecFormat.ADPCM)//can't seem to get ADPCM wav files to work :(
|
|
|
|
|
//{
|
|
|
|
|
// bitsPerSample = 4;
|
|
|
|
|
// formatcodec = 17;
|
|
|
|
|
// byteRate = (int)(SamplesPerSecond * 0.50685 * channels);
|
|
|
|
|
// blockAlign = 2048;// (short)(256 * (4 * channels));// (short)(36 * channels);//256;// 2048;//
|
|
|
|
|
// samplesPerBlock = 4088;// (short)(((blockAlign - (4 * channels)) * 8) / (bitsPerSample * channels) + 1); // 2044;//
|
|
|
|
|
// addextrafmt = true;
|
|
|
|
|
//}
|
|
|
|
|
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
|
|
|
|
MemoryStream stream = new MemoryStream();
|
|
|
|
|
BinaryWriter w = new BinaryWriter(stream);
|
2020-02-07 16:27:23 +08:00
|
|
|
|
int wavLength = 36 + dataPCM.Length;
|
|
|
|
|
if (addextrafmt)
|
|
|
|
|
{
|
|
|
|
|
wavLength += 4;
|
|
|
|
|
}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
|
|
|
|
// RIFF chunk
|
|
|
|
|
w.Write("RIFF".ToCharArray());
|
2020-02-07 16:27:23 +08:00
|
|
|
|
w.Write((int)wavLength);
|
2018-02-24 21:59:00 +08:00
|
|
|
|
w.Write("WAVE".ToCharArray());
|
|
|
|
|
|
|
|
|
|
// fmt sub-chunk
|
|
|
|
|
w.Write("fmt ".ToCharArray());
|
2020-02-07 16:27:23 +08:00
|
|
|
|
w.Write((int)(addextrafmt ? 20 : 16)); // fmt size
|
|
|
|
|
w.Write((short)formatcodec);
|
2020-02-07 01:02:55 +08:00
|
|
|
|
w.Write((short)channels);
|
2018-02-24 21:59:00 +08:00
|
|
|
|
w.Write((int)SamplesPerSecond);
|
|
|
|
|
w.Write((int)byteRate);
|
|
|
|
|
w.Write((short)blockAlign);
|
|
|
|
|
w.Write((short)bitsPerSample);
|
2020-02-07 16:27:23 +08:00
|
|
|
|
if (addextrafmt)
|
|
|
|
|
{
|
|
|
|
|
w.Write((ushort)0x0002);
|
|
|
|
|
w.Write((ushort)samplesPerBlock);
|
|
|
|
|
}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
|
|
|
|
// data sub-chunk
|
|
|
|
|
w.Write("data".ToCharArray());
|
|
|
|
|
w.Write((int)dataPCM.Length);
|
|
|
|
|
w.Write(dataPCM);
|
|
|
|
|
|
|
|
|
|
w.Flush();
|
|
|
|
|
stream.Position = 0;
|
|
|
|
|
return stream;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TC(typeof(EXP))] public class AwcAudioAnimClipDict
|
|
|
|
|
{
|
|
|
|
|
public byte[] Data { get; set; }
|
|
|
|
|
public ClipDictionary ClipDict { get; set; }
|
|
|
|
|
|
|
|
|
|
public AwcAudioAnimClipDict(DataReader r, AwcChunkInfo info)
|
|
|
|
|
{
|
|
|
|
|
Data = r.ReadBytes(info.Size);
|
|
|
|
|
|
|
|
|
|
if ((Data == null) || (Data.Length < 16)) return;
|
|
|
|
|
|
|
|
|
|
var data = Data;
|
|
|
|
|
|
|
|
|
|
RpfResourceFileEntry resentry = new RpfResourceFileEntry();
|
|
|
|
|
uint rsc7 = BitConverter.ToUInt32(data, 0);
|
|
|
|
|
int version = BitConverter.ToInt32(data, 4);
|
|
|
|
|
resentry.SystemFlags = BitConverter.ToUInt32(data, 8);
|
|
|
|
|
resentry.GraphicsFlags = BitConverter.ToUInt32(data, 12);
|
|
|
|
|
|
|
|
|
|
if (rsc7 != 0x37435352)
|
|
|
|
|
{ } //testing..
|
|
|
|
|
if (version != 46) //46 is Clip Dictionary...
|
|
|
|
|
{ }
|
|
|
|
|
|
|
|
|
|
int newlen = data.Length - 16; //trim the header from the data passed to the next step.
|
|
|
|
|
int arrlen = Math.Max(newlen, resentry.SystemSize + resentry.GraphicsSize);//expand it as necessary for the reader.
|
|
|
|
|
byte[] newdata = new byte[arrlen];
|
|
|
|
|
Buffer.BlockCopy(data, 16, newdata, 0, newlen);
|
|
|
|
|
data = newdata;
|
|
|
|
|
|
|
|
|
|
ResourceDataReader rd = new ResourceDataReader(resentry, data);
|
|
|
|
|
|
|
|
|
|
ClipDict = rd.ReadBlock<ClipDictionary>();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return (ClipDict?.ClipsMapEntries ?? 0).ToString() + " entries";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
[TC(typeof(EXP))] public class AwcAudioPeak
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-06 02:33:12 +08:00
|
|
|
|
public AwcChunkInfo ChunkInfo { get; set; }
|
2020-02-08 03:24:04 +08:00
|
|
|
|
public ushort[] Data { get; set; }
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
public AwcAudioPeak(DataReader r, AwcChunkInfo info)
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-06 02:33:12 +08:00
|
|
|
|
ChunkInfo = info;
|
2020-02-08 03:24:04 +08:00
|
|
|
|
|
|
|
|
|
//if ((info.Size % 2) != 0)
|
|
|
|
|
//{ }//no hit
|
|
|
|
|
var count = info.Size / 2;
|
|
|
|
|
Data = new ushort[count];
|
|
|
|
|
for (int i = 0; i < count; i++)
|
|
|
|
|
{
|
|
|
|
|
Data[i] = r.ReadUInt16();
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
if (Data == null) return "";
|
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
|
for (int i = 0; i < Data.Length; i++)
|
|
|
|
|
{
|
|
|
|
|
if (sb.Length > 0) sb.Append(' ');
|
|
|
|
|
sb.Append(Data[i].ToString());
|
|
|
|
|
}
|
|
|
|
|
return sb.ToString();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
[TC(typeof(EXP))] public class AwcAudioUnk2B
|
|
|
|
|
{
|
|
|
|
|
public AwcChunkInfo ChunkInfo { get; set; }
|
|
|
|
|
public Item[] Items { get; set; }
|
|
|
|
|
|
|
|
|
|
public class Item
|
|
|
|
|
{
|
|
|
|
|
public MetaHash Name { get; set; }
|
|
|
|
|
public uint UnkUint1 { get; set; }
|
|
|
|
|
public float UnkFloat1 { get; set; }
|
|
|
|
|
public float UnkFloat2 { get; set; }
|
|
|
|
|
public float UnkFloat3 { get; set; }
|
|
|
|
|
public float UnkFloat4 { get; set; }
|
|
|
|
|
public float UnkFloat5 { get; set; }
|
|
|
|
|
public float UnkFloat6 { get; set; }
|
|
|
|
|
public uint UnkUint2 { get; set; }
|
|
|
|
|
|
|
|
|
|
public Item(DataReader r)
|
|
|
|
|
{
|
|
|
|
|
Name = r.ReadUInt32();
|
|
|
|
|
UnkUint1 = r.ReadUInt32();
|
|
|
|
|
UnkFloat1 = r.ReadSingle();
|
|
|
|
|
UnkFloat2 = r.ReadSingle();
|
|
|
|
|
UnkFloat3 = r.ReadSingle();
|
|
|
|
|
UnkFloat4 = r.ReadSingle();
|
|
|
|
|
UnkFloat5 = r.ReadSingle();
|
|
|
|
|
UnkFloat6 = r.ReadSingle();
|
|
|
|
|
UnkUint2 = r.ReadUInt32();
|
|
|
|
|
|
|
|
|
|
//switch (Name)
|
|
|
|
|
//{
|
|
|
|
|
// case 0xceda50be: //
|
|
|
|
|
// case 0x452c06fc: //
|
|
|
|
|
// case 0xba377ce2: //
|
|
|
|
|
// case 0xbe4c6d06: //
|
|
|
|
|
// case 0x9db051b4: //
|
|
|
|
|
// case 0x8a726e9f: //
|
|
|
|
|
// case 0x1f60ea95: //
|
|
|
|
|
// case 0x14e63a65: //
|
|
|
|
|
// case 0x32b4abf4: //
|
|
|
|
|
// case 0xe2b1dd62: //
|
|
|
|
|
// case 0x482d3572: //
|
|
|
|
|
// case 0x32f8d7a7: //
|
|
|
|
|
// case 0x9144296b: //
|
|
|
|
|
// case 0x3c73a8a4: //
|
|
|
|
|
// case 0x057c10c5: //
|
|
|
|
|
// case 0x981a4da0: //
|
|
|
|
|
// case 0x519d5f74: //
|
|
|
|
|
// case 0x0de43bc8: //
|
|
|
|
|
// case 0x89c16359: //
|
|
|
|
|
// case 0xd8884a6b: //
|
|
|
|
|
// case 0xfec7eb20: //
|
|
|
|
|
// case 0x06f0f709: //
|
|
|
|
|
// case 0x788a8abd: //
|
|
|
|
|
// break;
|
|
|
|
|
// default:
|
|
|
|
|
// break;//and more...
|
|
|
|
|
//}
|
|
|
|
|
|
|
|
|
|
//switch (UnkUint2)
|
|
|
|
|
//{
|
|
|
|
|
// case 1:
|
|
|
|
|
// case 0:
|
|
|
|
|
// break;
|
|
|
|
|
// default:
|
|
|
|
|
// break;//no hit
|
|
|
|
|
//}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return Name.ToString() + ": " + UnkUint1.ToString() + ", " + UnkFloat1.ToString() + ", " + UnkFloat2.ToString() + ", " + UnkFloat3.ToString() + ", " + UnkFloat4.ToString() + ", " + UnkFloat5.ToString() + ", " + UnkFloat6.ToString() + ", " + UnkUint2.ToString();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public AwcAudioUnk2B(DataReader r, AwcChunkInfo info)
|
|
|
|
|
{
|
|
|
|
|
ChunkInfo = info;
|
|
|
|
|
|
|
|
|
|
// (hash, uint, 6x floats, uint) * n
|
|
|
|
|
|
|
|
|
|
//if ((info.Size % 36) != 0)
|
|
|
|
|
//{ }//no hit
|
|
|
|
|
var count = info.Size / 36;
|
|
|
|
|
Items = new Item[count];
|
|
|
|
|
for (int i = 0; i < count; i++)
|
|
|
|
|
{
|
|
|
|
|
Items[i] = new Item(r);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return (Items?.Length ?? 0).ToString() + " items";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TC(typeof(EXP))] public class AwcAudioUnk5A
|
|
|
|
|
{
|
|
|
|
|
public AwcChunkInfo ChunkInfo { get; set; }
|
|
|
|
|
public Item[] Items { get; set; }
|
|
|
|
|
public float UnkFloat1 { get; set; }
|
|
|
|
|
|
|
|
|
|
public class Item
|
|
|
|
|
{
|
|
|
|
|
public uint UnkUint1 { get; set; }
|
|
|
|
|
public float UnkFloat1 { get; set; }
|
|
|
|
|
public ushort UnkUshort1 { get; set; }
|
|
|
|
|
public ushort UnkUshort2 { get; set; }
|
|
|
|
|
|
|
|
|
|
public Item(DataReader r)
|
|
|
|
|
{
|
|
|
|
|
UnkUint1 = r.ReadUInt32();
|
|
|
|
|
UnkFloat1 = r.ReadSingle();
|
|
|
|
|
UnkUshort1 = r.ReadUInt16();
|
|
|
|
|
UnkUshort2 = r.ReadUInt16();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return UnkUint1.ToString() + ", " + UnkFloat1.ToString() + ", " + UnkUshort1.ToString() + ", " + UnkUshort2.ToString();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public AwcAudioUnk5A(DataReader r, AwcChunkInfo info)
|
|
|
|
|
{
|
|
|
|
|
ChunkInfo = info;
|
|
|
|
|
|
|
|
|
|
//int, (2x floats, int) * n ?
|
|
|
|
|
|
|
|
|
|
//if ((info.Size % 12) != 4)
|
|
|
|
|
//{ }//no hit
|
|
|
|
|
var count = (info.Size - 4) / 12;
|
|
|
|
|
Items = new Item[count];
|
|
|
|
|
for (int i = 0; i < count; i++)
|
|
|
|
|
{
|
|
|
|
|
Items[i] = new Item(r);
|
|
|
|
|
}
|
|
|
|
|
UnkFloat1 = r.ReadSingle();
|
|
|
|
|
|
|
|
|
|
//if (UnkFloat1 > 1.0f)
|
|
|
|
|
//{ }//no hit
|
|
|
|
|
//if (UnkFloat1 < 0.45833f)
|
|
|
|
|
//{ }//no hit
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return (Items?.Length ?? 0).ToString() + " items";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TC(typeof(EXP))] public class AwcAudioUnkD9
|
|
|
|
|
{
|
|
|
|
|
public AwcChunkInfo ChunkInfo { get; set; }
|
|
|
|
|
public uint ItemCount { get; set; }
|
|
|
|
|
public Item[] Items { get; set; }
|
|
|
|
|
|
|
|
|
|
public class Item
|
|
|
|
|
{
|
|
|
|
|
public uint UnkUint1 { get; set; } = 2;
|
|
|
|
|
public uint ItemCount { get; set; }
|
|
|
|
|
public MetaHash Hash { get; set; } = 0x4c633d07;
|
|
|
|
|
public uint[] Items { get; set; }
|
|
|
|
|
|
|
|
|
|
public Item(DataReader r)
|
|
|
|
|
{
|
|
|
|
|
UnkUint1 = r.ReadUInt32();
|
|
|
|
|
ItemCount = r.ReadUInt32();
|
|
|
|
|
Hash = r.ReadUInt32();
|
|
|
|
|
Items = new uint[ItemCount];
|
|
|
|
|
for (int i = 0; i < ItemCount; i++)
|
|
|
|
|
{
|
|
|
|
|
Items[i] = r.ReadUInt32();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch (UnkUint1)
|
|
|
|
|
{
|
|
|
|
|
case 2:
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
switch (Hash)
|
|
|
|
|
{
|
|
|
|
|
case 0x4c633d07:
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return Hash.ToString() + ": " + UnkUint1.ToString() + ": " + ItemCount.ToString() + " items";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public AwcAudioUnkD9(DataReader r, AwcChunkInfo info)
|
|
|
|
|
{
|
|
|
|
|
ChunkInfo = info;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//uint count
|
|
|
|
|
// [count*items]: uint(type?), uint(count2), hash, [count2*uint]
|
|
|
|
|
|
|
|
|
|
ItemCount = r.ReadUInt32();
|
|
|
|
|
Items = new Item[ItemCount];
|
|
|
|
|
for (int i = 0; i < ItemCount; i++)
|
|
|
|
|
{
|
|
|
|
|
Items[i] = new Item(r);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
////if ((info.Size % 4) != 0)
|
|
|
|
|
////{ }//no hit
|
|
|
|
|
//var count = info.Size / 4;
|
|
|
|
|
//Data = new uint[count];
|
|
|
|
|
//for (int i = 0; i < count; i++)
|
|
|
|
|
//{
|
|
|
|
|
// Data[i] = r.ReadUInt32();
|
|
|
|
|
//}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return (Items?.Length ?? 0).ToString() + " items";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[TC(typeof(EXP))] public class AwcAudioMarkers
|
|
|
|
|
{
|
|
|
|
|
public AwcChunkInfo ChunkInfo { get; set; }
|
|
|
|
|
public Marker[] Markers { get; set; }
|
|
|
|
|
|
|
|
|
|
public class Marker
|
|
|
|
|
{
|
|
|
|
|
public MetaHash Name { get; set; }
|
|
|
|
|
public MetaHash Value { get; set; }//usually a float, but in some cases a hash, or other value
|
|
|
|
|
public uint SampleOffset { get; set; }
|
|
|
|
|
public uint Unused { get; set; }
|
|
|
|
|
|
|
|
|
|
public Marker(DataReader r)
|
|
|
|
|
{
|
|
|
|
|
Name = r.ReadUInt32();
|
|
|
|
|
Value = r.ReadUInt32();
|
|
|
|
|
SampleOffset = r.ReadUInt32();
|
|
|
|
|
Unused = r.ReadUInt32();
|
|
|
|
|
|
|
|
|
|
//switch (Name)
|
|
|
|
|
//{
|
|
|
|
|
// case 0:
|
|
|
|
|
// case 0xa6d93246: // trackid
|
|
|
|
|
// case 0xe89ae78c: // beat
|
|
|
|
|
// case 0xf31b4f6a: // rockout
|
|
|
|
|
// case 0x08dba0f8: // dj
|
|
|
|
|
// case 0x7a495db3: // tempo
|
|
|
|
|
// case 0x14d857be: // g_s
|
|
|
|
|
// break;
|
|
|
|
|
// case 0xcd171e55: //
|
|
|
|
|
// case 0x806b80c9: // 1
|
|
|
|
|
// case 0x91aa2346: // 2
|
|
|
|
|
// case 0x11976678: // r_p
|
|
|
|
|
// case 0x91be54cb: //
|
|
|
|
|
// case 0xab2238c0: //
|
|
|
|
|
// case 0xdb599288: //
|
|
|
|
|
// case 0x2ce40eb5: //
|
|
|
|
|
// case 0xa35e1092: // 01
|
|
|
|
|
// case 0x1332b405: // tank_jump
|
|
|
|
|
// case 0x2b20b891: //
|
|
|
|
|
// case 0x8aa726e7: // tank_jump_land
|
|
|
|
|
// case 0xe0bfba99: // tank_turret_move
|
|
|
|
|
// case 0x1d91339e: //
|
|
|
|
|
// case 0xa5344b07: //
|
|
|
|
|
// case 0x7a7cba39: // tank_weapon_main_cannon_hit
|
|
|
|
|
// case 0xd66a90c3: //
|
|
|
|
|
// case 0x1fd18857: // 14
|
|
|
|
|
// case 0x65a52c67: //
|
|
|
|
|
// case 0xd8846402: // uihit
|
|
|
|
|
// case 0x8958bce4: // m_p
|
|
|
|
|
// if (Value != 0)
|
|
|
|
|
// { }//no hit
|
|
|
|
|
// break;
|
|
|
|
|
// default:
|
|
|
|
|
// break;//no hit
|
|
|
|
|
//}
|
|
|
|
|
|
|
|
|
|
//if (Unused != 0)
|
|
|
|
|
//{ }//no hit
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
var valstr = Value.Float.ToString();
|
|
|
|
|
switch (Name)
|
|
|
|
|
{
|
|
|
|
|
case 0xf31b4f6a: // rockout
|
|
|
|
|
case 0x08dba0f8: // dj
|
|
|
|
|
case 0x14d857be: // g_s
|
|
|
|
|
valstr = Value.ToString();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return Name.ToString() + ": " + valstr + ", " + SampleOffset.ToString() + ", " + Unused.ToString();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public AwcAudioMarkers(DataReader r, AwcChunkInfo info)
|
|
|
|
|
{
|
|
|
|
|
ChunkInfo = info;
|
|
|
|
|
|
|
|
|
|
//if ((info.Size % 16) != 0)
|
|
|
|
|
//{ }//no hit
|
|
|
|
|
var count = info.Size / 16;
|
|
|
|
|
Markers = new Marker[count];
|
|
|
|
|
for (int i = 0; i < count; i++)
|
|
|
|
|
{
|
|
|
|
|
Markers[i] = new Marker(r);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return (Markers?.Length ?? 0).ToString() + " markers";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
[TC(typeof(EXP))] public class AwcAudioMIDI
|
|
|
|
|
{
|
|
|
|
|
public AwcChunkInfo ChunkInfo { get; set; }
|
|
|
|
|
public byte[] Data { get; set; }
|
|
|
|
|
|
|
|
|
|
public AwcAudioMIDI(DataReader r, AwcChunkInfo info)
|
|
|
|
|
{
|
|
|
|
|
ChunkInfo = info;
|
|
|
|
|
Data = r.ReadBytes(info.Size);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return "MIDI - " + (Data?.Length??0).ToString() + " bytes";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-06 02:33:12 +08:00
|
|
|
|
[TC(typeof(EXP))] public class AwcChannelOffsetsInfo
|
|
|
|
|
{
|
|
|
|
|
public AwcChunkInfo ChunkInfo { get; set; }
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public uint[] SampleOffsets { get; set; }
|
2020-02-06 02:33:12 +08:00
|
|
|
|
|
|
|
|
|
public AwcChannelOffsetsInfo(DataReader r, AwcChunkInfo info)
|
|
|
|
|
{
|
|
|
|
|
ChunkInfo = info;
|
2020-02-08 03:24:04 +08:00
|
|
|
|
|
|
|
|
|
//if ((info.Size % 4) != 0)
|
|
|
|
|
//{ }//no hit
|
2020-02-06 02:33:12 +08:00
|
|
|
|
var count = info.Size / 4;
|
2020-02-07 01:02:55 +08:00
|
|
|
|
SampleOffsets = new uint[count];
|
2020-02-06 02:33:12 +08:00
|
|
|
|
for (int i = 0; i < count; i++)
|
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
SampleOffsets[i] = r.ReadUInt32();
|
2020-02-06 02:33:12 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
return (SampleOffsets?.Length ?? 0).ToString() + " items";
|
2020-02-06 02:33:12 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
[TC(typeof(EXP))] public class AwcChannelDataBlock
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public byte[] Data { get; set; }
|
|
|
|
|
public int Index { get; set; }
|
|
|
|
|
public int SampleOffset { get; set; }
|
|
|
|
|
|
|
|
|
|
public AwcChannelDataItem[] Channels { get; set; }
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public AwcChannelDataBlock(byte[] data, AwcChannelBlockInfo channelInfo, Endianess endianess, int index, int sampleOffset)
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
Data = data;
|
|
|
|
|
Index = index;
|
|
|
|
|
SampleOffset = sampleOffset;
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
using (var ms = new MemoryStream(data))
|
|
|
|
|
{
|
|
|
|
|
var r = new DataReader(ms, endianess);
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
var channels = channelInfo?.Channels;
|
|
|
|
|
var channelcount = channelInfo?.ChannelCount ?? 0;
|
|
|
|
|
|
|
|
|
|
var ilist = new List<AwcChannelDataItem>();
|
|
|
|
|
for (int i = 0; i < channelcount; i++)
|
|
|
|
|
{
|
|
|
|
|
var channel = new AwcChannelDataItem(r);
|
|
|
|
|
channel.ChannelInfo = ((channels != null) && (i < channels.Length)) ? channels[i] : null;
|
|
|
|
|
ilist.Add(channel);
|
|
|
|
|
}
|
|
|
|
|
Channels = ilist.ToArray();
|
|
|
|
|
|
|
|
|
|
foreach (var channel in Channels)
|
|
|
|
|
{
|
|
|
|
|
channel.ReadOffsets(r);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var padc = (0x800 - (r.Position % 0x800)) % 0x800;
|
|
|
|
|
var padb = r.ReadBytes((int)padc);
|
|
|
|
|
|
|
|
|
|
foreach (var channel in Channels)
|
|
|
|
|
{
|
|
|
|
|
var bcnt = channel.OffsetCount * 2048;
|
|
|
|
|
channel.ChannelData = r.ReadBytes(bcnt);
|
|
|
|
|
}
|
|
|
|
|
if (r.Position != r.Length)
|
|
|
|
|
{ }//still more, just padding?
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return (Data?.Length ?? 0).ToString() + " bytes";
|
|
|
|
|
}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
[TC(typeof(EXP))] public class AwcChannelDataItem
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public int Unk0 { get; set; }
|
|
|
|
|
public int OffsetCount { get; set; }
|
|
|
|
|
public int Unk2 { get; set; }
|
|
|
|
|
public int SampleCount { get; set; }
|
|
|
|
|
public int Unk4 { get; set; }
|
|
|
|
|
public int Unk5 { get; set; }
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public int[] SampleOffsets { get; set; }
|
|
|
|
|
|
|
|
|
|
public AwcChannelBlockItemInfo ChannelInfo { get; set; } //for convenience
|
|
|
|
|
|
|
|
|
|
public byte[] ChannelData { get; set; }
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public AwcChannelDataItem(DataReader r)
|
|
|
|
|
{
|
|
|
|
|
Unk0 = r.ReadInt32();
|
|
|
|
|
OffsetCount = r.ReadInt32();
|
|
|
|
|
Unk2 = r.ReadInt32();
|
|
|
|
|
SampleCount = r.ReadInt32();
|
|
|
|
|
Unk4 = r.ReadInt32();
|
|
|
|
|
Unk5 = r.ReadInt32();
|
|
|
|
|
}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public void ReadOffsets(DataReader r)
|
|
|
|
|
{
|
|
|
|
|
var olist = new List<int>();
|
|
|
|
|
for (int i = 0; i < OffsetCount; i++)
|
2018-02-24 21:59:00 +08:00
|
|
|
|
{
|
2020-02-07 01:02:55 +08:00
|
|
|
|
var v = r.ReadInt32();
|
|
|
|
|
olist.Add(v);
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
2020-02-07 01:02:55 +08:00
|
|
|
|
SampleOffsets = olist.ToArray();
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-02-07 01:02:55 +08:00
|
|
|
|
public override string ToString()
|
|
|
|
|
{
|
|
|
|
|
return Unk0.ToString() + ", " + OffsetCount.ToString() + ", " + Unk2.ToString() + ", " + SampleCount.ToString() + ", " + Unk4.ToString() + ", " + Unk5.ToString();
|
|
|
|
|
}
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
2020-02-07 16:27:23 +08:00
|
|
|
|
|
|
|
|
|
public class ADPCMCodec
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
private static int[] ima_index_table =
|
|
|
|
|
{
|
|
|
|
|
-1, -1, -1, -1, 2, 4, 6, 8,
|
|
|
|
|
-1, -1, -1, -1, 2, 4, 6, 8
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
private static short[] ima_step_table =
|
|
|
|
|
{
|
|
|
|
|
7, 8, 9, 10, 11, 12, 13, 14, 16, 17,
|
|
|
|
|
19, 21, 23, 25, 28, 31, 34, 37, 41, 45,
|
|
|
|
|
50, 55, 60, 66, 73, 80, 88, 97, 107, 118,
|
|
|
|
|
130, 143, 157, 173, 190, 209, 230, 253, 279, 307,
|
|
|
|
|
337, 371, 408, 449, 494, 544, 598, 658, 724, 796,
|
|
|
|
|
876, 963, 1060, 1166, 1282, 1411, 1552, 1707, 1878, 2066,
|
|
|
|
|
2272, 2499, 2749, 3024, 3327, 3660, 4026, 4428, 4871, 5358,
|
|
|
|
|
5894, 6484, 7132, 7845, 8630, 9493, 10442, 11487, 12635, 13899,
|
|
|
|
|
15289, 16818, 18500, 20350, 22385, 24623, 27086, 29794, 32767
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
private static int clip(int value, int min, int max)
|
|
|
|
|
{
|
|
|
|
|
if (value < min) return min;
|
|
|
|
|
if (value > max) return max;
|
|
|
|
|
return value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static byte[] DecodeADPCM(byte[] data, int sampleCount)
|
|
|
|
|
{
|
|
|
|
|
byte[] dataPCM = new byte[data.Length * 4];
|
2020-02-07 19:22:40 +08:00
|
|
|
|
int predictor = 0, stepIndex = 0;
|
2020-02-07 16:27:23 +08:00
|
|
|
|
int readingOffset = 0, writingOffset = 0, bytesInBlock = 0;
|
|
|
|
|
|
|
|
|
|
void parseNibble(byte nibble)
|
|
|
|
|
{
|
2020-02-07 19:22:40 +08:00
|
|
|
|
var step = ima_step_table[stepIndex];
|
|
|
|
|
int diff = ((((nibble & 7) << 1) + 1) * step) >> 3;
|
|
|
|
|
if ((nibble & 8) != 0) diff = -diff;
|
|
|
|
|
predictor = predictor + diff;
|
|
|
|
|
stepIndex = clip(stepIndex + ima_index_table[nibble], 0, 88);
|
2020-02-07 16:27:23 +08:00
|
|
|
|
int samplePCM = clip(predictor, -32768, 32767);
|
2020-02-07 19:22:40 +08:00
|
|
|
|
|
2020-02-07 16:27:23 +08:00
|
|
|
|
dataPCM[writingOffset] = (byte)(samplePCM & 0xFF);
|
|
|
|
|
dataPCM[writingOffset + 1] = (byte)((samplePCM >> 8) & 0xFF);
|
|
|
|
|
writingOffset += 2;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
while ((readingOffset < data.Length) && (sampleCount > 0))
|
|
|
|
|
{
|
|
|
|
|
if (bytesInBlock == 0)
|
|
|
|
|
{
|
2020-02-07 19:22:40 +08:00
|
|
|
|
stepIndex = clip(data[readingOffset], 0, 88);
|
2020-02-07 16:27:23 +08:00
|
|
|
|
predictor = BitConverter.ToInt16(data, readingOffset + 2);
|
|
|
|
|
bytesInBlock = 2044;
|
|
|
|
|
readingOffset += 4;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
parseNibble((byte)(data[readingOffset] & 0x0F));
|
|
|
|
|
parseNibble((byte)((data[readingOffset] >> 4) & 0x0F));
|
|
|
|
|
bytesInBlock--;
|
|
|
|
|
sampleCount -= 2;
|
|
|
|
|
readingOffset++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return dataPCM;
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-07 19:22:40 +08:00
|
|
|
|
public static byte[] EncodeADPCM(byte[] data, int sampleCount)
|
|
|
|
|
{
|
|
|
|
|
byte[] dataPCM = new byte[data.Length / 4];
|
|
|
|
|
|
|
|
|
|
int predictor = 0, stepIndex = 0;
|
|
|
|
|
int readingOffset = 0, writingOffset = 0, bytesInBlock = 0;
|
|
|
|
|
|
|
|
|
|
short readSample()
|
|
|
|
|
{
|
|
|
|
|
var s = BitConverter.ToInt16(data, readingOffset);
|
|
|
|
|
readingOffset += 2;
|
|
|
|
|
return s;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void writeInt16(short v)
|
|
|
|
|
{
|
|
|
|
|
var ba = BitConverter.GetBytes(v);
|
|
|
|
|
dataPCM[writingOffset++] = ba[0];
|
|
|
|
|
dataPCM[writingOffset++] = ba[1];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
byte encodeNibble(int pcm16)
|
|
|
|
|
{
|
|
|
|
|
int delta = pcm16 - predictor;
|
|
|
|
|
uint value = 0;
|
|
|
|
|
if (delta < 0)
|
|
|
|
|
{
|
|
|
|
|
value = 8;
|
|
|
|
|
delta = -delta;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var step = ima_step_table[stepIndex];
|
|
|
|
|
var diff = step >> 3;
|
|
|
|
|
if (delta > step)
|
|
|
|
|
{
|
|
|
|
|
value |= 4;
|
|
|
|
|
delta -= step;
|
|
|
|
|
diff += step;
|
|
|
|
|
}
|
|
|
|
|
step >>= 1;
|
|
|
|
|
if (delta > step)
|
|
|
|
|
{
|
|
|
|
|
value |= 2;
|
|
|
|
|
delta -= step;
|
|
|
|
|
diff += step;
|
|
|
|
|
}
|
|
|
|
|
step >>= 1;
|
|
|
|
|
if (delta > step)
|
|
|
|
|
{
|
|
|
|
|
value |= 1;
|
|
|
|
|
diff += step;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
predictor += (((value & 8) != 0) ? -diff : diff);
|
|
|
|
|
predictor = clip(predictor, short.MinValue, short.MaxValue);
|
|
|
|
|
|
|
|
|
|
stepIndex += ima_index_table[value & 7];
|
|
|
|
|
stepIndex = clip(stepIndex, 0, 88);
|
|
|
|
|
|
|
|
|
|
return (byte)value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
while ((writingOffset < dataPCM.Length) && (sampleCount > 0))
|
|
|
|
|
{
|
|
|
|
|
if (bytesInBlock == 0)
|
|
|
|
|
{
|
|
|
|
|
writeInt16((short)stepIndex);
|
|
|
|
|
writeInt16((short)predictor);
|
|
|
|
|
bytesInBlock = 2044;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
var s0 = readSample();
|
|
|
|
|
var s1 = readSample();
|
|
|
|
|
var b0 = encodeNibble(s0);
|
|
|
|
|
var b1 = encodeNibble(s1);
|
|
|
|
|
var b = (b0 & 0x0F) + ((b1 & 0x0F) << 4);
|
|
|
|
|
dataPCM[writingOffset++] = (byte)b;
|
|
|
|
|
bytesInBlock--;
|
|
|
|
|
sampleCount -= 2;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return dataPCM;
|
|
|
|
|
}
|
2020-02-07 16:27:23 +08:00
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-08 03:24:04 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public class AwcXml : MetaXmlBase
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
public static string GetXml(AwcFile awc, string outputFolder = "")
|
|
|
|
|
{
|
|
|
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
|
sb.AppendLine(XmlHeader);
|
|
|
|
|
|
|
|
|
|
if (awc != null)
|
|
|
|
|
{
|
|
|
|
|
AwcFile.WriteXmlNode(awc, sb, 0, outputFolder);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return sb.ToString();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class XmlAwc
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
public static AwcFile AwcYft(string xml, string inputFolder = "")
|
|
|
|
|
{
|
|
|
|
|
XmlDocument doc = new XmlDocument();
|
|
|
|
|
doc.LoadXml(xml);
|
|
|
|
|
return GetAwc(doc, inputFolder);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public static AwcFile GetAwc(XmlDocument doc, string inputFolder = "")
|
|
|
|
|
{
|
|
|
|
|
AwcFile r = null;
|
|
|
|
|
|
|
|
|
|
var node = doc.DocumentElement;
|
|
|
|
|
if (node != null)
|
|
|
|
|
{
|
|
|
|
|
r = AwcFile.ReadXmlNode(node, inputFolder);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
r.Name = Path.GetFileName(inputFolder);
|
|
|
|
|
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-02-24 21:59:00 +08:00
|
|
|
|
}
|