// 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 #include #include #include #include #include #include #include "base/at_exit.h" #include "base/command_line.h" #include "base/files/file_path.h" #include "base/lazy_instance.h" #include "base/logging.h" #include "base/strings/string16.h" #include "base/strings/string_util.h" #include "base/time/time.h" #include "base/win/windows_version.h" #include "chrome/app/chrome_crash_reporter_client.h" #include "chrome/app/main_dll_loader_win.h" #include "chrome/browser/chrome_process_finder_win.h" #include "chrome/browser/policy/policy_path_parser.h" #include "chrome/common/chrome_paths_internal.h" #include "chrome/common/chrome_switches.h" #include "chrome/installer/util/browser_distribution.h" #include "chrome_elf/chrome_elf_main.h" #include "components/crash/content/app/crash_reporter_client.h" #include "components/crash/content/app/crash_switches.h" #include "components/crash/content/app/crashpad.h" #include "components/crash/content/app/run_as_crashpad_handler_win.h" #include "components/startup_metric_utils/browser/startup_metric_utils.h" #include "components/startup_metric_utils/common/pre_read_field_trial_utils_win.h" #include "content/public/common/content_switches.h" #include "content/public/common/result_codes.h" #include "ui/gfx/win/dpi.h" namespace { base::LazyInstance::Leaky g_chrome_crash_client = LAZY_INSTANCE_INITIALIZER; base::LazyInstance>::Leaky g_uploaded_reports = LAZY_INSTANCE_INITIALIZER; // List of switches that it's safe to rendezvous early with. Fast start should // not be done if a command line contains a switch not in this set. // Note this is currently stored as a list of two because it's probably faster // to iterate over this small array than building a map for constant time // lookups. const char* const kFastStartSwitches[] = { switches::kProfileDirectory, switches::kShowAppList, }; bool IsFastStartSwitch(const std::string& command_line_switch) { for (size_t i = 0; i < arraysize(kFastStartSwitches); ++i) { if (command_line_switch == kFastStartSwitches[i]) return true; } return false; } bool ContainsNonFastStartFlag(const base::CommandLine& command_line) { const base::CommandLine::SwitchMap& switches = command_line.GetSwitches(); if (switches.size() > arraysize(kFastStartSwitches)) return true; for (base::CommandLine::SwitchMap::const_iterator it = switches.begin(); it != switches.end(); ++it) { if (!IsFastStartSwitch(it->first)) return true; } return false; } bool AttemptFastNotify(const base::CommandLine& command_line) { if (ContainsNonFastStartFlag(command_line)) return false; base::FilePath user_data_dir; if (!chrome::GetDefaultUserDataDirectory(&user_data_dir)) return false; policy::path_parser::CheckUserDataDirPolicy(&user_data_dir); HWND chrome = chrome::FindRunningChromeWindow(user_data_dir); if (!chrome) return false; return chrome::AttemptToNotifyRunningChrome(chrome, true) == chrome::NOTIFY_SUCCESS; } // Win8.1 supports monitor-specific DPI scaling. bool SetProcessDpiAwarenessWrapper(PROCESS_DPI_AWARENESS value) { typedef HRESULT(WINAPI *SetProcessDpiAwarenessPtr)(PROCESS_DPI_AWARENESS); SetProcessDpiAwarenessPtr set_process_dpi_awareness_func = reinterpret_cast( GetProcAddress(GetModuleHandleA("user32.dll"), "SetProcessDpiAwarenessInternal")); if (set_process_dpi_awareness_func) { HRESULT hr = set_process_dpi_awareness_func(value); if (SUCCEEDED(hr)) { VLOG(1) << "SetProcessDpiAwareness succeeded."; return true; } else if (hr == E_ACCESSDENIED) { LOG(ERROR) << "Access denied error from SetProcessDpiAwareness. " "Function called twice, or manifest was used."; } } return false; } // This function works for Windows Vista through Win8. Win8.1 must use // SetProcessDpiAwareness[Wrapper]. BOOL SetProcessDPIAwareWrapper() { typedef BOOL(WINAPI *SetProcessDPIAwarePtr)(VOID); SetProcessDPIAwarePtr set_process_dpi_aware_func = reinterpret_cast( GetProcAddress(GetModuleHandleA("user32.dll"), "SetProcessDPIAware")); return set_process_dpi_aware_func && set_process_dpi_aware_func(); } void EnableHighDPISupport() { if (!SetProcessDpiAwarenessWrapper(PROCESS_SYSTEM_DPI_AWARE)) { SetProcessDPIAwareWrapper(); } } void SwitchToLFHeap() { // Only needed on XP but harmless on other Windows flavors. auto crt_heap = _get_heap_handle(); ULONG enable_LFH = 2; if (HeapSetInformation(reinterpret_cast(crt_heap), HeapCompatibilityInformation, &enable_LFH, sizeof(enable_LFH))) { VLOG(1) << "Low fragmentation heap enabled."; } } // Returns true if |command_line| contains a /prefetch:# argument where # is in // [1, 8]. bool HasValidWindowsPrefetchArgument(const base::CommandLine& command_line) { const base::char16 kPrefetchArgumentPrefix[] = L"/prefetch:"; for (const auto& arg : command_line.argv()) { if (arg.size() == arraysize(kPrefetchArgumentPrefix) && base::StartsWith(arg, kPrefetchArgumentPrefix, base::CompareCase::SENSITIVE)) { return arg[arraysize(kPrefetchArgumentPrefix) - 1] >= L'1' && arg[arraysize(kPrefetchArgumentPrefix) - 1] <= L'8'; } } return false; } } // namespace // This helper is looked up in the browser to retrieve the crash reports. See // CrashUploadListCrashpad. Note that we do not pass an std::vector here, // because we do not want to allocate/free in different modules. The returned // pointer is read-only. extern "C" __declspec(dllexport) void GetUploadedReportsImpl( const crash_reporter::UploadedReport** reports, size_t* report_count) { crash_reporter::GetUploadedReports(g_uploaded_reports.Pointer()); *reports = g_uploaded_reports.Pointer()->data(); *report_count = g_uploaded_reports.Pointer()->size(); } #if !defined(WIN_CONSOLE_APP) int APIENTRY wWinMain(HINSTANCE instance, HINSTANCE prev, wchar_t*, int) { #else int main() { HINSTANCE instance = GetModuleHandle(nullptr); #endif // Initialize the CommandLine singleton from the environment. base::CommandLine::Init(0, nullptr); const base::CommandLine* command_line = base::CommandLine::ForCurrentProcess(); const std::string process_type = command_line->GetSwitchValueASCII(switches::kProcessType); startup_metric_utils::InitializePreReadOptions( BrowserDistribution::GetDistribution()->GetRegistryPath()); // Confirm that an explicit prefetch profile is used for all process types // except for the browser process. Any new process type will have to assign // itself a prefetch id. See kPrefetchArgument* constants in // content_switches.cc for details. DCHECK(!startup_metric_utils::GetPreReadOptions().use_prefetch_argument || process_type.empty() || HasValidWindowsPrefetchArgument(*command_line)); if (process_type == crash_reporter::switches::kCrashpadHandler) { return crash_reporter::RunAsCrashpadHandler( *base::CommandLine::ForCurrentProcess()); } crash_reporter::SetCrashReporterClient(g_chrome_crash_client.Pointer()); crash_reporter::InitializeCrashpadWithEmbeddedHandler(process_type.empty(), process_type); SwitchToLFHeap(); startup_metric_utils::RecordExeMainEntryPointTime(base::Time::Now()); // Signal Chrome Elf that Chrome has begun to start. SignalChromeElf(); // The exit manager is in charge of calling the dtors of singletons. base::AtExitManager exit_manager; // We don't want to set DPI awareness on pre-Win7 because we don't support // DirectWrite there. GDI fonts are kerned very badly, so better to leave // DPI-unaware and at effective 1.0. See also ShouldUseDirectWrite(). if (base::win::GetVersion() >= base::win::VERSION_WIN7) EnableHighDPISupport(); if (AttemptFastNotify(*command_line)) return 0; // Load and launch the chrome dll. *Everything* happens inside. VLOG(1) << "About to load main DLL."; MainDllLoader* loader = MakeMainDllLoader(); int rc = loader->Launch(instance); loader->RelaunchChromeBrowserWithNewCommandLineIfNeeded(); delete loader; return rc; }