Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(188)

Side by Side Diff: chrome_elf/create_file/chrome_create_file_unittest.cc

Issue 138593004: Use an alternate mechanism for CreateFile calls in Chrome (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: CreateFile and tests only, gyp changes in separate CL Created 6 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « chrome_elf/create_file/chrome_create_file.cc ('k') | sandbox/win/src/nt_internals.h » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
(Empty)
1 // Copyright 2014 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 #include "chrome_elf/create_file/chrome_create_file.h"
6
7 #include <windows.h>
8
9 #include <bitset>
10 #include <string>
11
12 #include "base/base_paths_win.h"
13 #include "base/file_util.h"
14 #include "base/files/file_path.h"
15 #include "base/files/scoped_temp_dir.h"
16 #include "base/path_service.h"
17 #include "base/threading/platform_thread.h"
18 #include "base/win/iat_patch_function.h"
19 #include "base/win/scoped_handle.h"
20 #include "base/win/windows_version.h"
21 #include "chrome_elf/chrome_elf_constants.h"
22 #include "chrome_elf/ntdll_cache.h"
23 #include "sandbox/win/src/nt_internals.h"
24 #include "testing/gtest/include/gtest/gtest.h"
25 #include "testing/platform_test.h"
26
27
28 namespace {
29
30 // Test fixtures -------------------------------------------------------------
31
32 class ChromeCreateFileTest : public PlatformTest {
33 protected:
34 struct NtCreateFileParams {
35 ACCESS_MASK desired_access;
36 OBJECT_ATTRIBUTES object_attributes;
37 PLARGE_INTEGER allocation_size;
38 ULONG file_attributes;
39 ULONG share_access;
40 ULONG create_disposition;
41 ULONG create_options;
42 PVOID ea_buffer;
43 ULONG ea_length;
44 };
45
46 enum CallPath {
47 ELF,
48 KERNEL
49 };
50
51 template<CallPath path>
52 static NTSTATUS WINAPI FakeNtCreateFile(
53 PHANDLE file_handle,
54 ACCESS_MASK desired_access,
55 POBJECT_ATTRIBUTES object_attributes,
56 PIO_STATUS_BLOCK io_status_block,
57 PLARGE_INTEGER allocation_size,
58 ULONG file_attributes,
59 ULONG share_access,
60 ULONG create_disposition,
61 ULONG create_options,
62 PVOID ea_buffer,
63 ULONG ea_length) {
64 return self_->HandleCreateFileCall(file_handle,
65 desired_access,
66 object_attributes,
67 io_status_block,
68 allocation_size,
69 file_attributes,
70 share_access,
71 create_disposition,
72 create_options,
73 ea_buffer,
74 ea_length,
75 path);
76 }
77
78 virtual void SetUp() OVERRIDE {
79 original_thread_ = base::PlatformThread::CurrentId();
80 InitCache();
81 PlatformTest::SetUp();
82
83 base::FilePath user_data_dir;
84 PathService::Get(base::DIR_LOCAL_APP_DATA, &user_data_dir);
85 ASSERT_TRUE(temp_dir_.CreateUniqueTempDirUnderPath(user_data_dir));
86 ASSERT_TRUE(temp_dir2_.CreateUniqueTempDir());
87 self_ = this;
88 }
89
90 void RedirectNtCreateFileCalls() {
91 old_func_ptr_ =
92 reinterpret_cast<NtCreateFileFunction>(g_ntdll_lookup["NtCreateFile"]);
93
94 // KernelBase.dll only exists for Win7 and later, prior to that, kernel32
95 // imports from ntdll directly.
96 if (base::win::GetVersion() < base::win::VERSION_WIN7) {
97 patcher_.Patch(L"kernel32.dll", "ntdll.dll", "NtCreateFile",
98 reinterpret_cast<void(*)()>(&FakeNtCreateFile<KERNEL>));
99 } else {
100 patcher_.Patch(L"kernelbase.dll", "ntdll.dll", "NtCreateFile",
101 reinterpret_cast<void(*)()>(&FakeNtCreateFile<KERNEL>));
102 }
103
104 g_ntdll_lookup["NtCreateFile"] = reinterpret_cast<void(*)()>(
105 &ChromeCreateFileTest::FakeNtCreateFile<ELF>);
106 }
107
108 void ResetNtCreateFileCalls() {
109 g_ntdll_lookup["NtCreateFile"] = reinterpret_cast<void*>(old_func_ptr_);
110 patcher_.Unpatch();
111 }
112
113 NTSTATUS HandleCreateFileCall(PHANDLE file_handle,
114 ACCESS_MASK desired_access,
115 POBJECT_ATTRIBUTES object_attributes,
116 PIO_STATUS_BLOCK io_status_block,
117 PLARGE_INTEGER allocation_size,
118 ULONG file_attributes,
119 ULONG share_access,
120 ULONG create_disposition,
121 ULONG create_options,
122 PVOID ea_buffer,
123 ULONG ea_length,
124 CallPath call_path) {
125 if (original_thread_ == base::PlatformThread::CurrentId()) {
126 SetParams(desired_access,
127 object_attributes,
128 allocation_size,
129 file_attributes,
130 share_access,
131 create_disposition,
132 create_options,
133 ea_buffer,
134 ea_length,
135 call_path == ELF ? &elf_params_ : &kernel_params_);
136 }
137
138 // Forward the call to the real NTCreateFile.
139 return old_func_ptr_(file_handle,
140 desired_access,
141 object_attributes,
142 io_status_block,
143 allocation_size,
144 file_attributes,
145 share_access,
146 create_disposition,
147 create_options,
148 ea_buffer,
149 ea_length);
150 }
151
152 void SetParams(ACCESS_MASK desired_access,
153 POBJECT_ATTRIBUTES object_attributes,
154 PLARGE_INTEGER allocation_size,
155 ULONG file_attributes,
156 ULONG share_access,
157 ULONG create_disposition,
158 ULONG create_options,
159 PVOID ea_buffer,
160 ULONG ea_length,
161 NtCreateFileParams* params) {
162 params->desired_access = desired_access;
163 params->object_attributes.Length = object_attributes->Length;
164 params->object_attributes.ObjectName = object_attributes->ObjectName;
165 params->object_attributes.RootDirectory = object_attributes->RootDirectory;
166 params->object_attributes.Attributes = object_attributes->Attributes;
167 params->object_attributes.SecurityDescriptor =
168 object_attributes->SecurityDescriptor;
169 params->object_attributes.SecurityQualityOfService =
170 object_attributes->SecurityQualityOfService;
171 params->allocation_size = allocation_size;
172 params->file_attributes = file_attributes;
173 params->share_access = share_access;
174 params->create_disposition = create_disposition;
175 params->create_options = create_options;
176 params->ea_buffer = ea_buffer;
177 params->ea_length = ea_length;
178 }
179
180 void CheckParams() {
181 std::bitset<32> elf((int) elf_params_.desired_access);
182 std::bitset<32> ker((int) kernel_params_.desired_access);
183
184 EXPECT_EQ(kernel_params_.desired_access, elf_params_.desired_access)
185 << elf << "\n" << ker;
186 EXPECT_EQ(kernel_params_.object_attributes.Length,
187 elf_params_.object_attributes.Length);
188 EXPECT_EQ(kernel_params_.object_attributes.RootDirectory,
189 elf_params_.object_attributes.RootDirectory);
190 EXPECT_EQ(kernel_params_.object_attributes.Attributes,
191 elf_params_.object_attributes.Attributes);
192 EXPECT_EQ(kernel_params_.object_attributes.SecurityDescriptor,
193 elf_params_.object_attributes.SecurityDescriptor);
194 EXPECT_EQ(kernel_params_.allocation_size, elf_params_.allocation_size);
195 EXPECT_EQ(kernel_params_.file_attributes, elf_params_.file_attributes);
196 EXPECT_EQ(kernel_params_.share_access, elf_params_.share_access);
197 EXPECT_EQ(kernel_params_.create_disposition,
198 elf_params_.create_disposition);
199 EXPECT_EQ(kernel_params_.create_options, elf_params_.create_options);
200 EXPECT_EQ(kernel_params_.ea_buffer, elf_params_.ea_buffer);
201 EXPECT_EQ(kernel_params_.ea_length, elf_params_.ea_length);
202 }
203
204 void DoWriteCheck(const base::FilePath& path, DWORD flag, bool is_system) {
205 base::win::ScopedHandle file_handle;
206 const char kTestData[] = "0123456789";
207 int buffer_size = sizeof(kTestData) - 1;
208 DWORD bytes_written;
209
210 if (is_system) {
211 file_handle.Set(::CreateFileW(path.value().c_str(),
212 GENERIC_WRITE,
213 FILE_SHARE_READ,
214 NULL,
215 CREATE_ALWAYS,
216 FILE_ATTRIBUTE_NORMAL | flag,
217 NULL));
218 } else {
219 file_handle.Set(CreateFileNTDLL(path.value().c_str(),
220 GENERIC_WRITE,
221 FILE_SHARE_READ,
222 NULL,
223 CREATE_ALWAYS,
224 FILE_ATTRIBUTE_NORMAL | flag,
225 NULL));
226 }
227
228
229 EXPECT_FALSE(file_handle == INVALID_HANDLE_VALUE);
230 ::WriteFile(file_handle, kTestData, buffer_size, &bytes_written, NULL);
231 EXPECT_EQ(buffer_size, bytes_written);
232 }
233
234 void DoReadCheck(const base::FilePath& path, DWORD flag, bool is_system) {
235 base::win::ScopedHandle file_handle;
236 const char kTestData[] = "0123456789";
237 int buffer_size = sizeof(kTestData) - 1;
238 DWORD bytes_read;
239 char read_buffer[10];
240
241 if (is_system) {
242 file_handle.Set(::CreateFileW(path.value().c_str(),
243 GENERIC_READ,
244 0,
245 NULL,
246 OPEN_ALWAYS,
247 FILE_ATTRIBUTE_NORMAL | flag,
248 NULL));
249 } else {
250 file_handle.Set(CreateFileNTDLL(path.value().c_str(),
251 GENERIC_READ,
252 0,
253 NULL,
254 OPEN_ALWAYS,
255 FILE_ATTRIBUTE_NORMAL | flag,
256 NULL));
257 }
258
259 EXPECT_FALSE(file_handle == INVALID_HANDLE_VALUE);
260 ::ReadFile(file_handle, read_buffer, buffer_size, &bytes_read, NULL);
261 EXPECT_EQ(buffer_size, bytes_read);
262 EXPECT_EQ(0, memcmp(kTestData, read_buffer, bytes_read));
263 }
264
265 void RunChecks(DWORD flag, bool check_reads) {
266 // Make sure we can write to this file handle when called via the system.
267 base::FilePath junk_path_1 = temp_dir_.path().Append(L"junk_1.txt");
268 base::FilePath junk_path_2 = temp_dir_.path().Append(L"junk_2.txt");
269 DoWriteCheck(junk_path_1, flag, true);
270 DoWriteCheck(junk_path_2, flag, false);
271 CheckParams();
272
273 if (check_reads) {
274 // Make sure we can read from this file handle when called via the system.
275 DoReadCheck(junk_path_1, flag, true);
276 DoReadCheck(junk_path_2, flag, false);
277 CheckParams();
278 }
279 base::DeleteFile(junk_path_1, false);
280 base::DeleteFile(junk_path_2, false);
281
282 }
283
284 static ChromeCreateFileTest* self_;
285
286 NtCreateFileFunction old_func_ptr_;
287 base::ScopedTempDir temp_dir_;
288 base::ScopedTempDir temp_dir2_;
289 base::win::IATPatchFunction patcher_;
290 NtCreateFileParams kernel_params_;
291 NtCreateFileParams elf_params_;
292 base::PlatformThreadId original_thread_;
293 };
294
295 ChromeCreateFileTest* ChromeCreateFileTest::self_ = NULL;
296
297 // Tests ---------------------------------------------------------------------
298 TEST_F(ChromeCreateFileTest, CheckParams_FILE_ATTRIBUTE_NORMAL) {
299 RedirectNtCreateFileCalls();
300 RunChecks(FILE_ATTRIBUTE_NORMAL, true);
301 ResetNtCreateFileCalls();
302 }
303
304 TEST_F(ChromeCreateFileTest, CheckParams_FILE_FLAG_WRITE_THROUGH) {
305 RedirectNtCreateFileCalls();
306 RunChecks(FILE_FLAG_WRITE_THROUGH, true);
307 ResetNtCreateFileCalls();
308 }
309
310 TEST_F(ChromeCreateFileTest, CheckParams_FILE_FLAG_RANDOM_ACCESS) {
311 RedirectNtCreateFileCalls();
312 RunChecks(FILE_FLAG_RANDOM_ACCESS, true);
313 ResetNtCreateFileCalls();
314 }
315
316 TEST_F(ChromeCreateFileTest, CheckParams_FILE_FLAG_SEQUENTIAL_SCAN) {
317 RedirectNtCreateFileCalls();
318 RunChecks(FILE_FLAG_SEQUENTIAL_SCAN, true);
319 ResetNtCreateFileCalls();
320 }
321
322 TEST_F(ChromeCreateFileTest, CheckParams_FILE_FLAG_DELETE_ON_CLOSE) {
323 RedirectNtCreateFileCalls();
324 RunChecks(FILE_FLAG_DELETE_ON_CLOSE, false);
325 ResetNtCreateFileCalls();
326 }
327
328 TEST_F(ChromeCreateFileTest, CheckParams_FILE_FLAG_BACKUP_SEMANTICS) {
329 RedirectNtCreateFileCalls();
330 RunChecks(FILE_FLAG_BACKUP_SEMANTICS, true);
331 ResetNtCreateFileCalls();
332 }
333
334 TEST_F(ChromeCreateFileTest, CheckParams_FILE_FLAG_OPEN_REPARSE_POINT) {
335 RedirectNtCreateFileCalls();
336 RunChecks(FILE_FLAG_OPEN_REPARSE_POINT, true);
337 ResetNtCreateFileCalls();
338 }
339
340 TEST_F(ChromeCreateFileTest, CheckParams_FILE_FLAG_OPEN_NO_RECALL) {
341 RedirectNtCreateFileCalls();
342 RunChecks(FILE_FLAG_OPEN_NO_RECALL, true);
343 ResetNtCreateFileCalls();
344 }
345
346 TEST_F(ChromeCreateFileTest, BypassTest) {
347 std::wstring UNC_filepath_file(L"\\\\.\\some_file.txt");
348
349 base::FilePath local_path;
350 PathService::Get(base::DIR_LOCAL_APP_DATA, &local_path);
351
352 base::FilePath local_prefs_path = local_path.Append(kAppDataDirName).Append(
353 kUserDataDirName).Append(L"default\\Preferences");
354 base::FilePath local_state_path = local_path.Append(kAppDataDirName).Append(
355 kUserDataDirName).Append(L"ninja\\Local State");
356 base::FilePath local_junk_path = local_path.Append(kAppDataDirName).Append(
357 kUserDataDirName).Append(L"default\\Junk");
358
359 base::FilePath desktop_path;
360 PathService::Get(base::DIR_USER_DESKTOP, &desktop_path);
361 base::FilePath desktop_junk_path =
362 desktop_path.Append(L"Downloads\\junk.txt");
363 base::FilePath desktop_prefs_path =
364 desktop_path.Append(L"Downloads\\Preferences");
365
366 // Don't redirect UNC files.
367 EXPECT_FALSE(ShouldBypass(UNC_filepath_file.c_str()));
368
369 // Don't redirect if file is not in UserData directory.
370 EXPECT_FALSE(ShouldBypass(desktop_junk_path.value().c_str()));
371 EXPECT_FALSE(ShouldBypass(desktop_prefs_path.value().c_str()));
372
373 // Only redirect "Preferences" and "Local State" files.
374 EXPECT_TRUE(ShouldBypass(local_prefs_path.value().c_str()));
375 EXPECT_TRUE(ShouldBypass(local_state_path.value().c_str()));
376 EXPECT_FALSE(ShouldBypass(local_junk_path.value().c_str()));
377 }
378
379 TEST_F(ChromeCreateFileTest, NtCreateFileAddressCheck) {
380 HMODULE ntdll_handle = ::GetModuleHandle(L"ntdll.dll");
381 EXPECT_EQ(::GetProcAddress(ntdll_handle, "NtCreateFile"),
382 g_ntdll_lookup["NtCreateFile"]);
383 }
384
385 TEST_F(ChromeCreateFileTest, ReadWriteFromNtDll) {
386 base::FilePath file_name = temp_dir_.path().Append(L"some_file.txt");
387 DoWriteCheck(file_name, FILE_ATTRIBUTE_NORMAL, false);
388 DoReadCheck(file_name, FILE_ATTRIBUTE_NORMAL, false);
389 }
390
391 } // namespace
OLDNEW
« no previous file with comments | « chrome_elf/create_file/chrome_create_file.cc ('k') | sandbox/win/src/nt_internals.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698