1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
|
// Copyright (c) 2006-2009 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.
// mini_installer.exe is the first exe that is run when chrome is being
// installed or upgraded. It is designed to be extremely small (~5KB with no
// extra resources linked) and it has two main jobs:
// 1) unpack the resources (possibly decompressing some)
// 2) run the real installer (setup.exe) with appropiate flags.
//
// In order to be really small we don't link against the CRT and we define the
// following compiler/linker flags:
// EnableIntrinsicFunctions="true" compiler: /Oi
// BasicRuntimeChecks="0"
// BufferSecurityCheck="false" compiler: /GS-
// EntryPointSymbol="MainEntryPoint" linker: /ENTRY
// IgnoreAllDefaultLibraries="true" linker: /NODEFAULTLIB
// OptimizeForWindows98="1" liker: /OPT:NOWIN98
// linker: /SAFESEH:NO
// Also some built-in code that the compiler relies on is not defined so we
// are forced to manually link against it. It comes in the form of two
// object files that exist in $(VCInstallDir)\crt\src which are memset.obj and
// P4_memset.obj. These two object files rely on the existence of a static
// variable named __sse2_available which indicates the presence of intel sse2
// extensions. We define it to false which causes a slower but safe code for
// memcpy and memset intrinsics.
// having the linker merge the sections is saving us ~500 bytes.
#pragma comment(linker, "/MERGE:.rdata=.text")
#include <windows.h>
#include <Shellapi.h>
#include <shlwapi.h>
#include "chrome/installer/mini_installer/mini_installer.h"
#include "chrome/installer/mini_installer/pe_resource.h"
// Required linker symbol. See remarks above.
extern "C" unsigned int __sse2_available = 0;
namespace mini_installer {
// This structure passes data back and forth for the processing
// of resource callbacks.
struct Context {
// Input to the call back method. Specifies the dir to save resources.
const wchar_t* base_path;
// First output from call back method. Full path of Chrome archive.
wchar_t* chrome_resource_path;
// Size of chrome_resource_path buffer
size_t chrome_resource_path_size;
// Second output from call back method. Full path of Setup archive/exe.
wchar_t* setup_resource_path;
// Size of setup_resource_path buffer
size_t setup_resource_path_size;
};
// Returns true if the given two ASCII characters are same (ignoring case).
bool EqualASCIICharI(wchar_t a, wchar_t b) {
if (a >= L'A' && a <= L'Z')
a = a + (L'a' - L'A');
if (b >= L'A' && b <= L'Z')
b = b + (L'a' - L'A');
return (a == b);
}
// Takes the path to file and returns a pointer to the filename component. For
// exmaple for input of c:\full\path\to\file.ext it returns pointer to file.ext.
// It returns NULL if extension or path separator is not found.
wchar_t* GetNameFromPathExt(wchar_t* path, size_t size) {
if (size <= 1)
return NULL;
wchar_t* current = &path[size - 1];
while (current != path && L'\\' != *current)
--current;
return (current == path) ? NULL : (current + 1);
}
// Simple replacement for CRT string copy method that does not overflow.
// Returns true if the source was copied successfully otherwise returns false.
// Parameter src is assumed to be NULL terminated and the NULL character is
// copied over to string dest.
bool SafeStrCopy(wchar_t* dest, size_t dest_size, const wchar_t* src) {
for (size_t length = 0; length < dest_size; ++dest, ++src, ++length) {
*dest = *src;
if (L'\0' == *src)
return true;
}
return false;
}
// Safer replacement for lstrcat function.
bool SafeStrCat(wchar_t* dest, size_t dest_size, const wchar_t* src) {
int str_len = ::lstrlen(dest);
return SafeStrCopy(dest + str_len, dest_size - str_len, src);
}
// Function to check if a string (specified by str) ends with another string
// (specified by end_str).
bool StrEndsWith(const wchar_t *str, const wchar_t *end_str) {
if (str == NULL || end_str == NULL)
return false;
for (int i = lstrlen(str) - 1, j = lstrlen(end_str) - 1; j >= 0; --i, --j) {
if (i < 0 || !EqualASCIICharI(str[i], end_str[j]))
return false;
}
return true;
}
// Function to check if a string (specified by str) starts with another string
// (specified by start_str).
bool StrStartsWith(const wchar_t *str, const wchar_t *start_str) {
if (str == NULL || start_str == NULL)
return false;
for (int i = 0; start_str[i] != L'\0'; ++i) {
if (!EqualASCIICharI(str[i], start_str[i]))
return false;
}
return true;
}
// Helper function to read a value from registry. Returns true if value
// is read successfully and stored in parameter value. Returns false otherwise.
bool ReadValueFromRegistry(HKEY root_key, const wchar_t *sub_key,
const wchar_t *value_name, wchar_t *value,
size_t size) {
HKEY key;
if ((::RegOpenKeyEx(root_key, sub_key, NULL,
KEY_READ, &key) == ERROR_SUCCESS) &&
(::RegQueryValueEx(key, value_name, NULL, NULL,
reinterpret_cast<LPBYTE>(value),
reinterpret_cast<LPDWORD>(&size)) == ERROR_SUCCESS)) {
::RegCloseKey(key);
return true;
}
return false;
}
// Gets the setup.exe path from Registry by looking the value of Uninstall
// string, strips the arguments for uninstall and returns only the full path
// to setup.exe.
bool GetSetupExePathFromRegistry(wchar_t *path, size_t size) {
if (!ReadValueFromRegistry(HKEY_CURRENT_USER, kUninstallRegistryKey,
kUninstallRegistryValueName, path, size)) {
if (!ReadValueFromRegistry(HKEY_LOCAL_MACHINE, kUninstallRegistryKey,
kUninstallRegistryValueName, path, size)) {
return false;
}
}
wchar_t *tmp = StrStr(path, L" --");
if (tmp) {
*tmp = L'\0';
} else {
return false;
}
return true;
}
// Calls CreateProcess with good default parameters and waits for the process
// to terminate returning the process exit code.
bool RunProcessAndWait(const wchar_t* exe_path, wchar_t* cmdline,
int* exit_code) {
STARTUPINFOW si = {sizeof(si)};
PROCESS_INFORMATION pi = {0};
if (!::CreateProcess(exe_path, cmdline, NULL, NULL, FALSE, CREATE_NO_WINDOW,
NULL, NULL, &si, &pi))
return false;
DWORD wr = ::WaitForSingleObject(pi.hProcess, INFINITE);
if (WAIT_OBJECT_0 != wr)
return false;
bool ret = true;
if (exit_code) {
if (!::GetExitCodeProcess(pi.hProcess,
reinterpret_cast<DWORD*>(exit_code)))
ret = false;
}
::CloseHandle(pi.hProcess);
::CloseHandle(pi.hThread);
return ret;
}
// Windows defined callback used in the EnumResourceNames call. For each
// matching resource found, the callback is invoked and at this point we write
// it to disk. We expect resource names to start with 'chrome' or 'setup'. Any
// other name is treated as an error.
BOOL CALLBACK OnResourceFound(HMODULE module, const wchar_t* type,
wchar_t* name, LONG_PTR context) {
if (NULL == context) {
return FALSE;
}
Context* ctx = reinterpret_cast<Context*>(context);
PEResource resource(name, type, module);
if ((!resource.IsValid()) ||
(resource.Size() < 1 ) ||
(resource.Size() > kMaxResourceSize)) {
return FALSE;
}
wchar_t full_path[MAX_PATH];
if (!SafeStrCopy(full_path, _countof(full_path), ctx->base_path) ||
!SafeStrCat(full_path, _countof(full_path), name) ||
!resource.WriteToDisk(full_path))
return FALSE;
if (StrStartsWith(name, kChromePrefix)) {
if (!SafeStrCopy(ctx->chrome_resource_path,
ctx->chrome_resource_path_size, full_path))
return FALSE;
} else if (StrStartsWith(name, kSetupPrefix)) {
if (!SafeStrCopy(ctx->setup_resource_path,
ctx->setup_resource_path_size, full_path))
return FALSE;
} else {
// Resources should either start with 'chrome' or 'setup'. We dont handle
// anything else.
return FALSE;
}
return TRUE;
}
// Finds and writes to disk resources of various types. Returns false
// if there is a problem in writing any resource to disk. setup.exe resource
// can come in one of three possible forms:
// - Resource type 'B7', compressed using LZMA (*.7z)
// - Resource type 'BL', compressed using LZ (*.ex_)
// - Resource type 'BN', uncompressed (*.exe)
// If setup.exe is present in more than one form, the precedence order is
// BN < BL < B7
bool UnpackBinaryResources(HMODULE module, const wchar_t* base_path,
wchar_t* archive_path, size_t archive_path_size,
wchar_t* setup_path, size_t setup_path_size) {
// Prepare the input to OnResourceFound method that needs a location where
// it will write all the resources.
Context context = {base_path, archive_path, archive_path_size,
setup_path, setup_path_size};
// Get the resources of type 'B7'.
// We need a chrome archive to do the installation. So if there
// is a problem in fetching B7 resource, just return error.
if ((!::EnumResourceNames(module, kLZMAResourceType, OnResourceFound,
LONG_PTR(&context))) ||
(::lstrlen(archive_path) <= 0))
return false;
// Generate the setup.exe path where we patch/uncompress setup resource.
wchar_t setup_dest_path[MAX_PATH] = {0};
if (!SafeStrCopy(setup_dest_path, _countof(setup_dest_path),
context.base_path) ||
!SafeStrCat(setup_dest_path, _countof(setup_dest_path), kSetupName))
return false;
// If we found setup 'B7' resource, handle it.
if (::lstrlen(setup_path) > 0) {
wchar_t cmd_line[MAX_PATH * 3] = {0};
// Get the path to setup.exe first.
if (!GetSetupExePathFromRegistry(cmd_line, _countof(cmd_line)))
return false;
if (!SafeStrCat(cmd_line, _countof(cmd_line), kCmdUpdateSetupExe) ||
!SafeStrCat(cmd_line, _countof(cmd_line), L"=\"") ||
!SafeStrCat(cmd_line, _countof(cmd_line), setup_path) ||
!SafeStrCat(cmd_line, _countof(cmd_line), L"\"") ||
!SafeStrCat(cmd_line, _countof(cmd_line), kCmdNewSetupExe) ||
!SafeStrCat(cmd_line, _countof(cmd_line), L"=\"") ||
!SafeStrCat(cmd_line, _countof(cmd_line), setup_dest_path) ||
!SafeStrCat(cmd_line, _countof(cmd_line), L"\""))
return false;
int exit_code = 0;
if (!RunProcessAndWait(NULL, cmd_line, &exit_code) ||
(exit_code != 0))
return FALSE;
if (!SafeStrCopy(setup_path, setup_path_size, setup_dest_path))
return false;
return true;
}
// setup.exe wasn't sent as 'B7', lets see if it was sent as 'BL'
if ((!::EnumResourceNames(module, kLZCResourceType, OnResourceFound,
LONG_PTR(&context))) &&
(::GetLastError() != ERROR_RESOURCE_TYPE_NOT_FOUND))
return false;
if (::lstrlen(setup_path) > 0) {
// Uncompress LZ compressed resource using the existing
// program in the system32 folder named 'expand.exe'.
wchar_t expand_cmd[MAX_PATH * 3] = {0};
if (!SafeStrCopy(expand_cmd, _countof(expand_cmd), UNCOMPRESS_CMD) ||
!SafeStrCat(expand_cmd, _countof(expand_cmd), L"\"") ||
!SafeStrCat(expand_cmd, _countof(expand_cmd), setup_path) ||
!SafeStrCat(expand_cmd, _countof(expand_cmd), L"\" \"") ||
!SafeStrCat(expand_cmd, _countof(expand_cmd), setup_dest_path) ||
!SafeStrCat(expand_cmd, _countof(expand_cmd), L"\""))
return false;
// If we fail to uncompress the file, exit now and leave the file
// behind for postmortem analysis.
int exit_code = 0;
if (!RunProcessAndWait(NULL, expand_cmd, &exit_code) ||
(exit_code != 0))
return false;
// Uncompression was successful, delete the source but it is not critical
// if that fails.
::DeleteFile(setup_path);
if (!SafeStrCopy(setup_path, setup_path_size, setup_dest_path))
return false;
return true;
}
// setup.exe still not found. So finally check is it was sent as 'BN'
if ((!::EnumResourceNames(module, kBinResourceType, OnResourceFound,
LONG_PTR(&context))) &&
(::GetLastError() != ERROR_RESOURCE_TYPE_NOT_FOUND))
return false;
if (::lstrlen(setup_path) > 0) {
if (!::lstrcmpi(setup_path, setup_dest_path)) {
::CopyFile(setup_path, setup_dest_path, false);
if (!SafeStrCopy(setup_path, setup_path_size, setup_dest_path))
return false;
}
return true;
}
return true;
}
// Append any command line params passed to mini_installer to the given buffer
// so that they can be passed on to setup.exe. We do not return any error from
// this method and simply skip making any changes in case of error.
void AppendCommandLineFlags(wchar_t* buffer, int size) {
wchar_t full_exe_path[MAX_PATH];
int len = ::GetModuleFileName(NULL, full_exe_path, _countof(full_exe_path));
if (len <= 0 || len >= _countof(full_exe_path))
return;
wchar_t* exe_name = GetNameFromPathExt(full_exe_path, len);
if (exe_name == NULL)
return;
int args_num;
wchar_t* cmd_line = ::GetCommandLine();
wchar_t** args = ::CommandLineToArgvW(cmd_line, &args_num);
if (args_num <= 0)
return;
wchar_t* cmd_to_append = NULL;
if (!StrEndsWith(args[0], exe_name)) {
// Current executable name not in the command line so just append
// the whole command line.
cmd_to_append = cmd_line;
} else if (args_num > 1 ) {
wchar_t* tmp = StrStr(cmd_line, exe_name);
tmp = StrStr(tmp, L" ");
cmd_to_append = tmp;
}
if (size > ::lstrlen(cmd_to_append))
::lstrcat(buffer, cmd_to_append);
LocalFree(args);
}
// Executes setup.exe, waits for it to finish and returns the exit code.
bool RunSetup(const wchar_t* archive_path, const wchar_t* setup_path,
int* exit_code) {
// There could be three full paths in the command line for setup.exe (path
// to exe itself, path to archive and path to log file), so we declare
// total size as three + one additional to hold command line options.
wchar_t cmd_line[MAX_PATH * 4];
// Get the path to setup.exe first.
if (::lstrlen(setup_path) > 0) {
if (!SafeStrCopy(cmd_line, _countof(cmd_line), L"\"") ||
!SafeStrCat(cmd_line, _countof(cmd_line), setup_path) ||
!SafeStrCat(cmd_line, _countof(cmd_line), L"\""))
return false;
} else if (!GetSetupExePathFromRegistry(cmd_line, _countof(cmd_line))) {
return false;
}
// Append the command line param for chrome archive file
if (!SafeStrCat(cmd_line, _countof(cmd_line), kCmdInstallArchive) ||
!SafeStrCat(cmd_line, _countof(cmd_line), L"=\"") ||
!SafeStrCat(cmd_line, _countof(cmd_line), archive_path) ||
!SafeStrCat(cmd_line, _countof(cmd_line), L"\""))
return false;
// Get any command line option specified for mini_installer and pass them
// on to setup.exe
AppendCommandLineFlags(cmd_line, _countof(cmd_line) - lstrlen(cmd_line));
return (RunProcessAndWait(NULL, cmd_line, exit_code));
}
// Deletes given files and working dir.
void DeleteExtractedFiles(const wchar_t* base_path,
const wchar_t* archive_path,
const wchar_t* setup_path) {
::DeleteFile(archive_path);
::DeleteFile(setup_path);
// Delete the temp dir (if it is empty, otherwise fail).
::RemoveDirectory(base_path);
}
// Creates and returns a temporary directory that can be used to extract
// mini_installer payload.
bool GetWorkDir(HMODULE module, wchar_t* work_dir) {
wchar_t base_path[MAX_PATH];
DWORD len = ::GetTempPath(_countof(base_path), base_path);
if (len >= _countof(base_path) || len <= 0) {
// Problem in getting TEMP path so just use current directory as base path
len = ::GetModuleFileName(module, base_path, _countof(base_path));
if (len >= _countof(base_path) || len <= 0)
return false; // Can't even get current directory? Return with error.
wchar_t* name = GetNameFromPathExt(base_path, len);
*name = L'\0';
}
wchar_t temp_name[MAX_PATH];
if (!GetTempFileName(base_path, kTempPrefix, 0, temp_name))
return false; // Didn't get any temp name to use. Return error.
len = GetLongPathName(temp_name, work_dir, _countof(temp_name));
if (len >= _countof(temp_name) || len <= 0)
return false; // Couldn't get full path to temp dir. Return error.
// GetTempFileName creates the file as well so delete it before creating
// the directory in its place.
if (!::DeleteFile(work_dir) || !::CreateDirectory(work_dir, NULL))
return false; // What's the use of temp dir if we can not create it?
::lstrcat(work_dir, L"\\");
return true;
}
// Main function. First gets a working dir, unpacks the resources and finally
// executes setup.exe to do the install/upgrade.
int WMain(HMODULE module) {
// First get a path where we can extract payload
wchar_t base_path[MAX_PATH];
if (!GetWorkDir(module, base_path))
return 101;
wchar_t archive_path[MAX_PATH] = {0};
wchar_t setup_path[MAX_PATH] = {0};
if (!UnpackBinaryResources(module, base_path, archive_path, MAX_PATH,
setup_path, MAX_PATH))
return 102;
int exit_code = 103;
if (!RunSetup(archive_path, setup_path, &exit_code))
return exit_code;
wchar_t value[4];
if ((!ReadValueFromRegistry(HKEY_CURRENT_USER, kCleanupRegistryKey,
kCleanupRegistryValueName, value, 4)) ||
(value[0] != L'0'))
DeleteExtractedFiles(base_path, archive_path, setup_path);
return exit_code;
}
} // namespace mini_installer
int MainEntryPoint() {
int result = mini_installer::WMain(::GetModuleHandle(NULL));
::ExitProcess(result);
}
|