532 lines
14 KiB
C#
Raw Normal View History

//
// 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);
}
}
}