diff options
author | Sean Hall <r.sean.hall@gmail.com> | 2018-12-31 23:52:58 -0600 |
---|---|---|
committer | Sean Hall <r.sean.hall@gmail.com> | 2018-12-31 23:52:58 -0600 |
commit | 2c568de11510b453f499827919964badef22304e (patch) | |
tree | dd342fa59ad9b986e375c4128726b505ae95ad67 /src/mbahost | |
parent | cd23c2cba4239f32abf9743ecec9fef58136e0e8 (diff) | |
download | wix-2c568de11510b453f499827919964badef22304e.tar.gz wix-2c568de11510b453f499827919964badef22304e.tar.bz2 wix-2c568de11510b453f499827919964badef22304e.zip |
Import code from old v4 repo
Diffstat (limited to 'src/mbahost')
-rw-r--r-- | src/mbahost/host.cpp | 656 | ||||
-rw-r--r-- | src/mbahost/host.def | 6 | ||||
-rw-r--r-- | src/mbahost/host.vcxproj | 51 | ||||
-rw-r--r-- | src/mbahost/precomp.h | 25 |
4 files changed, 738 insertions, 0 deletions
diff --git a/src/mbahost/host.cpp b/src/mbahost/host.cpp new file mode 100644 index 00000000..694db357 --- /dev/null +++ b/src/mbahost/host.cpp | |||
@@ -0,0 +1,656 @@ | |||
1 | // Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. | ||
2 | |||
3 | #include "precomp.h" | ||
4 | #include <BootstrapperCore.h> // includes the generated assembly name macros. | ||
5 | #include "BalBaseBootstrapperApplicationProc.h" | ||
6 | |||
7 | static const DWORD NET452_RELEASE = 379893; | ||
8 | |||
9 | using namespace mscorlib; | ||
10 | |||
11 | extern "C" typedef HRESULT (WINAPI *PFN_CORBINDTOCURRENTRUNTIME)( | ||
12 | __in LPCWSTR pwszFileName, | ||
13 | __in REFCLSID rclsid, | ||
14 | __in REFIID riid, | ||
15 | __out LPVOID *ppv | ||
16 | ); | ||
17 | |||
18 | extern "C" typedef HRESULT(WINAPI *PFN_MBAPREQ_BOOTSTRAPPER_APPLICATION_CREATE)( | ||
19 | __in HRESULT hrHostInitialization, | ||
20 | __in IBootstrapperEngine* pEngine, | ||
21 | __in const BOOTSTRAPPER_CREATE_ARGS* pArgs, | ||
22 | __inout BOOTSTRAPPER_CREATE_RESULTS* pResults | ||
23 | ); | ||
24 | |||
25 | static HINSTANCE vhInstance = NULL; | ||
26 | static ICorRuntimeHost *vpCLRHost = NULL; | ||
27 | static _AppDomain *vpAppDomain = NULL; | ||
28 | static HMODULE vhMbapreqModule = NULL; | ||
29 | |||
30 | |||
31 | // internal function declarations | ||
32 | |||
33 | static HRESULT GetAppDomain( | ||
34 | __out _AppDomain** ppAppDomain | ||
35 | ); | ||
36 | static HRESULT GetAppBase( | ||
37 | __out LPWSTR* psczAppBase | ||
38 | ); | ||
39 | static HRESULT CheckSupportedFrameworks( | ||
40 | __in LPCWSTR wzConfigPath | ||
41 | ); | ||
42 | static HRESULT UpdateSupportedRuntime( | ||
43 | __in IXMLDOMDocument* pixdManifest, | ||
44 | __in IXMLDOMNode* pixnSupportedFramework, | ||
45 | __out BOOL* pfUpdatedManifest | ||
46 | ); | ||
47 | static HRESULT GetCLRHost( | ||
48 | __in LPCWSTR wzConfigPath, | ||
49 | __out ICorRuntimeHost** ppCLRHost | ||
50 | ); | ||
51 | static HRESULT CreateManagedBootstrapperApplication( | ||
52 | __in _AppDomain* pAppDomain, | ||
53 | __in IBootstrapperEngine* pEngine, | ||
54 | __in const BOOTSTRAPPER_CREATE_ARGS* pArgs, | ||
55 | __inout BOOTSTRAPPER_CREATE_RESULTS* pResults | ||
56 | ); | ||
57 | static HRESULT CreateManagedBootstrapperApplicationFactory( | ||
58 | __in _AppDomain* pAppDomain, | ||
59 | __out IBootstrapperApplicationFactory** ppAppFactory | ||
60 | ); | ||
61 | static HRESULT CreatePrerequisiteBA( | ||
62 | __in HRESULT hrHostInitialization, | ||
63 | __in IBootstrapperEngine* pEngine, | ||
64 | __in const BOOTSTRAPPER_CREATE_ARGS* pArgs, | ||
65 | __inout BOOTSTRAPPER_CREATE_RESULTS* pResults | ||
66 | ); | ||
67 | static HRESULT VerifyNET4RuntimeIsSupported( | ||
68 | ); | ||
69 | |||
70 | |||
71 | // function definitions | ||
72 | |||
73 | extern "C" BOOL WINAPI DllMain( | ||
74 | IN HINSTANCE hInstance, | ||
75 | IN DWORD dwReason, | ||
76 | IN LPVOID /* pvReserved */ | ||
77 | ) | ||
78 | { | ||
79 | switch (dwReason) | ||
80 | { | ||
81 | case DLL_PROCESS_ATTACH: | ||
82 | ::DisableThreadLibraryCalls(hInstance); | ||
83 | vhInstance = hInstance; | ||
84 | break; | ||
85 | |||
86 | case DLL_PROCESS_DETACH: | ||
87 | vhInstance = NULL; | ||
88 | break; | ||
89 | } | ||
90 | |||
91 | return TRUE; | ||
92 | } | ||
93 | |||
94 | // Note: This function assumes that COM was already initialized on the thread. | ||
95 | extern "C" HRESULT WINAPI BootstrapperApplicationCreate( | ||
96 | __in const BOOTSTRAPPER_CREATE_ARGS* pArgs, | ||
97 | __inout BOOTSTRAPPER_CREATE_RESULTS* pResults | ||
98 | ) | ||
99 | { | ||
100 | HRESULT hr = S_OK; | ||
101 | HRESULT hrHostInitialization = S_OK; | ||
102 | IBootstrapperEngine* pEngine = NULL; | ||
103 | |||
104 | hr = BalInitializeFromCreateArgs(pArgs, &pEngine); | ||
105 | ExitOnFailure(hr, "Failed to initialize Bal."); | ||
106 | |||
107 | hr = GetAppDomain(&vpAppDomain); | ||
108 | if (SUCCEEDED(hr)) | ||
109 | { | ||
110 | BalLog(BOOTSTRAPPER_LOG_LEVEL_STANDARD, "Loading managed bootstrapper application."); | ||
111 | |||
112 | hr = CreateManagedBootstrapperApplication(vpAppDomain, pEngine, pArgs, pResults); | ||
113 | BalExitOnFailure(hr, "Failed to create the managed bootstrapper application."); | ||
114 | } | ||
115 | else // fallback to the prerequisite BA. | ||
116 | { | ||
117 | if (E_MBAHOST_NET452_ON_WIN7RTM == hr) | ||
118 | { | ||
119 | BalLogError(hr, "The Burn engine cannot run with an MBA under the .NET 4 CLR on Windows 7 RTM with .NET 4.5.2 (or greater) installed."); | ||
120 | hrHostInitialization = hr; | ||
121 | } | ||
122 | else | ||
123 | { | ||
124 | hrHostInitialization = S_OK; | ||
125 | } | ||
126 | |||
127 | BalLog(BOOTSTRAPPER_LOG_LEVEL_STANDARD, "Loading prerequisite bootstrapper application because managed host could not be loaded, error: 0x%08x.", hr); | ||
128 | |||
129 | hr = CreatePrerequisiteBA(hrHostInitialization, pEngine, pArgs, pResults); | ||
130 | BalExitOnFailure(hr, "Failed to create the pre-requisite bootstrapper application."); | ||
131 | } | ||
132 | |||
133 | LExit: | ||
134 | return hr; | ||
135 | } | ||
136 | |||
137 | extern "C" void WINAPI BootstrapperApplicationDestroy() | ||
138 | { | ||
139 | if (vpAppDomain) | ||
140 | { | ||
141 | HRESULT hr = vpCLRHost->UnloadDomain(vpAppDomain); | ||
142 | if (FAILED(hr)) | ||
143 | { | ||
144 | BalLogError(hr, "Failed to unload app domain."); | ||
145 | } | ||
146 | |||
147 | vpAppDomain->Release(); | ||
148 | } | ||
149 | |||
150 | if (vpCLRHost) | ||
151 | { | ||
152 | vpCLRHost->Stop(); | ||
153 | vpCLRHost->Release(); | ||
154 | } | ||
155 | |||
156 | if (vhMbapreqModule) | ||
157 | { | ||
158 | PFN_BOOTSTRAPPER_APPLICATION_DESTROY pfnDestroy = reinterpret_cast<PFN_BOOTSTRAPPER_APPLICATION_DESTROY>(::GetProcAddress(vhMbapreqModule, "MbaPrereqBootstrapperApplicationDestroy")); | ||
159 | if (pfnDestroy) | ||
160 | { | ||
161 | (*pfnDestroy)(); | ||
162 | } | ||
163 | |||
164 | ::FreeLibrary(vhMbapreqModule); | ||
165 | vhMbapreqModule = NULL; | ||
166 | } | ||
167 | |||
168 | BalUninitialize(); | ||
169 | } | ||
170 | |||
171 | // Gets the custom AppDomain for loading managed BA. | ||
172 | static HRESULT GetAppDomain( | ||
173 | __out _AppDomain **ppAppDomain | ||
174 | ) | ||
175 | { | ||
176 | HRESULT hr = S_OK; | ||
177 | ICorRuntimeHost *pCLRHost = NULL; | ||
178 | IUnknown *pUnk = NULL; | ||
179 | LPWSTR sczAppBase = NULL; | ||
180 | LPWSTR sczConfigPath = NULL; | ||
181 | IAppDomainSetup *pAppDomainSetup; | ||
182 | BSTR bstrAppBase = NULL; | ||
183 | BSTR bstrConfigPath = NULL; | ||
184 | |||
185 | hr = GetAppBase(&sczAppBase); | ||
186 | ExitOnFailure(hr, "Failed to get the host base path."); | ||
187 | |||
188 | hr = PathConcat(sczAppBase, L"BootstrapperCore.config", &sczConfigPath); | ||
189 | ExitOnFailure(hr, "Failed to get the full path to the application configuration file."); | ||
190 | |||
191 | // Check that the supported framework is installed. | ||
192 | hr = CheckSupportedFrameworks(sczConfigPath); | ||
193 | ExitOnFailure(hr, "Failed to find supported framework."); | ||
194 | |||
195 | // Load the CLR. | ||
196 | hr = GetCLRHost(sczConfigPath, &pCLRHost); | ||
197 | ExitOnFailure(hr, "Failed to create the CLR host."); | ||
198 | |||
199 | hr = pCLRHost->Start(); | ||
200 | ExitOnRootFailure(hr, "Failed to start the CLR host."); | ||
201 | |||
202 | // Create the setup information for a new AppDomain to set the app base and config. | ||
203 | hr = pCLRHost->CreateDomainSetup(&pUnk); | ||
204 | ExitOnRootFailure(hr, "Failed to create the AppDomainSetup object."); | ||
205 | |||
206 | hr = pUnk->QueryInterface(__uuidof(IAppDomainSetup), reinterpret_cast<LPVOID*>(&pAppDomainSetup)); | ||
207 | ExitOnRootFailure(hr, "Failed to query for the IAppDomainSetup interface."); | ||
208 | ReleaseNullObject(pUnk); | ||
209 | |||
210 | // Set properties on the AppDomainSetup object. | ||
211 | bstrAppBase = ::SysAllocString(sczAppBase); | ||
212 | ExitOnNull(bstrAppBase, hr, E_OUTOFMEMORY, "Failed to allocate the application base path for the AppDomainSetup."); | ||
213 | |||
214 | hr = pAppDomainSetup->put_ApplicationBase(bstrAppBase); | ||
215 | ExitOnRootFailure(hr, "Failed to set the application base path for the AppDomainSetup."); | ||
216 | |||
217 | bstrConfigPath = ::SysAllocString(sczConfigPath); | ||
218 | ExitOnNull(bstrConfigPath, hr, E_OUTOFMEMORY, "Failed to allocate the application configuration file for the AppDomainSetup."); | ||
219 | |||
220 | hr = pAppDomainSetup->put_ConfigurationFile(bstrConfigPath); | ||
221 | ExitOnRootFailure(hr, "Failed to set the configuration file path for the AppDomainSetup."); | ||
222 | |||
223 | // Create the AppDomain to load the factory type. | ||
224 | hr = pCLRHost->CreateDomainEx(L"MBA", pAppDomainSetup, NULL, &pUnk); | ||
225 | ExitOnRootFailure(hr, "Failed to create the MBA AppDomain."); | ||
226 | |||
227 | hr = pUnk->QueryInterface(__uuidof(_AppDomain), reinterpret_cast<LPVOID*>(ppAppDomain)); | ||
228 | ExitOnRootFailure(hr, "Failed to query for the _AppDomain interface."); | ||
229 | |||
230 | LExit: | ||
231 | ReleaseBSTR(bstrConfigPath); | ||
232 | ReleaseBSTR(bstrAppBase); | ||
233 | ReleaseStr(sczConfigPath); | ||
234 | ReleaseStr(sczAppBase); | ||
235 | ReleaseNullObject(pUnk); | ||
236 | ReleaseNullObject(pCLRHost); | ||
237 | |||
238 | return hr; | ||
239 | } | ||
240 | |||
241 | static HRESULT GetAppBase( | ||
242 | __out LPWSTR *psczAppBase | ||
243 | ) | ||
244 | { | ||
245 | HRESULT hr = S_OK; | ||
246 | LPWSTR sczFullPath = NULL; | ||
247 | |||
248 | hr = PathForCurrentProcess(&sczFullPath, vhInstance); | ||
249 | ExitOnFailure(hr, "Failed to get the full host path."); | ||
250 | |||
251 | hr = PathGetDirectory(sczFullPath, psczAppBase); | ||
252 | ExitOnFailure(hr, "Failed to get the directory of the full process path."); | ||
253 | |||
254 | LExit: | ||
255 | ReleaseStr(sczFullPath); | ||
256 | |||
257 | return hr; | ||
258 | } | ||
259 | |||
260 | // Checks whether at least one of required supported frameworks is installed via the NETFX registry keys. | ||
261 | static HRESULT CheckSupportedFrameworks( | ||
262 | __in LPCWSTR wzConfigPath | ||
263 | ) | ||
264 | { | ||
265 | HRESULT hr = S_OK; | ||
266 | IXMLDOMDocument* pixdManifest = NULL; | ||
267 | IXMLDOMNodeList* pNodeList = NULL; | ||
268 | IXMLDOMNode* pNode = NULL; | ||
269 | DWORD cSupportedFrameworks = 0; | ||
270 | LPWSTR sczSupportedFrameworkVersion = NULL; | ||
271 | LPWSTR sczFrameworkRegistryKey = NULL; | ||
272 | HKEY hkFramework = NULL; | ||
273 | DWORD dwFrameworkInstalled = 0; | ||
274 | BOOL fUpdatedManifest = FALSE; | ||
275 | |||
276 | hr = XmlInitialize(); | ||
277 | ExitOnFailure(hr, "Failed to initialize XML."); | ||
278 | |||
279 | hr = XmlLoadDocumentFromFile(wzConfigPath, &pixdManifest); | ||
280 | ExitOnFailure(hr, "Failed to load bootstrapper config file from path: %ls", wzConfigPath); | ||
281 | |||
282 | hr = XmlSelectNodes(pixdManifest, L"/configuration/wix.bootstrapper/host/supportedFramework", &pNodeList); | ||
283 | ExitOnFailure(hr, "Failed to select all supportedFramework elements."); | ||
284 | |||
285 | hr = pNodeList->get_length(reinterpret_cast<long*>(&cSupportedFrameworks)); | ||
286 | ExitOnFailure(hr, "Failed to get the supported framework count."); | ||
287 | |||
288 | if (cSupportedFrameworks) | ||
289 | { | ||
290 | while (S_OK == (hr = XmlNextElement(pNodeList, &pNode, NULL))) | ||
291 | { | ||
292 | hr = XmlGetAttributeEx(pNode, L"version", &sczSupportedFrameworkVersion); | ||
293 | ExitOnFailure(hr, "Failed to get supportedFramework/@version."); | ||
294 | |||
295 | hr = StrAllocFormatted(&sczFrameworkRegistryKey, L"SOFTWARE\\Microsoft\\NET Framework Setup\\NDP\\%ls", sczSupportedFrameworkVersion); | ||
296 | ExitOnFailure(hr, "Failed to allocate path to supported framework Install registry key."); | ||
297 | |||
298 | hr = RegOpen(HKEY_LOCAL_MACHINE, sczFrameworkRegistryKey, KEY_READ, &hkFramework); | ||
299 | if (SUCCEEDED(hr)) | ||
300 | { | ||
301 | hr = RegReadNumber(hkFramework, L"Install", &dwFrameworkInstalled); | ||
302 | if (dwFrameworkInstalled) | ||
303 | { | ||
304 | hr = S_OK; | ||
305 | break; | ||
306 | } | ||
307 | } | ||
308 | |||
309 | ReleaseNullObject(pNode); | ||
310 | } | ||
311 | |||
312 | // If we looped through all the supported frameworks but didn't find anything, ensure we return a failure. | ||
313 | if (S_FALSE == hr) | ||
314 | { | ||
315 | hr = E_NOTFOUND; | ||
316 | ExitOnRootFailure(hr, "Failed to find a supported framework."); | ||
317 | } | ||
318 | |||
319 | hr = UpdateSupportedRuntime(pixdManifest, pNode, &fUpdatedManifest); | ||
320 | ExitOnFailure(hr, "Failed to update supportedRuntime."); | ||
321 | } | ||
322 | // else no supported frameworks specified, so the startup/supportedRuntime must be enough. | ||
323 | |||
324 | if (fUpdatedManifest) | ||
325 | { | ||
326 | hr = XmlSaveDocument(pixdManifest, wzConfigPath); | ||
327 | ExitOnFailure(hr, "Failed to save updated manifest over config file: %ls", wzConfigPath); | ||
328 | } | ||
329 | |||
330 | LExit: | ||
331 | ReleaseRegKey(hkFramework); | ||
332 | ReleaseStr(sczFrameworkRegistryKey); | ||
333 | ReleaseStr(sczSupportedFrameworkVersion); | ||
334 | ReleaseObject(pNode); | ||
335 | ReleaseObject(pNodeList); | ||
336 | ReleaseObject(pixdManifest); | ||
337 | |||
338 | XmlUninitialize(); | ||
339 | |||
340 | return hr; | ||
341 | } | ||
342 | |||
343 | // Fixes the supportedRuntime element if necessary. | ||
344 | static HRESULT UpdateSupportedRuntime( | ||
345 | __in IXMLDOMDocument* pixdManifest, | ||
346 | __in IXMLDOMNode* pixnSupportedFramework, | ||
347 | __out BOOL* pfUpdatedManifest | ||
348 | ) | ||
349 | { | ||
350 | HRESULT hr = S_OK; | ||
351 | LPWSTR sczSupportedRuntimeVersion = NULL; | ||
352 | IXMLDOMNode* pixnStartup = NULL; | ||
353 | IXMLDOMNode* pixnSupportedRuntime = NULL; | ||
354 | |||
355 | *pfUpdatedManifest = FALSE; | ||
356 | |||
357 | // If the runtime version attribute is not specified, don't update the manifest. | ||
358 | hr = XmlGetAttributeEx(pixnSupportedFramework, L"runtimeVersion", &sczSupportedRuntimeVersion); | ||
359 | if (E_NOTFOUND == hr) | ||
360 | { | ||
361 | ExitFunction1(hr = S_OK); | ||
362 | } | ||
363 | ExitOnFailure(hr, "Failed to get supportedFramework/@runtimeVersion."); | ||
364 | |||
365 | // Get the startup element. Fail if we can't find it since it'll be necessary to load the | ||
366 | // correct runtime. | ||
367 | hr = XmlSelectSingleNode(pixdManifest, L"/configuration/startup", &pixnStartup); | ||
368 | ExitOnFailure(hr, "Failed to get startup element."); | ||
369 | |||
370 | if (S_FALSE == hr) | ||
371 | { | ||
372 | hr = E_NOTFOUND; | ||
373 | ExitOnRootFailure(hr, "Failed to find startup element in bootstrapper application config."); | ||
374 | } | ||
375 | |||
376 | // Remove any pre-existing supported runtimes because they'll just get in the way and create our new one. | ||
377 | hr = XmlRemoveChildren(pixnStartup, L"supportedRuntime"); | ||
378 | ExitOnFailure(hr, "Failed to remove pre-existing supportedRuntime elements."); | ||
379 | |||
380 | hr = XmlCreateChild(pixnStartup, L"supportedRuntime", &pixnSupportedRuntime); | ||
381 | ExitOnFailure(hr, "Failed to create supportedRuntime element."); | ||
382 | |||
383 | hr = XmlSetAttribute(pixnSupportedRuntime, L"version", sczSupportedRuntimeVersion); | ||
384 | ExitOnFailure(hr, "Failed to set supportedRuntime/@version to '%ls'.", sczSupportedRuntimeVersion); | ||
385 | |||
386 | *pfUpdatedManifest = TRUE; | ||
387 | |||
388 | LExit: | ||
389 | ReleaseObject(pixnSupportedRuntime); | ||
390 | ReleaseObject(pixnStartup); | ||
391 | ReleaseStr(sczSupportedRuntimeVersion); | ||
392 | |||
393 | return hr; | ||
394 | } | ||
395 | |||
396 | // Gets the CLR host and caches it. | ||
397 | static HRESULT GetCLRHost( | ||
398 | __in LPCWSTR wzConfigPath, | ||
399 | __out ICorRuntimeHost **ppCLRHost | ||
400 | ) | ||
401 | { | ||
402 | HRESULT hr = S_OK; | ||
403 | UINT uiMode = 0; | ||
404 | HMODULE hModule = NULL; | ||
405 | BOOL fFallbackToCorBindToCurrentRuntime = TRUE; | ||
406 | CLRCreateInstanceFnPtr pfnCLRCreateInstance = NULL; | ||
407 | ICLRMetaHostPolicy* pCLRMetaHostPolicy = NULL; | ||
408 | IStream* pCfgStream = NULL; | ||
409 | LPWSTR pwzVersion = NULL; | ||
410 | DWORD cchVersion = 0; | ||
411 | DWORD dwConfigFlags = 0; | ||
412 | ICLRRuntimeInfo* pCLRRuntimeInfo = NULL; | ||
413 | PFN_CORBINDTOCURRENTRUNTIME pfnCorBindToCurrentRuntime = NULL; | ||
414 | |||
415 | // Always set the error mode because we will always restore it below. | ||
416 | uiMode = ::SetErrorMode(0); | ||
417 | |||
418 | // Cache the CLR host to be shutdown later. This can occur on a different thread. | ||
419 | if (!vpCLRHost) | ||
420 | { | ||
421 | // Disable message boxes from being displayed on error and blocking execution. | ||
422 | ::SetErrorMode(uiMode | SEM_FAILCRITICALERRORS); | ||
423 | |||
424 | hr = LoadSystemLibrary(L"mscoree.dll", &hModule); | ||
425 | ExitOnFailure(hr, "Failed to load mscoree.dll"); | ||
426 | |||
427 | pfnCLRCreateInstance = reinterpret_cast<CLRCreateInstanceFnPtr>(::GetProcAddress(hModule, "CLRCreateInstance")); | ||
428 | |||
429 | if (pfnCLRCreateInstance) | ||
430 | { | ||
431 | hr = pfnCLRCreateInstance(CLSID_CLRMetaHostPolicy, IID_ICLRMetaHostPolicy, reinterpret_cast<LPVOID*>(&pCLRMetaHostPolicy)); | ||
432 | if (E_NOTIMPL != hr) | ||
433 | { | ||
434 | ExitOnRootFailure(hr, "Failed to create instance of ICLRMetaHostPolicy."); | ||
435 | |||
436 | fFallbackToCorBindToCurrentRuntime = FALSE; | ||
437 | } | ||
438 | } | ||
439 | |||
440 | if (fFallbackToCorBindToCurrentRuntime) | ||
441 | { | ||
442 | pfnCorBindToCurrentRuntime = reinterpret_cast<PFN_CORBINDTOCURRENTRUNTIME>(::GetProcAddress(hModule, "CorBindToCurrentRuntime")); | ||
443 | ExitOnNullWithLastError(pfnCorBindToCurrentRuntime, hr, "Failed to get procedure address for CorBindToCurrentRuntime."); | ||
444 | |||
445 | hr = pfnCorBindToCurrentRuntime(wzConfigPath, CLSID_CorRuntimeHost, IID_ICorRuntimeHost, reinterpret_cast<LPVOID*>(&vpCLRHost)); | ||
446 | ExitOnRootFailure(hr, "Failed to create the CLR host using the application configuration file path."); | ||
447 | } | ||
448 | else | ||
449 | { | ||
450 | |||
451 | hr = SHCreateStreamOnFileEx(wzConfigPath, STGM_READ | STGM_SHARE_DENY_WRITE, 0, FALSE, NULL, &pCfgStream); | ||
452 | ExitOnFailure(hr, "Failed to load bootstrapper config file from path: %ls", wzConfigPath); | ||
453 | |||
454 | hr = pCLRMetaHostPolicy->GetRequestedRuntime(METAHOST_POLICY_HIGHCOMPAT, NULL, pCfgStream, NULL, &cchVersion, NULL, NULL, &dwConfigFlags, IID_ICLRRuntimeInfo, reinterpret_cast<LPVOID*>(&pCLRRuntimeInfo)); | ||
455 | ExitOnRootFailure(hr, "Failed to get the CLR runtime info using the application configuration file path."); | ||
456 | |||
457 | // .NET 4 RTM had a bug where it wouldn't set pcchVersion if pwzVersion was NULL. | ||
458 | if (!cchVersion) | ||
459 | { | ||
460 | hr = pCLRRuntimeInfo->GetVersionString(NULL, &cchVersion); | ||
461 | if (HRESULT_FROM_WIN32(ERROR_INSUFFICIENT_BUFFER) != hr) | ||
462 | { | ||
463 | ExitOnFailure(hr, "Failed to get the length of the CLR version string."); | ||
464 | } | ||
465 | } | ||
466 | |||
467 | hr = StrAlloc(&pwzVersion, cchVersion); | ||
468 | ExitOnFailure(hr, "Failed to allocate the CLR version string."); | ||
469 | |||
470 | hr = pCLRRuntimeInfo->GetVersionString(pwzVersion, &cchVersion); | ||
471 | ExitOnFailure(hr, "Failed to get the CLR version string."); | ||
472 | |||
473 | if (CSTR_EQUAL == CompareString(LOCALE_NEUTRAL, 0, L"v4.0.30319", -1, pwzVersion, cchVersion)) | ||
474 | { | ||
475 | hr = VerifyNET4RuntimeIsSupported(); | ||
476 | ExitOnFailure(hr, "Found unsupported .NET 4 Runtime."); | ||
477 | } | ||
478 | |||
479 | if (METAHOST_CONFIG_FLAGS_LEGACY_V2_ACTIVATION_POLICY_TRUE == (METAHOST_CONFIG_FLAGS_LEGACY_V2_ACTIVATION_POLICY_MASK & dwConfigFlags)) | ||
480 | { | ||
481 | hr = pCLRRuntimeInfo->BindAsLegacyV2Runtime(); | ||
482 | ExitOnRootFailure(hr, "Failed to bind as legacy V2 runtime."); | ||
483 | } | ||
484 | |||
485 | hr = pCLRRuntimeInfo->GetInterface(CLSID_CorRuntimeHost, IID_ICorRuntimeHost, reinterpret_cast<LPVOID*>(&vpCLRHost)); | ||
486 | ExitOnRootFailure(hr, "Failed to get instance of ICorRuntimeHost."); | ||
487 | |||
488 | // TODO: use ICLRRuntimeHost instead of ICorRuntimeHost on .NET 4 since the former is faster and the latter is deprecated | ||
489 | //hr = pCLRRuntimeInfo->GetInterface(CLSID_CLRRuntimeHost, IID_ICLRRuntimeHost, reinterpret_cast<LPVOID*>(&pCLRRuntimeHost)); | ||
490 | //ExitOnRootFailure(hr, "Failed to get instance of ICLRRuntimeHost."); | ||
491 | } | ||
492 | } | ||
493 | |||
494 | vpCLRHost->AddRef(); | ||
495 | *ppCLRHost = vpCLRHost; | ||
496 | |||
497 | LExit: | ||
498 | ReleaseStr(pwzVersion); | ||
499 | ReleaseNullObject(pCLRRuntimeInfo); | ||
500 | ReleaseNullObject(pCfgStream); | ||
501 | ReleaseNullObject(pCLRMetaHostPolicy); | ||
502 | |||
503 | // Unload the module so it's not in use when we install .NET. | ||
504 | if (FAILED(hr)) | ||
505 | { | ||
506 | ::FreeLibrary(hModule); | ||
507 | } | ||
508 | |||
509 | ::SetErrorMode(uiMode); // restore the previous error mode. | ||
510 | |||
511 | return hr; | ||
512 | } | ||
513 | |||
514 | // Creates the bootstrapper app and returns it for the engine. | ||
515 | static HRESULT CreateManagedBootstrapperApplication( | ||
516 | __in _AppDomain* pAppDomain, | ||
517 | __in IBootstrapperEngine* pEngine, | ||
518 | __in const BOOTSTRAPPER_CREATE_ARGS* pArgs, | ||
519 | __inout BOOTSTRAPPER_CREATE_RESULTS* pResults | ||
520 | ) | ||
521 | { | ||
522 | HRESULT hr = S_OK; | ||
523 | IBootstrapperApplicationFactory* pAppFactory = NULL; | ||
524 | IBootstrapperApplication* pApp = NULL; | ||
525 | |||
526 | hr = CreateManagedBootstrapperApplicationFactory(pAppDomain, &pAppFactory); | ||
527 | ExitOnFailure(hr, "Failed to create the factory to create the bootstrapper application."); | ||
528 | |||
529 | hr = pAppFactory->Create(pEngine, pArgs->pCommand, &pApp); | ||
530 | ExitOnFailure(hr, "Failed to create the bootstrapper application."); | ||
531 | |||
532 | pResults->pfnBootstrapperApplicationProc = BalBaseBootstrapperApplicationProc; | ||
533 | pResults->pvBootstrapperApplicationProcContext = pApp; | ||
534 | pApp = NULL; | ||
535 | |||
536 | LExit: | ||
537 | ReleaseNullObject(pApp); | ||
538 | ReleaseNullObject(pAppFactory); | ||
539 | |||
540 | return hr; | ||
541 | } | ||
542 | |||
543 | // Creates the app factory to create the managed app in the default AppDomain. | ||
544 | static HRESULT CreateManagedBootstrapperApplicationFactory( | ||
545 | __in _AppDomain* pAppDomain, | ||
546 | __out IBootstrapperApplicationFactory** ppAppFactory | ||
547 | ) | ||
548 | { | ||
549 | HRESULT hr = S_OK; | ||
550 | BSTR bstrAssemblyName = NULL; | ||
551 | BSTR bstrTypeName = NULL; | ||
552 | _ObjectHandle* pObj = NULL; | ||
553 | VARIANT vtBAFactory; | ||
554 | |||
555 | ::VariantInit(&vtBAFactory); | ||
556 | |||
557 | bstrAssemblyName = ::SysAllocString(MUX_ASSEMBLY_FULL_NAME); | ||
558 | ExitOnNull(bstrAssemblyName, hr, E_OUTOFMEMORY, "Failed to allocate the full assembly name for the bootstrapper application factory."); | ||
559 | |||
560 | bstrTypeName = ::SysAllocString(L"WixToolset.Bootstrapper.BootstrapperApplicationFactory"); | ||
561 | ExitOnNull(bstrTypeName, hr, E_OUTOFMEMORY, "Failed to allocate the full type name for the BA factory."); | ||
562 | |||
563 | hr = pAppDomain->CreateInstance(bstrAssemblyName, bstrTypeName, &pObj); | ||
564 | ExitOnRootFailure(hr, "Failed to create the BA factory object."); | ||
565 | |||
566 | hr = pObj->Unwrap(&vtBAFactory); | ||
567 | ExitOnRootFailure(hr, "Failed to unwrap the BA factory object into the host domain."); | ||
568 | ExitOnNull(vtBAFactory.punkVal, hr, E_UNEXPECTED, "The variant did not contain the expected IUnknown pointer."); | ||
569 | |||
570 | hr = vtBAFactory.punkVal->QueryInterface(__uuidof(IBootstrapperApplicationFactory), reinterpret_cast<LPVOID*>(ppAppFactory)); | ||
571 | ExitOnRootFailure(hr, "Failed to query for the bootstrapper app factory interface."); | ||
572 | |||
573 | LExit: | ||
574 | ReleaseVariant(vtBAFactory); | ||
575 | ReleaseNullObject(pObj); | ||
576 | ReleaseBSTR(bstrTypeName); | ||
577 | ReleaseBSTR(bstrAssemblyName); | ||
578 | |||
579 | return hr; | ||
580 | } | ||
581 | |||
582 | static HRESULT CreatePrerequisiteBA( | ||
583 | __in HRESULT hrHostInitialization, | ||
584 | __in IBootstrapperEngine* pEngine, | ||
585 | __in const BOOTSTRAPPER_CREATE_ARGS* pArgs, | ||
586 | __inout BOOTSTRAPPER_CREATE_RESULTS* pResults | ||
587 | ) | ||
588 | { | ||
589 | HRESULT hr = S_OK; | ||
590 | LPWSTR sczMbapreqPath = NULL; | ||
591 | HMODULE hModule = NULL; | ||
592 | |||
593 | hr = PathRelativeToModule(&sczMbapreqPath, L"mbapreq.dll", vhInstance); | ||
594 | ExitOnFailure(hr, "Failed to get path to pre-requisite BA."); | ||
595 | |||
596 | hModule = ::LoadLibraryW(sczMbapreqPath); | ||
597 | ExitOnNullWithLastError(hModule, hr, "Failed to load pre-requisite BA DLL."); | ||
598 | |||
599 | PFN_MBAPREQ_BOOTSTRAPPER_APPLICATION_CREATE pfnCreate = reinterpret_cast<PFN_MBAPREQ_BOOTSTRAPPER_APPLICATION_CREATE>(::GetProcAddress(hModule, "MbaPrereqBootstrapperApplicationCreate")); | ||
600 | ExitOnNullWithLastError(pfnCreate, hr, "Failed to get MbaPrereqBootstrapperApplicationCreate entry-point from: %ls", sczMbapreqPath); | ||
601 | |||
602 | hr = pfnCreate(hrHostInitialization, pEngine, pArgs, pResults); | ||
603 | ExitOnFailure(hr, "Failed to create prequisite bootstrapper app."); | ||
604 | |||
605 | vhMbapreqModule = hModule; | ||
606 | hModule = NULL; | ||
607 | |||
608 | LExit: | ||
609 | if (hModule) | ||
610 | { | ||
611 | ::FreeLibrary(hModule); | ||
612 | } | ||
613 | ReleaseStr(sczMbapreqPath); | ||
614 | |||
615 | return hr; | ||
616 | } | ||
617 | |||
618 | static HRESULT VerifyNET4RuntimeIsSupported( | ||
619 | ) | ||
620 | { | ||
621 | HRESULT hr = S_OK; | ||
622 | OS_VERSION osv = OS_VERSION_UNKNOWN; | ||
623 | DWORD dwServicePack = 0; | ||
624 | HKEY hKey = NULL; | ||
625 | DWORD er = ERROR_SUCCESS; | ||
626 | DWORD dwRelease = 0; | ||
627 | DWORD cchRelease = sizeof(dwRelease); | ||
628 | |||
629 | OsGetVersion(&osv, &dwServicePack); | ||
630 | if (OS_VERSION_WIN7 == osv && 0 == dwServicePack) | ||
631 | { | ||
632 | hr = RegOpen(HKEY_LOCAL_MACHINE, L"SOFTWARE\\Microsoft\\NET Framework Setup\\NDP\\v4\\Full", KEY_QUERY_VALUE, &hKey); | ||
633 | if (E_FILENOTFOUND == hr) | ||
634 | { | ||
635 | ExitFunction1(hr = S_OK); | ||
636 | } | ||
637 | ExitOnFailure(hr, "Failed to open registry key for .NET 4."); | ||
638 | |||
639 | er = ::RegQueryValueExW(hKey, L"Release", NULL, NULL, reinterpret_cast<LPBYTE>(&dwRelease), &cchRelease); | ||
640 | if (ERROR_FILE_NOT_FOUND == er) | ||
641 | { | ||
642 | ExitFunction1(hr = S_OK); | ||
643 | } | ||
644 | ExitOnWin32Error(er, hr, "Failed to get Release value."); | ||
645 | |||
646 | if (NET452_RELEASE <= dwRelease) | ||
647 | { | ||
648 | hr = E_MBAHOST_NET452_ON_WIN7RTM; | ||
649 | } | ||
650 | } | ||
651 | |||
652 | LExit: | ||
653 | ReleaseRegKey(hKey); | ||
654 | |||
655 | return hr; | ||
656 | } | ||
diff --git a/src/mbahost/host.def b/src/mbahost/host.def new file mode 100644 index 00000000..4488df94 --- /dev/null +++ b/src/mbahost/host.def | |||
@@ -0,0 +1,6 @@ | |||
1 | ; Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. | ||
2 | |||
3 | |||
4 | EXPORTS | ||
5 | BootstrapperApplicationCreate | ||
6 | BootstrapperApplicationDestroy | ||
diff --git a/src/mbahost/host.vcxproj b/src/mbahost/host.vcxproj new file mode 100644 index 00000000..a4c3ea16 --- /dev/null +++ b/src/mbahost/host.vcxproj | |||
@@ -0,0 +1,51 @@ | |||
1 | <?xml version="1.0" encoding="utf-8"?> | ||
2 | <!-- Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. --> | ||
3 | |||
4 | |||
5 | <Project DefaultTargets="Build" ToolsVersion="4.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003"> | ||
6 | <ItemGroup Label="ProjectConfigurations"> | ||
7 | <ProjectConfiguration Include="Debug|Win32"> | ||
8 | <Configuration>Debug</Configuration> | ||
9 | <Platform>Win32</Platform> | ||
10 | </ProjectConfiguration> | ||
11 | <ProjectConfiguration Include="Release|Win32"> | ||
12 | <Configuration>Release</Configuration> | ||
13 | <Platform>Win32</Platform> | ||
14 | </ProjectConfiguration> | ||
15 | </ItemGroup> | ||
16 | |||
17 | <PropertyGroup Label="Globals"> | ||
18 | <ProjectGuid>{12C87C77-3547-44F8-8134-29BC915CB19D}</ProjectGuid> | ||
19 | <ConfigurationType>DynamicLibrary</ConfigurationType> | ||
20 | <CharacterSet>Unicode</CharacterSet> | ||
21 | <TargetName>mbahost</TargetName> | ||
22 | <ProjectModuleDefinitionFile>host.def</ProjectModuleDefinitionFile> | ||
23 | </PropertyGroup> | ||
24 | |||
25 | <Import Project="$([MSBuild]::GetDirectoryNameOfFileAbove($(MSBuildProjectDirectory), wix.proj))\tools\WixBuild.props" /> | ||
26 | |||
27 | <PropertyGroup> | ||
28 | <ProjectAdditionalIncludeDirectories>$(WixRoot)src\libs\dutil\inc;$(WixRoot)src\burn\inc;$(WixRoot)src\libs\balutil\inc;$(BaseIntermediateOutputPath)\core</ProjectAdditionalIncludeDirectories> | ||
29 | <ProjectAdditionalLinkLibraries>dutil.lib;balutil.lib;shlwapi.lib</ProjectAdditionalLinkLibraries> | ||
30 | </PropertyGroup> | ||
31 | |||
32 | <ItemGroup> | ||
33 | <ClCompile Include="host.cpp" /> | ||
34 | </ItemGroup> | ||
35 | <ItemGroup> | ||
36 | <ClInclude Include="IBootstrapperApplicationFactory.h" /> | ||
37 | <ClInclude Include="precomp.h" /> | ||
38 | </ItemGroup> | ||
39 | <ItemGroup> | ||
40 | <None Include="host.def" /> | ||
41 | </ItemGroup> | ||
42 | <ItemGroup> | ||
43 | <ResourceCompile Include="host.rc" /> | ||
44 | </ItemGroup> | ||
45 | |||
46 | <ItemGroup> | ||
47 | <ProjectReference Include="..\core\core.csproj" /> | ||
48 | </ItemGroup> | ||
49 | |||
50 | <Import Project="$([MSBuild]::GetDirectoryNameOfFileAbove($(MSBuildProjectDirectory), wix.proj))\tools\WixBuild.targets" /> | ||
51 | </Project> | ||
diff --git a/src/mbahost/precomp.h b/src/mbahost/precomp.h new file mode 100644 index 00000000..d29a23f3 --- /dev/null +++ b/src/mbahost/precomp.h | |||
@@ -0,0 +1,25 @@ | |||
1 | #pragma once | ||
2 | // Copyright (c) .NET Foundation and contributors. All rights reserved. Licensed under the Microsoft Reciprocal License. See LICENSE.TXT file in the project root for full license information. | ||
3 | |||
4 | |||
5 | #include <windows.h> | ||
6 | #include <msiquery.h> | ||
7 | #include <metahost.h> | ||
8 | #include <shlwapi.h> | ||
9 | |||
10 | #import <mscorlib.tlb> raw_interfaces_only rename("ReportEvent", "mscorlib_ReportEvent") | ||
11 | |||
12 | #include <dutil.h> | ||
13 | #include <osutil.h> | ||
14 | #include <pathutil.h> | ||
15 | #include <regutil.h> | ||
16 | #include <strutil.h> | ||
17 | #include <xmlutil.h> | ||
18 | |||
19 | #include "BootstrapperEngine.h" | ||
20 | #include "BootstrapperApplication.h" | ||
21 | #include "IBootstrapperEngine.h" | ||
22 | #include "IBootstrapperApplication.h" | ||
23 | #include "IBootstrapperApplicationFactory.h" | ||
24 | |||
25 | #include "balutil.h" | ||