de4dot-cex/de4dot.code/resources/ResourceReader.cs

181 lines
6.9 KiB
C#
Raw Normal View History

2012-05-03 20:34:58 +08:00
/*
2014-03-12 05:15:43 +08:00
Copyright (C) 2011-2014 de4dot@gmail.com
2012-05-03 20:34:58 +08:00
This file is part of de4dot.
de4dot is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
de4dot is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with de4dot. If not, see <http://www.gnu.org/licenses/>.
*/
using System;
using System.Collections.Generic;
using System.IO;
using System.Text;
using System.Text.RegularExpressions;
using dnlib.DotNet;
using dnlib.IO;
2012-05-03 20:34:58 +08:00
namespace de4dot.code.resources {
[Serializable]
public class ResourceReaderException : Exception {
2012-05-03 20:34:58 +08:00
public ResourceReaderException(string msg)
: base(msg) {
}
}
public struct ResourceReader {
2012-11-04 07:50:24 +08:00
IBinaryReader reader;
2012-05-03 20:34:58 +08:00
ResourceDataCreator resourceDataCreator;
ResourceReader(ModuleDef module, IBinaryReader reader) {
2012-11-04 07:50:24 +08:00
this.reader = reader;
2012-05-03 20:34:58 +08:00
this.resourceDataCreator = new ResourceDataCreator(module);
}
public static ResourceElementSet Read(ModuleDef module, IBinaryReader reader) {
2013-01-19 20:03:57 +08:00
return new ResourceReader(module, reader).Read();
2012-05-03 20:34:58 +08:00
}
2013-01-19 20:03:57 +08:00
ResourceElementSet Read() {
2012-05-03 20:34:58 +08:00
ResourceElementSet resources = new ResourceElementSet();
uint sig = reader.ReadUInt32();
if (sig != 0xBEEFCACE)
throw new ResourceReaderException(string.Format("Invalid resource sig: {0:X8}", sig));
2013-01-19 20:03:57 +08:00
if (!CheckReaders())
2012-05-03 20:34:58 +08:00
throw new ResourceReaderException("Invalid resource reader");
int version = reader.ReadInt32();
if (version != 2)
throw new ResourceReaderException(string.Format("Invalid resource version: {0}", version));
int numResources = reader.ReadInt32();
if (numResources < 0)
throw new ResourceReaderException(string.Format("Invalid number of resources: {0}", numResources));
int numUserTypes = reader.ReadInt32();
if (numUserTypes < 0)
throw new ResourceReaderException(string.Format("Invalid number of user types: {0}", numUserTypes));
var userTypes = new List<UserResourceType>();
for (int i = 0; i < numUserTypes; i++)
userTypes.Add(new UserResourceType(reader.ReadString(), ResourceTypeCode.UserTypes + i));
2012-11-04 07:50:24 +08:00
reader.Position = (reader.Position + 7) & ~7;
2012-05-03 20:34:58 +08:00
var hashes = new int[numResources];
for (int i = 0; i < numResources; i++)
hashes[i] = reader.ReadInt32();
var offsets = new int[numResources];
for (int i = 0; i < numResources; i++)
offsets[i] = reader.ReadInt32();
2012-11-04 07:50:24 +08:00
long baseOffset = reader.Position;
2012-05-03 20:34:58 +08:00
long dataBaseOffset = reader.ReadInt32();
2012-11-04 07:50:24 +08:00
long nameBaseOffset = reader.Position;
long end = reader.Length;
2012-05-03 20:34:58 +08:00
var infos = new List<ResourceInfo>(numResources);
for (int i = 0; i < numResources; i++) {
2012-11-04 07:50:24 +08:00
reader.Position = nameBaseOffset + offsets[i];
var name = reader.ReadString(Encoding.Unicode);
long offset = dataBaseOffset + reader.ReadInt32();
2012-05-03 20:34:58 +08:00
infos.Add(new ResourceInfo(name, offset));
}
2012-11-04 07:50:24 +08:00
infos.Sort((a, b) => a.offset.CompareTo(b.offset));
2012-05-03 20:34:58 +08:00
for (int i = 0; i < infos.Count; i++) {
var info = infos[i];
var element = new ResourceElement();
element.Name = info.name;
2012-11-04 07:50:24 +08:00
reader.Position = info.offset;
2012-05-03 20:34:58 +08:00
long nextDataOffset = i == infos.Count - 1 ? end : infos[i + 1].offset;
int size = (int)(nextDataOffset - info.offset);
2013-01-19 20:03:57 +08:00
element.ResourceData = ReadResourceData(userTypes, size);
2012-05-03 20:34:58 +08:00
2013-01-19 20:03:57 +08:00
resources.Add(element);
2012-05-03 20:34:58 +08:00
}
return resources;
}
class ResourceInfo {
public string name;
public long offset;
public ResourceInfo(string name, long offset) {
this.name = name;
this.offset = offset;
}
public override string ToString() {
return string.Format("{0:X8} - {1}", offset, name);
}
}
2013-01-19 20:03:57 +08:00
IResourceData ReadResourceData(List<UserResourceType> userTypes, int size) {
uint code = ReadUInt32(reader);
2012-05-03 20:34:58 +08:00
switch ((ResourceTypeCode)code) {
2013-01-19 20:03:57 +08:00
case ResourceTypeCode.Null: return resourceDataCreator.CreateNull();
case ResourceTypeCode.String: return resourceDataCreator.Create(reader.ReadString());
case ResourceTypeCode.Boolean: return resourceDataCreator.Create(reader.ReadBoolean());
case ResourceTypeCode.Char: return resourceDataCreator.Create((char)reader.ReadUInt16());
case ResourceTypeCode.Byte: return resourceDataCreator.Create(reader.ReadByte());
case ResourceTypeCode.SByte: return resourceDataCreator.Create(reader.ReadSByte());
case ResourceTypeCode.Int16: return resourceDataCreator.Create(reader.ReadInt16());
case ResourceTypeCode.UInt16: return resourceDataCreator.Create(reader.ReadUInt16());
case ResourceTypeCode.Int32: return resourceDataCreator.Create(reader.ReadInt32());
case ResourceTypeCode.UInt32: return resourceDataCreator.Create(reader.ReadUInt32());
case ResourceTypeCode.Int64: return resourceDataCreator.Create(reader.ReadInt64());
case ResourceTypeCode.UInt64: return resourceDataCreator.Create(reader.ReadUInt64());
case ResourceTypeCode.Single: return resourceDataCreator.Create(reader.ReadSingle());
case ResourceTypeCode.Double: return resourceDataCreator.Create(reader.ReadDouble());
case ResourceTypeCode.Decimal: return resourceDataCreator.Create(reader.ReadDecimal());
case ResourceTypeCode.DateTime: return resourceDataCreator.Create(new DateTime(reader.ReadInt64()));
case ResourceTypeCode.TimeSpan: return resourceDataCreator.Create(new TimeSpan(reader.ReadInt64()));
case ResourceTypeCode.ByteArray: return resourceDataCreator.Create(reader.ReadBytes(reader.ReadInt32()));
2013-11-18 03:24:55 +08:00
case ResourceTypeCode.Stream: return resourceDataCreator.CreateStream(reader.ReadBytes(reader.ReadInt32()));
2012-05-03 20:34:58 +08:00
default:
int userTypeIndex = (int)(code - (uint)ResourceTypeCode.UserTypes);
if (userTypeIndex < 0 || userTypeIndex >= userTypes.Count)
throw new ResourceReaderException(string.Format("Invalid resource data code: {0}", code));
2013-01-19 20:03:57 +08:00
return resourceDataCreator.CreateSerialized(reader.ReadBytes(size));
2012-05-03 20:34:58 +08:00
}
}
2013-01-19 20:03:57 +08:00
static uint ReadUInt32(IBinaryReader reader) {
2012-06-03 22:07:05 +08:00
try {
2012-11-04 07:50:24 +08:00
return reader.Read7BitEncodedUInt32();
2012-06-03 22:07:05 +08:00
}
catch {
throw new ResourceReaderException("Invalid encoded int32");
2012-05-03 20:34:58 +08:00
}
}
2013-01-19 20:03:57 +08:00
bool CheckReaders() {
2012-05-03 20:34:58 +08:00
bool validReader = false;
int numReaders = reader.ReadInt32();
if (numReaders < 0)
throw new ResourceReaderException(string.Format("Invalid number of readers: {0}", numReaders));
int readersSize = reader.ReadInt32();
if (readersSize < 0)
throw new ResourceReaderException(string.Format("Invalid readers size: {0:X8}", readersSize));
for (int i = 0; i < numReaders; i++) {
var resourceReaderFullName = reader.ReadString();
var resourceSetFullName = reader.ReadString();
if (Regex.IsMatch(resourceReaderFullName, @"^System\.Resources\.ResourceReader,\s*mscorlib,"))
validReader = true;
}
return validReader;
}
}
}