| // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "sandbox/win/src/win_utils.h" |
| |
| #include <windows.h> |
| |
| #include <psapi.h> |
| |
| #include <vector> |
| |
| #include "base/files/file_path.h" |
| #include "base/files/file_util.h" |
| #include "base/numerics/safe_conversions.h" |
| #include "base/path_service.h" |
| #include "base/win/scoped_handle.h" |
| #include "base/win/scoped_process_information.h" |
| #include "sandbox/win/src/nt_internals.h" |
| #include "sandbox/win/tests/common/test_utils.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| |
| namespace { |
| |
| class ScopedTerminateProcess { |
| public: |
| ScopedTerminateProcess(HANDLE process) : process_(process) {} |
| |
| ~ScopedTerminateProcess() { ::TerminateProcess(process_, 0); } |
| |
| private: |
| HANDLE process_; |
| }; |
| |
| bool GetModuleList(HANDLE process, std::vector<HMODULE>* result) { |
| std::vector<HMODULE> modules(256); |
| DWORD size_needed = 0; |
| if (EnumProcessModules( |
| process, &modules[0], |
| base::checked_cast<DWORD>(modules.size() * sizeof(HMODULE)), |
| &size_needed)) { |
| result->assign(modules.begin(), |
| modules.begin() + (size_needed / sizeof(HMODULE))); |
| return true; |
| } |
| modules.resize(size_needed / sizeof(HMODULE)); |
| if (EnumProcessModules( |
| process, &modules[0], |
| base::checked_cast<DWORD>(modules.size() * sizeof(HMODULE)), |
| &size_needed)) { |
| result->assign(modules.begin(), |
| modules.begin() + (size_needed / sizeof(HMODULE))); |
| return true; |
| } |
| return false; |
| } |
| |
| } // namespace |
| |
| TEST(WinUtils, IsReparsePoint) { |
| using sandbox::IsReparsePoint; |
| |
| // Create a temp file because we need write access to it. |
| wchar_t temp_directory[MAX_PATH]; |
| wchar_t my_folder[MAX_PATH]; |
| ASSERT_NE(::GetTempPath(MAX_PATH, temp_directory), 0u); |
| ASSERT_NE(::GetTempFileName(temp_directory, L"test", 0, my_folder), 0u); |
| |
| // Delete the file and create a directory instead. |
| ASSERT_TRUE(::DeleteFile(my_folder)); |
| ASSERT_TRUE(::CreateDirectory(my_folder, nullptr)); |
| |
| EXPECT_EQ(static_cast<DWORD>(ERROR_NOT_A_REPARSE_POINT), |
| IsReparsePoint(my_folder)); |
| |
| base::string16 not_found = base::string16(my_folder) + L"\\foo\\bar"; |
| EXPECT_EQ(static_cast<DWORD>(ERROR_NOT_A_REPARSE_POINT), |
| IsReparsePoint(not_found)); |
| |
| base::string16 new_file = base::string16(my_folder) + L"\\foo"; |
| EXPECT_EQ(static_cast<DWORD>(ERROR_NOT_A_REPARSE_POINT), |
| IsReparsePoint(new_file)); |
| |
| // Replace the directory with a reparse point to %temp%. |
| HANDLE dir = ::CreateFile(my_folder, FILE_ALL_ACCESS, |
| FILE_SHARE_READ | FILE_SHARE_WRITE, nullptr, |
| OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, nullptr); |
| EXPECT_NE(INVALID_HANDLE_VALUE, dir); |
| |
| base::string16 temp_dir_nt = base::string16(L"\\??\\") + temp_directory; |
| EXPECT_TRUE(SetReparsePoint(dir, temp_dir_nt.c_str())); |
| |
| EXPECT_EQ(static_cast<DWORD>(ERROR_SUCCESS), IsReparsePoint(new_file)); |
| |
| EXPECT_TRUE(DeleteReparsePoint(dir)); |
| EXPECT_TRUE(::CloseHandle(dir)); |
| EXPECT_TRUE(::RemoveDirectory(my_folder)); |
| } |
| |
| TEST(WinUtils, SameObject) { |
| using sandbox::SameObject; |
| |
| // Create a temp file because we need write access to it. |
| wchar_t temp_directory[MAX_PATH]; |
| wchar_t my_folder[MAX_PATH]; |
| ASSERT_NE(::GetTempPath(MAX_PATH, temp_directory), 0u); |
| ASSERT_NE(::GetTempFileName(temp_directory, L"test", 0, my_folder), 0u); |
| |
| // Delete the file and create a directory instead. |
| ASSERT_TRUE(::DeleteFile(my_folder)); |
| ASSERT_TRUE(::CreateDirectory(my_folder, nullptr)); |
| |
| base::string16 folder(my_folder); |
| base::string16 file_name = folder + L"\\foo.txt"; |
| const ULONG kSharing = FILE_SHARE_WRITE | FILE_SHARE_READ | FILE_SHARE_DELETE; |
| base::win::ScopedHandle file(CreateFile(file_name.c_str(), GENERIC_WRITE, |
| kSharing, nullptr, CREATE_ALWAYS, |
| FILE_FLAG_DELETE_ON_CLOSE, nullptr)); |
| |
| EXPECT_TRUE(file.IsValid()); |
| base::string16 file_name_nt1 = base::string16(L"\\??\\") + file_name; |
| base::string16 file_name_nt2 = |
| base::string16(L"\\??\\") + folder + L"\\FOO.txT"; |
| EXPECT_TRUE(SameObject(file.Get(), file_name_nt1.c_str())); |
| EXPECT_TRUE(SameObject(file.Get(), file_name_nt2.c_str())); |
| |
| file.Close(); |
| EXPECT_TRUE(::RemoveDirectory(my_folder)); |
| } |
| |
| TEST(WinUtils, IsPipe) { |
| using sandbox::IsPipe; |
| |
| base::string16 pipe_name = L"\\??\\pipe\\mypipe"; |
| EXPECT_TRUE(IsPipe(pipe_name)); |
| |
| pipe_name = L"\\??\\PiPe\\mypipe"; |
| EXPECT_TRUE(IsPipe(pipe_name)); |
| |
| pipe_name = L"\\??\\pipe"; |
| EXPECT_FALSE(IsPipe(pipe_name)); |
| |
| pipe_name = L"\\??\\_pipe_\\mypipe"; |
| EXPECT_FALSE(IsPipe(pipe_name)); |
| |
| pipe_name = L"\\??\\ABCD\\mypipe"; |
| EXPECT_FALSE(IsPipe(pipe_name)); |
| |
| // Written as two strings to prevent trigraph '?' '?' '/'. |
| pipe_name = |
| L"/?" |
| L"?/pipe/mypipe"; |
| EXPECT_FALSE(IsPipe(pipe_name)); |
| |
| pipe_name = L"\\XX\\pipe\\mypipe"; |
| EXPECT_FALSE(IsPipe(pipe_name)); |
| |
| pipe_name = L"\\Device\\NamedPipe\\mypipe"; |
| EXPECT_FALSE(IsPipe(pipe_name)); |
| } |
| |
| TEST(WinUtils, NtStatusToWin32Error) { |
| using sandbox::GetLastErrorFromNtStatus; |
| EXPECT_EQ(static_cast<DWORD>(ERROR_SUCCESS), |
| GetLastErrorFromNtStatus(STATUS_SUCCESS)); |
| EXPECT_EQ(static_cast<DWORD>(ERROR_NOT_SUPPORTED), |
| GetLastErrorFromNtStatus(STATUS_NOT_SUPPORTED)); |
| EXPECT_EQ(static_cast<DWORD>(ERROR_ALREADY_EXISTS), |
| GetLastErrorFromNtStatus(STATUS_OBJECT_NAME_COLLISION)); |
| EXPECT_EQ(static_cast<DWORD>(ERROR_ACCESS_DENIED), |
| GetLastErrorFromNtStatus(STATUS_ACCESS_DENIED)); |
| } |
| |
| TEST(WinUtils, GetProcessBaseAddress) { |
| using sandbox::GetProcessBaseAddress; |
| STARTUPINFO start_info = {}; |
| PROCESS_INFORMATION proc_info = {}; |
| WCHAR command_line[] = L"notepad"; |
| start_info.cb = sizeof(start_info); |
| start_info.dwFlags = STARTF_USESHOWWINDOW; |
| start_info.wShowWindow = SW_HIDE; |
| ASSERT_TRUE(::CreateProcessW(nullptr, command_line, nullptr, nullptr, false, |
| CREATE_SUSPENDED, nullptr, nullptr, &start_info, |
| &proc_info)); |
| base::win::ScopedProcessInformation scoped_proc_info(proc_info); |
| ScopedTerminateProcess process_terminate(scoped_proc_info.process_handle()); |
| void* base_address = GetProcessBaseAddress(scoped_proc_info.process_handle()); |
| ASSERT_NE(nullptr, base_address); |
| ASSERT_NE(static_cast<DWORD>(-1), |
| ::ResumeThread(scoped_proc_info.thread_handle())); |
| ::WaitForInputIdle(scoped_proc_info.process_handle(), 1000); |
| ASSERT_NE(static_cast<DWORD>(-1), |
| ::SuspendThread(scoped_proc_info.thread_handle())); |
| |
| std::vector<HMODULE> modules; |
| // Compare against the loader's module list (which should now be initialized). |
| ASSERT_TRUE(GetModuleList(scoped_proc_info.process_handle(), &modules)); |
| ASSERT_GT(modules.size(), 0U); |
| EXPECT_EQ(base_address, modules[0]); |
| } |
| |
| // This test requires an elevated prompt to setup. |
| TEST(WinUtils, ConvertToLongPath) { |
| // Test setup. |
| base::FilePath orig_path; |
| ASSERT_TRUE(base::PathService::Get(base::DIR_SYSTEM, &orig_path)); |
| orig_path = orig_path.Append(L"calc.exe"); |
| |
| base::FilePath temp_path; |
| ASSERT_TRUE(base::PathService::Get(base::DIR_PROGRAM_FILES, &temp_path)); |
| temp_path = temp_path.Append(L"test_calc.exe"); |
| |
| ASSERT_TRUE(base::CopyFile(orig_path, temp_path)); |
| // No more asserts until cleanup. |
| |
| // WIN32 long path: "c:\Program Files\test_calc.exe" |
| wchar_t short_path[MAX_PATH] = {}; |
| DWORD size = |
| ::GetShortPathNameW(temp_path.value().c_str(), short_path, MAX_PATH); |
| EXPECT_TRUE(size > 0 && size < MAX_PATH); |
| // WIN32 short path: "C:\PROGRA~1\TEST_C~1.exe" |
| |
| // Sanity check that we actually got a short path above! Small chance |
| // it was disabled in the filesystem setup. |
| EXPECT_NE(temp_path.value().length(), ::wcslen(short_path)); |
| |
| base::string16 short_form_native_path; |
| EXPECT_TRUE(sandbox::GetNtPathFromWin32Path(base::string16(short_path), |
| &short_form_native_path)); |
| // NT short path: "\Device\HarddiskVolume4\PROGRA~1\TEST_C~1.EXE" |
| |
| // Test 1: convert win32 short path to long: |
| base::string16 test1(short_path); |
| EXPECT_TRUE(sandbox::ConvertToLongPath(&test1)); |
| EXPECT_TRUE(::wcsicmp(temp_path.value().c_str(), test1.c_str()) == 0); |
| // Expected result: "c:\Program Files\test_calc.exe" |
| |
| // Test 2: convert native short path to long: |
| base::string16 drive_letter = temp_path.value().substr(0, 3); |
| base::string16 test2(short_form_native_path); |
| EXPECT_TRUE(sandbox::ConvertToLongPath(&test2, &drive_letter)); |
| |
| size_t index = short_form_native_path.find_first_of( |
| L'\\', ::wcslen(L"\\Device\\HarddiskVolume")); |
| EXPECT_TRUE(index != base::string16::npos); |
| base::string16 expected_result = short_form_native_path.substr(0, index + 1); |
| expected_result.append(temp_path.value().substr(3)); |
| EXPECT_TRUE(::wcsicmp(expected_result.c_str(), test2.c_str()) == 0); |
| // Expected result: "\Device\HarddiskVolumeX\Program Files\test_calc.exe" |
| |
| // clean up |
| EXPECT_TRUE(base::DeleteFileW(temp_path, false)); |
| |
| return; |
| } |