Pin Chrome's shortcut to the Win10 Start menu on install and OS upgrade.
[chromium-blink-merge.git] / rlz / win / lib / machine_deal.cc
blob8d6d369e095aec24a4d619e02587932df0e3393b
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.
4 //
5 // Library functions related to the OEM Deal Confirmation Code.
7 #include "rlz/win/lib/machine_deal.h"
9 #include <windows.h>
10 #include <vector>
12 #include "base/basictypes.h"
13 #include "base/memory/scoped_ptr.h"
14 #include "base/strings/string_split.h"
15 #include "base/strings/string_util.h"
16 #include "base/strings/stringprintf.h"
17 #include "base/win/registry.h"
18 #include "rlz/lib/assert.h"
19 #include "rlz/lib/lib_values.h"
20 #include "rlz/win/lib/lib_mutex.h"
21 #include "rlz/win/lib/registry_util.h"
22 #include "rlz/win/lib/rlz_value_store_registry.h"
24 namespace {
26 const wchar_t kDccValueName[] = L"DCC";
28 // Current DCC can only uses [a-zA-Z0-9_-!@$*();.<>,:]
29 // We will be more liberal and allow some additional chars, but not url meta
30 // chars.
31 bool IsGoodDccChar(char ch) {
32 if (base::IsAsciiAlpha(ch) || base::IsAsciiDigit(ch))
33 return true;
35 switch (ch) {
36 case '_':
37 case '-':
38 case '!':
39 case '@':
40 case '$':
41 case '*':
42 case '(':
43 case ')':
44 case ';':
45 case '.':
46 case '<':
47 case '>':
48 case ',':
49 case ':':
50 return true;
53 return false;
56 // This function will remove bad rlz chars and also limit the max rlz to some
57 // reasonable size. It also assumes that normalized_dcc is at least
58 // kMaxDccLength+1 long.
59 void NormalizeDcc(const char* raw_dcc, char* normalized_dcc) {
60 int index = 0;
61 for (; raw_dcc[index] != 0 && index < rlz_lib::kMaxDccLength; ++index) {
62 char current = raw_dcc[index];
63 if (IsGoodDccChar(current)) {
64 normalized_dcc[index] = current;
65 } else {
66 normalized_dcc[index] = '.';
70 normalized_dcc[index] = 0;
73 bool GetResponseLine(const char* response_text, int response_length,
74 int* search_index, std::string* response_line) {
75 if (!response_line || !search_index || *search_index > response_length)
76 return false;
78 response_line->clear();
80 if (*search_index < 0)
81 return false;
83 int line_begin = *search_index;
84 const char* line_end = strchr(response_text + line_begin, '\n');
86 if (line_end == NULL || line_end - response_text > response_length) {
87 line_end = response_text + response_length;
88 *search_index = -1;
89 } else {
90 *search_index = line_end - response_text + 1;
93 response_line->assign(response_text + line_begin,
94 line_end - response_text - line_begin);
95 return true;
98 bool GetResponseValue(const std::string& response_line,
99 const std::string& response_key,
100 std::string* value) {
101 if (!value)
102 return false;
104 value->clear();
106 if (!base::StartsWith(response_line, response_key,
107 base::CompareCase::SENSITIVE))
108 return false;
110 std::vector<std::string> tokens;
111 base::SplitString(response_line, ':', &tokens);
112 if (tokens.size() != 2)
113 return false;
115 // The first token is the key, the second is the value. The value is already
116 // trimmed for whitespace.
117 *value = tokens[1];
118 return true;
121 } // namespace
123 namespace rlz_lib {
125 bool MachineDealCode::Set(const char* dcc) {
126 LibMutex lock;
127 if (lock.failed())
128 return false;
130 // TODO: if (!ProcessInfo::CanWriteMachineKey()) return false;
132 // Validate the new dcc value.
133 size_t length = strlen(dcc);
134 if (length > kMaxDccLength) {
135 ASSERT_STRING("MachineDealCode::Set: DCC length is exceeds max allowed.");
136 return false;
139 base::win::RegKey hklm_key(HKEY_LOCAL_MACHINE,
140 RlzValueStoreRegistry::GetWideLibKeyName().c_str(),
141 KEY_READ | KEY_WRITE | KEY_WOW64_32KEY);
142 if (!hklm_key.Valid()) {
143 ASSERT_STRING("MachineDealCode::Set: Unable to create / open machine key."
144 " Did you call rlz_lib::CreateMachineState()?");
145 return false;
148 char normalized_dcc[kMaxDccLength + 1];
149 NormalizeDcc(dcc, normalized_dcc);
150 VERIFY(length == strlen(normalized_dcc));
152 // Write the DCC to HKLM. Note that we need to include the null character
153 // when writing the string.
154 if (!RegKeyWriteValue(&hklm_key, kDccValueName, normalized_dcc)) {
155 ASSERT_STRING("MachineDealCode::Set: Could not write the DCC value");
156 return false;
159 return true;
162 bool MachineDealCode::GetNewCodeFromPingResponse(const char* response,
163 bool* has_new_dcc, char* new_dcc, int new_dcc_size) {
164 if (!has_new_dcc || !new_dcc || !new_dcc_size)
165 return false;
167 *has_new_dcc = false;
168 new_dcc[0] = 0;
170 int response_length = -1;
171 if (!IsPingResponseValid(response, &response_length))
172 return false;
174 // Get the current DCC value to compare to later)
175 char stored_dcc[kMaxDccLength + 1];
176 if (!Get(stored_dcc, arraysize(stored_dcc)))
177 stored_dcc[0] = 0;
179 int search_index = 0;
180 std::string response_line;
181 std::string new_dcc_value;
182 bool old_dcc_confirmed = false;
183 const std::string dcc_cgi(kDccCgiVariable);
184 const std::string dcc_cgi_response(kSetDccResponseVariable);
185 while (GetResponseLine(response, response_length, &search_index,
186 &response_line)) {
187 std::string value;
189 if (!old_dcc_confirmed &&
190 GetResponseValue(response_line, dcc_cgi, &value)) {
191 // This is the old DCC confirmation - should match value in registry.
192 if (value != stored_dcc)
193 return false; // Corrupted DCC - ignore this response.
194 else
195 old_dcc_confirmed = true;
196 continue;
199 if (!(*has_new_dcc) &&
200 GetResponseValue(response_line, dcc_cgi_response, &value)) {
201 // This is the new DCC.
202 if (value.size() > kMaxDccLength) continue; // Too long
203 *has_new_dcc = true;
204 new_dcc_value = value;
208 old_dcc_confirmed |= (NULL == stored_dcc[0]);
210 base::strlcpy(new_dcc, new_dcc_value.c_str(), new_dcc_size);
211 return old_dcc_confirmed;
214 bool MachineDealCode::SetFromPingResponse(const char* response) {
215 bool has_new_dcc = false;
216 char new_dcc[kMaxDccLength + 1];
218 bool response_valid = GetNewCodeFromPingResponse(
219 response, &has_new_dcc, new_dcc, arraysize(new_dcc));
221 if (response_valid && has_new_dcc)
222 return Set(new_dcc);
224 return response_valid;
227 bool MachineDealCode::GetAsCgi(char* cgi, int cgi_size) {
228 if (!cgi || cgi_size <= 0) {
229 ASSERT_STRING("MachineDealCode::GetAsCgi: Invalid buffer");
230 return false;
233 cgi[0] = 0;
235 std::string cgi_arg;
236 base::StringAppendF(&cgi_arg, "%s=", kDccCgiVariable);
237 int cgi_arg_length = cgi_arg.size();
239 if (cgi_arg_length >= cgi_size) {
240 ASSERT_STRING("MachineDealCode::GetAsCgi: Insufficient buffer size");
241 return false;
244 base::strlcpy(cgi, cgi_arg.c_str(), cgi_size);
246 if (!Get(cgi + cgi_arg_length, cgi_size - cgi_arg_length)) {
247 cgi[0] = 0;
248 return false;
250 return true;
253 bool MachineDealCode::Get(char* dcc, int dcc_size) {
254 LibMutex lock;
255 if (lock.failed())
256 return false;
258 if (!dcc || dcc_size <= 0) {
259 ASSERT_STRING("MachineDealCode::Get: Invalid buffer");
260 return false;
263 dcc[0] = 0;
265 base::win::RegKey dcc_key(HKEY_LOCAL_MACHINE,
266 RlzValueStoreRegistry::GetWideLibKeyName().c_str(),
267 KEY_READ | KEY_WOW64_32KEY);
268 if (!dcc_key.Valid())
269 return false; // no DCC key.
271 size_t size = dcc_size;
272 if (!RegKeyReadValue(dcc_key, kDccValueName, dcc, &size)) {
273 ASSERT_STRING("MachineDealCode::Get: Insufficient buffer size");
274 dcc[0] = 0;
275 return false;
278 return true;
281 bool MachineDealCode::Clear() {
282 base::win::RegKey dcc_key(HKEY_LOCAL_MACHINE,
283 RlzValueStoreRegistry::GetWideLibKeyName().c_str(),
284 KEY_READ | KEY_WRITE | KEY_WOW64_32KEY);
285 if (!dcc_key.Valid())
286 return false; // no DCC key.
288 dcc_key.DeleteValue(kDccValueName);
290 // Verify deletion.
291 wchar_t dcc[kMaxDccLength + 1];
292 DWORD dcc_size = arraysize(dcc);
293 if (dcc_key.ReadValue(kDccValueName, dcc, &dcc_size, NULL) == ERROR_SUCCESS) {
294 ASSERT_STRING("MachineDealCode::Clear: Could not delete the DCC value.");
295 return false;
298 return true;
301 } // namespace rlz_lib