532 lines
14 KiB
C#
532 lines
14 KiB
C#
|
//
|
||
|
// MetadataWriter.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.Metadata {
|
||
|
|
||
|
using System;
|
||
|
using System.Collections;
|
||
|
using System.IO;
|
||
|
using System.Text;
|
||
|
|
||
|
using Mono.Cecil;
|
||
|
using Mono.Cecil.Binary;
|
||
|
|
||
|
internal sealed class MetadataWriter : BaseMetadataVisitor {
|
||
|
|
||
|
AssemblyDefinition m_assembly;
|
||
|
MetadataRoot m_root;
|
||
|
TargetRuntime m_runtime;
|
||
|
ImageWriter m_imgWriter;
|
||
|
MetadataTableWriter m_tableWriter;
|
||
|
MemoryBinaryWriter m_binaryWriter;
|
||
|
|
||
|
IDictionary m_stringCache;
|
||
|
MemoryBinaryWriter m_stringWriter;
|
||
|
|
||
|
IDictionary m_guidCache;
|
||
|
MemoryBinaryWriter m_guidWriter;
|
||
|
|
||
|
IDictionary m_usCache;
|
||
|
MemoryBinaryWriter m_usWriter;
|
||
|
|
||
|
IDictionary m_blobCache;
|
||
|
MemoryBinaryWriter m_blobWriter;
|
||
|
|
||
|
MemoryBinaryWriter m_tWriter;
|
||
|
|
||
|
MemoryBinaryWriter m_cilWriter;
|
||
|
|
||
|
MemoryBinaryWriter m_fieldDataWriter;
|
||
|
MemoryBinaryWriter m_resWriter;
|
||
|
|
||
|
uint m_mdStart, m_mdSize;
|
||
|
uint m_resStart, m_resSize;
|
||
|
uint m_snsStart, m_snsSize;
|
||
|
uint m_debugHeaderStart;
|
||
|
uint m_imporTableStart;
|
||
|
|
||
|
uint m_entryPointToken;
|
||
|
|
||
|
RVA m_cursor = new RVA (0x2050);
|
||
|
|
||
|
public MemoryBinaryWriter CilWriter {
|
||
|
get { return m_cilWriter; }
|
||
|
}
|
||
|
|
||
|
public MemoryBinaryWriter StringWriter {
|
||
|
get { return m_stringWriter; }
|
||
|
}
|
||
|
|
||
|
public MemoryBinaryWriter GuidWriter {
|
||
|
get { return m_guidWriter; }
|
||
|
}
|
||
|
|
||
|
public MemoryBinaryWriter UserStringWriter {
|
||
|
get { return m_usWriter; }
|
||
|
}
|
||
|
|
||
|
public MemoryBinaryWriter BlobWriter {
|
||
|
get { return m_blobWriter; }
|
||
|
}
|
||
|
|
||
|
public uint DebugHeaderPosition {
|
||
|
get { return m_debugHeaderStart; }
|
||
|
}
|
||
|
|
||
|
public uint ImportTablePosition {
|
||
|
get { return m_imporTableStart; }
|
||
|
}
|
||
|
|
||
|
public uint EntryPointToken {
|
||
|
get { return m_entryPointToken; }
|
||
|
set { m_entryPointToken = value; }
|
||
|
}
|
||
|
|
||
|
public TargetRuntime TargetRuntime {
|
||
|
get { return m_runtime; }
|
||
|
}
|
||
|
|
||
|
public MetadataWriter (AssemblyDefinition asm, MetadataRoot root,
|
||
|
AssemblyKind kind, TargetRuntime rt, BinaryWriter writer)
|
||
|
{
|
||
|
m_assembly = asm;
|
||
|
m_root = root;
|
||
|
m_runtime = rt;
|
||
|
m_imgWriter = new ImageWriter (this, kind, writer);
|
||
|
m_binaryWriter = m_imgWriter.GetTextWriter ();
|
||
|
|
||
|
m_stringCache = new Hashtable ();
|
||
|
m_stringWriter = new MemoryBinaryWriter (Encoding.UTF8);
|
||
|
m_stringWriter.Write ((byte) 0);
|
||
|
|
||
|
m_guidCache = new Hashtable ();
|
||
|
m_guidWriter = new MemoryBinaryWriter ();
|
||
|
|
||
|
m_usCache = new Hashtable ();
|
||
|
m_usWriter = new MemoryBinaryWriter (Encoding.Unicode);
|
||
|
m_usWriter.Write ((byte) 0);
|
||
|
|
||
|
m_blobCache = new Hashtable (ByteArrayEqualityComparer.Instance, ByteArrayEqualityComparer.Instance);
|
||
|
m_blobWriter = new MemoryBinaryWriter ();
|
||
|
m_blobWriter.Write ((byte) 0);
|
||
|
|
||
|
m_tWriter = new MemoryBinaryWriter ();
|
||
|
m_tableWriter = new MetadataTableWriter (this, m_tWriter);
|
||
|
|
||
|
m_cilWriter = new MemoryBinaryWriter ();
|
||
|
|
||
|
m_fieldDataWriter = new MemoryBinaryWriter ();
|
||
|
m_resWriter = new MemoryBinaryWriter ();
|
||
|
}
|
||
|
|
||
|
public MetadataRoot GetMetadataRoot ()
|
||
|
{
|
||
|
return m_root;
|
||
|
}
|
||
|
|
||
|
public ImageWriter GetImageWriter ()
|
||
|
{
|
||
|
return m_imgWriter;
|
||
|
}
|
||
|
|
||
|
public MemoryBinaryWriter GetWriter ()
|
||
|
{
|
||
|
return m_binaryWriter;
|
||
|
}
|
||
|
|
||
|
public MetadataTableWriter GetTableVisitor ()
|
||
|
{
|
||
|
return m_tableWriter;
|
||
|
}
|
||
|
|
||
|
public void AddData (int length)
|
||
|
{
|
||
|
m_cursor += new RVA ((uint) length);
|
||
|
}
|
||
|
|
||
|
public RVA GetDataCursor ()
|
||
|
{
|
||
|
return m_cursor;
|
||
|
}
|
||
|
|
||
|
public uint AddString (string str)
|
||
|
{
|
||
|
if (str == null || str.Length == 0)
|
||
|
return 0;
|
||
|
|
||
|
if (m_stringCache.Contains (str))
|
||
|
return (uint) m_stringCache [str];
|
||
|
|
||
|
uint pointer = (uint) m_stringWriter.BaseStream.Position;
|
||
|
m_stringCache [str] = pointer;
|
||
|
m_stringWriter.Write (Encoding.UTF8.GetBytes (str));
|
||
|
m_stringWriter.Write ('\0');
|
||
|
return pointer;
|
||
|
}
|
||
|
|
||
|
public uint AddBlob (byte [] data)
|
||
|
{
|
||
|
if (data == null || data.Length == 0)
|
||
|
return 0;
|
||
|
|
||
|
object cached = m_blobCache [data];
|
||
|
if (cached != null)
|
||
|
return (uint) cached;
|
||
|
|
||
|
uint pointer = (uint) m_blobWriter.BaseStream.Position;
|
||
|
m_blobCache [data] = pointer;
|
||
|
Utilities.WriteCompressedInteger (m_blobWriter, data.Length);
|
||
|
m_blobWriter.Write (data);
|
||
|
return pointer;
|
||
|
}
|
||
|
|
||
|
public uint AddGuid (Guid g)
|
||
|
{
|
||
|
if (m_guidCache.Contains (g))
|
||
|
return (uint) m_guidCache [g];
|
||
|
|
||
|
uint pointer = (uint) m_guidWriter.BaseStream.Position;
|
||
|
m_guidCache [g] = pointer;
|
||
|
m_guidWriter.Write (g.ToByteArray ());
|
||
|
return pointer + 1;
|
||
|
}
|
||
|
|
||
|
public uint AddUserString (string str)
|
||
|
{
|
||
|
if (str == null)
|
||
|
return 0;
|
||
|
|
||
|
if (m_usCache.Contains (str))
|
||
|
return (uint) m_usCache [str];
|
||
|
|
||
|
uint pointer = (uint) m_usWriter.BaseStream.Position;
|
||
|
m_usCache [str] = pointer;
|
||
|
byte [] us = Encoding.Unicode.GetBytes (str);
|
||
|
Utilities.WriteCompressedInteger (m_usWriter, us.Length + 1);
|
||
|
m_usWriter.Write (us);
|
||
|
m_usWriter.Write ((byte) (RequiresSpecialHandling (us) ? 1 : 0));
|
||
|
return pointer;
|
||
|
}
|
||
|
|
||
|
static bool RequiresSpecialHandling (byte [] chars)
|
||
|
{
|
||
|
for (int i = 0; i < chars.Length; i++) {
|
||
|
byte c = chars [i];
|
||
|
if ((i % 2) == 1)
|
||
|
if (c != 0)
|
||
|
return true;
|
||
|
|
||
|
if (InRange (0x01, 0x08, c) ||
|
||
|
InRange (0x0e, 0x1f, c) ||
|
||
|
c == 0x27 ||
|
||
|
c == 0x2d ||
|
||
|
c == 0x7f) {
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
static bool InRange (int left, int right, int value)
|
||
|
{
|
||
|
return left <= value && value <= right;
|
||
|
}
|
||
|
|
||
|
void CreateStream (string name)
|
||
|
{
|
||
|
MetadataStream stream = new MetadataStream ();
|
||
|
stream.Header.Name = name;
|
||
|
stream.Heap = MetadataHeap.HeapFactory (stream);
|
||
|
m_root.Streams.Add (stream);
|
||
|
}
|
||
|
|
||
|
void SetHeapSize (MetadataHeap heap, MemoryBinaryWriter data, byte flag)
|
||
|
{
|
||
|
if (data.BaseStream.Length > 65536) {
|
||
|
m_root.Streams.TablesHeap.HeapSizes |= flag;
|
||
|
heap.IndexSize = 4;
|
||
|
} else
|
||
|
heap.IndexSize = 2;
|
||
|
}
|
||
|
|
||
|
public uint AddResource (byte [] data)
|
||
|
{
|
||
|
uint offset = (uint) m_resWriter.BaseStream.Position;
|
||
|
m_resWriter.Write (data.Length);
|
||
|
m_resWriter.Write (data);
|
||
|
m_resWriter.QuadAlign ();
|
||
|
return offset;
|
||
|
}
|
||
|
|
||
|
public void AddFieldInitData (byte [] data)
|
||
|
{
|
||
|
m_fieldDataWriter.Write (data);
|
||
|
m_fieldDataWriter.QuadAlign ();
|
||
|
}
|
||
|
|
||
|
uint GetStrongNameSignatureSize ()
|
||
|
{
|
||
|
if (m_assembly.Name.PublicKey != null) {
|
||
|
// in fx 2.0 the key may be from 384 to 16384 bits
|
||
|
// so we must calculate the signature size based on
|
||
|
// the size of the public key (minus the 32 byte header)
|
||
|
int size = m_assembly.Name.PublicKey.Length;
|
||
|
if (size > 32)
|
||
|
return (uint) (size - 32);
|
||
|
// note: size == 16 for the ECMA "key" which is replaced
|
||
|
// by the runtime with a 1024 bits key (128 bytes)
|
||
|
}
|
||
|
return 128; // default strongname signature size
|
||
|
}
|
||
|
|
||
|
public override void VisitMetadataRoot (MetadataRoot root)
|
||
|
{
|
||
|
WriteMemStream (m_cilWriter);
|
||
|
WriteMemStream (m_fieldDataWriter);
|
||
|
m_resStart = (uint) m_binaryWriter.BaseStream.Position;
|
||
|
WriteMemStream (m_resWriter);
|
||
|
m_resSize = (uint) (m_binaryWriter.BaseStream.Position - m_resStart);
|
||
|
|
||
|
// for now, we only reserve the place for the strong name signature
|
||
|
if ((m_assembly.Name.Flags & AssemblyFlags.PublicKey) > 0) {
|
||
|
m_snsStart = (uint) m_binaryWriter.BaseStream.Position;
|
||
|
m_snsSize = GetStrongNameSignatureSize ();
|
||
|
m_binaryWriter.Write (new byte [m_snsSize]);
|
||
|
m_binaryWriter.QuadAlign ();
|
||
|
}
|
||
|
|
||
|
// save place for debug header
|
||
|
if (m_imgWriter.GetImage ().DebugHeader != null) {
|
||
|
m_debugHeaderStart = (uint) m_binaryWriter.BaseStream.Position;
|
||
|
m_binaryWriter.Write (new byte [m_imgWriter.GetImage ().DebugHeader.GetSize ()]);
|
||
|
m_binaryWriter.QuadAlign ();
|
||
|
}
|
||
|
|
||
|
m_mdStart = (uint) m_binaryWriter.BaseStream.Position;
|
||
|
|
||
|
if (m_stringWriter.BaseStream.Length > 1) {
|
||
|
CreateStream (MetadataStream.Strings);
|
||
|
SetHeapSize (root.Streams.StringsHeap, m_stringWriter, 0x01);
|
||
|
m_stringWriter.QuadAlign ();
|
||
|
}
|
||
|
|
||
|
if (m_guidWriter.BaseStream.Length > 0) {
|
||
|
CreateStream (MetadataStream.GUID);
|
||
|
SetHeapSize (root.Streams.GuidHeap, m_guidWriter, 0x02);
|
||
|
}
|
||
|
|
||
|
if (m_blobWriter.BaseStream.Length > 1) {
|
||
|
CreateStream (MetadataStream.Blob);
|
||
|
SetHeapSize (root.Streams.BlobHeap, m_blobWriter, 0x04);
|
||
|
m_blobWriter.QuadAlign ();
|
||
|
}
|
||
|
|
||
|
if (m_usWriter.BaseStream.Length > 2) {
|
||
|
CreateStream (MetadataStream.UserStrings);
|
||
|
m_usWriter.QuadAlign ();
|
||
|
}
|
||
|
|
||
|
m_root.Header.MajorVersion = 1;
|
||
|
m_root.Header.MinorVersion = 1;
|
||
|
|
||
|
switch (m_runtime) {
|
||
|
case TargetRuntime.NET_1_0 :
|
||
|
m_root.Header.Version = "v1.0.3705";
|
||
|
break;
|
||
|
case TargetRuntime.NET_1_1 :
|
||
|
m_root.Header.Version = "v1.1.4322";
|
||
|
break;
|
||
|
case TargetRuntime.NET_2_0 :
|
||
|
m_root.Header.Version = "v2.0.50727";
|
||
|
break;
|
||
|
case TargetRuntime.NET_4_0 :
|
||
|
m_root.Header.Version = "v4.0.30319";
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
m_root.Streams.TablesHeap.Tables.Accept (m_tableWriter);
|
||
|
|
||
|
if (m_tWriter.BaseStream.Length == 0)
|
||
|
m_root.Streams.Remove (m_root.Streams.TablesHeap.GetStream ());
|
||
|
}
|
||
|
|
||
|
public override void VisitMetadataRootHeader (MetadataRoot.MetadataRootHeader header)
|
||
|
{
|
||
|
m_binaryWriter.Write (header.Signature);
|
||
|
m_binaryWriter.Write (header.MajorVersion);
|
||
|
m_binaryWriter.Write (header.MinorVersion);
|
||
|
m_binaryWriter.Write (header.Reserved);
|
||
|
m_binaryWriter.Write (header.Version.Length + 3 & (~3));
|
||
|
m_binaryWriter.Write (Encoding.ASCII.GetBytes (header.Version));
|
||
|
m_binaryWriter.QuadAlign ();
|
||
|
m_binaryWriter.Write (header.Flags);
|
||
|
m_binaryWriter.Write ((ushort) m_root.Streams.Count);
|
||
|
}
|
||
|
|
||
|
public override void VisitMetadataStreamCollection (MetadataStreamCollection streams)
|
||
|
{
|
||
|
foreach (MetadataStream stream in streams) {
|
||
|
MetadataStream.MetadataStreamHeader header = stream.Header;
|
||
|
|
||
|
header.Offset = (uint) (m_binaryWriter.BaseStream.Position);
|
||
|
m_binaryWriter.Write (header.Offset);
|
||
|
MemoryBinaryWriter container;
|
||
|
string name = header.Name;
|
||
|
uint size = 0;
|
||
|
switch (header.Name) {
|
||
|
case MetadataStream.Tables :
|
||
|
container = m_tWriter;
|
||
|
size += 24; // header
|
||
|
break;
|
||
|
case MetadataStream.Strings :
|
||
|
name += "\0\0\0\0";
|
||
|
container = m_stringWriter;
|
||
|
break;
|
||
|
case MetadataStream.GUID :
|
||
|
container = m_guidWriter;
|
||
|
break;
|
||
|
case MetadataStream.Blob :
|
||
|
container = m_blobWriter;
|
||
|
break;
|
||
|
case MetadataStream.UserStrings :
|
||
|
container = m_usWriter;
|
||
|
break;
|
||
|
default :
|
||
|
throw new MetadataFormatException ("Unknown stream kind");
|
||
|
}
|
||
|
|
||
|
size += (uint) (container.BaseStream.Length + 3 & (~3));
|
||
|
m_binaryWriter.Write (size);
|
||
|
m_binaryWriter.Write (Encoding.ASCII.GetBytes (name));
|
||
|
m_binaryWriter.QuadAlign ();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void WriteMemStream (MemoryBinaryWriter writer)
|
||
|
{
|
||
|
m_binaryWriter.Write (writer);
|
||
|
m_binaryWriter.QuadAlign ();
|
||
|
}
|
||
|
|
||
|
void PatchStreamHeaderOffset (MetadataHeap heap)
|
||
|
{
|
||
|
long pos = m_binaryWriter.BaseStream.Position;
|
||
|
m_binaryWriter.BaseStream.Position = heap.GetStream ().Header.Offset;
|
||
|
m_binaryWriter.Write ((uint) (pos - m_mdStart));
|
||
|
m_binaryWriter.BaseStream.Position = pos;
|
||
|
}
|
||
|
|
||
|
public override void VisitGuidHeap (GuidHeap heap)
|
||
|
{
|
||
|
PatchStreamHeaderOffset (heap);
|
||
|
WriteMemStream (m_guidWriter);
|
||
|
}
|
||
|
|
||
|
public override void VisitStringsHeap (StringsHeap heap)
|
||
|
{
|
||
|
PatchStreamHeaderOffset (heap);
|
||
|
WriteMemStream (m_stringWriter);
|
||
|
}
|
||
|
|
||
|
public override void VisitTablesHeap (TablesHeap heap)
|
||
|
{
|
||
|
PatchStreamHeaderOffset (heap);
|
||
|
m_binaryWriter.Write (heap.Reserved);
|
||
|
switch (m_runtime) {
|
||
|
case TargetRuntime.NET_1_0 :
|
||
|
case TargetRuntime.NET_1_1 :
|
||
|
heap.MajorVersion = 1;
|
||
|
heap.MinorVersion = 0;
|
||
|
break;
|
||
|
case TargetRuntime.NET_2_0 :
|
||
|
case TargetRuntime.NET_4_0 :
|
||
|
heap.MajorVersion = 2;
|
||
|
heap.MinorVersion = 0;
|
||
|
break;
|
||
|
}
|
||
|
m_binaryWriter.Write (heap.MajorVersion);
|
||
|
m_binaryWriter.Write (heap.MinorVersion);
|
||
|
m_binaryWriter.Write (heap.HeapSizes);
|
||
|
m_binaryWriter.Write (heap.Reserved2);
|
||
|
m_binaryWriter.Write (heap.Valid);
|
||
|
m_binaryWriter.Write (heap.Sorted);
|
||
|
WriteMemStream (m_tWriter);
|
||
|
}
|
||
|
|
||
|
public override void VisitBlobHeap (BlobHeap heap)
|
||
|
{
|
||
|
PatchStreamHeaderOffset (heap);
|
||
|
WriteMemStream (m_blobWriter);
|
||
|
}
|
||
|
|
||
|
public override void VisitUserStringsHeap (UserStringsHeap heap)
|
||
|
{
|
||
|
PatchStreamHeaderOffset (heap);
|
||
|
WriteMemStream (m_usWriter);
|
||
|
}
|
||
|
|
||
|
void PatchHeader ()
|
||
|
{
|
||
|
Image img = m_imgWriter.GetImage ();
|
||
|
|
||
|
img.CLIHeader.EntryPointToken = m_entryPointToken;
|
||
|
|
||
|
if ((m_assembly.Name.Flags & AssemblyFlags.PublicKey) == 0)
|
||
|
img.CLIHeader.Flags &= ~RuntimeImage.StrongNameSigned;
|
||
|
|
||
|
if (m_mdSize > 0)
|
||
|
img.CLIHeader.Metadata = new DataDirectory (
|
||
|
img.TextSection.VirtualAddress + m_mdStart, m_imporTableStart - m_mdStart);
|
||
|
|
||
|
if (m_resSize > 0)
|
||
|
img.CLIHeader.Resources = new DataDirectory (
|
||
|
img.TextSection.VirtualAddress + m_resStart, m_resSize);
|
||
|
|
||
|
if (m_snsStart > 0)
|
||
|
img.CLIHeader.StrongNameSignature = new DataDirectory (
|
||
|
img.TextSection.VirtualAddress + m_snsStart, m_snsSize);
|
||
|
|
||
|
if (m_debugHeaderStart > 0)
|
||
|
img.PEOptionalHeader.DataDirectories.Debug = new DataDirectory (
|
||
|
img.TextSection.VirtualAddress + m_debugHeaderStart, 0x1c);
|
||
|
}
|
||
|
|
||
|
public override void TerminateMetadataRoot (MetadataRoot root)
|
||
|
{
|
||
|
m_mdSize = (uint) (m_binaryWriter.BaseStream.Position - m_mdStart);
|
||
|
m_imporTableStart = (uint) m_binaryWriter.BaseStream.Position;
|
||
|
m_binaryWriter.Write (new byte [0x60]); // imports
|
||
|
m_imgWriter.Initialize ();
|
||
|
PatchHeader ();
|
||
|
root.GetImage ().Accept (m_imgWriter);
|
||
|
}
|
||
|
}
|
||
|
}
|