1 // Copyright (c) 2010 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 // This file contains the implementation for an iterator over a portable
6 // executable file's resources.
7
8 #include "chrome/installer/test/pe_image_resources.h"
9
10 #include "base/logging.h"
11 #include "base/win/pe_image.h"
12
13 namespace {
14
15 // Performs a cast to type |T| of |data| iff |data_size| is sufficient to hold
16 // an instance of type |T|. Returns true on success.
17 template<class T>
StructureAt(const uint8 * data,size_t data_size,const T ** structure)18 bool StructureAt(const uint8* data, size_t data_size, const T** structure) {
19 if (sizeof(T) <= data_size) {
20 *structure = reinterpret_cast<const T*>(data);
21 return true;
22 }
23 return false;
24 }
25
26 // Recursive function for enumerating entries in an image's resource segment.
27 // static
EnumResourcesWorker(const base::win::PEImage & image,const uint8 * tree_base,DWORD tree_size,DWORD directory_offset,upgrade_test::EntryPath * path,upgrade_test::EnumResource_Fn callback,uintptr_t context)28 bool EnumResourcesWorker(
29 const base::win::PEImage& image, const uint8* tree_base, DWORD tree_size,
30 DWORD directory_offset, upgrade_test::EntryPath* path,
31 upgrade_test::EnumResource_Fn callback, uintptr_t context) {
32 bool success = true;
33 const IMAGE_RESOURCE_DIRECTORY* resource_directory;
34
35 if (!StructureAt(tree_base + directory_offset, tree_size - directory_offset,
36 &resource_directory) ||
37 directory_offset + sizeof(IMAGE_RESOURCE_DIRECTORY) +
38 (resource_directory->NumberOfNamedEntries +
39 resource_directory->NumberOfIdEntries) *
40 sizeof(IMAGE_RESOURCE_DIRECTORY_ENTRY) > tree_size) {
41 LOG(DFATAL) << "Insufficient room in resource segment for directory entry.";
42 return false;
43 }
44
45 const IMAGE_RESOURCE_DIRECTORY_ENTRY* scan =
46 reinterpret_cast<const IMAGE_RESOURCE_DIRECTORY_ENTRY*>(
47 tree_base + directory_offset +
48 sizeof(IMAGE_RESOURCE_DIRECTORY));
49 const IMAGE_RESOURCE_DIRECTORY_ENTRY* end = scan +
50 resource_directory->NumberOfNamedEntries +
51 resource_directory->NumberOfIdEntries;
52 for (; success && scan != end; ++scan) {
53 if ((scan->NameIsString != 0) !=
54 (scan - reinterpret_cast<const IMAGE_RESOURCE_DIRECTORY_ENTRY*>(
55 tree_base + directory_offset +
56 sizeof(IMAGE_RESOURCE_DIRECTORY)) <
57 resource_directory->NumberOfNamedEntries)) {
58 LOG(DFATAL) << "Inconsistent number of named or numbered entries.";
59 success = false;
60 break;
61 }
62 if (scan->NameIsString) {
63 const IMAGE_RESOURCE_DIR_STRING_U* dir_string;
64 if (!StructureAt(tree_base + scan->NameOffset,
65 tree_size - scan->NameOffset, &dir_string) ||
66 scan->NameOffset + sizeof(WORD) +
67 dir_string->Length * sizeof(wchar_t) > tree_size) {
68 LOG(DFATAL) << "Insufficient room in resource segment for entry name.";
69 success = false;
70 break;
71 }
72 path->push_back(
73 upgrade_test::EntryId(std::wstring(&dir_string->NameString[0],
74 dir_string->Length)));
75 } else {
76 path->push_back(upgrade_test::EntryId(scan->Id));
77 }
78 if (scan->DataIsDirectory) {
79 success = EnumResourcesWorker(image, tree_base, tree_size,
80 scan->OffsetToDirectory, path, callback,
81 context);
82 } else {
83 const IMAGE_RESOURCE_DATA_ENTRY* data_entry;
84 if (StructureAt(tree_base + scan->OffsetToData,
85 tree_size - scan->OffsetToData, &data_entry) &&
86 reinterpret_cast<uint8*>(
87 image.RVAToAddr(data_entry->OffsetToData)) + data_entry->Size <=
88 tree_base + tree_size) {
89 // Despite what winnt.h says, OffsetToData is an RVA.
90 callback(
91 *path,
92 reinterpret_cast<uint8*>(image.RVAToAddr(data_entry->OffsetToData)),
93 data_entry->Size, data_entry->CodePage, context);
94 } else {
95 LOG(DFATAL) << "Insufficient room in resource segment for data entry.";
96 success = false;
97 }
98 }
99 path->pop_back();
100 }
101
102 return success;
103 }
104
105 } // namespace
106
107 namespace upgrade_test {
108
109 // static
EnumResources(const base::win::PEImage & image,EnumResource_Fn callback,uintptr_t context)110 bool EnumResources(const base::win::PEImage& image, EnumResource_Fn callback,
111 uintptr_t context) {
112 DWORD resources_size =
113 image.GetImageDirectoryEntrySize(IMAGE_DIRECTORY_ENTRY_RESOURCE);
114 if (resources_size != 0) {
115 EntryPath path_storage;
116 return EnumResourcesWorker(
117 image,
118 reinterpret_cast<uint8*>(
119 image.GetImageDirectoryEntryAddr(IMAGE_DIRECTORY_ENTRY_RESOURCE)),
120 resources_size, 0, &path_storage, callback, context);
121 }
122 return true;
123 }
124
125 } // namespace upgrade_test
126