OLD | NEW |
| (Empty) |
1 // Copyright (c) 2013 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 #include "chrome/browser/component_updater/component_patcher_win.h" | |
6 | |
7 #include <string> | |
8 | |
9 #include "base/base_paths.h" | |
10 #include "base/command_line.h" | |
11 #include "base/file_util.h" | |
12 #include "base/path_service.h" | |
13 #include "base/process/kill.h" | |
14 #include "base/process/launch.h" | |
15 #include "base/strings/string_util.h" | |
16 #include "base/win/scoped_handle.h" | |
17 #include "chrome/installer/util/util_constants.h" | |
18 | |
19 namespace component_updater { | |
20 | |
21 namespace { | |
22 | |
23 std::string PatchTypeToCommandLineSwitch( | |
24 ComponentPatcher::PatchType patch_type) { | |
25 if (patch_type == ComponentPatcher::kPatchTypeCourgette) | |
26 return std::string(installer::kCourgette); | |
27 else if (patch_type == ComponentPatcher::kPatchTypeBsdiff) | |
28 return std::string(installer::kBsdiff); | |
29 else | |
30 return std::string(); | |
31 } | |
32 | |
33 // Finds the path to the setup.exe. First, it looks for the program in the | |
34 // "installer" directory. If the program is not found there, it tries to find it | |
35 // in the directory where chrome.dll lives. Returns the path to the setup.exe, | |
36 // if the path exists, otherwise it returns an an empty path. | |
37 base::FilePath FindSetupProgram() { | |
38 base::FilePath exe_dir; | |
39 if (!PathService::Get(base::DIR_MODULE, &exe_dir)) | |
40 return base::FilePath(); | |
41 | |
42 const std::string installer_dir(WideToASCII(installer::kInstallerDir)); | |
43 const std::string setup_exe(WideToASCII(installer::kSetupExe)); | |
44 | |
45 base::FilePath setup_path = exe_dir; | |
46 setup_path = setup_path.AppendASCII(installer_dir); | |
47 setup_path = setup_path.AppendASCII(setup_exe); | |
48 if (base::PathExists(setup_path)) | |
49 return setup_path; | |
50 | |
51 setup_path = exe_dir; | |
52 setup_path = setup_path.AppendASCII(setup_exe); | |
53 if (base::PathExists(setup_path)) | |
54 return setup_path; | |
55 | |
56 return base::FilePath(); | |
57 } | |
58 | |
59 } // namespace | |
60 | |
61 // Applies the patch to the input file. Returns kNone if the patch was | |
62 // successfully applied, kDeltaOperationFailure if the patch operation | |
63 // encountered errors, and kDeltaPatchProcessFailure if there was an error | |
64 // when running the patch code out of process. In the error case, detailed error | |
65 // information could be returned in the error parameter. | |
66 ComponentUnpacker::Error ComponentPatcherWin::Patch( | |
67 PatchType patch_type, | |
68 const base::FilePath& input_file, | |
69 const base::FilePath& patch_file, | |
70 const base::FilePath& output_file, | |
71 int* error) { | |
72 *error = 0; | |
73 | |
74 const base::FilePath exe_path = FindSetupProgram(); | |
75 if (exe_path.empty()) | |
76 return ComponentUnpacker::kDeltaPatchProcessFailure; | |
77 | |
78 const std::string patch_type_str(PatchTypeToCommandLineSwitch(patch_type)); | |
79 | |
80 CommandLine cl(CommandLine::NO_PROGRAM); | |
81 cl.AppendSwitchASCII(installer::switches::kPatch, patch_type_str.c_str()); | |
82 cl.AppendSwitchPath(installer::switches::kInputFile, input_file); | |
83 cl.AppendSwitchPath(installer::switches::kPatchFile, patch_file); | |
84 cl.AppendSwitchPath(installer::switches::kOutputFile, output_file); | |
85 | |
86 // Create the child process in a job object. The job object prevents leaving | |
87 // child processes around when the parent process exits, either gracefully or | |
88 // accidentally. | |
89 base::win::ScopedHandle job(CreateJobObject(NULL, NULL)); | |
90 if (!job || | |
91 !base::SetJobObjectLimitFlags(job, JOB_OBJECT_LIMIT_KILL_ON_JOB_CLOSE)) { | |
92 *error = GetLastError(); | |
93 return ComponentUnpacker::kDeltaPatchProcessFailure; | |
94 } | |
95 | |
96 base::LaunchOptions launch_options; | |
97 launch_options.wait = true; | |
98 launch_options.job_handle = job; | |
99 launch_options.start_hidden = true; | |
100 CommandLine setup_path(exe_path); | |
101 setup_path.AppendArguments(cl, false); | |
102 | |
103 // |ph| is closed by WaitForExitCode. | |
104 base::ProcessHandle ph = base::kNullProcessHandle; | |
105 int exit_code = 0; | |
106 if (!base::LaunchProcess(setup_path, launch_options, &ph) || | |
107 !base::WaitForExitCode(ph, &exit_code)) { | |
108 *error = GetLastError(); | |
109 return ComponentUnpacker::kDeltaPatchProcessFailure; | |
110 } | |
111 | |
112 *error = exit_code; | |
113 return *error ? ComponentUnpacker::kDeltaOperationFailure : | |
114 ComponentUnpacker::kNone; | |
115 } | |
116 | |
117 } // namespace component_updater | |
OLD | NEW |