2018-12-20 01:28:43 +08:00
|
|
|
|
using SharpDX;
|
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
|
|
|
|
|
namespace CodeWalker.GameFiles
|
|
|
|
|
{
|
|
|
|
|
public class PsoBuilder
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public PsoBuilderPointer RootPointer { get; set; }
|
|
|
|
|
|
|
|
|
|
List<string> STRFStrings = new List<string>();
|
|
|
|
|
List<string> STRSStrings = new List<string>();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Dictionary<MetaName, PsoStructureInfo> StructureInfos = new Dictionary<MetaName, PsoStructureInfo>();
|
|
|
|
|
Dictionary<MetaName, PsoEnumInfo> EnumInfos = new Dictionary<MetaName, PsoEnumInfo>();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
List<PsoBuilderBlock> Blocks = new List<PsoBuilderBlock>();
|
|
|
|
|
int MaxBlockLength = 0x100000; //TODO: figure what this should be!
|
|
|
|
|
|
|
|
|
|
public PsoBuilderBlock EnsureBlock(MetaName type)
|
|
|
|
|
{
|
|
|
|
|
foreach (var block in Blocks)
|
|
|
|
|
{
|
|
|
|
|
if (block.StructureNameHash == type)
|
|
|
|
|
{
|
|
|
|
|
if (block.TotalSize < MaxBlockLength)
|
|
|
|
|
{
|
|
|
|
|
return block;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
PsoBuilderBlock b = new PsoBuilderBlock();
|
|
|
|
|
b.StructureNameHash = type;
|
|
|
|
|
b.Index = Blocks.Count;
|
|
|
|
|
Blocks.Add(b);
|
|
|
|
|
return b;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public PsoBuilderPointer AddItem<T>(MetaName type, T item) where T : struct
|
|
|
|
|
{
|
|
|
|
|
byte[] data = MetaTypes.ConvertToBytes(item);
|
|
|
|
|
return AddItem(type, data);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PsoBuilderPointer AddItem(MetaName type, byte[] data)
|
|
|
|
|
{
|
|
|
|
|
PsoBuilderBlock block = EnsureBlock(type);
|
|
|
|
|
int brem = data.Length % 16;
|
2019-01-14 09:25:03 +08:00
|
|
|
|
//if (brem > 0)
|
|
|
|
|
//{
|
|
|
|
|
// int newlen = data.Length - brem + 16;
|
|
|
|
|
// byte[] newdata = new byte[newlen];
|
|
|
|
|
// Buffer.BlockCopy(data, 0, newdata, 0, data.Length);
|
|
|
|
|
// data = newdata; //make sure item size is multiple of 16... so pointers don't need sub offsets!
|
|
|
|
|
//}
|
2018-12-20 01:28:43 +08:00
|
|
|
|
int idx = block.AddItem(data);
|
|
|
|
|
PsoBuilderPointer r = new PsoBuilderPointer();
|
|
|
|
|
r.BlockID = block.Index + 1;
|
|
|
|
|
r.Offset = (idx * data.Length);
|
|
|
|
|
r.Length = data.Length;
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PsoBuilderPointer AddItemArray<T>(MetaName type, T[] items) where T : struct
|
|
|
|
|
{
|
|
|
|
|
byte[] data = MetaTypes.ConvertArrayToBytes(items);
|
|
|
|
|
return AddItemArray(type, data, items.Length);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PsoBuilderPointer AddItemArray(MetaName type, byte[] data, int length)
|
|
|
|
|
{
|
|
|
|
|
PsoBuilderBlock block = EnsureBlock(type);
|
|
|
|
|
int datalen = data.Length;
|
|
|
|
|
int newlen = datalen;
|
|
|
|
|
//int lenrem = newlen % 16;
|
|
|
|
|
//if (lenrem != 0)
|
|
|
|
|
//{
|
|
|
|
|
// newlen += (16 - lenrem);
|
|
|
|
|
//}
|
|
|
|
|
byte[] newdata = new byte[newlen];
|
|
|
|
|
Buffer.BlockCopy(data, 0, newdata, 0, datalen);
|
|
|
|
|
int offs = block.TotalSize;
|
|
|
|
|
int idx = block.AddItem(newdata);
|
|
|
|
|
PsoBuilderPointer r = new PsoBuilderPointer();
|
|
|
|
|
r.BlockID = block.Index + 1;
|
|
|
|
|
r.Offset = offs; //(idx * data.Length);;
|
|
|
|
|
r.Length = length;
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public PsoPOINTER AddItemPtr<T>(MetaName type, T item) where T : struct //helper method for AddItem<T>
|
|
|
|
|
{
|
|
|
|
|
var ptr = AddItem(type, item);
|
|
|
|
|
return new PsoPOINTER(ptr.BlockID, ptr.Offset, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PsoPOINTER AddItemPtr(MetaName type, byte[] data)//helper method for AddItem<T>
|
|
|
|
|
{
|
|
|
|
|
var ptr = AddItem(type, data);
|
|
|
|
|
return new PsoPOINTER(ptr.BlockID, ptr.Offset, 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Array_Structure AddItemArrayPtr<T>(MetaName type, T[] items) where T : struct //helper method for AddItemArray<T>
|
|
|
|
|
{
|
|
|
|
|
if ((items == null) || (items.Length == 0)) return new Array_Structure();
|
|
|
|
|
var ptr = AddItemArray(type, items);
|
|
|
|
|
return new Array_Structure(ptr.Pointer, ptr.Length);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Array_Structure AddItemArrayPtr(MetaName type, byte[][] data) //helper method for AddItemArray<T>
|
|
|
|
|
{
|
|
|
|
|
if ((data == null) || (data.Length == 0)) return new Array_Structure();
|
|
|
|
|
|
|
|
|
|
int len = 0;
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < data.Length; i++)
|
|
|
|
|
{
|
|
|
|
|
len += data[i].Length;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var newdata = new byte[len];
|
|
|
|
|
|
|
|
|
|
int offset = 0;
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < data.Length; i++)
|
|
|
|
|
{
|
|
|
|
|
Buffer.BlockCopy(data[i], 0, newdata, offset, data[i].Length);
|
|
|
|
|
offset += data[i].Length;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var ptr = AddItemArray(type, newdata, data.Length);
|
|
|
|
|
return new Array_Structure(ptr.Pointer, ptr.Length);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public Array_StructurePointer AddPointerArray(PsoPOINTER[] arr)
|
|
|
|
|
{
|
|
|
|
|
if ((arr == null) || (arr.Length == 0)) return new Array_StructurePointer();
|
2019-10-31 14:11:12 +08:00
|
|
|
|
var ptr = AddItemArray((MetaName)MetaTypeName.PsoPOINTER, arr);
|
2018-12-20 01:28:43 +08:00
|
|
|
|
Array_StructurePointer sp = new Array_StructurePointer();
|
|
|
|
|
sp.Count1 = (ushort)arr.Length;
|
|
|
|
|
sp.Count2 = sp.Count1;
|
|
|
|
|
sp.Pointer = ptr.Pointer;
|
|
|
|
|
return sp;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public PsoBuilderPointer AddString(string str)
|
|
|
|
|
{
|
|
|
|
|
PsoBuilderBlock block = EnsureBlock((MetaName)1); //PsoSTRING seems to be 1
|
|
|
|
|
byte[] data = Encoding.ASCII.GetBytes(str + (char)0);
|
|
|
|
|
int datalen = data.Length;
|
|
|
|
|
int newlen = datalen;
|
|
|
|
|
//int lenrem = newlen % 16;
|
|
|
|
|
//if (lenrem != 0) //pad the data length up to multiple of 16.
|
|
|
|
|
//{
|
|
|
|
|
// newlen += (16 - lenrem);
|
|
|
|
|
//}
|
|
|
|
|
byte[] newdata = new byte[newlen];
|
|
|
|
|
Buffer.BlockCopy(data, 0, newdata, 0, datalen);
|
|
|
|
|
int offs = block.TotalSize;
|
|
|
|
|
int idx = block.AddItem(newdata);
|
|
|
|
|
PsoBuilderPointer r = new PsoBuilderPointer();
|
|
|
|
|
r.BlockID = block.Index + 1;
|
|
|
|
|
r.Offset = offs;// (idx * data.Length);
|
|
|
|
|
r.Length = datalen; //actual length of string.
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public Array_Vector3 AddPaddedVector3ArrayPtr(Vector4[] items)
|
|
|
|
|
{
|
|
|
|
|
if ((items == null) || (items.Length == 0)) return new Array_Vector3();
|
2019-10-31 14:11:12 +08:00
|
|
|
|
var ptr = AddItemArray((MetaName)1, items); //(MetaName)MetaTypeName.VECTOR4 padded to vec4...
|
2018-12-20 01:28:43 +08:00
|
|
|
|
return new Array_Vector3(ptr.Pointer, items.Length);
|
|
|
|
|
}
|
|
|
|
|
public Array_Vector3 AddVector2ArrayPtr(Vector2[] items)
|
|
|
|
|
{
|
|
|
|
|
if ((items == null) || (items.Length == 0)) return new Array_Vector3();
|
2019-10-31 14:11:12 +08:00
|
|
|
|
var ptr = AddItemArray((MetaName)1, items); //(MetaName)MetaTypeName.VECTOR4 padded to vec4...?
|
2018-12-20 01:28:43 +08:00
|
|
|
|
return new Array_Vector3(ptr.Pointer, items.Length);
|
|
|
|
|
}
|
|
|
|
|
public Array_uint AddHashArrayPtr(MetaHash[] items)
|
|
|
|
|
{
|
|
|
|
|
if ((items == null) || (items.Length == 0)) return new Array_uint();
|
2019-10-31 14:11:12 +08:00
|
|
|
|
var ptr = AddItemArray((MetaName)6, items); //(MetaName)MetaTypeName.HASH
|
2018-12-20 01:28:43 +08:00
|
|
|
|
return new Array_uint(ptr.Pointer, items.Length);
|
|
|
|
|
}
|
|
|
|
|
public Array_uint AddUIntArrayPtr(uint[] items)
|
|
|
|
|
{
|
|
|
|
|
if ((items == null) || (items.Length == 0)) return new Array_uint();
|
|
|
|
|
var ptr = AddItemArray((MetaName)6, items);
|
|
|
|
|
return new Array_uint(ptr.Pointer, items.Length);
|
|
|
|
|
}
|
|
|
|
|
public Array_uint AddSIntArrayPtr(int[] items)
|
|
|
|
|
{
|
|
|
|
|
if ((items == null) || (items.Length == 0)) return new Array_uint();
|
|
|
|
|
var ptr = AddItemArray((MetaName)5, items);
|
|
|
|
|
return new Array_uint(ptr.Pointer, items.Length);
|
|
|
|
|
}
|
|
|
|
|
public Array_ushort AddUShortArrayPtr(ushort[] items)
|
|
|
|
|
{
|
|
|
|
|
if ((items == null) || (items.Length == 0)) return new Array_ushort();
|
|
|
|
|
var ptr = AddItemArray((MetaName)4, items);
|
|
|
|
|
return new Array_ushort(ptr.Pointer, items.Length);
|
|
|
|
|
}
|
|
|
|
|
public Array_byte AddByteArrayPtr(byte[] items)
|
|
|
|
|
{
|
|
|
|
|
if ((items == null) || (items.Length == 0)) return new Array_byte();
|
|
|
|
|
var ptr = AddItemArray((MetaName)2, items);
|
|
|
|
|
return new Array_byte(ptr.Pointer, items.Length);
|
|
|
|
|
}
|
|
|
|
|
public Array_float AddFloatArrayPtr(float[] items)
|
|
|
|
|
{
|
|
|
|
|
if ((items == null) || (items.Length == 0)) return new Array_float();
|
2019-10-31 14:11:12 +08:00
|
|
|
|
var ptr = AddItemArray((MetaName)7, items); //MetaName.PsoFLOAT ? comes up as (MetaName)MetaTypeName.POINTER due to RSC meta values
|
2018-12-20 01:28:43 +08:00
|
|
|
|
return new Array_float(ptr.Pointer, items.Length);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void AddStringToSTRF(string str)
|
|
|
|
|
{
|
|
|
|
|
STRFStrings.Add(str);
|
|
|
|
|
}
|
|
|
|
|
public void AddStringToSTRS(string str)
|
|
|
|
|
{
|
|
|
|
|
STRSStrings.Add(str);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void AddStructureInfo(MetaName name)
|
|
|
|
|
{
|
|
|
|
|
if (!StructureInfos.ContainsKey(name))
|
|
|
|
|
{
|
|
|
|
|
PsoStructureInfo si = PsoTypes.GetStructureInfo(name);
|
|
|
|
|
if (si != null)
|
|
|
|
|
{
|
|
|
|
|
StructureInfos[name] = si;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
public void AddEnumInfo(MetaName name)
|
|
|
|
|
{
|
|
|
|
|
if (!EnumInfos.ContainsKey(name))
|
|
|
|
|
{
|
|
|
|
|
PsoEnumInfo ei = PsoTypes.GetEnumInfo(name);
|
|
|
|
|
if (ei != null)
|
|
|
|
|
{
|
|
|
|
|
EnumInfos[name] = ei;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public PsoStructureInfo AddMapNodeStructureInfo(MetaName valType)
|
|
|
|
|
{
|
|
|
|
|
PsoStructureInfo inf = null;
|
|
|
|
|
|
|
|
|
|
if (valType == 0)
|
|
|
|
|
{
|
2019-10-31 14:11:12 +08:00
|
|
|
|
inf = PsoTypes.GetStructureInfo((MetaName)MetaTypeName.ARRAYINFO); //default ARRAYINFO with pointer
|
2018-12-20 01:28:43 +08:00
|
|
|
|
if (!StructureInfos.ContainsKey(inf.IndexInfo.NameHash))
|
|
|
|
|
{
|
|
|
|
|
StructureInfos[inf.IndexInfo.NameHash] = inf;
|
|
|
|
|
}
|
|
|
|
|
return inf;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var structInfo = PsoTypes.GetStructureInfo(valType);
|
|
|
|
|
if (structInfo == null)
|
|
|
|
|
{ }//error?
|
|
|
|
|
|
2019-10-31 14:11:12 +08:00
|
|
|
|
MetaName xName = (MetaName)MetaTypeName.ARRAYINFO + 1; //257
|
2018-12-20 01:28:43 +08:00
|
|
|
|
bool nameOk = !StructureInfos.ContainsKey(xName);
|
|
|
|
|
while (!nameOk)
|
|
|
|
|
{
|
|
|
|
|
var exInfo = StructureInfos[xName];
|
|
|
|
|
var exInfoItem = exInfo.FindEntry(MetaName.Item);
|
|
|
|
|
if (((MetaName)(exInfoItem?.ReferenceKey ?? 0) == valType))
|
|
|
|
|
{
|
|
|
|
|
return exInfo; //this one already exists.. use it!
|
|
|
|
|
}
|
|
|
|
|
xName++;
|
|
|
|
|
nameOk = !StructureInfos.ContainsKey(xName);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
inf = new PsoStructureInfo(xName, 0, 2, 8 + structInfo.StructureLength,
|
|
|
|
|
new PsoStructureEntryInfo(MetaName.Key, PsoDataType.String, 0, 7, 0),
|
|
|
|
|
new PsoStructureEntryInfo(MetaName.Item, PsoDataType.Structure, 8, 0, valType)
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if (!StructureInfos.ContainsKey(xName))
|
|
|
|
|
{
|
|
|
|
|
StructureInfos[xName] = inf;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return inf;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//switch (valType)
|
|
|
|
|
//{
|
2018-12-20 17:52:42 +08:00
|
|
|
|
// case MetaName.fwClipDictionaryMetadata: return PsoTypes.GetStructureInfo((MetaName)257);
|
|
|
|
|
// case MetaName.fwMemoryGroupMetadata: return PsoTypes.GetStructureInfo((MetaName)258);
|
2018-12-20 01:28:43 +08:00
|
|
|
|
// case (MetaName)3219912345: return PsoTypes.GetStructureInfo((MetaName)259);
|
2019-10-31 14:11:12 +08:00
|
|
|
|
// case (MetaName)0: return PsoTypes.GetStructureInfo((MetaName)MetaTypeName.ARRAYINFO);
|
2018-12-20 01:28:43 +08:00
|
|
|
|
// default:
|
2019-10-31 14:11:12 +08:00
|
|
|
|
// return PsoTypes.GetStructureInfo((MetaName)MetaTypeName.ARRAYINFO);//error?
|
2018-12-20 01:28:43 +08:00
|
|
|
|
//}
|
|
|
|
|
//case (MetaName)257:
|
|
|
|
|
// return new PsoStructureInfo((MetaName)257, 0, 2, 32,
|
|
|
|
|
// new PsoStructureEntryInfo(MetaName.Key, PsoDataType.String, 0, 7, 0),
|
2018-12-20 17:52:42 +08:00
|
|
|
|
// new PsoStructureEntryInfo(MetaName.Item, PsoDataType.Structure, 8, 0, MetaName.fwClipDictionaryMetadata)
|
2018-12-20 01:28:43 +08:00
|
|
|
|
// );
|
|
|
|
|
//case (MetaName)258:
|
|
|
|
|
// return new PsoStructureInfo((MetaName)258, 0, 2, 24,
|
|
|
|
|
// new PsoStructureEntryInfo(MetaName.Key, PsoDataType.String, 0, 7, 0),
|
2018-12-20 17:52:42 +08:00
|
|
|
|
// new PsoStructureEntryInfo(MetaName.Item, PsoDataType.Structure, 8, 0, MetaName.fwMemoryGroupMetadata)
|
2018-12-20 01:28:43 +08:00
|
|
|
|
// );
|
|
|
|
|
//case (MetaName)259:
|
|
|
|
|
// return new PsoStructureInfo((MetaName)259, 0, 2, 32,
|
|
|
|
|
// new PsoStructureEntryInfo(MetaName.Key, PsoDataType.String, 0, 7, 0),
|
|
|
|
|
// new PsoStructureEntryInfo(MetaName.Item, PsoDataType.Structure, 8, 0, (MetaName)3219912345)
|
|
|
|
|
// );
|
|
|
|
|
//case (MetaName)3219912345:
|
|
|
|
|
// return new PsoStructureInfo((MetaName)3219912345, 0, 0, 24,
|
2019-10-31 14:11:12 +08:00
|
|
|
|
// new PsoStructureEntryInfo((MetaName)MetaTypeName.ARRAYINFO, PsoDataType.Structure, 0, 0, (MetaName)2356519750),
|
2019-01-12 14:45:19 +08:00
|
|
|
|
// new PsoStructureEntryInfo(MetaName.Adjustments, PsoDataType.Array, 8, 0, 0)
|
2018-12-20 01:28:43 +08:00
|
|
|
|
// );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public byte[] GetData()
|
|
|
|
|
{
|
|
|
|
|
int totlen = 16;
|
|
|
|
|
for (int i = 0; i < Blocks.Count; i++)
|
|
|
|
|
{
|
|
|
|
|
totlen += Blocks[i].TotalSize;
|
|
|
|
|
}
|
|
|
|
|
byte[] data = new byte[totlen];
|
|
|
|
|
int offset = 16; //reserved space for headers
|
|
|
|
|
for (int i = 0; i < Blocks.Count; i++)
|
|
|
|
|
{
|
|
|
|
|
var block = Blocks[i];
|
|
|
|
|
for (int j = 0; j < block.Items.Count; j++)
|
|
|
|
|
{
|
|
|
|
|
var bdata = block.Items[j];
|
|
|
|
|
Buffer.BlockCopy(bdata, 0, data, offset, bdata.Length);
|
|
|
|
|
offset += bdata.Length;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (offset != data.Length)
|
|
|
|
|
{ }
|
|
|
|
|
return data;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public PsoFile GetPso()
|
|
|
|
|
{
|
|
|
|
|
PsoFile pso = new PsoFile();
|
|
|
|
|
pso.SchemaSection = new PsoSchemaSection();
|
|
|
|
|
|
|
|
|
|
var schEntries = new List<PsoElementInfo>();
|
|
|
|
|
foreach (var structInfo in StructureInfos.Values)
|
|
|
|
|
{
|
|
|
|
|
schEntries.Add(structInfo);
|
|
|
|
|
}
|
|
|
|
|
foreach (var enumInfo in EnumInfos.Values)
|
|
|
|
|
{
|
|
|
|
|
schEntries.Add(enumInfo);
|
|
|
|
|
}
|
|
|
|
|
pso.SchemaSection.Entries = schEntries.ToArray();
|
|
|
|
|
pso.SchemaSection.EntriesIdx = new PsoElementIndexInfo[schEntries.Count];
|
|
|
|
|
for (int i = 0; i < schEntries.Count; i++)
|
|
|
|
|
{
|
|
|
|
|
pso.SchemaSection.EntriesIdx[i] = new PsoElementIndexInfo();
|
|
|
|
|
pso.SchemaSection.EntriesIdx[i].NameHash = schEntries[i].IndexInfo.NameHash;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (STRFStrings.Count > 0)
|
|
|
|
|
{
|
|
|
|
|
pso.STRFSection = new PsoSTRFSection();
|
|
|
|
|
pso.STRFSection.Strings = STRFStrings.ToArray();
|
|
|
|
|
}
|
|
|
|
|
if (STRSStrings.Count > 0)
|
|
|
|
|
{
|
|
|
|
|
pso.STRSSection = new PsoSTRSSection();
|
|
|
|
|
pso.STRSSection.Strings = STRSStrings.ToArray();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
pso.DataSection = new PsoDataSection();
|
|
|
|
|
pso.DataSection.Data = GetData();
|
|
|
|
|
|
|
|
|
|
pso.DataMapSection = new PsoDataMapSection();
|
|
|
|
|
pso.DataMapSection.Entries = new PsoDataMappingEntry[Blocks.Count];
|
|
|
|
|
pso.DataMapSection.RootId = RootPointer.BlockID;
|
|
|
|
|
var offset = 16;
|
|
|
|
|
for (int i = 0; i < Blocks.Count; i++)
|
|
|
|
|
{
|
|
|
|
|
var b = Blocks[i];
|
|
|
|
|
var e = new PsoDataMappingEntry();
|
|
|
|
|
e.NameHash = b.StructureNameHash;
|
|
|
|
|
e.Length = b.TotalSize;
|
|
|
|
|
e.Offset = offset;
|
|
|
|
|
offset += b.TotalSize;
|
|
|
|
|
pso.DataMapSection.Entries[i] = e;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return pso;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public class PsoBuilderBlock
|
|
|
|
|
{
|
|
|
|
|
public MetaName StructureNameHash { get; set; }
|
|
|
|
|
public List<byte[]> Items { get; set; } = new List<byte[]>();
|
|
|
|
|
public int TotalSize { get; set; } = 0;
|
|
|
|
|
public int Index { get; set; } = 0;
|
|
|
|
|
|
|
|
|
|
public int AddItem(byte[] item)
|
|
|
|
|
{
|
|
|
|
|
int idx = Items.Count;
|
|
|
|
|
Items.Add(item);
|
|
|
|
|
TotalSize += item.Length;
|
|
|
|
|
return idx;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public uint BasePointer
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
return (((uint)Index + 1) & 0xFFF);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//public MetaDataBlock GetMetaDataBlock()
|
|
|
|
|
//{
|
|
|
|
|
// if (TotalSize <= 0) return null;
|
|
|
|
|
// byte[] data = new byte[TotalSize];
|
|
|
|
|
// int offset = 0;
|
|
|
|
|
// for (int j = 0; j < Items.Count; j++)
|
|
|
|
|
// {
|
|
|
|
|
// var bdata = Items[j];
|
|
|
|
|
// Buffer.BlockCopy(bdata, 0, data, offset, bdata.Length);
|
|
|
|
|
// offset += bdata.Length;
|
|
|
|
|
// }
|
|
|
|
|
// MetaDataBlock db = new MetaDataBlock();
|
|
|
|
|
// db.StructureNameHash = StructureNameHash;
|
|
|
|
|
// db.DataLength = TotalSize;
|
|
|
|
|
// db.Data = data;
|
|
|
|
|
// return db;
|
|
|
|
|
//}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public struct PsoBuilderPointer
|
|
|
|
|
{
|
|
|
|
|
public int BlockID { get; set; } //1-based id
|
|
|
|
|
public int Offset { get; set; } //byte offset
|
|
|
|
|
public int Length { get; set; } //for temp use...
|
|
|
|
|
public uint Pointer
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
uint bidx = (((uint)BlockID) & 0xFFF);
|
|
|
|
|
uint offs = (((uint)Offset) & 0xFFFFF) << 12;
|
|
|
|
|
return bidx + offs;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
}
|