// Copyright (c) 2012 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 "chrome/common/child_process_logging.h" #include #include "base/command_line.h" #include "base/string_util.h" #include "base/stringprintf.h" #include "base/strings/string_number_conversions.h" #include "base/utf_string_conversions.h" #include "chrome/common/chrome_constants.h" #include "chrome/common/metrics/variations/variations_util.h" #include "chrome/installer/util/google_update_settings.h" #include "content/public/common/gpu_info.h" #include "googleurl/src/gurl.h" namespace child_process_logging { namespace { // exported in breakpad_win.cc: void __declspec(dllexport) __cdecl SetActiveURL. typedef void (__cdecl *MainSetActiveURL)(const wchar_t*); // exported in breakpad_win.cc: void __declspec(dllexport) __cdecl SetClientId. typedef void (__cdecl *MainSetClientId)(const wchar_t*); // exported in breakpad_win.cc: // void __declspec(dllexport) __cdecl SetNumberOfExtensions. typedef void (__cdecl *MainSetNumberOfExtensions)(int); // exported in breakpad_win.cc: // void __declspec(dllexport) __cdecl SetExtensionID. typedef void (__cdecl *MainSetExtensionID)(size_t, const wchar_t*); // exported in breakpad_win.cc: void __declspec(dllexport) __cdecl SetGpuInfo. typedef void (__cdecl *MainSetGpuInfo)(const wchar_t*, const wchar_t*, const wchar_t*, const wchar_t*, const wchar_t*); // exported in breakpad_win.cc: // void __declspec(dllexport) __cdecl SetPrinterInfo. typedef void (__cdecl *MainSetPrinterInfo)(const wchar_t*); // exported in breakpad_win.cc: // void __declspec(dllexport) __cdecl SetNumberOfViews. typedef void (__cdecl *MainSetNumberOfViews)(int); // exported in breakpad_win.cc: // void __declspec(dllexport) __cdecl SetCommandLine2 typedef void (__cdecl *MainSetCommandLine)(const wchar_t**, size_t); // exported in breakpad_field_trial_win.cc: // void __declspec(dllexport) __cdecl SetExperimentList3 typedef void (__cdecl *MainSetExperimentList)(const wchar_t**, size_t, size_t); // Copied from breakpad_win.cc. void StringVectorToCStringVector(const std::vector& wstrings, std::vector* cstrings) { cstrings->clear(); cstrings->reserve(wstrings.size()); for (size_t i = 0; i < wstrings.size(); ++i) cstrings->push_back(wstrings[i].c_str()); } } // namespace void SetActiveURL(const GURL& url) { static MainSetActiveURL set_active_url = NULL; // note: benign race condition on set_active_url. if (!set_active_url) { HMODULE exe_module = GetModuleHandle(chrome::kBrowserProcessExecutableName); if (!exe_module) return; set_active_url = reinterpret_cast( GetProcAddress(exe_module, "SetActiveURL")); if (!set_active_url) return; } (set_active_url)(UTF8ToWide(url.possibly_invalid_spec()).c_str()); } void SetClientId(const std::string& client_id) { std::string str(client_id); // Remove all instance of '-' char from the GUID. So BCD-WXY becomes BCDWXY. ReplaceSubstringsAfterOffset(&str, 0, "-", ""); if (str.empty()) return; std::wstring wstr = ASCIIToWide(str); std::wstring old_wstr; if (!GoogleUpdateSettings::GetMetricsId(&old_wstr) || wstr != old_wstr) GoogleUpdateSettings::SetMetricsId(wstr); static MainSetClientId set_client_id = NULL; // note: benign race condition on set_client_id. if (!set_client_id) { HMODULE exe_module = GetModuleHandle(chrome::kBrowserProcessExecutableName); if (!exe_module) return; set_client_id = reinterpret_cast( GetProcAddress(exe_module, "SetClientId")); if (!set_client_id) return; } (set_client_id)(wstr.c_str()); } std::string GetClientId() { std::wstring wstr_client_id; if (GoogleUpdateSettings::GetMetricsId(&wstr_client_id)) return WideToASCII(wstr_client_id); else return std::string(); } void SetActiveExtensions(const std::set& extension_ids) { static MainSetNumberOfExtensions set_number_of_extensions = NULL; // note: benign race condition on set_number_of_extensions. if (!set_number_of_extensions) { HMODULE exe_module = GetModuleHandle(chrome::kBrowserProcessExecutableName); if (!exe_module) return; set_number_of_extensions = reinterpret_cast( GetProcAddress(exe_module, "SetNumberOfExtensions")); if (!set_number_of_extensions) return; } static MainSetExtensionID set_extension_id = NULL; // note: benign race condition on set_extension_id. if (!set_extension_id) { HMODULE exe_module = GetModuleHandle(chrome::kBrowserProcessExecutableName); if (!exe_module) return; set_extension_id = reinterpret_cast( GetProcAddress(exe_module, "SetExtensionID")); if (!set_extension_id) return; } (set_number_of_extensions)(static_cast(extension_ids.size())); std::set::const_iterator iter = extension_ids.begin(); for (size_t i = 0; i < kMaxReportedActiveExtensions; ++i) { if (iter != extension_ids.end()) { (set_extension_id)(i, ASCIIToWide(iter->c_str()).c_str()); ++iter; } else { (set_extension_id)(i, L""); } } } void SetGpuInfo(const content::GPUInfo& gpu_info) { static MainSetGpuInfo set_gpu_info = NULL; // note: benign race condition on set_gpu_info. if (!set_gpu_info) { HMODULE exe_module = GetModuleHandle(chrome::kBrowserProcessExecutableName); if (!exe_module) return; set_gpu_info = reinterpret_cast( GetProcAddress(exe_module, "SetGpuInfo")); if (!set_gpu_info) return; } (set_gpu_info)( base::StringPrintf(L"0x%04x", gpu_info.gpu.vendor_id).c_str(), base::StringPrintf(L"0x%04x", gpu_info.gpu.device_id).c_str(), UTF8ToUTF16(gpu_info.driver_version).c_str(), UTF8ToUTF16(gpu_info.pixel_shader_version).c_str(), UTF8ToUTF16(gpu_info.vertex_shader_version).c_str()); } void SetPrinterInfo(const char* printer_info) { static MainSetPrinterInfo set_printer_info = NULL; // note: benign race condition on set_printer_info. if (!set_printer_info) { HMODULE exe_module = GetModuleHandle(chrome::kBrowserProcessExecutableName); if (!exe_module) return; set_printer_info = reinterpret_cast( GetProcAddress(exe_module, "SetPrinterInfo")); if (!set_printer_info) return; } (set_printer_info)(UTF8ToWide(printer_info).c_str()); } void SetCommandLine(const CommandLine* command_line) { static MainSetCommandLine set_command_line = NULL; // note: benign race condition on set_command_line. if (!set_command_line) { HMODULE exe_module = GetModuleHandle(chrome::kBrowserProcessExecutableName); if (!exe_module) return; set_command_line = reinterpret_cast( GetProcAddress(exe_module, "SetCommandLine2")); if (!set_command_line) return; } if (command_line->argv().empty()) return; std::vector cstrings; StringVectorToCStringVector(command_line->argv(), &cstrings); (set_command_line)(&cstrings[0], cstrings.size()); } void SetExperimentList(const std::vector& experiments) { static MainSetExperimentList set_experiment_list = NULL; // note: benign race condition on set_experiment_list. if (!set_experiment_list) { HMODULE exe_module = GetModuleHandle(chrome::kBrowserProcessExecutableName); if (!exe_module) return; set_experiment_list = reinterpret_cast( GetProcAddress(exe_module, "SetExperimentList3")); if (!set_experiment_list) return; } std::vector chunks; chrome_variations::GenerateVariationChunks(experiments, &chunks); // If the list is empty, notify the child process of the number of experiments // and exit early. if (chunks.empty()) { (set_experiment_list)(NULL, 0, 0); return; } std::vector cstrings; StringVectorToCStringVector(chunks, &cstrings); (set_experiment_list)(&cstrings[0], cstrings.size(), experiments.size()); } void SetNumberOfViews(int number_of_views) { static MainSetNumberOfViews set_number_of_views = NULL; // note: benign race condition on set_number_of_views. if (!set_number_of_views) { HMODULE exe_module = GetModuleHandle(chrome::kBrowserProcessExecutableName); if (!exe_module) return; set_number_of_views = reinterpret_cast( GetProcAddress(exe_module, "SetNumberOfViews")); if (!set_number_of_views) return; } (set_number_of_views)(number_of_views); } } // namespace child_process_logging