2015-09-01 09:32:09 -07:00
|
|
|
// Copyright 2015 The Crashpad Authors. All rights reserved.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
#include "snapshot/win/pe_image_reader.h"
|
|
|
|
|
2015-09-16 12:42:20 -07:00
|
|
|
#define PSAPI_VERSION 1
|
2015-09-01 09:32:09 -07:00
|
|
|
#include <psapi.h>
|
|
|
|
|
2015-12-01 17:06:37 -05:00
|
|
|
#include "base/files/file_path.h"
|
|
|
|
#include "base/strings/utf_string_conversions.h"
|
2015-09-01 09:32:09 -07:00
|
|
|
#include "gtest/gtest.h"
|
|
|
|
#include "snapshot/win/process_reader_win.h"
|
2015-12-01 17:06:37 -05:00
|
|
|
#include "test/errors.h"
|
2016-01-18 15:05:20 -05:00
|
|
|
#include "util/win/get_module_information.h"
|
2015-12-01 17:06:37 -05:00
|
|
|
#include "util/win/module_version.h"
|
2015-12-02 11:11:54 -05:00
|
|
|
#include "util/win/process_info.h"
|
2015-09-01 09:32:09 -07:00
|
|
|
|
|
|
|
extern "C" IMAGE_DOS_HEADER __ImageBase;
|
|
|
|
|
|
|
|
namespace crashpad {
|
|
|
|
namespace test {
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
TEST(PEImageReader, DebugDirectory) {
|
|
|
|
PEImageReader pe_image_reader;
|
|
|
|
ProcessReaderWin process_reader;
|
2015-09-09 12:29:29 -07:00
|
|
|
ASSERT_TRUE(process_reader.Initialize(GetCurrentProcess(),
|
|
|
|
ProcessSuspensionState::kRunning));
|
2015-09-01 09:32:09 -07:00
|
|
|
HMODULE self = reinterpret_cast<HMODULE>(&__ImageBase);
|
|
|
|
MODULEINFO module_info;
|
2015-09-16 12:42:20 -07:00
|
|
|
ASSERT_TRUE(CrashpadGetModuleInformation(
|
2015-12-01 17:06:37 -05:00
|
|
|
GetCurrentProcess(), self, &module_info, sizeof(module_info)))
|
|
|
|
<< ErrorMessage("GetModuleInformation");
|
2015-09-01 09:32:09 -07:00
|
|
|
EXPECT_EQ(self, module_info.lpBaseOfDll);
|
2015-12-01 17:06:37 -05:00
|
|
|
ASSERT_TRUE(pe_image_reader.Initialize(&process_reader,
|
2015-09-01 09:32:09 -07:00
|
|
|
reinterpret_cast<WinVMAddress>(self),
|
|
|
|
module_info.SizeOfImage,
|
|
|
|
"self"));
|
|
|
|
UUID uuid;
|
|
|
|
DWORD age;
|
|
|
|
std::string pdbname;
|
|
|
|
EXPECT_TRUE(pe_image_reader.DebugDirectoryInformation(&uuid, &age, &pdbname));
|
2015-09-02 18:35:19 -07:00
|
|
|
EXPECT_NE(std::string::npos, pdbname.find("crashpad_snapshot_test"));
|
2015-09-01 09:32:09 -07:00
|
|
|
const std::string suffix(".pdb");
|
|
|
|
EXPECT_EQ(
|
|
|
|
0,
|
|
|
|
pdbname.compare(pdbname.size() - suffix.size(), suffix.size(), suffix));
|
|
|
|
}
|
|
|
|
|
2015-12-02 11:11:54 -05:00
|
|
|
void TestVSFixedFileInfo(ProcessReaderWin* process_reader,
|
|
|
|
const ProcessInfo::Module& module,
|
|
|
|
bool known_dll) {
|
|
|
|
PEImageReader pe_image_reader;
|
|
|
|
ASSERT_TRUE(pe_image_reader.Initialize(process_reader,
|
|
|
|
module.dll_base,
|
|
|
|
module.size,
|
|
|
|
base::UTF16ToUTF8(module.name)));
|
|
|
|
|
|
|
|
VS_FIXEDFILEINFO observed;
|
|
|
|
const bool observed_rv = pe_image_reader.VSFixedFileInfo(&observed);
|
|
|
|
ASSERT_TRUE(observed_rv || !known_dll);
|
|
|
|
|
|
|
|
if (observed_rv) {
|
|
|
|
EXPECT_EQ(VS_FFI_SIGNATURE, observed.dwSignature);
|
|
|
|
EXPECT_EQ(VS_FFI_STRUCVERSION, observed.dwStrucVersion);
|
|
|
|
EXPECT_EQ(0, observed.dwFileFlags & ~observed.dwFileFlagsMask);
|
|
|
|
EXPECT_EQ(VOS_NT_WINDOWS32, observed.dwFileOS);
|
|
|
|
if (known_dll) {
|
|
|
|
EXPECT_EQ(VFT_DLL, observed.dwFileType);
|
|
|
|
} else {
|
|
|
|
EXPECT_TRUE(observed.dwFileType == VFT_APP ||
|
|
|
|
observed.dwFileType == VFT_DLL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-02 17:48:20 -05:00
|
|
|
base::FilePath module_path(module.name);
|
|
|
|
|
|
|
|
const DWORD version = GetVersion();
|
|
|
|
const int major_version = LOBYTE(LOWORD(version));
|
|
|
|
const int minor_version = HIBYTE(LOWORD(version));
|
|
|
|
if (major_version > 6 || (major_version == 6 && minor_version >= 2)) {
|
|
|
|
// Windows 8 or later.
|
|
|
|
//
|
|
|
|
// Use BaseName() to ensure that GetModuleVersionAndType() finds the
|
|
|
|
// already-loaded module with the specified name. Otherwise, dwFileVersionMS
|
|
|
|
// may not match. This appears to be related to the changes made in Windows
|
|
|
|
// 8.1 to GetVersion() and GetVersionEx() for non-manifested applications
|
|
|
|
module_path = module_path.BaseName();
|
|
|
|
}
|
|
|
|
|
2015-12-02 11:11:54 -05:00
|
|
|
VS_FIXEDFILEINFO expected;
|
2015-12-02 17:48:20 -05:00
|
|
|
const bool expected_rv = GetModuleVersionAndType(module_path, &expected);
|
2015-12-02 11:11:54 -05:00
|
|
|
ASSERT_TRUE(expected_rv || !known_dll);
|
|
|
|
|
|
|
|
EXPECT_EQ(expected_rv, observed_rv);
|
|
|
|
|
|
|
|
if (observed_rv && expected_rv) {
|
|
|
|
EXPECT_EQ(expected.dwSignature, observed.dwSignature);
|
|
|
|
EXPECT_EQ(expected.dwStrucVersion, observed.dwStrucVersion);
|
|
|
|
EXPECT_EQ(expected.dwFileVersionMS, observed.dwFileVersionMS);
|
|
|
|
EXPECT_EQ(expected.dwFileVersionLS, observed.dwFileVersionLS);
|
|
|
|
EXPECT_EQ(expected.dwProductVersionMS, observed.dwProductVersionMS);
|
|
|
|
EXPECT_EQ(expected.dwProductVersionLS, observed.dwProductVersionLS);
|
|
|
|
EXPECT_EQ(expected.dwFileFlagsMask, observed.dwFileFlagsMask);
|
|
|
|
EXPECT_EQ(expected.dwFileFlags, observed.dwFileFlags);
|
|
|
|
EXPECT_EQ(expected.dwFileOS, observed.dwFileOS);
|
|
|
|
EXPECT_EQ(expected.dwFileType, observed.dwFileType);
|
|
|
|
EXPECT_EQ(expected.dwFileSubtype, observed.dwFileSubtype);
|
|
|
|
EXPECT_EQ(expected.dwFileDateMS, observed.dwFileDateMS);
|
|
|
|
EXPECT_EQ(expected.dwFileDateLS, observed.dwFileDateLS);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(PEImageReader, VSFixedFileInfo_OneModule) {
|
2015-12-01 17:06:37 -05:00
|
|
|
ProcessReaderWin process_reader;
|
|
|
|
ASSERT_TRUE(process_reader.Initialize(GetCurrentProcess(),
|
|
|
|
ProcessSuspensionState::kRunning));
|
|
|
|
|
|
|
|
const wchar_t kModuleName[] = L"kernel32.dll";
|
2015-12-02 11:11:54 -05:00
|
|
|
const HMODULE module_handle = GetModuleHandle(kModuleName);
|
2015-12-01 17:06:37 -05:00
|
|
|
ASSERT_TRUE(module_handle) << ErrorMessage("GetModuleHandle");
|
|
|
|
|
|
|
|
MODULEINFO module_info;
|
|
|
|
ASSERT_TRUE(CrashpadGetModuleInformation(
|
|
|
|
GetCurrentProcess(), module_handle, &module_info, sizeof(module_info)))
|
|
|
|
<< ErrorMessage("GetModuleInformation");
|
|
|
|
EXPECT_EQ(module_handle, module_info.lpBaseOfDll);
|
|
|
|
|
2015-12-02 11:11:54 -05:00
|
|
|
ProcessInfo::Module module;
|
|
|
|
module.name = kModuleName;
|
|
|
|
module.dll_base = reinterpret_cast<WinVMAddress>(module_info.lpBaseOfDll);
|
|
|
|
module.size = module_info.SizeOfImage;
|
2015-12-01 17:06:37 -05:00
|
|
|
|
2015-12-02 11:11:54 -05:00
|
|
|
TestVSFixedFileInfo(&process_reader, module, true);
|
|
|
|
}
|
2015-12-01 17:06:37 -05:00
|
|
|
|
2015-12-02 11:11:54 -05:00
|
|
|
TEST(PEImageReader, VSFixedFileInfo_AllModules) {
|
|
|
|
ProcessReaderWin process_reader;
|
|
|
|
ASSERT_TRUE(process_reader.Initialize(GetCurrentProcess(),
|
|
|
|
ProcessSuspensionState::kRunning));
|
2015-12-01 17:06:37 -05:00
|
|
|
|
2015-12-02 11:11:54 -05:00
|
|
|
const std::vector<ProcessInfo::Module>& modules = process_reader.Modules();
|
|
|
|
EXPECT_GT(modules.size(), 2u);
|
|
|
|
|
|
|
|
for (const auto& module : modules) {
|
|
|
|
SCOPED_TRACE(base::UTF16ToUTF8(module.name));
|
|
|
|
TestVSFixedFileInfo(&process_reader, module, false);
|
|
|
|
}
|
2015-12-01 17:06:37 -05:00
|
|
|
}
|
|
|
|
|
2015-09-01 09:32:09 -07:00
|
|
|
} // namespace
|
|
|
|
} // namespace test
|
|
|
|
} // namespace crashpad
|