| 1 | // Licensed to the .NET Foundation under one or more agreements. |
| 2 | // The .NET Foundation licenses this file to you under the MIT license. |
| 3 | // See the LICENSE file in the project root for more information. |
| 4 | |
| 5 | // |
| 6 | |
| 7 | /*============================================================ |
| 8 | ** |
| 9 | ** CorImage.cpp |
| 10 | ** |
| 11 | ** IMAGEHLP routines so we can avoid early binding to that DLL. |
| 12 | ** |
| 13 | ===========================================================*/ |
| 14 | |
| 15 | #include "stdafx.h" |
| 16 | #include "contract.h" |
| 17 | #include <daccess.h> |
| 18 | #include "corimage.h" |
| 19 | #include "safemath.h" |
| 20 | |
| 21 | #define RTL_MEG (1024UL * 1024UL) |
| 22 | #define ( 256UL * RTL_MEG) |
| 23 | |
| 24 | // IMAGE_FIRST_SECTION doesn't need 32/64 versions since the file header is |
| 25 | // the same either way. |
| 26 | |
| 27 | #define PTR_IMAGE_FIRST_SECTION( ntheader ) \ |
| 28 | PTR_IMAGE_SECTION_HEADER \ |
| 29 | (dac_cast<TADDR>(ntheader) + \ |
| 30 | FIELD_OFFSET( IMAGE_NT_HEADERS, OptionalHeader ) + \ |
| 31 | VAL16((ntheader)->FileHeader.SizeOfOptionalHeader) \ |
| 32 | ) |
| 33 | |
| 34 | #ifndef DACCESS_COMPILE |
| 35 | |
| 36 | IMAGE_NT_HEADERS *Cor_RtlImageNtHeader(VOID *pvBase, ULONG FileLength) |
| 37 | { |
| 38 | LIMITED_METHOD_CONTRACT; |
| 39 | IMAGE_NT_HEADERS *pNtHeaders = NULL; |
| 40 | if (pvBase && (pvBase != (VOID*)-1)) { |
| 41 | struct Param |
| 42 | { |
| 43 | IMAGE_DOS_HEADER *pDos; |
| 44 | ULONG FileLength; |
| 45 | IMAGE_NT_HEADERS *pNtHeaders; |
| 46 | } param; |
| 47 | param.pDos = (IMAGE_DOS_HEADER*)pvBase; |
| 48 | param.FileLength = FileLength; |
| 49 | param.pNtHeaders = pNtHeaders; |
| 50 | |
| 51 | PAL_TRY(Param *, pParam, ¶m) { |
| 52 | if ( (pParam->pDos->e_magic == VAL16(IMAGE_DOS_SIGNATURE)) |
| 53 | && ((DWORD)VAL32(pParam->pDos->e_lfanew) < RTLP_IMAGE_MAX_DOS_HEADER) |
| 54 | && ovadd_lt((DWORD)VAL32(pParam->pDos->e_lfanew), sizeof(IMAGE_FILE_HEADER) + sizeof(DWORD), pParam->FileLength)) { |
| 55 | pParam->pNtHeaders = (IMAGE_NT_HEADERS*)((BYTE*)pParam->pDos + VAL32(pParam->pDos->e_lfanew)); |
| 56 | if (pParam->pNtHeaders->Signature != VAL32(IMAGE_NT_SIGNATURE)) |
| 57 | pParam->pNtHeaders = NULL; |
| 58 | } |
| 59 | } |
| 60 | PAL_EXCEPT(EXCEPTION_EXECUTE_HANDLER) { |
| 61 | param.pNtHeaders = NULL; |
| 62 | } |
| 63 | PAL_ENDTRY |
| 64 | |
| 65 | pNtHeaders = param.pNtHeaders; |
| 66 | } |
| 67 | return pNtHeaders; |
| 68 | } |
| 69 | |
| 70 | #endif // #ifndef DACCESS_COMPILE |
| 71 | |
| 72 | EXTERN_C PIMAGE_SECTION_HEADER |
| 73 | Cor_RtlImageRvaToSection32(PTR_IMAGE_NT_HEADERS32 , |
| 74 | ULONG Rva, |
| 75 | ULONG FileLength) |
| 76 | { |
| 77 | LIMITED_METHOD_CONTRACT; |
| 78 | ULONG i; |
| 79 | PTR_IMAGE_SECTION_HEADER NtSection; |
| 80 | |
| 81 | NtSection = PTR_IMAGE_FIRST_SECTION( NtHeaders ); |
| 82 | for (i=0; i<NtHeaders->FileHeader.NumberOfSections; i++) { |
| 83 | if (FileLength && |
| 84 | (((VAL32(NtSection->PointerToRawData) > FileLength)) || |
| 85 | (VAL32(NtSection->SizeOfRawData) > FileLength - VAL32(NtSection->PointerToRawData)))) |
| 86 | return NULL; |
| 87 | if (Rva >= VAL32(NtSection->VirtualAddress) && |
| 88 | Rva < VAL32(NtSection->VirtualAddress) + VAL32(NtSection->SizeOfRawData)) |
| 89 | return NtSection; |
| 90 | |
| 91 | ++NtSection; |
| 92 | } |
| 93 | |
| 94 | return NULL; |
| 95 | } |
| 96 | |
| 97 | EXTERN_C PIMAGE_SECTION_HEADER |
| 98 | Cor_RtlImageRvaToSection64(PTR_IMAGE_NT_HEADERS64 , |
| 99 | ULONG Rva, |
| 100 | ULONG FileLength) |
| 101 | { |
| 102 | LIMITED_METHOD_CONTRACT; |
| 103 | ULONG i; |
| 104 | PTR_IMAGE_SECTION_HEADER NtSection; |
| 105 | |
| 106 | NtSection = PTR_IMAGE_FIRST_SECTION( NtHeaders ); |
| 107 | for (i=0; i<VAL16(NtHeaders->FileHeader.NumberOfSections); i++) { |
| 108 | if (FileLength && |
| 109 | (((VAL32(NtSection->PointerToRawData) > FileLength)) || |
| 110 | (VAL32(NtSection->SizeOfRawData) > FileLength - VAL32(NtSection->PointerToRawData)))) |
| 111 | return NULL; |
| 112 | if (Rva >= VAL32(NtSection->VirtualAddress) && |
| 113 | Rva < VAL32(NtSection->VirtualAddress) + VAL32(NtSection->SizeOfRawData)) |
| 114 | return NtSection; |
| 115 | |
| 116 | ++NtSection; |
| 117 | } |
| 118 | |
| 119 | return NULL; |
| 120 | } |
| 121 | |
| 122 | EXTERN_C PIMAGE_SECTION_HEADER |
| 123 | Cor_RtlImageRvaToSection(PTR_IMAGE_NT_HEADERS , |
| 124 | ULONG Rva, |
| 125 | ULONG FileLength) |
| 126 | { |
| 127 | LIMITED_METHOD_CONTRACT; |
| 128 | if (NtHeaders->OptionalHeader.Magic == VAL16(IMAGE_NT_OPTIONAL_HDR32_MAGIC)) |
| 129 | return Cor_RtlImageRvaToSection32((PTR_IMAGE_NT_HEADERS32)NtHeaders, |
| 130 | Rva, FileLength); |
| 131 | else if(NtHeaders->OptionalHeader.Magic == VAL16(IMAGE_NT_OPTIONAL_HDR64_MAGIC)) |
| 132 | return Cor_RtlImageRvaToSection64((PTR_IMAGE_NT_HEADERS64)NtHeaders, |
| 133 | Rva, FileLength); |
| 134 | else { |
| 135 | _ASSERTE(!"Invalid File Type" ); |
| 136 | return NULL; |
| 137 | } |
| 138 | } |
| 139 | |
| 140 | EXTERN_C PBYTE Cor_RtlImageRvaToVa32(PTR_IMAGE_NT_HEADERS32 , |
| 141 | PBYTE Base, |
| 142 | ULONG Rva, |
| 143 | ULONG FileLength) |
| 144 | { |
| 145 | LIMITED_METHOD_CONTRACT; |
| 146 | PIMAGE_SECTION_HEADER NtSection = |
| 147 | Cor_RtlImageRvaToSection32(NtHeaders, |
| 148 | Rva, |
| 149 | FileLength); |
| 150 | |
| 151 | if (NtSection != NULL) |
| 152 | return (Base + |
| 153 | (Rva - VAL32(NtSection->VirtualAddress)) + |
| 154 | VAL32(NtSection->PointerToRawData)); |
| 155 | else |
| 156 | return NULL; |
| 157 | } |
| 158 | |
| 159 | EXTERN_C PBYTE Cor_RtlImageRvaToVa64(PTR_IMAGE_NT_HEADERS64 , |
| 160 | PBYTE Base, |
| 161 | ULONG Rva, |
| 162 | ULONG FileLength) |
| 163 | { |
| 164 | LIMITED_METHOD_CONTRACT; |
| 165 | PIMAGE_SECTION_HEADER NtSection = |
| 166 | Cor_RtlImageRvaToSection64(NtHeaders, |
| 167 | Rva, |
| 168 | FileLength); |
| 169 | |
| 170 | if (NtSection != NULL) |
| 171 | return (Base + |
| 172 | (Rva - VAL32(NtSection->VirtualAddress)) + |
| 173 | VAL32(NtSection->PointerToRawData)); |
| 174 | else |
| 175 | return NULL; |
| 176 | } |
| 177 | |
| 178 | EXTERN_C PBYTE Cor_RtlImageRvaToVa(PTR_IMAGE_NT_HEADERS , |
| 179 | PBYTE Base, |
| 180 | ULONG Rva, |
| 181 | ULONG FileLength) |
| 182 | { |
| 183 | LIMITED_METHOD_CONTRACT; |
| 184 | if (NtHeaders->OptionalHeader.Magic == VAL16(IMAGE_NT_OPTIONAL_HDR32_MAGIC)) |
| 185 | return Cor_RtlImageRvaToVa32((PTR_IMAGE_NT_HEADERS32)NtHeaders, |
| 186 | Base, Rva, FileLength); |
| 187 | else if(NtHeaders->OptionalHeader.Magic == VAL16(IMAGE_NT_OPTIONAL_HDR64_MAGIC)) |
| 188 | return Cor_RtlImageRvaToVa64((PTR_IMAGE_NT_HEADERS64)NtHeaders, |
| 189 | Base, Rva, FileLength); |
| 190 | else { |
| 191 | _ASSERTE(!"Invalid File Type" ); |
| 192 | return NULL; |
| 193 | } |
| 194 | } |
| 195 | |
| 196 | EXTERN_C PBYTE Cor_RtlImageDirToVa(PTR_IMAGE_NT_HEADERS , |
| 197 | PBYTE Base, |
| 198 | UINT DirIndex, |
| 199 | ULONG FileLength) |
| 200 | { |
| 201 | LIMITED_METHOD_CONTRACT; |
| 202 | if (NtHeaders->OptionalHeader.Magic == VAL16(IMAGE_NT_OPTIONAL_HDR32_MAGIC)) |
| 203 | return Cor_RtlImageRvaToVa32((PTR_IMAGE_NT_HEADERS32)NtHeaders, Base, |
| 204 | VAL32(((PTR_IMAGE_NT_HEADERS32)NtHeaders)->OptionalHeader.DataDirectory[DirIndex].VirtualAddress), |
| 205 | FileLength); |
| 206 | else if(NtHeaders->OptionalHeader.Magic == VAL16(IMAGE_NT_OPTIONAL_HDR64_MAGIC)) |
| 207 | return Cor_RtlImageRvaToVa64((PTR_IMAGE_NT_HEADERS64)NtHeaders, Base, |
| 208 | VAL32(((PTR_IMAGE_NT_HEADERS64)NtHeaders)->OptionalHeader.DataDirectory[DirIndex].VirtualAddress), |
| 209 | FileLength); |
| 210 | else { |
| 211 | _ASSERTE(!"Invalid File Type" ); |
| 212 | return NULL; |
| 213 | } |
| 214 | } |
| 215 | |
| 216 | EXTERN_C PIMAGE_SECTION_HEADER |
| 217 | Cor_RtlImageRvaRangeToSection(PTR_IMAGE_NT_HEADERS , |
| 218 | ULONG Rva, |
| 219 | ULONG Range, |
| 220 | ULONG FileLength) |
| 221 | { |
| 222 | LIMITED_METHOD_CONTRACT; |
| 223 | ULONG i; |
| 224 | PTR_IMAGE_SECTION_HEADER NtSection; |
| 225 | |
| 226 | if (!Range) |
| 227 | return Cor_RtlImageRvaToSection(NtHeaders, Rva, FileLength); |
| 228 | |
| 229 | NtSection = PTR_IMAGE_FIRST_SECTION( NtHeaders ); |
| 230 | for (i=0; i<VAL16(NtHeaders->FileHeader.NumberOfSections); i++) { |
| 231 | if (FileLength && |
| 232 | ((VAL32(NtSection->PointerToRawData) > FileLength) || |
| 233 | (VAL32(NtSection->SizeOfRawData) > FileLength - VAL32(NtSection->PointerToRawData)))) |
| 234 | return NULL; |
| 235 | if (Rva >= VAL32(NtSection->VirtualAddress) && |
| 236 | Rva + Range <= VAL32(NtSection->VirtualAddress) + VAL32(NtSection->SizeOfRawData)) |
| 237 | return NtSection; |
| 238 | |
| 239 | ++NtSection; |
| 240 | } |
| 241 | |
| 242 | return NULL; |
| 243 | } |
| 244 | |
| 245 | EXTERN_C DWORD Cor_RtlImageRvaToOffset(PTR_IMAGE_NT_HEADERS , |
| 246 | ULONG Rva, |
| 247 | ULONG FileLength) |
| 248 | { |
| 249 | LIMITED_METHOD_CONTRACT; |
| 250 | PIMAGE_SECTION_HEADER NtSection = |
| 251 | Cor_RtlImageRvaToSection(NtHeaders, |
| 252 | Rva, |
| 253 | FileLength); |
| 254 | |
| 255 | if (NtSection) |
| 256 | return ((Rva - VAL32(NtSection->VirtualAddress)) + |
| 257 | VAL32(NtSection->PointerToRawData)); |
| 258 | else |
| 259 | return NULL; |
| 260 | } |
| 261 | |