Jo Shields a575963da9 Imported Upstream version 3.6.0
Former-commit-id: da6be194a6b1221998fc28233f2503bd61dd9d14
2014-08-13 10:39:27 +01:00

435 lines
16 KiB
C#

//
// ImageWriter.cs
//
// Author:
// Jb Evain (jbevain@gmail.com)
//
// (C) 2005 Jb Evain
//
// Permission is hereby granted, free of charge, to any person obtaining
// a copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
//
namespace Mono.Cecil.Binary {
using System.IO;
using System.Text;
using Mono.Cecil.Metadata;
sealed class ImageWriter : BaseImageVisitor {
Image m_img;
AssemblyKind m_kind;
MetadataWriter m_mdWriter;
BinaryWriter m_binaryWriter;
Section m_textSect;
MemoryBinaryWriter m_textWriter;
Section m_relocSect;
MemoryBinaryWriter m_relocWriter;
Section m_rsrcSect;
MemoryBinaryWriter m_rsrcWriter;
public ImageWriter (MetadataWriter writer, AssemblyKind kind, BinaryWriter bw)
{
m_mdWriter= writer;
m_img = writer.GetMetadataRoot ().GetImage ();
m_kind = kind;
m_binaryWriter = bw;
m_textWriter = new MemoryBinaryWriter ();
m_textWriter.BaseStream.Position = 80;
m_relocWriter = new MemoryBinaryWriter ();
}
public Image GetImage ()
{
return m_img;
}
public MemoryBinaryWriter GetTextWriter ()
{
return m_textWriter;
}
public uint GetAligned (uint integer, uint alignWith)
{
return (integer + alignWith - 1) & ~(alignWith - 1);
}
public void Initialize ()
{
Image img = m_img;
ResourceWriter resWriter = null;
uint sectAlign = img.PEOptionalHeader.NTSpecificFields.SectionAlignment;
uint fileAlign = img.PEOptionalHeader.NTSpecificFields.FileAlignment;
m_textSect = img.TextSection;
foreach (Section s in img.Sections) {
if (s.Name == Section.Relocs)
m_relocSect = s;
else if (s.Name == Section.Resources) {
m_rsrcSect = s;
m_rsrcWriter = new MemoryBinaryWriter ();
resWriter = new ResourceWriter (img, m_rsrcSect, m_rsrcWriter);
resWriter.Write ();
}
}
// size computations, fields setting, etc.
uint nbSects = (uint) img.Sections.Count;
img.PEFileHeader.NumberOfSections = (ushort) nbSects;
// build the reloc section data
uint relocSize = 12;
m_relocWriter.Write ((uint) 0);
m_relocWriter.Write (relocSize);
m_relocWriter.Write ((ushort) 0);
m_relocWriter.Write ((ushort) 0);
m_textSect.VirtualSize = (uint) m_textWriter.BaseStream.Length;
m_relocSect.VirtualSize = (uint) m_relocWriter.BaseStream.Length;
if (m_rsrcSect != null)
m_rsrcSect.VirtualSize = (uint) m_rsrcWriter.BaseStream.Length;
// start counting before sections headers
// section start + section header sixe * number of sections
uint headersEnd = 0x178 + 0x28 * nbSects;
uint fileOffset = headersEnd;
uint sectOffset = sectAlign;
uint imageSize = 0;
foreach (Section sect in img.Sections) {
fileOffset = GetAligned (fileOffset, fileAlign);
sectOffset = GetAligned (sectOffset, sectAlign);
sect.PointerToRawData = new RVA (fileOffset);
sect.VirtualAddress = new RVA (sectOffset);
sect.SizeOfRawData = GetAligned (sect.VirtualSize, fileAlign);
fileOffset += sect.SizeOfRawData;
sectOffset += sect.SizeOfRawData;
imageSize += GetAligned (sect.SizeOfRawData, sectAlign);
}
if (m_textSect.VirtualAddress.Value != 0x2000)
throw new ImageFormatException ("Wrong RVA for .text section");
if (resWriter != null)
resWriter.Patch ();
img.PEOptionalHeader.StandardFields.CodeSize = GetAligned (
m_textSect.SizeOfRawData, fileAlign);
img.PEOptionalHeader.StandardFields.InitializedDataSize = m_textSect.SizeOfRawData;
if (m_rsrcSect != null)
img.PEOptionalHeader.StandardFields.InitializedDataSize += m_rsrcSect.SizeOfRawData;
img.PEOptionalHeader.StandardFields.BaseOfCode = m_textSect.VirtualAddress;
img.PEOptionalHeader.StandardFields.BaseOfData = m_relocSect.VirtualAddress;
imageSize += headersEnd;
img.PEOptionalHeader.NTSpecificFields.ImageSize = GetAligned (imageSize, sectAlign);
img.PEOptionalHeader.DataDirectories.BaseRelocationTable = new DataDirectory (
m_relocSect.VirtualAddress, m_relocSect.VirtualSize);
if (m_rsrcSect != null)
img.PEOptionalHeader.DataDirectories.ResourceTable = new DataDirectory (
m_rsrcSect.VirtualAddress, (uint) m_rsrcWriter.BaseStream.Length);
if (m_kind == AssemblyKind.Dll) {
img.PEFileHeader.Characteristics = ImageCharacteristics.CILOnlyDll;
img.HintNameTable.RuntimeMain = HintNameTable.RuntimeMainDll;
img.PEOptionalHeader.NTSpecificFields.DLLFlags = 0x400;
} else {
img.PEFileHeader.Characteristics = ImageCharacteristics.CILOnlyExe;
img.HintNameTable.RuntimeMain = HintNameTable.RuntimeMainExe;
}
switch (m_kind) {
case AssemblyKind.Dll :
case AssemblyKind.Console :
img.PEOptionalHeader.NTSpecificFields.SubSystem = SubSystem.WindowsCui;
break;
case AssemblyKind.Windows :
img.PEOptionalHeader.NTSpecificFields.SubSystem = SubSystem.WindowsGui;
break;
}
RVA importTable = new RVA (img.TextSection.VirtualAddress + m_mdWriter.ImportTablePosition);
img.PEOptionalHeader.DataDirectories.ImportTable = new DataDirectory (importTable, 0x57);
img.ImportTable.ImportLookupTable = new RVA ((uint) importTable + 0x28);
img.ImportLookupTable.HintNameRVA = img.ImportAddressTable.HintNameTableRVA =
new RVA ((uint) img.ImportTable.ImportLookupTable + 0x14);
img.ImportTable.Name = new RVA ((uint) img.ImportLookupTable.HintNameRVA + 0xe);
}
public override void VisitDOSHeader (DOSHeader header)
{
m_binaryWriter.Write (header.Start);
m_binaryWriter.Write (header.Lfanew);
m_binaryWriter.Write (header.End);
m_binaryWriter.Write ((ushort) 0x4550);
m_binaryWriter.Write ((ushort) 0);
}
public override void VisitPEFileHeader (PEFileHeader header)
{
m_binaryWriter.Write (header.Machine);
m_binaryWriter.Write (header.NumberOfSections);
m_binaryWriter.Write (header.TimeDateStamp);
m_binaryWriter.Write (header.PointerToSymbolTable);
m_binaryWriter.Write (header.NumberOfSymbols);
m_binaryWriter.Write (header.OptionalHeaderSize);
m_binaryWriter.Write ((ushort) header.Characteristics);
}
public override void VisitNTSpecificFieldsHeader (PEOptionalHeader.NTSpecificFieldsHeader header)
{
WriteIntOrLong (header.ImageBase);
m_binaryWriter.Write (header.SectionAlignment);
m_binaryWriter.Write (header.FileAlignment);
m_binaryWriter.Write (header.OSMajor);
m_binaryWriter.Write (header.OSMinor);
m_binaryWriter.Write (header.UserMajor);
m_binaryWriter.Write (header.UserMinor);
m_binaryWriter.Write (header.SubSysMajor);
m_binaryWriter.Write (header.SubSysMinor);
m_binaryWriter.Write (header.Reserved);
m_binaryWriter.Write (header.ImageSize);
m_binaryWriter.Write (header.HeaderSize);
m_binaryWriter.Write (header.FileChecksum);
m_binaryWriter.Write ((ushort) header.SubSystem);
m_binaryWriter.Write (header.DLLFlags);
WriteIntOrLong (header.StackReserveSize);
WriteIntOrLong (header.StackCommitSize);
WriteIntOrLong (header.HeapReserveSize);
WriteIntOrLong (header.HeapCommitSize);
m_binaryWriter.Write (header.LoaderFlags);
m_binaryWriter.Write (header.NumberOfDataDir);
}
public override void VisitStandardFieldsHeader (PEOptionalHeader.StandardFieldsHeader header)
{
m_binaryWriter.Write (header.Magic);
m_binaryWriter.Write (header.LMajor);
m_binaryWriter.Write (header.LMinor);
m_binaryWriter.Write (header.CodeSize);
m_binaryWriter.Write (header.InitializedDataSize);
m_binaryWriter.Write (header.UninitializedDataSize);
m_binaryWriter.Write (header.EntryPointRVA.Value);
m_binaryWriter.Write (header.BaseOfCode.Value);
if (!header.IsPE64)
m_binaryWriter.Write (header.BaseOfData.Value);
}
void WriteIntOrLong (ulong value)
{
if (m_img.PEOptionalHeader.StandardFields.IsPE64)
m_binaryWriter.Write (value);
else
m_binaryWriter.Write ((uint) value);
}
public override void VisitDataDirectoriesHeader (PEOptionalHeader.DataDirectoriesHeader header)
{
m_binaryWriter.Write (header.ExportTable.VirtualAddress);
m_binaryWriter.Write (header.ExportTable.Size);
m_binaryWriter.Write (header.ImportTable.VirtualAddress);
m_binaryWriter.Write (header.ImportTable.Size);
m_binaryWriter.Write (header.ResourceTable.VirtualAddress);
m_binaryWriter.Write (header.ResourceTable.Size);
m_binaryWriter.Write (header.ExceptionTable.VirtualAddress);
m_binaryWriter.Write (header.ExceptionTable.Size);
m_binaryWriter.Write (header.CertificateTable.VirtualAddress);
m_binaryWriter.Write (header.CertificateTable.Size);
m_binaryWriter.Write (header.BaseRelocationTable.VirtualAddress);
m_binaryWriter.Write (header.BaseRelocationTable.Size);
m_binaryWriter.Write (header.Debug.VirtualAddress);
m_binaryWriter.Write (header.Debug.Size);
m_binaryWriter.Write (header.Copyright.VirtualAddress);
m_binaryWriter.Write (header.Copyright.Size);
m_binaryWriter.Write (header.GlobalPtr.VirtualAddress);
m_binaryWriter.Write (header.GlobalPtr.Size);
m_binaryWriter.Write (header.TLSTable.VirtualAddress);
m_binaryWriter.Write (header.TLSTable.Size);
m_binaryWriter.Write (header.LoadConfigTable.VirtualAddress);
m_binaryWriter.Write (header.LoadConfigTable.Size);
m_binaryWriter.Write (header.BoundImport.VirtualAddress);
m_binaryWriter.Write (header.BoundImport.Size);
m_binaryWriter.Write (header.IAT.VirtualAddress);
m_binaryWriter.Write (header.IAT.Size);
m_binaryWriter.Write (header.DelayImportDescriptor.VirtualAddress);
m_binaryWriter.Write (header.DelayImportDescriptor.Size);
m_binaryWriter.Write (header.CLIHeader.VirtualAddress);
m_binaryWriter.Write (header.CLIHeader.Size);
m_binaryWriter.Write (header.Reserved.VirtualAddress);
m_binaryWriter.Write (header.Reserved.Size);
}
public override void VisitSection (Section sect)
{
m_binaryWriter.Write (Encoding.ASCII.GetBytes (sect.Name));
int more = 8 - sect.Name.Length;
for (int i = 0; i < more; i++)
m_binaryWriter.Write ((byte) 0);
m_binaryWriter.Write (sect.VirtualSize);
m_binaryWriter.Write (sect.VirtualAddress.Value);
m_binaryWriter.Write (sect.SizeOfRawData);
m_binaryWriter.Write (sect.PointerToRawData.Value);
m_binaryWriter.Write (sect.PointerToRelocations.Value);
m_binaryWriter.Write (sect.PointerToLineNumbers.Value);
m_binaryWriter.Write (sect.NumberOfRelocations);
m_binaryWriter.Write (sect.NumberOfLineNumbers);
m_binaryWriter.Write ((uint) sect.Characteristics);
}
public override void VisitImportAddressTable (ImportAddressTable iat)
{
m_textWriter.BaseStream.Position = 0;
m_textWriter.Write (iat.HintNameTableRVA.Value);
m_textWriter.Write (new byte [4]);
}
public override void VisitCLIHeader (CLIHeader header)
{
m_textWriter.Write (header.Cb);
if (m_mdWriter.TargetRuntime >= TargetRuntime.NET_2_0) {
m_textWriter.Write ((ushort) 2);
m_textWriter.Write ((ushort) 5);
} else {
m_textWriter.Write ((ushort) 2);
m_textWriter.Write ((ushort) 0);
}
m_textWriter.Write (header.Metadata.VirtualAddress);
m_textWriter.Write (header.Metadata.Size);
m_textWriter.Write ((uint) header.Flags);
m_textWriter.Write (header.EntryPointToken);
m_textWriter.Write (header.Resources.VirtualAddress);
m_textWriter.Write (header.Resources.Size);
m_textWriter.Write (header.StrongNameSignature.VirtualAddress);
m_textWriter.Write (header.StrongNameSignature.Size);
m_textWriter.Write (header.CodeManagerTable.VirtualAddress);
m_textWriter.Write (header.CodeManagerTable.Size);
m_textWriter.Write (header.VTableFixups.VirtualAddress);
m_textWriter.Write (header.VTableFixups.Size);
m_textWriter.Write (header.ExportAddressTableJumps.VirtualAddress);
m_textWriter.Write (header.ExportAddressTableJumps.Size);
m_textWriter.Write (header.ManagedNativeHeader.VirtualAddress);
m_textWriter.Write (header.ManagedNativeHeader.Size);
}
public override void VisitDebugHeader (DebugHeader header)
{
m_textWriter.BaseStream.Position = m_mdWriter.DebugHeaderPosition;
uint sizeUntilData = 0x1c;
header.AddressOfRawData = m_img.TextSection.VirtualAddress + m_mdWriter.DebugHeaderPosition + sizeUntilData;
header.PointerToRawData = 0x200 + m_mdWriter.DebugHeaderPosition + sizeUntilData;
header.SizeOfData = 0x18 + (uint) header.FileName.Length + 1;
m_textWriter.Write (header.Characteristics);
m_textWriter.Write (header.TimeDateStamp);
m_textWriter.Write (header.MajorVersion);
m_textWriter.Write (header.MinorVersion);
m_textWriter.Write ((uint) header.Type);
m_textWriter.Write (header.SizeOfData);
m_textWriter.Write (header.AddressOfRawData.Value);
m_textWriter.Write (header.PointerToRawData);
m_textWriter.Write (header.Magic);
m_textWriter.Write (header.Signature.ToByteArray ());
m_textWriter.Write (header.Age);
m_textWriter.Write (Encoding.ASCII.GetBytes (header.FileName));
m_textWriter.Write ((byte) 0);
}
public override void VisitImportTable (ImportTable it)
{
m_textWriter.BaseStream.Position = m_mdWriter.ImportTablePosition;
m_textWriter.Write (it.ImportLookupTable.Value);
m_textWriter.Write (it.DateTimeStamp);
m_textWriter.Write (it.ForwardChain);
m_textWriter.Write (it.Name.Value);
m_textWriter.Write (it.ImportAddressTable.Value);
m_textWriter.Write (new byte [20]);
}
public override void VisitImportLookupTable (ImportLookupTable ilt)
{
m_textWriter.Write (ilt.HintNameRVA.Value);
m_textWriter.Write (new byte [16]);
}
public override void VisitHintNameTable (HintNameTable hnt)
{
m_textWriter.Write (hnt.Hint);
m_textWriter.Write (Encoding.ASCII.GetBytes (hnt.RuntimeMain));
m_textWriter.Write ('\0');
m_textWriter.Write (Encoding.ASCII.GetBytes (hnt.RuntimeLibrary));
m_textWriter.Write ('\0');
m_textWriter.Write (new byte [4]);
// patch header with ep rva
RVA ep = m_img.TextSection.VirtualAddress +
(uint) m_textWriter.BaseStream.Position;
long pos = m_binaryWriter.BaseStream.Position;
m_binaryWriter.BaseStream.Position = 0xa8;
m_binaryWriter.Write (ep.Value);
m_binaryWriter.BaseStream.Position = pos;
// patch reloc Sect with ep
uint reloc = (ep.Value + 2) % 0x1000;
uint rva = (ep.Value + 2) - reloc;
m_relocWriter.BaseStream.Position = 0;
m_relocWriter.Write (rva);
m_relocWriter.BaseStream.Position = 8;
m_relocWriter.Write ((ushort) ((3 << 12) | reloc));
m_textWriter.Write (hnt.EntryPoint);
m_textWriter.Write (hnt.RVA);
}
public override void TerminateImage (Image img)
{
m_binaryWriter.BaseStream.Position = 0x200;
WriteSection (m_textSect, m_textWriter);
WriteSection (m_relocSect, m_relocWriter);
if (m_rsrcSect != null)
WriteSection (m_rsrcSect, m_rsrcWriter);
}
void WriteSection (Section sect, MemoryBinaryWriter sectWriter)
{
sectWriter.MemoryStream.WriteTo (m_binaryWriter.BaseStream);
m_binaryWriter.Write (new byte [
sect.SizeOfRawData - sectWriter.BaseStream.Length]);
}
}
}