169 lines
4.2 KiB
C#
169 lines
4.2 KiB
C#
|
// ****************************************************************
|
||
|
// Copyright 2007, Charlie Poole
|
||
|
// This is free software licensed under the NUnit license. You may
|
||
|
// obtain a copy of the license at http://nunit.org/?p=license&r=2.4
|
||
|
// ****************************************************************
|
||
|
using System;
|
||
|
using System.Reflection;
|
||
|
using System.Text;
|
||
|
using System.IO;
|
||
|
|
||
|
namespace NUnit.Core
|
||
|
{
|
||
|
/// <summary>
|
||
|
/// AssemblyReader knows how to find various things in an assembly header
|
||
|
/// </summary>
|
||
|
public class AssemblyReader : IDisposable
|
||
|
{
|
||
|
private string assemblyPath;
|
||
|
private BinaryReader rdr;
|
||
|
private FileStream fs;
|
||
|
|
||
|
UInt16 dos_magic = 0xffff;
|
||
|
uint pe_signature = 0xffffffff;
|
||
|
UInt16 numDataSections;
|
||
|
UInt16 optionalHeaderSize;
|
||
|
|
||
|
private uint peHeader = 0;
|
||
|
private uint fileHeader = 0;
|
||
|
private uint optionalHeader = 0;
|
||
|
private uint dataDirectory = 0;
|
||
|
private uint dataSections = 0;
|
||
|
|
||
|
private struct Section
|
||
|
{
|
||
|
public uint virtualAddress;
|
||
|
public uint virtualSize;
|
||
|
public uint fileOffset;
|
||
|
};
|
||
|
|
||
|
private Section[] sections;
|
||
|
|
||
|
public AssemblyReader( string assemblyPath )
|
||
|
{
|
||
|
this.assemblyPath = assemblyPath;
|
||
|
CalcHeaderOffsets();
|
||
|
}
|
||
|
|
||
|
public AssemblyReader( Assembly assembly )
|
||
|
{
|
||
|
this.assemblyPath = TestFixtureBuilder.GetAssemblyPath( assembly );
|
||
|
CalcHeaderOffsets();
|
||
|
}
|
||
|
|
||
|
private void CalcHeaderOffsets()
|
||
|
{
|
||
|
this.fs = new FileStream( assemblyPath, FileMode.Open, FileAccess.Read );
|
||
|
this.rdr = new BinaryReader( fs );
|
||
|
dos_magic = rdr.ReadUInt16();
|
||
|
if ( dos_magic == 0x5a4d )
|
||
|
{
|
||
|
fs.Position = 0x3c;
|
||
|
peHeader = rdr.ReadUInt32();
|
||
|
fileHeader = peHeader + 4;
|
||
|
optionalHeader = fileHeader + 20;
|
||
|
dataDirectory = optionalHeader + 96;
|
||
|
// dotNetDirectoryEntry = dataDirectory + 14 * 8;
|
||
|
|
||
|
fs.Position = peHeader;
|
||
|
pe_signature = rdr.ReadUInt32();
|
||
|
rdr.ReadUInt16(); // machine
|
||
|
numDataSections = rdr.ReadUInt16();
|
||
|
fs.Position += 12;
|
||
|
optionalHeaderSize = rdr.ReadUInt16();
|
||
|
dataSections = optionalHeader + optionalHeaderSize;
|
||
|
|
||
|
sections = new Section[numDataSections];
|
||
|
fs.Position = dataSections;
|
||
|
for( int i = 0; i < numDataSections; i++ )
|
||
|
{
|
||
|
fs.Position += 8;
|
||
|
sections[i].virtualSize = rdr.ReadUInt32();
|
||
|
sections[i].virtualAddress = rdr.ReadUInt32();
|
||
|
uint rawDataSize = rdr.ReadUInt32();
|
||
|
sections[i].fileOffset = rdr.ReadUInt32();
|
||
|
if ( sections[i].virtualSize == 0 )
|
||
|
sections[i].virtualSize = rawDataSize;
|
||
|
|
||
|
fs.Position += 16;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private uint DataDirectoryRva( int n )
|
||
|
{
|
||
|
fs.Position = dataDirectory + n * 8;
|
||
|
return rdr.ReadUInt32();
|
||
|
}
|
||
|
|
||
|
private uint RvaToLfa( uint rva )
|
||
|
{
|
||
|
for( int i = 0; i < numDataSections; i++ )
|
||
|
if ( rva >= sections[i].virtualAddress && rva < sections[i].virtualAddress + sections[i].virtualSize )
|
||
|
return rva - sections[i].virtualAddress + sections[i].fileOffset;
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
public string AssemblyPath
|
||
|
{
|
||
|
get { return assemblyPath; }
|
||
|
}
|
||
|
|
||
|
public bool IsValidPeFile
|
||
|
{
|
||
|
get { return dos_magic == 0x5a4d && pe_signature == 0x00004550; }
|
||
|
}
|
||
|
|
||
|
public bool IsDotNetFile
|
||
|
{
|
||
|
get { return IsValidPeFile && DataDirectoryRva(14) != 0; }
|
||
|
}
|
||
|
|
||
|
public string ImageRuntimeVersion
|
||
|
{
|
||
|
get
|
||
|
{
|
||
|
string runtimeVersion = string.Empty;
|
||
|
|
||
|
uint rva = DataDirectoryRva(14);
|
||
|
if ( rva != 0 )
|
||
|
{
|
||
|
fs.Position = RvaToLfa( rva ) + 8;
|
||
|
uint metadata = rdr.ReadUInt32();
|
||
|
fs.Position = RvaToLfa( metadata );
|
||
|
if ( rdr.ReadUInt32() == 0x424a5342 )
|
||
|
{
|
||
|
// Copy string representing runtime version
|
||
|
fs.Position += 12;
|
||
|
StringBuilder sb = new StringBuilder();
|
||
|
char c;
|
||
|
while ((c = rdr.ReadChar())!= '\0')
|
||
|
sb.Append(c);
|
||
|
|
||
|
if (sb[0] == 'v') // Last sanity check
|
||
|
runtimeVersion = sb.ToString();
|
||
|
|
||
|
// Could do fixups here for bad values in older files
|
||
|
// like 1.x86, 1.build, etc. But we are only using
|
||
|
// the major version anyway
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return runtimeVersion;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public void Dispose()
|
||
|
{
|
||
|
if ( fs != null )
|
||
|
fs.Close();
|
||
|
if ( rdr != null )
|
||
|
rdr.Close();
|
||
|
|
||
|
fs = null;
|
||
|
rdr = null;
|
||
|
}
|
||
|
}
|
||
|
}
|