2 * Copyright (c) 2002 Sergey Chaban <serge@wildwestsoftware.com>
\r
7 using System.Collections;
\r
8 using System.Runtime.InteropServices;
\r
10 using Mono.PEToolkit.Metadata;
\r
12 namespace Mono.PEToolkit {
\r
14 public class Image : IDisposable {
\r
16 internal DOSHeader dosHdr;
\r
17 internal COFFHeader coffHdr;
\r
18 internal PEHeader peHdr;
\r
20 internal CorHeader corHdr;
\r
22 internal Hashtable sections;
\r
23 // File position right after PEHeader (NT Optional Header).
\r
24 protected long sectionsPos;
\r
26 private MetaDataRoot mdRoot;
\r
28 private string name;
\r
30 internal BinaryReader reader;
\r
32 public Image(string name)
\r
40 dosHdr = new DOSHeader();
\r
41 coffHdr = new COFFHeader();
\r
42 peHdr = new PEHeader();
\r
43 corHdr = new CorHeader();
\r
45 sections = new Hashtable();
\r
55 public Hashtable Sections {
\r
63 lock (this) if (!open) {
\r
64 FileInfo pe = new FileInfo(name);
\r
66 throw new Exception("Invalid file path.");
\r
69 reader = new BinaryReader(new BufferedStream(pe.OpenRead()));
\r
70 if (!reader.BaseStream.CanSeek) {
\r
71 throw new Exception("Can't seek.");
\r
80 lock (this) if (open) {
\r
87 public void Dispose()
\r
95 return peHdr.IsCLIImage;
\r
99 public MetaDataRoot MetadataRoot {
\r
107 public void ReadHeaders()
\r
110 throw new Exception("You must open image before trying to read it.");
\r
113 dosHdr.Read(reader);
\r
114 reader.BaseStream.Position = dosHdr.Lfanew;
\r
115 ExeSignature peSig = (ExeSignature) reader.ReadUInt16();
\r
116 if (peSig != ExeSignature.NT) {
\r
117 throw new Exception ("Invalid image format: cannot find PE signature.");
\r
119 peSig = (ExeSignature) reader.ReadUInt16();
\r
120 if (peSig != ExeSignature.NT2) {
\r
121 throw new Exception ("Invalid image format: cannot find PE signature.");
\r
124 coffHdr.Read(reader);
\r
125 peHdr.Read(reader);
\r
127 sectionsPos = reader.BaseStream.Position;
\r
132 reader.BaseStream.Position = RVAToVA(peHdr.CLIHdrDir.virtAddr);
\r
133 corHdr.Read (reader);
\r
135 mdRoot = new MetaDataRoot(this);
\r
136 reader.BaseStream.Position = RVAToVA(corHdr.MetaData.virtAddr);
\r
137 mdRoot.Read(reader);
\r
142 public void WriteHeaders (BinaryWriter writer)
\r
144 dosHdr.Write (writer);
\r
145 writer.BaseStream.Position = dosHdr.Lfanew;
\r
146 writer.Write ((ushort)ExeSignature.NT);
\r
147 writer.Write ((ushort)ExeSignature.NT2);
\r
149 coffHdr.Write (writer);
\r
150 peHdr.Write (writer);
\r
152 WriteSections (writer);
\r
156 writer.BaseStream.Position = RVAToVA (peHdr.CLIHdrDir.virtAddr);
\r
157 corHdr.Write (writer);
\r
159 long pos = RVAToVA (corHdr.MetaData.virtAddr);
\r
160 writer.BaseStream.Position = pos;
\r
161 mdRoot.Write (writer);
\r
169 protected void ReadSections()
\r
171 if (sectionsPos < 0) {
\r
172 throw new Exception("Read headers first.");
\r
174 reader.BaseStream.Position = sectionsPos;
\r
176 int n = coffHdr.NumberOfSections;
\r
177 for (int i = n; --i >=0;) {
\r
178 Section sect = new Section();
\r
180 sections [sect.Name] = sect;
\r
184 protected void WriteSections (BinaryWriter writer)
\r
186 foreach (Section section in sections.Values) {
\r
187 section.Write (writer);
\r
194 /// <param name="writer"></param>
\r
195 public void Dump(TextWriter writer)
\r
198 "COFF Header:" + Environment.NewLine +
\r
199 coffHdr.ToString() + Environment.NewLine +
\r
200 "PE Header:" + Environment.NewLine +
\r
201 peHdr.ToString() + Environment.NewLine +
\r
202 "Core Header:" + Environment.NewLine +
\r
210 /// <returns></returns>
\r
211 public override string ToString()
\r
213 StringWriter sw = new StringWriter();
\r
215 return sw.ToString();
\r
220 /// Returns name of the section for the given RVA.
\r
222 /// <param name="rva"></param>
\r
223 /// <returns></returns>
\r
224 public string RVAToSectionName(RVA rva)
\r
227 foreach (Section s in Sections.Values) {
\r
228 RVA sva = s.VirtualAddress;
\r
229 if (rva >= sva && rva < sva + s.SizeOfRawData) {
\r
237 public long RVAToVA(RVA rva)
\r
239 string sectName = RVAToSectionName(rva);
\r
241 if (sectName != null) {
\r
242 Section s = (Section) Sections [sectName];
\r
243 res = rva + (s.PointerToRawData - s.VirtualAddress);
\r
248 public MetaDataRoot MetaDataRoot {
\r
254 public void DumpStreamHeader(TextWriter writer, string name)
\r
256 if (mdRoot == null || name == null || name == String.Empty || writer == null) return;
\r
257 writer.Write(name + " header: ");
\r
258 MDStream s = MetaDataRoot.Streams[name] as MDStream;
\r
260 writer.WriteLine();
\r
261 writer.WriteLine(s);
\r
263 writer.WriteLine("not present.");
\r
264 writer.WriteLine();
\r
268 public void DumpStreamHeader(string name)
\r
270 DumpStreamHeader(Console.Out, name);
\r