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
|
// 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 <atlbase.h>
#include <atlcom.h>
#include <atlctl.h>
#include <initguid.h>
#include <shellapi.h>
#include "base/at_exit.h"
#include "base/command_line.h"
#include "base/file_util.h"
#include "base/string16.h"
#include "base/string_number_conversions.h"
#include "base/utf_string_conversions.h"
#include "base/win/scoped_com_initializer.h"
#include "base/win/scoped_handle.h"
#include "chrome/common/chrome_switches.h"
#include "command_execute_impl.h"
#include "win8/delegate_execute/delegate_execute_operation.h"
#include "win8/delegate_execute/resource.h"
using namespace ATL;
class DelegateExecuteModule
: public ATL::CAtlExeModuleT< DelegateExecuteModule > {
public :
typedef ATL::CAtlExeModuleT<DelegateExecuteModule> ParentClass;
DECLARE_REGISTRY_APPID_RESOURCEID(IDR_DELEGATEEXECUTE,
"{B1935DA1-112F-479A-975B-AB8588ABA636}")
virtual HRESULT AddCommonRGSReplacements(IRegistrarBase* registrar) throw() {
AtlTrace(L"In %hs\n", __FUNCTION__);
HRESULT hr = ParentClass::AddCommonRGSReplacements(registrar);
if (FAILED(hr))
return hr;
wchar_t delegate_execute_clsid[MAX_PATH] = {0};
if (!StringFromGUID2(__uuidof(CommandExecuteImpl), delegate_execute_clsid,
ARRAYSIZE(delegate_execute_clsid))) {
ATLASSERT(false);
return E_FAIL;
}
hr = registrar->AddReplacement(L"DELEGATE_EXECUTE_CLSID",
delegate_execute_clsid);
ATLASSERT(SUCCEEDED(hr));
return hr;
}
};
DelegateExecuteModule _AtlModule;
// Relaunch metro Chrome by ShellExecute on |shortcut| with --force-immersive.
// |handle_value| is an optional handle on which this function will wait before
// performing the relaunch.
int RelaunchChrome(const FilePath& shortcut, const string16& handle_value) {
base::win::ScopedHandle handle;
if (!handle_value.empty()) {
uint32 the_handle = 0;
if (!base::StringToUint(handle_value, &the_handle)) {
// Failed to parse the handle value. Skip the wait but proceed with the
// relaunch.
AtlTrace(L"Failed to parse handle value %ls\n", handle_value.c_str());
} else {
handle.Set(reinterpret_cast<HANDLE>(the_handle));
}
}
if (handle.IsValid()) {
AtlTrace(L"Waiting for chrome.exe to exit.\n");
DWORD result = ::WaitForSingleObject(handle, 10 * 1000);
AtlTrace(L"And we're back.\n");
if (result != WAIT_OBJECT_0) {
AtlTrace(L"Failed to wait for parent to exit; result=%u.\n", result);
// This could mean that Chrome has hung. Conservatively proceed with
// the relaunch anyway.
}
handle.Close();
}
base::win::ScopedCOMInitializer com_initializer;
AtlTrace(L"Launching Chrome via %ls.\n", shortcut.value().c_str());
int ser = reinterpret_cast<int>(
::ShellExecute(NULL, NULL, shortcut.value().c_str(),
ASCIIToWide(switches::kForceImmersive).c_str(), NULL,
SW_SHOWNORMAL));
AtlTrace(L"ShellExecute returned %d.\n", ser);
return ser <= 32;
}
//
extern "C" int WINAPI _tWinMain(HINSTANCE , HINSTANCE,
LPTSTR, int nShowCmd) {
using delegate_execute::DelegateExecuteOperation;
base::AtExitManager exit_manager;
CommandLine::Init(0, NULL);
CommandLine* cmd_line = CommandLine::ForCurrentProcess();
DelegateExecuteOperation operation;
operation.Initialize(cmd_line);
switch (operation.operation_type()) {
case DelegateExecuteOperation::EXE_MODULE:
return _AtlModule.WinMain(nShowCmd);
case DelegateExecuteOperation::RELAUNCH_CHROME:
return RelaunchChrome(operation.relaunch_shortcut(),
cmd_line->GetSwitchValueNative(switches::kWaitForHandle));
default:
NOTREACHED();
}
return 1;
}
|