ResourceDataReader pool improvement

This commit is contained in:
dexy 2020-03-10 13:46:18 +11:00
parent e608912977
commit 9d41d329c4
3 changed files with 37 additions and 36 deletions

View File

@ -454,7 +454,7 @@ namespace CodeWalker.GameFiles
} }
} }
[TypeConverter(typeof(ExpandableObjectConverter))] public class ShaderParametersBlock : ResourceSystemBlock [TypeConverter(typeof(ExpandableObjectConverter))] public class ShaderParametersBlock : ResourceSystemBlock, IResourceNoCacheBlock
{ {
public override long BlockLength public override long BlockLength

View File

@ -636,7 +636,7 @@ namespace CodeWalker.GameFiles
[TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleArray<T> : ListBase<T> where T : IResourceSystemBlock, new() [TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleArray<T> : ListBase<T>, IResourceNoCacheBlock where T : IResourceSystemBlock, new()
{ {
/// <summary> /// <summary>
/// Gets the length of the data block. /// Gets the length of the data block.
@ -722,7 +722,7 @@ namespace CodeWalker.GameFiles
} }
} }
[TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64<T> : ResourceSystemBlock where T : IResourceSystemBlock, new() [TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64<T> : ResourceSystemBlock, IResourceNoCacheBlock where T : IResourceSystemBlock, new()
{ {
public override long BlockLength public override long BlockLength
{ {
@ -814,7 +814,7 @@ namespace CodeWalker.GameFiles
return "(Count: " + EntriesCount.ToString() + ")"; return "(Count: " + EntriesCount.ToString() + ")";
} }
} }
[TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64_s<T> : ResourceSystemBlock where T : struct [TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64_s<T> : ResourceSystemBlock, IResourceNoCacheBlock where T : struct
{ {
public override long BlockLength public override long BlockLength
{ {
@ -892,7 +892,7 @@ namespace CodeWalker.GameFiles
return "(Count: " + EntriesCount.ToString() + ")"; return "(Count: " + EntriesCount.ToString() + ")";
} }
} }
[TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64b_s<T> : ResourceSystemBlock where T : struct [TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64b_s<T> : ResourceSystemBlock, IResourceNoCacheBlock where T : struct
{ {
//this version uses uints for the count/cap! //this version uses uints for the count/cap!
@ -972,7 +972,7 @@ namespace CodeWalker.GameFiles
return "(Count: " + EntriesCount.ToString() + ")"; return "(Count: " + EntriesCount.ToString() + ")";
} }
} }
[TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64_byte : ResourceSystemBlock [TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64_byte : ResourceSystemBlock, IResourceNoCacheBlock
{ {
public override long BlockLength public override long BlockLength
{ {
@ -1050,7 +1050,7 @@ namespace CodeWalker.GameFiles
return "(Count: " + EntriesCount.ToString() + ")"; return "(Count: " + EntriesCount.ToString() + ")";
} }
} }
[TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64_ushort : ResourceSystemBlock [TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64_ushort : ResourceSystemBlock, IResourceNoCacheBlock
{ {
public override long BlockLength public override long BlockLength
{ {
@ -1128,7 +1128,7 @@ namespace CodeWalker.GameFiles
return "(Count: " + EntriesCount.ToString() + ")"; return "(Count: " + EntriesCount.ToString() + ")";
} }
} }
[TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64_uint : ResourceSystemBlock [TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64_uint : ResourceSystemBlock, IResourceNoCacheBlock
{ {
public override long BlockLength public override long BlockLength
{ {
@ -1206,7 +1206,7 @@ namespace CodeWalker.GameFiles
return "(Count: " + EntriesCount.ToString() + ")"; return "(Count: " + EntriesCount.ToString() + ")";
} }
} }
[TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64_ulong : ResourceSystemBlock [TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64_ulong : ResourceSystemBlock, IResourceNoCacheBlock
{ {
public override long BlockLength public override long BlockLength
{ {
@ -1284,7 +1284,7 @@ namespace CodeWalker.GameFiles
return "(Count: " + EntriesCount.ToString() + ")"; return "(Count: " + EntriesCount.ToString() + ")";
} }
} }
[TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64_float : ResourceSystemBlock [TypeConverter(typeof(ExpandableObjectConverter))] public class ResourceSimpleList64_float : ResourceSystemBlock, IResourceNoCacheBlock
{ {
public override long BlockLength public override long BlockLength
{ {

View File

@ -50,7 +50,7 @@ namespace CodeWalker.GameFiles
// this is a dictionary that contains all the resource blocks // this is a dictionary that contains all the resource blocks
// which were read from this resource reader // which were read from this resource reader
private Dictionary<long, List<IResourceBlock>> blockPool; public Dictionary<long, IResourceBlock> blockPool = new Dictionary<long, IResourceBlock>();
/// <summary> /// <summary>
/// Gets the length of the underlying stream. /// Gets the length of the underlying stream.
@ -80,7 +80,6 @@ namespace CodeWalker.GameFiles
{ {
this.systemStream = systemStream; this.systemStream = systemStream;
this.graphicsStream = graphicsStream; this.graphicsStream = graphicsStream;
this.blockPool = new Dictionary<long, List<IResourceBlock>>();
} }
public ResourceDataReader(RpfResourceFileEntry resentry, byte[] data, Endianess endianess = Endianess.LittleEndian) public ResourceDataReader(RpfResourceFileEntry resentry, byte[] data, Endianess endianess = Endianess.LittleEndian)
@ -105,16 +104,14 @@ namespace CodeWalker.GameFiles
this.systemStream = new MemoryStream(data, 0, systemSize); this.systemStream = new MemoryStream(data, 0, systemSize);
this.graphicsStream = new MemoryStream(data, systemSize, graphicsSize); this.graphicsStream = new MemoryStream(data, systemSize, graphicsSize);
this.blockPool = new Dictionary<long, List<IResourceBlock>>();
Position = 0x50000000; Position = 0x50000000;
} }
public ResourceDataReader(int systemSize, int graphicsSize, byte[] data, Endianess endianess = Endianess.LittleEndian) public ResourceDataReader(int systemSize, int graphicsSize, byte[] data, Endianess endianess = Endianess.LittleEndian)
: base((Stream)null, endianess) : base((Stream)null, endianess)
{ {
this.systemStream = new MemoryStream(data, 0, systemSize); this.systemStream = new MemoryStream(data, 0, systemSize);
this.graphicsStream = new MemoryStream(data, systemSize, graphicsSize); this.graphicsStream = new MemoryStream(data, systemSize, graphicsSize);
this.blockPool = new Dictionary<long, List<IResourceBlock>>();
Position = 0x50000000; Position = 0x50000000;
} }
@ -171,20 +168,24 @@ namespace CodeWalker.GameFiles
/// </summary> /// </summary>
public T ReadBlock<T>(params object[] parameters) where T : IResourceBlock, new() public T ReadBlock<T>(params object[] parameters) where T : IResourceBlock, new()
{ {
// make sure to return the same object if the same var usepool = !typeof(IResourceNoCacheBlock).IsAssignableFrom(typeof(T));
// block is read again... if (usepool)
if (blockPool.ContainsKey(Position))
{ {
var blocks = blockPool[Position]; // make sure to return the same object if the same
foreach (var block in blocks) // block is read again...
if (block is T) if (blockPool.ContainsKey(Position))
{
var block = blockPool[Position];
if (block is T tblk)
{ {
Position += block.BlockLength; Position += block.BlockLength;
return tblk;
// since a resource block of the same type
// has been found at the same address, return it
return (T)block;
} }
else
{
usepool = false;
}
}
} }
var result = new T(); var result = new T();
@ -192,30 +193,25 @@ namespace CodeWalker.GameFiles
// replace with correct type... // replace with correct type...
if (result is IResourceXXSystemBlock) if (result is IResourceXXSystemBlock)
{
result = (T)((IResourceXXSystemBlock)result).GetType(this, parameters); result = (T)((IResourceXXSystemBlock)result).GetType(this, parameters);
}
if (result == null) if (result == null)
{ {
return default(T); return default(T);
} }
if (usepool)
// add block to the block pool...
if (blockPool.ContainsKey(Position))
{ {
blockPool[Position].Add(result); blockPool[Position] = result;
}
else
{
var blocks = new List<IResourceBlock>();
blocks.Add(result);
blockPool.Add(Position, blocks);
} }
var classPosition = Position; var classPosition = Position;
result.Read(this, parameters); result.Read(this, parameters);
//result.Position = classPosition; //TODO: need this if writing stuff! //result.Position = classPosition; //TODO: need this if writing stuff!
return (T)result;
return result;
} }
/// <summary> /// <summary>
@ -595,6 +591,11 @@ namespace CodeWalker.GameFiles
{ } { }
/// <summary>
/// Represents a data block that won't get cached while loading. Used for 0-length object parts
/// </summary>
public interface IResourceNoCacheBlock : IResourceBlock
{ }