ozone: evdev: Sync caps lock LED state to evdev
[chromium-blink-merge.git] / chrome / installer / util / google_update_settings.cc
blobb3467bc2efc40ea6db3e98acd6c7a9e70c3efa5e
1 // Copyright (c) 2012 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.
5 #include "chrome/installer/util/google_update_settings.h"
7 #include <algorithm>
8 #include <limits>
10 #include "base/command_line.h"
11 #include "base/files/file_path.h"
12 #include "base/logging.h"
13 #include "base/metrics/histogram.h"
14 #include "base/path_service.h"
15 #include "base/strings/string_number_conversions.h"
16 #include "base/strings/string_util.h"
17 #include "base/strings/utf_string_conversions.h"
18 #include "base/threading/thread_restrictions.h"
19 #include "base/time/time.h"
20 #include "base/win/registry.h"
21 #include "base/win/win_util.h"
22 #include "chrome/common/chrome_switches.h"
23 #include "chrome/installer/util/app_registration_data.h"
24 #include "chrome/installer/util/browser_distribution.h"
25 #include "chrome/installer/util/channel_info.h"
26 #include "chrome/installer/util/google_update_constants.h"
27 #include "chrome/installer/util/google_update_experiment_util.h"
28 #include "chrome/installer/util/install_util.h"
29 #include "chrome/installer/util/installation_state.h"
30 #include "chrome/installer/util/product.h"
32 using base::win::RegKey;
33 using installer::InstallationState;
35 const wchar_t GoogleUpdateSettings::kPoliciesKey[] =
36 L"SOFTWARE\\Policies\\Google\\Update";
37 const wchar_t GoogleUpdateSettings::kUpdatePolicyValue[] = L"UpdateDefault";
38 const wchar_t GoogleUpdateSettings::kUpdateOverrideValuePrefix[] = L"Update";
39 const wchar_t GoogleUpdateSettings::kCheckPeriodOverrideMinutes[] =
40 L"AutoUpdateCheckPeriodMinutes";
42 // Don't allow update periods longer than six weeks.
43 const int GoogleUpdateSettings::kCheckPeriodOverrideMinutesMax =
44 60 * 24 * 7 * 6;
46 const GoogleUpdateSettings::UpdatePolicy
47 GoogleUpdateSettings::kDefaultUpdatePolicy =
48 #if defined(GOOGLE_CHROME_BUILD)
49 GoogleUpdateSettings::AUTOMATIC_UPDATES;
50 #else
51 GoogleUpdateSettings::UPDATES_DISABLED;
52 #endif
54 namespace {
56 bool ReadGoogleUpdateStrKey(const wchar_t* const name, base::string16* value) {
57 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
58 base::string16 reg_path = dist->GetStateKey();
59 RegKey key(HKEY_CURRENT_USER, reg_path.c_str(), KEY_READ | KEY_WOW64_32KEY);
60 if (key.ReadValue(name, value) != ERROR_SUCCESS) {
61 RegKey hklm_key(
62 HKEY_LOCAL_MACHINE, reg_path.c_str(), KEY_READ | KEY_WOW64_32KEY);
63 return (hklm_key.ReadValue(name, value) == ERROR_SUCCESS);
65 return true;
68 // Writes |value| into a user-specific value in the key |name| under
69 // |app_reg_data|'s ClientStateMedium key in HKLM along with the aggregation
70 // method |aggregate|. This function is solely for use by system-level installs.
71 bool WriteGoogleUpdateAggregateNumKeyInternal(
72 const AppRegistrationData& app_reg_data,
73 const wchar_t* const name,
74 size_t value,
75 const wchar_t* const aggregate) {
76 DCHECK(aggregate);
77 DCHECK(GoogleUpdateSettings::IsSystemInstall());
78 const REGSAM kAccess = KEY_SET_VALUE | KEY_WOW64_32KEY;
80 // Machine installs require each OS user to write a unique key under a
81 // named key in HKLM as well as an "aggregation" function that describes
82 // how the values of multiple users are to be combined.
83 base::string16 uniquename;
84 if (!base::win::GetUserSidString(&uniquename)) {
85 NOTREACHED();
86 return false;
89 base::string16 reg_path(app_reg_data.GetStateMediumKey());
90 reg_path.append(L"\\");
91 reg_path.append(name);
92 RegKey key(HKEY_LOCAL_MACHINE, reg_path.c_str(), kAccess);
93 key.WriteValue(google_update::kRegAggregateMethod, aggregate);
95 DWORD dword_value = (value > std::numeric_limits<DWORD>::max() ?
96 std::numeric_limits<DWORD>::max() :
97 static_cast<DWORD>(value));
98 return (key.WriteValue(uniquename.c_str(), dword_value) == ERROR_SUCCESS);
101 // Updates a registry key |name| to be |value| for the given |app_reg_data|.
102 bool WriteGoogleUpdateStrKeyInternal(const AppRegistrationData& app_reg_data,
103 const wchar_t* const name,
104 const base::string16& value) {
105 const REGSAM kAccess = KEY_SET_VALUE | KEY_WOW64_32KEY;
106 RegKey key(HKEY_CURRENT_USER, app_reg_data.GetStateKey().c_str(), kAccess);
107 return (key.WriteValue(name, value.c_str()) == ERROR_SUCCESS);
110 bool WriteGoogleUpdateStrKey(const wchar_t* const name,
111 const base::string16& value) {
112 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
113 return WriteGoogleUpdateStrKeyInternal(
114 dist->GetAppRegistrationData(), name, value);
117 bool ClearGoogleUpdateStrKey(const wchar_t* const name) {
118 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
119 base::string16 reg_path = dist->GetStateKey();
120 RegKey key(HKEY_CURRENT_USER,
121 reg_path.c_str(),
122 KEY_READ | KEY_WRITE | KEY_WOW64_32KEY);
123 base::string16 value;
124 if (key.ReadValue(name, &value) != ERROR_SUCCESS)
125 return false;
126 return (key.WriteValue(name, L"") == ERROR_SUCCESS);
129 bool RemoveGoogleUpdateStrKey(const wchar_t* const name) {
130 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
131 base::string16 reg_path = dist->GetStateKey();
132 RegKey key(HKEY_CURRENT_USER,
133 reg_path.c_str(),
134 KEY_READ | KEY_WRITE | KEY_WOW64_32KEY);
135 if (!key.HasValue(name))
136 return true;
137 return (key.DeleteValue(name) == ERROR_SUCCESS);
140 bool GetChromeChannelInternal(bool system_install,
141 bool add_multi_modifier,
142 base::string16* channel) {
143 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
145 // Shortcut in case this distribution knows what channel it is (canary).
146 if (dist->GetChromeChannel(channel))
147 return true;
149 // Determine whether or not chrome is multi-install. If so, updates are
150 // delivered under the binaries' app guid, so that's where the relevant
151 // channel is found.
152 installer::ProductState state;
153 installer::ChannelInfo channel_info;
154 ignore_result(state.Initialize(system_install, dist));
155 if (!state.is_multi_install()) {
156 // Use the channel info that was just read for this single-install chrome.
157 channel_info = state.channel();
158 } else {
159 // Read the channel info from the binaries' state key.
160 HKEY root_key = system_install ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER;
161 dist = BrowserDistribution::GetSpecificDistribution(
162 BrowserDistribution::CHROME_BINARIES);
163 RegKey key(root_key, dist->GetStateKey().c_str(),
164 KEY_READ | KEY_WOW64_32KEY);
166 if (!channel_info.Initialize(key)) {
167 channel->assign(installer::kChromeChannelUnknown);
168 return false;
172 if (!channel_info.GetChannelName(channel))
173 channel->assign(installer::kChromeChannelUnknown);
175 // Tag the channel name if this is a multi-install.
176 if (add_multi_modifier && state.is_multi_install()) {
177 if (!channel->empty())
178 channel->push_back(L'-');
179 channel->push_back(L'm');
182 return true;
185 // Populates |update_policy| with the UpdatePolicy enum value corresponding to a
186 // DWORD read from the registry and returns true if |value| is within range.
187 // If |value| is out of range, returns false without modifying |update_policy|.
188 bool GetUpdatePolicyFromDword(
189 const DWORD value,
190 GoogleUpdateSettings::UpdatePolicy* update_policy) {
191 switch (value) {
192 case GoogleUpdateSettings::UPDATES_DISABLED:
193 case GoogleUpdateSettings::AUTOMATIC_UPDATES:
194 case GoogleUpdateSettings::MANUAL_UPDATES_ONLY:
195 case GoogleUpdateSettings::AUTO_UPDATES_ONLY:
196 *update_policy = static_cast<GoogleUpdateSettings::UpdatePolicy>(value);
197 return true;
198 default:
199 LOG(WARNING) << "Unexpected update policy override value: " << value;
201 return false;
204 // Convenience routine: GoogleUpdateSettings::UpdateDidRunStateForApp()
205 // specialized for Chrome Binaries.
206 bool UpdateDidRunStateForBinaries(bool did_run) {
207 BrowserDistribution* dist = BrowserDistribution::GetSpecificDistribution(
208 BrowserDistribution::CHROME_BINARIES);
209 return GoogleUpdateSettings::UpdateDidRunStateForApp(
210 dist->GetAppRegistrationData(), did_run);
213 } // namespace
215 bool GoogleUpdateSettings::IsSystemInstall() {
216 bool system_install = false;
217 base::FilePath module_dir;
218 if (!PathService::Get(base::DIR_MODULE, &module_dir)) {
219 LOG(WARNING)
220 << "Failed to get directory of module; assuming per-user install.";
221 } else {
222 system_install = !InstallUtil::IsPerUserInstall(module_dir);
224 return system_install;
227 bool GoogleUpdateSettings::GetCollectStatsConsent() {
228 return GetCollectStatsConsentAtLevel(IsSystemInstall());
231 // Older versions of Chrome unconditionally read from HKCU\...\ClientState\...
232 // and then HKLM\...\ClientState\.... This means that system-level Chrome
233 // never checked ClientStateMedium (which has priority according to Google
234 // Update) and gave preference to a value in HKCU (which was never checked by
235 // Google Update). From now on, Chrome follows Google Update's policy.
236 bool GoogleUpdateSettings::GetCollectStatsConsentAtLevel(bool system_install) {
237 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
239 // Consent applies to all products in a multi-install package.
240 if (InstallUtil::IsMultiInstall(dist, system_install)) {
241 dist = BrowserDistribution::GetSpecificDistribution(
242 BrowserDistribution::CHROME_BINARIES);
245 RegKey key;
246 DWORD value = 0;
247 bool have_value = false;
248 const REGSAM kAccess = KEY_QUERY_VALUE | KEY_WOW64_32KEY;
250 // For system-level installs, try ClientStateMedium first.
251 have_value =
252 system_install &&
253 key.Open(HKEY_LOCAL_MACHINE, dist->GetStateMediumKey().c_str(),
254 kAccess) == ERROR_SUCCESS &&
255 key.ReadValueDW(google_update::kRegUsageStatsField,
256 &value) == ERROR_SUCCESS;
258 // Otherwise, try ClientState.
259 if (!have_value) {
260 have_value =
261 key.Open(system_install ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER,
262 dist->GetStateKey().c_str(),
263 kAccess) == ERROR_SUCCESS &&
264 key.ReadValueDW(google_update::kRegUsageStatsField,
265 &value) == ERROR_SUCCESS;
268 // Google Update specifically checks that the value is 1, so we do the same.
269 return have_value && value == 1;
272 bool GoogleUpdateSettings::SetCollectStatsConsent(bool consented) {
273 return SetCollectStatsConsentAtLevel(IsSystemInstall(), consented);
276 bool GoogleUpdateSettings::SetCollectStatsConsentAtLevel(bool system_install,
277 bool consented) {
278 // Google Update writes and expects 1 for true, 0 for false.
279 DWORD value = consented ? 1 : 0;
281 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
283 // Consent applies to all products in a multi-install package.
284 if (InstallUtil::IsMultiInstall(dist, system_install)) {
285 dist = BrowserDistribution::GetSpecificDistribution(
286 BrowserDistribution::CHROME_BINARIES);
289 // Write to ClientStateMedium for system-level; ClientState otherwise.
290 HKEY root_key = system_install ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER;
291 base::string16 reg_path =
292 system_install ? dist->GetStateMediumKey() : dist->GetStateKey();
293 RegKey key;
294 LONG result = key.Create(
295 root_key, reg_path.c_str(), KEY_SET_VALUE | KEY_WOW64_32KEY);
296 if (result != ERROR_SUCCESS) {
297 LOG(ERROR) << "Failed opening key " << reg_path << " to set "
298 << google_update::kRegUsageStatsField << "; result: " << result;
299 } else {
300 result = key.WriteValue(google_update::kRegUsageStatsField, value);
301 LOG_IF(ERROR, result != ERROR_SUCCESS) << "Failed setting "
302 << google_update::kRegUsageStatsField << " in key " << reg_path
303 << "; result: " << result;
305 return (result == ERROR_SUCCESS);
308 scoped_ptr<metrics::ClientInfo> GoogleUpdateSettings::LoadMetricsClientInfo() {
309 base::string16 client_id_16;
310 if (!ReadGoogleUpdateStrKey(google_update::kRegMetricsId, &client_id_16) ||
311 client_id_16.empty()) {
312 return scoped_ptr<metrics::ClientInfo>();
315 scoped_ptr<metrics::ClientInfo> client_info(new metrics::ClientInfo);
316 client_info->client_id = base::UTF16ToUTF8(client_id_16);
318 base::string16 installation_date_str;
319 if (ReadGoogleUpdateStrKey(google_update::kRegMetricsIdInstallDate,
320 &installation_date_str)) {
321 base::StringToInt64(installation_date_str, &client_info->installation_date);
324 base::string16 reporting_enbaled_date_date_str;
325 if (ReadGoogleUpdateStrKey(google_update::kRegMetricsIdEnabledDate,
326 &reporting_enbaled_date_date_str)) {
327 base::StringToInt64(reporting_enbaled_date_date_str,
328 &client_info->reporting_enabled_date);
331 return client_info.Pass();
334 void GoogleUpdateSettings::StoreMetricsClientInfo(
335 const metrics::ClientInfo& client_info) {
336 // Attempt a best-effort at backing |client_info| in the registry (but don't
337 // handle/report failures).
338 WriteGoogleUpdateStrKey(google_update::kRegMetricsId,
339 base::UTF8ToUTF16(client_info.client_id));
340 WriteGoogleUpdateStrKey(google_update::kRegMetricsIdInstallDate,
341 base::Int64ToString16(client_info.installation_date));
342 WriteGoogleUpdateStrKey(
343 google_update::kRegMetricsIdEnabledDate,
344 base::Int64ToString16(client_info.reporting_enabled_date));
347 // EULA consent is only relevant for system-level installs.
348 bool GoogleUpdateSettings::SetEULAConsent(
349 const InstallationState& machine_state,
350 BrowserDistribution* dist,
351 bool consented) {
352 DCHECK(dist);
353 const DWORD eula_accepted = consented ? 1 : 0;
354 const REGSAM kAccess = KEY_SET_VALUE | KEY_WOW64_32KEY;
355 base::string16 reg_path = dist->GetStateMediumKey();
356 bool succeeded = true;
357 RegKey key;
359 // Write the consent value into the product's ClientStateMedium key.
360 if (key.Create(HKEY_LOCAL_MACHINE, reg_path.c_str(),
361 kAccess) != ERROR_SUCCESS ||
362 key.WriteValue(google_update::kRegEULAAceptedField,
363 eula_accepted) != ERROR_SUCCESS) {
364 succeeded = false;
367 // If this is a multi-install, also write it into the binaries' key.
368 // --mutli-install is not provided on the command-line, so deduce it from
369 // the product's state.
370 const installer::ProductState* product_state =
371 machine_state.GetProductState(true, dist->GetType());
372 if (product_state != NULL && product_state->is_multi_install()) {
373 dist = BrowserDistribution::GetSpecificDistribution(
374 BrowserDistribution::CHROME_BINARIES);
375 reg_path = dist->GetStateMediumKey();
376 if (key.Create(HKEY_LOCAL_MACHINE, reg_path.c_str(),
377 kAccess) != ERROR_SUCCESS ||
378 key.WriteValue(google_update::kRegEULAAceptedField,
379 eula_accepted) != ERROR_SUCCESS) {
380 succeeded = false;
384 return succeeded;
387 int GoogleUpdateSettings::GetLastRunTime() {
388 base::string16 time_s;
389 if (!ReadGoogleUpdateStrKey(google_update::kRegLastRunTimeField, &time_s))
390 return -1;
391 int64 time_i;
392 if (!base::StringToInt64(time_s, &time_i))
393 return -1;
394 base::TimeDelta td =
395 base::Time::NowFromSystemTime() - base::Time::FromInternalValue(time_i);
396 return td.InDays();
399 bool GoogleUpdateSettings::SetLastRunTime() {
400 int64 time = base::Time::NowFromSystemTime().ToInternalValue();
401 return WriteGoogleUpdateStrKey(google_update::kRegLastRunTimeField,
402 base::Int64ToString16(time));
405 bool GoogleUpdateSettings::RemoveLastRunTime() {
406 return RemoveGoogleUpdateStrKey(google_update::kRegLastRunTimeField);
409 bool GoogleUpdateSettings::GetBrowser(base::string16* browser) {
410 return ReadGoogleUpdateStrKey(google_update::kRegBrowserField, browser);
413 bool GoogleUpdateSettings::GetLanguage(base::string16* language) {
414 return ReadGoogleUpdateStrKey(google_update::kRegLangField, language);
417 bool GoogleUpdateSettings::GetBrand(base::string16* brand) {
418 return ReadGoogleUpdateStrKey(google_update::kRegRLZBrandField, brand);
421 bool GoogleUpdateSettings::GetReactivationBrand(base::string16* brand) {
422 return ReadGoogleUpdateStrKey(google_update::kRegRLZReactivationBrandField,
423 brand);
426 bool GoogleUpdateSettings::GetClient(base::string16* client) {
427 return ReadGoogleUpdateStrKey(google_update::kRegClientField, client);
430 bool GoogleUpdateSettings::SetClient(const base::string16& client) {
431 return WriteGoogleUpdateStrKey(google_update::kRegClientField, client);
434 bool GoogleUpdateSettings::GetReferral(base::string16* referral) {
435 return ReadGoogleUpdateStrKey(google_update::kRegReferralField, referral);
438 bool GoogleUpdateSettings::ClearReferral() {
439 return ClearGoogleUpdateStrKey(google_update::kRegReferralField);
442 bool GoogleUpdateSettings::UpdateDidRunStateForApp(
443 const AppRegistrationData& app_reg_data,
444 bool did_run) {
445 return WriteGoogleUpdateStrKeyInternal(app_reg_data,
446 google_update::kRegDidRunField,
447 did_run ? L"1" : L"0");
450 bool GoogleUpdateSettings::UpdateDidRunState(bool did_run, bool system_level) {
451 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
452 bool result = UpdateDidRunStateForApp(dist->GetAppRegistrationData(),
453 did_run);
454 // Update state for binaries, even if the previous call was unsuccessful.
455 if (InstallUtil::IsMultiInstall(dist, system_level))
456 result = UpdateDidRunStateForBinaries(did_run) && result;
457 return result;
460 base::string16 GoogleUpdateSettings::GetChromeChannel(bool system_install) {
461 base::string16 channel;
462 GetChromeChannelInternal(system_install, false, &channel);
463 return channel;
466 bool GoogleUpdateSettings::GetChromeChannelAndModifiers(
467 bool system_install,
468 base::string16* channel) {
469 return GetChromeChannelInternal(system_install, true, channel);
472 void GoogleUpdateSettings::UpdateInstallStatus(bool system_install,
473 installer::ArchiveType archive_type, int install_return_code,
474 const base::string16& product_guid) {
475 DCHECK(archive_type != installer::UNKNOWN_ARCHIVE_TYPE ||
476 install_return_code != 0);
477 HKEY reg_root = (system_install) ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER;
479 RegKey key;
480 installer::ChannelInfo channel_info;
481 base::string16 reg_key(google_update::kRegPathClientState);
482 reg_key.append(L"\\");
483 reg_key.append(product_guid);
484 LONG result = key.Open(reg_root,
485 reg_key.c_str(),
486 KEY_QUERY_VALUE | KEY_SET_VALUE | KEY_WOW64_32KEY);
487 if (result == ERROR_SUCCESS)
488 channel_info.Initialize(key);
489 else if (result != ERROR_FILE_NOT_FOUND)
490 LOG(ERROR) << "Failed to open " << reg_key << "; Error: " << result;
492 if (UpdateGoogleUpdateApKey(archive_type, install_return_code,
493 &channel_info)) {
494 // We have a modified channel_info value to write.
495 // Create the app's ClientState key if it doesn't already exist.
496 if (!key.Valid()) {
497 result = key.Open(reg_root,
498 google_update::kRegPathClientState,
499 KEY_CREATE_SUB_KEY | KEY_WOW64_32KEY);
500 if (result == ERROR_SUCCESS)
501 result = key.CreateKey(product_guid.c_str(),
502 KEY_SET_VALUE | KEY_WOW64_32KEY);
504 if (result != ERROR_SUCCESS) {
505 LOG(ERROR) << "Failed to create " << reg_key << "; Error: " << result;
506 return;
509 if (!channel_info.Write(&key)) {
510 LOG(ERROR) << "Failed to write to application's ClientState key "
511 << google_update::kRegApField << " = " << channel_info.value();
516 bool GoogleUpdateSettings::UpdateGoogleUpdateApKey(
517 installer::ArchiveType archive_type, int install_return_code,
518 installer::ChannelInfo* value) {
519 DCHECK(archive_type != installer::UNKNOWN_ARCHIVE_TYPE ||
520 install_return_code != 0);
521 bool modified = false;
523 if (archive_type == installer::FULL_ARCHIVE_TYPE || !install_return_code) {
524 if (value->SetFullSuffix(false)) {
525 VLOG(1) << "Removed incremental installer failure key; "
526 "switching to channel: "
527 << value->value();
528 modified = true;
530 } else if (archive_type == installer::INCREMENTAL_ARCHIVE_TYPE) {
531 if (value->SetFullSuffix(true)) {
532 VLOG(1) << "Incremental installer failed; switching to channel: "
533 << value->value();
534 modified = true;
535 } else {
536 VLOG(1) << "Incremental installer failure; already on channel: "
537 << value->value();
539 } else {
540 // It's okay if we don't know the archive type. In this case, leave the
541 // "-full" suffix as we found it.
542 DCHECK_EQ(installer::UNKNOWN_ARCHIVE_TYPE, archive_type);
545 if (value->SetMultiFailSuffix(false)) {
546 VLOG(1) << "Removed multi-install failure key; switching to channel: "
547 << value->value();
548 modified = true;
551 return modified;
554 void GoogleUpdateSettings::UpdateProfileCounts(size_t profiles_active,
555 size_t profiles_signedin) {
556 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
557 // System-level installs must write into the ClientStateMedium key shared by
558 // all users. Special treatment is used to aggregate across those users.
559 if (IsSystemInstall()) {
560 // Write the counts as ints that get aggregated across all users via
561 // summation for system-level installs.
562 WriteGoogleUpdateAggregateNumKeyInternal(
563 dist->GetAppRegistrationData(),
564 google_update::kRegProfilesActive,
565 profiles_active,
566 L"sum()");
567 WriteGoogleUpdateAggregateNumKeyInternal(
568 dist->GetAppRegistrationData(),
569 google_update::kRegProfilesSignedIn,
570 profiles_signedin,
571 L"sum()");
572 } else {
573 // Write the counts as strings since no aggregation function is needed for
574 // user-level installs.
575 WriteGoogleUpdateStrKeyInternal(dist->GetAppRegistrationData(),
576 google_update::kRegProfilesActive,
577 base::SizeTToString16(profiles_active));
578 WriteGoogleUpdateStrKeyInternal(dist->GetAppRegistrationData(),
579 google_update::kRegProfilesSignedIn,
580 base::SizeTToString16(profiles_signedin));
584 int GoogleUpdateSettings::DuplicateGoogleUpdateSystemClientKey() {
585 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
586 base::string16 reg_path = dist->GetStateKey();
588 // Minimum access needed is to be able to write to this key.
589 RegKey reg_key(
590 HKEY_LOCAL_MACHINE, reg_path.c_str(), KEY_SET_VALUE | KEY_WOW64_32KEY);
591 if (!reg_key.Valid())
592 return 0;
594 HANDLE target_handle = 0;
595 if (!DuplicateHandle(GetCurrentProcess(), reg_key.Handle(),
596 GetCurrentProcess(), &target_handle, KEY_SET_VALUE,
597 TRUE, DUPLICATE_SAME_ACCESS)) {
598 return 0;
600 return reinterpret_cast<int>(target_handle);
603 bool GoogleUpdateSettings::WriteGoogleUpdateSystemClientKey(
604 int handle, const base::string16& key, const base::string16& value) {
605 HKEY reg_key = reinterpret_cast<HKEY>(reinterpret_cast<void*>(handle));
606 DWORD size = static_cast<DWORD>(value.size()) * sizeof(wchar_t);
607 LSTATUS status = RegSetValueEx(reg_key, key.c_str(), 0, REG_SZ,
608 reinterpret_cast<const BYTE*>(value.c_str()), size);
609 return status == ERROR_SUCCESS;
612 GoogleUpdateSettings::UpdatePolicy GoogleUpdateSettings::GetAppUpdatePolicy(
613 const base::string16& app_guid,
614 bool* is_overridden) {
615 bool found_override = false;
616 UpdatePolicy update_policy = kDefaultUpdatePolicy;
618 #if defined(GOOGLE_CHROME_BUILD)
619 DCHECK(!app_guid.empty());
620 RegKey policy_key;
622 // Google Update Group Policy settings are always in HKLM.
623 // TODO(wfh): Check if policies should go into Wow6432Node or not.
624 if (policy_key.Open(HKEY_LOCAL_MACHINE, kPoliciesKey, KEY_QUERY_VALUE) ==
625 ERROR_SUCCESS) {
626 DWORD value = 0;
627 base::string16 app_update_override(kUpdateOverrideValuePrefix);
628 app_update_override.append(app_guid);
629 // First try to read and comprehend the app-specific override.
630 found_override = (policy_key.ReadValueDW(app_update_override.c_str(),
631 &value) == ERROR_SUCCESS &&
632 GetUpdatePolicyFromDword(value, &update_policy));
634 // Failing that, try to read and comprehend the default override.
635 if (!found_override &&
636 policy_key.ReadValueDW(kUpdatePolicyValue, &value) == ERROR_SUCCESS) {
637 GetUpdatePolicyFromDword(value, &update_policy);
640 #endif // defined(GOOGLE_CHROME_BUILD)
642 if (is_overridden != NULL)
643 *is_overridden = found_override;
645 return update_policy;
648 // static
649 bool GoogleUpdateSettings::AreAutoupdatesEnabled() {
650 #if defined(GOOGLE_CHROME_BUILD)
651 // Check the auto-update check period override. If it is 0 or exceeds the
652 // maximum timeout, then for all intents and purposes auto updates are
653 // disabled.
654 RegKey policy_key;
655 DWORD value = 0;
656 if (policy_key.Open(HKEY_LOCAL_MACHINE, kPoliciesKey,
657 KEY_QUERY_VALUE) == ERROR_SUCCESS &&
658 policy_key.ReadValueDW(kCheckPeriodOverrideMinutes,
659 &value) == ERROR_SUCCESS &&
660 (value == 0 || value > kCheckPeriodOverrideMinutesMax)) {
661 return false;
664 // Auto updates are subtly broken when Chrome and the binaries have different
665 // overrides in place. If this Chrome cannot possibly be multi-install by
666 // virtue of being a side-by-side installation, simply check Chrome's policy.
667 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
668 UpdatePolicy app_policy = GetAppUpdatePolicy(dist->GetAppGuid(), nullptr);
669 if (InstallUtil::IsChromeSxSProcess())
670 return app_policy == AUTOMATIC_UPDATES || app_policy == AUTO_UPDATES_ONLY;
672 // Otherwise, check for consistency between Chrome and the binaries regardless
673 // of whether or not this Chrome is multi-install since the next update likely
674 // will attempt to migrate it to such.
675 BrowserDistribution* binaries = BrowserDistribution::GetSpecificDistribution(
676 BrowserDistribution::CHROME_BINARIES);
677 return (GetAppUpdatePolicy(binaries->GetAppGuid(), nullptr) == app_policy &&
678 (app_policy == AUTOMATIC_UPDATES || app_policy == AUTO_UPDATES_ONLY));
679 #else // defined(GOOGLE_CHROME_BUILD)
680 // Chromium does not auto update.
681 return false;
682 #endif // !defined(GOOGLE_CHROME_BUILD)
685 // static
686 bool GoogleUpdateSettings::ReenableAutoupdates() {
687 #if defined(GOOGLE_CHROME_BUILD)
688 int needs_reset_count = 0;
689 int did_reset_count = 0;
691 // Reset overrides for Chrome and for the binaries if this Chrome supports
692 // multi-install.
693 std::vector<base::string16> app_guids;
694 app_guids.push_back(BrowserDistribution::GetDistribution()->GetAppGuid());
695 if (!InstallUtil::IsChromeSxSProcess()) {
696 app_guids.push_back(BrowserDistribution::GetSpecificDistribution(
697 BrowserDistribution::CHROME_BINARIES)->GetAppGuid());
700 UpdatePolicy update_policy = kDefaultUpdatePolicy;
701 RegKey policy_key;
702 if (policy_key.Open(HKEY_LOCAL_MACHINE, kPoliciesKey,
703 KEY_SET_VALUE | KEY_QUERY_VALUE) == ERROR_SUCCESS) {
704 // Set to true while app-specific overrides are present that allow automatic
705 // updates. When this is the case, the defaults are irrelevant and don't
706 // need to be checked or reset.
707 bool automatic_updates_allowed_by_overrides = true;
708 DWORD value = 0;
709 for (const base::string16& app_guid : app_guids) {
710 // First check the app-specific override value and reset that if needed.
711 // Note that this intentionally sets the override to AUTOMATIC_UPDATES
712 // even if it was previously AUTO_UPDATES_ONLY. The thinking is that
713 // AUTOMATIC_UPDATES is marginally more likely to let a user update and
714 // this code is only called when a stuck user asks for updates.
715 base::string16 app_update_override(kUpdateOverrideValuePrefix);
716 app_update_override.append(app_guid);
717 if (policy_key.ReadValueDW(app_update_override.c_str(),
718 &value) != ERROR_SUCCESS) {
719 automatic_updates_allowed_by_overrides = false;
720 } else if (!GetUpdatePolicyFromDword(value, &update_policy) ||
721 update_policy != GoogleUpdateSettings::AUTOMATIC_UPDATES) {
722 automatic_updates_allowed_by_overrides = false;
723 ++needs_reset_count;
724 if (policy_key.WriteValue(
725 app_update_override.c_str(),
726 static_cast<DWORD>(GoogleUpdateSettings::AUTOMATIC_UPDATES)) ==
727 ERROR_SUCCESS) {
728 ++did_reset_count;
733 // If there were no app-specific override policies, see if there's a global
734 // policy preventing updates and delete it if so.
735 if (!automatic_updates_allowed_by_overrides &&
736 policy_key.ReadValueDW(kUpdatePolicyValue, &value) == ERROR_SUCCESS &&
737 (!GetUpdatePolicyFromDword(value, &update_policy) ||
738 update_policy != GoogleUpdateSettings::AUTOMATIC_UPDATES)) {
739 ++needs_reset_count;
740 if (policy_key.DeleteValue(kUpdatePolicyValue) == ERROR_SUCCESS)
741 ++did_reset_count;
744 // Check the auto-update check period override. If it is 0 or exceeds
745 // the maximum timeout, delete the override value.
746 if (policy_key.ReadValueDW(kCheckPeriodOverrideMinutes,
747 &value) == ERROR_SUCCESS &&
748 (value == 0 || value > kCheckPeriodOverrideMinutesMax)) {
749 ++needs_reset_count;
750 if (policy_key.DeleteValue(kCheckPeriodOverrideMinutes) == ERROR_SUCCESS)
751 ++did_reset_count;
754 // Return whether the number of successful resets is the same as the
755 // number of things that appeared to need resetting.
756 return (needs_reset_count == did_reset_count);
757 } else {
758 // For some reason we couldn't open the policy key with the desired
759 // permissions to make changes (the most likely reason is that there is no
760 // policy set). Simply return whether or not we think updates are enabled.
761 return AreAutoupdatesEnabled();
764 #endif
765 // Non Google Chrome isn't going to autoupdate.
766 return true;
769 void GoogleUpdateSettings::RecordChromeUpdatePolicyHistograms() {
770 const bool is_multi_install = InstallUtil::IsMultiInstall(
771 BrowserDistribution::GetDistribution(), IsSystemInstall());
772 const base::string16 app_guid =
773 BrowserDistribution::GetSpecificDistribution(
774 is_multi_install ? BrowserDistribution::CHROME_BINARIES :
775 BrowserDistribution::CHROME_BROWSER)->GetAppGuid();
777 bool is_overridden = false;
778 const UpdatePolicy update_policy = GetAppUpdatePolicy(app_guid,
779 &is_overridden);
780 UMA_HISTOGRAM_BOOLEAN("GoogleUpdate.UpdatePolicyIsOverridden", is_overridden);
781 UMA_HISTOGRAM_ENUMERATION("GoogleUpdate.EffectivePolicy", update_policy,
782 UPDATE_POLICIES_COUNT);
785 base::string16 GoogleUpdateSettings::GetUninstallCommandLine(
786 bool system_install) {
787 const HKEY root_key = system_install ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER;
788 base::string16 cmd_line;
789 RegKey update_key;
791 if (update_key.Open(root_key, google_update::kRegPathGoogleUpdate,
792 KEY_QUERY_VALUE | KEY_WOW64_32KEY) == ERROR_SUCCESS) {
793 update_key.ReadValue(google_update::kRegUninstallCmdLine, &cmd_line);
796 return cmd_line;
799 Version GoogleUpdateSettings::GetGoogleUpdateVersion(bool system_install) {
800 const HKEY root_key = system_install ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER;
801 base::string16 version;
802 RegKey key;
804 if (key.Open(root_key,
805 google_update::kRegPathGoogleUpdate,
806 KEY_QUERY_VALUE | KEY_WOW64_32KEY) == ERROR_SUCCESS &&
807 key.ReadValue(google_update::kRegGoogleUpdateVersion, &version) ==
808 ERROR_SUCCESS) {
809 return Version(base::UTF16ToUTF8(version));
812 return Version();
815 base::Time GoogleUpdateSettings::GetGoogleUpdateLastStartedAU(
816 bool system_install) {
817 const HKEY root_key = system_install ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER;
818 RegKey update_key;
820 if (update_key.Open(root_key,
821 google_update::kRegPathGoogleUpdate,
822 KEY_QUERY_VALUE | KEY_WOW64_32KEY) == ERROR_SUCCESS) {
823 DWORD last_start;
824 if (update_key.ReadValueDW(google_update::kRegLastStartedAUField,
825 &last_start) == ERROR_SUCCESS) {
826 return base::Time::FromTimeT(last_start);
830 return base::Time();
833 base::Time GoogleUpdateSettings::GetGoogleUpdateLastChecked(
834 bool system_install) {
835 const HKEY root_key = system_install ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER;
836 RegKey update_key;
838 if (update_key.Open(root_key,
839 google_update::kRegPathGoogleUpdate,
840 KEY_QUERY_VALUE | KEY_WOW64_32KEY) == ERROR_SUCCESS) {
841 DWORD last_check;
842 if (update_key.ReadValueDW(google_update::kRegLastCheckedField,
843 &last_check) == ERROR_SUCCESS) {
844 return base::Time::FromTimeT(last_check);
848 return base::Time();
851 bool GoogleUpdateSettings::GetUpdateDetailForApp(bool system_install,
852 const wchar_t* app_guid,
853 ProductData* data) {
854 DCHECK(app_guid);
855 DCHECK(data);
857 bool product_found = false;
859 const HKEY root_key = system_install ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER;
860 base::string16 clientstate_reg_path(google_update::kRegPathClientState);
861 clientstate_reg_path.append(L"\\");
862 clientstate_reg_path.append(app_guid);
864 RegKey clientstate;
865 if (clientstate.Open(root_key,
866 clientstate_reg_path.c_str(),
867 KEY_QUERY_VALUE | KEY_WOW64_32KEY) == ERROR_SUCCESS) {
868 base::string16 version;
869 DWORD dword_value;
870 if ((clientstate.ReadValueDW(google_update::kRegLastCheckSuccessField,
871 &dword_value) == ERROR_SUCCESS) &&
872 (clientstate.ReadValue(google_update::kRegVersionField,
873 &version) == ERROR_SUCCESS)) {
874 product_found = true;
875 data->version = base::UTF16ToASCII(version);
876 data->last_success = base::Time::FromTimeT(dword_value);
877 data->last_result = 0;
878 data->last_error_code = 0;
879 data->last_extra_code = 0;
881 if (clientstate.ReadValueDW(google_update::kRegLastInstallerResultField,
882 &dword_value) == ERROR_SUCCESS) {
883 // Google Update convention is that if an installer writes an result
884 // code that is invalid, it is clamped to an exit code result.
885 const DWORD kMaxValidInstallResult = 4; // INSTALLER_RESULT_EXIT_CODE
886 data->last_result = std::min(dword_value, kMaxValidInstallResult);
888 if (clientstate.ReadValueDW(google_update::kRegLastInstallerErrorField,
889 &dword_value) == ERROR_SUCCESS) {
890 data->last_error_code = dword_value;
892 if (clientstate.ReadValueDW(google_update::kRegLastInstallerExtraField,
893 &dword_value) == ERROR_SUCCESS) {
894 data->last_extra_code = dword_value;
899 return product_found;
902 bool GoogleUpdateSettings::GetUpdateDetailForGoogleUpdate(bool system_install,
903 ProductData* data) {
904 return GetUpdateDetailForApp(system_install,
905 google_update::kGoogleUpdateUpgradeCode,
906 data);
909 bool GoogleUpdateSettings::GetUpdateDetail(bool system_install,
910 ProductData* data) {
911 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
912 return GetUpdateDetailForApp(system_install,
913 dist->GetAppGuid().c_str(),
914 data);
917 bool GoogleUpdateSettings::SetExperimentLabels(
918 bool system_install,
919 const base::string16& experiment_labels) {
920 HKEY reg_root = system_install ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER;
922 // Use the browser distribution and install level to write to the correct
923 // client state/app guid key.
924 bool success = false;
925 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
926 if (dist->ShouldSetExperimentLabels()) {
927 base::string16 client_state_path(
928 system_install ? dist->GetStateMediumKey() : dist->GetStateKey());
929 RegKey client_state(
930 reg_root, client_state_path.c_str(), KEY_SET_VALUE | KEY_WOW64_32KEY);
931 if (experiment_labels.empty()) {
932 success = client_state.DeleteValue(google_update::kExperimentLabels)
933 == ERROR_SUCCESS;
934 } else {
935 success = client_state.WriteValue(google_update::kExperimentLabels,
936 experiment_labels.c_str()) == ERROR_SUCCESS;
940 return success;
943 bool GoogleUpdateSettings::ReadExperimentLabels(
944 bool system_install,
945 base::string16* experiment_labels) {
946 HKEY reg_root = system_install ? HKEY_LOCAL_MACHINE : HKEY_CURRENT_USER;
948 // If this distribution does not set the experiment labels, don't bother
949 // reading.
950 BrowserDistribution* dist = BrowserDistribution::GetDistribution();
951 if (!dist->ShouldSetExperimentLabels())
952 return false;
954 base::string16 client_state_path(
955 system_install ? dist->GetStateMediumKey() : dist->GetStateKey());
957 RegKey client_state;
958 LONG result = client_state.Open(
959 reg_root, client_state_path.c_str(), KEY_QUERY_VALUE | KEY_WOW64_32KEY);
960 if (result == ERROR_SUCCESS) {
961 result = client_state.ReadValue(google_update::kExperimentLabels,
962 experiment_labels);
965 // If the key or value was not present, return the empty string.
966 if (result == ERROR_FILE_NOT_FOUND || result == ERROR_PATH_NOT_FOUND) {
967 experiment_labels->clear();
968 return true;
971 return result == ERROR_SUCCESS;