1 | // Licensed to the .NET Foundation under one or more agreements. |
2 | // The .NET Foundation licenses this file to you under the MIT license. |
3 | // See the LICENSE file in the project root for more information. |
4 | //***************************************************************************** |
5 | // CorDB.cpp |
6 | // |
7 | |
8 | // |
9 | // Dll* routines for entry points, and support for COM framework. The class |
10 | // factory and other routines live in this module. |
11 | // |
12 | //***************************************************************************** |
13 | #include "stdafx.h" |
14 | #include "classfactory.h" |
15 | #include "corsym.h" |
16 | #include "contract.h" |
17 | #include "metadataexports.h" |
18 | #if defined(FEATURE_DBGIPC_TRANSPORT_DI) |
19 | #include "dbgtransportsession.h" |
20 | #include "dbgtransportmanager.h" |
21 | #endif // FEATURE_DBGIPC_TRANSPORT_DI |
22 | |
23 | #if defined(PLATFORM_UNIX) || defined(__ANDROID__) |
24 | // Local (in-process) debugging is not supported for UNIX and Android. |
25 | #define SUPPORT_LOCAL_DEBUGGING 0 |
26 | #else |
27 | #define SUPPORT_LOCAL_DEBUGGING 1 |
28 | #endif |
29 | |
30 | //********** Globals. ********************************************************* |
31 | #ifndef FEATURE_PAL |
32 | HINSTANCE g_hInst; // Instance handle to this piece of code. |
33 | #endif |
34 | |
35 | //----------------------------------------------------------------------------- |
36 | // SxS Versioning story for Mscordbi (ICorDebug + friends) |
37 | //----------------------------------------------------------------------------- |
38 | |
39 | //----------------------------------------------------------------------------- |
40 | // In v1.0, we declared that mscordbi was a "shared" component, which means |
41 | // that we promised to provide it from now until the end of time. So every CLR implementation |
42 | // needs an Mscordbi that implements the everett guids for CorDebug + CorPublish. |
43 | // |
44 | // This works fine for CorPublish, which is truly shared. |
45 | // CorDebug however is "versioned" not "shared" - each version of the CLR has its own disjoint copy. |
46 | // |
47 | // Thus creating a CorDebug object requires a version parameter. |
48 | // CoCreateInstance doesn't have a the version param, so we use the new (v2.0+) |
49 | // shim interface CreateDebuggingInterfaceFromVersion. |
50 | // |
51 | // ** So in summary: ** |
52 | // - Dlls don't do self-registration; they're registered by setup using .vrg files. |
53 | // - All CLR versions (past + future) must have the same registry footprint w.r.t mscordbi. |
54 | // This just means that all CLRs have the same mscordbi.vrg file. |
55 | // - CorDebug is in fact versioned and each CLR version has its own copy. |
56 | // - In v1.0/1.1, CorDebug was a CoClass. In v2.0+, it is not a CoClass and is created via the |
57 | // CreateDebuggingInterfaceFromVersion shim API, which takes a version parameter. |
58 | // - CorDebug must be SxS. V1.1 must only get the V1.1 version, and V2.0 must only get the V2.0 version. |
59 | // V1.1: Clients will cocreate to get CorDebug. v1.1 will be the only mscordbi!DllGetClassObject |
60 | // that provides a CorDebug, so CoCreateInstance will guarantee getting a v1.1 object. |
61 | // V2.0: Clients use the new version-aware shim API, so it's not an issue. |
62 | // |
63 | // ** Preparing for Life in a Single-CLR world: ** |
64 | // In Orcas (v3), we expect to run on single-CLR. There will only be 1 mscordbi, and it will service all versions. |
65 | // For whidbey (v2), we want to be able to flip a knob and pretend to be orcas (for testing purposes). |
66 | // |
67 | // Here's how to do that: |
68 | // - copy whidbey mscordbi & dac over the everett mscordbi. |
69 | // - When VS cocreates w/ the everett-guid, it will load the mscordbi on the everett path ( |
70 | // which will be whidbey dll), and ask for the everett guid. |
71 | // - re-add CorDebug to the g_CoClasses list. |
72 | |
73 | |
74 | //********** Locals. ********************************************************** |
75 | |
76 | |
77 | //********** Code. ************************************************************ |
78 | |
79 | |
80 | //***************************************************************************** |
81 | // Standard public helper to create a Cordb object (ICorDebug instance). |
82 | // This is used by the shim to get the Cordb object out of this module. |
83 | // This is the creation path for V2.0+ for CorDebug using the in-process debugging |
84 | // architecture (ICorDebug). In CLR v4+ debugger may choose to use the out-of-process |
85 | // architecture to get an ICorDebugProcess directly (IClrDebugging::OpenVirtualProcess). |
86 | // |
87 | // This was used by the Mix07 release of Silverlight, but it didn't properly support versioning |
88 | // and we no longer support it's debugger protocol so we require callers to use |
89 | // code:CoreCLRCreateCordbObject instead. |
90 | //***************************************************************************** |
91 | STDAPI CreateCordbObject(int iDebuggerVersion, IUnknown ** ppCordb) |
92 | { |
93 | #if !defined(FEATURE_DBGIPC_TRANSPORT_DI) && !defined(FEATURE_CORESYSTEM) |
94 | // This API should not be called for Windows CoreCLR unless we are doing interop-debugging |
95 | // (which is only supported internally). Use code:CoreCLRCreateCordbObject instead. |
96 | if (CLRConfig::GetConfigValue(CLRConfig::INTERNAL_DbgEnableMixedModeDebugging) == 0) |
97 | { |
98 | _ASSERTE(!"Deprecated entry point CreateCordbObject() is called on Windows CoreCLR\n" ); |
99 | return E_NOTIMPL; |
100 | } |
101 | #endif // !defined(FEATURE_DBGIPC_TRANSPORT_DI) && !defined(FEATURE_CORESYSTEM) |
102 | |
103 | if (ppCordb == NULL) |
104 | { |
105 | return E_INVALIDARG; |
106 | } |
107 | if (iDebuggerVersion != CorDebugVersion_2_0 && iDebuggerVersion != CorDebugVersion_4_0) |
108 | { |
109 | return E_INVALIDARG; |
110 | } |
111 | |
112 | return Cordb::CreateObject( |
113 | (CorDebugInterfaceVersion)iDebuggerVersion, ProcessDescriptor::UNINITIALIZED_PID, /*lpApplicationGroupId*/ NULL, IID_ICorDebug, (void **) ppCordb); |
114 | } |
115 | |
116 | // |
117 | // Public API. |
118 | // Telesto Creation path with Mac sandbox support - only way to debug a sandboxed application on Mac. |
119 | // This supercedes code:CoreCLRCreateCordbObject |
120 | // |
121 | // Arguments: |
122 | // iDebuggerVersion - version of ICorDebug interfaces that the debugger is requesting |
123 | // pid - pid of debuggee that we're attaching to. |
124 | // lpApplicationGroupId - A string representing the application group ID of a sandboxed |
125 | // process running in Mac. Pass NULL if the process is not |
126 | // running in a sandbox and other platforms. |
127 | // hmodTargetCLR - module handle to clr in target pid that we're attaching to. |
128 | // ppCordb - (out) the resulting ICorDebug object. |
129 | // |
130 | // Notes: |
131 | // It's inconsistent that this takes a (handle, pid) but hands back an ICorDebug instead of an ICorDebugProcess. |
132 | // Callers will need to call *ppCordb->DebugActiveProcess(pid). |
133 | STDAPI CoreCLRCreateCordbObjectEx(int iDebuggerVersion, DWORD pid, LPCWSTR lpApplicationGroupId, HMODULE hmodTargetCLR, IUnknown ** ppCordb) |
134 | { |
135 | if (ppCordb == NULL) |
136 | { |
137 | return E_INVALIDARG; |
138 | } |
139 | if ((iDebuggerVersion < CorDebugVersion_2_0) || |
140 | (iDebuggerVersion > CorDebugLatestVersion)) |
141 | { |
142 | return E_INVALIDARG; |
143 | } |
144 | |
145 | // |
146 | // Create the ICorDebug object |
147 | // |
148 | RSExtSmartPtr<ICorDebug> pCordb; |
149 | Cordb::CreateObject((CorDebugInterfaceVersion)iDebuggerVersion, pid, lpApplicationGroupId, IID_ICorDebug, (void **) &pCordb); |
150 | |
151 | // |
152 | // Associate it with the target instance |
153 | // |
154 | HRESULT hr = static_cast<Cordb*>(pCordb.GetValue())->SetTargetCLR(hmodTargetCLR); |
155 | if (FAILED(hr)) |
156 | { |
157 | return hr; |
158 | } |
159 | |
160 | // |
161 | // Assign to out parameter. |
162 | // |
163 | hr = pCordb->QueryInterface(IID_IUnknown, (void**) ppCordb); |
164 | |
165 | // Implicit release of pUnk, pCordb |
166 | return hr; |
167 | } |
168 | |
169 | // |
170 | // Public API. |
171 | // Telesto Creation path - only way to debug multi-instance. |
172 | // This supercedes code:CreateCordbObject |
173 | // |
174 | // Arguments: |
175 | // iDebuggerVersion - version of ICorDebug interfaces that the debugger is requesting |
176 | // pid - pid of debuggee that we're attaching to. |
177 | // hmodTargetCLR - module handle to clr in target pid that we're attaching to. |
178 | // ppCordb - (out) the resulting ICorDebug object. |
179 | // |
180 | // Notes: |
181 | // It's inconsistent that this takes a (handle, pid) but hands back an ICorDebug instead of an ICorDebugProcess. |
182 | // Callers will need to call *ppCordb->DebugActiveProcess(pid). |
183 | STDAPI CoreCLRCreateCordbObject(int iDebuggerVersion, DWORD pid, HMODULE hmodTargetCLR, IUnknown ** ppCordb) |
184 | { |
185 | return CoreCLRCreateCordbObjectEx(iDebuggerVersion, pid, NULL, hmodTargetCLR, ppCordb); |
186 | } |
187 | |
188 | |
189 | |
190 | |
191 | |
192 | //***************************************************************************** |
193 | // The main dll entry point for this module. This routine is called by the |
194 | // OS when the dll gets loaded. Control is simply deferred to the main code. |
195 | //***************************************************************************** |
196 | BOOL WINAPI DbgDllMain(HINSTANCE hInstance, DWORD dwReason, LPVOID lpReserved) |
197 | { |
198 | // Save off the instance handle for later use. |
199 | switch (dwReason) |
200 | { |
201 | |
202 | case DLL_PROCESS_ATTACH: |
203 | { |
204 | #ifndef FEATURE_PAL |
205 | g_hInst = hInstance; |
206 | #else |
207 | int err = PAL_InitializeDLL(); |
208 | if(err != 0) |
209 | { |
210 | return FALSE; |
211 | } |
212 | #endif |
213 | |
214 | #if defined(_DEBUG) |
215 | static int BreakOnDILoad = -1; |
216 | if (BreakOnDILoad == -1) |
217 | BreakOnDILoad = CLRConfig::GetConfigValue(CLRConfig::INTERNAL_BreakOnDILoad); |
218 | |
219 | if (BreakOnDILoad) |
220 | { |
221 | _ASSERTE(!"DI Loaded" ); |
222 | } |
223 | #endif |
224 | |
225 | #if defined(LOGGING) |
226 | { |
227 | PathString rcFile; |
228 | WszGetModuleFileName(hInstance, rcFile); |
229 | LOG((LF_CORDB, LL_INFO10000, |
230 | "DI::DbgDllMain: load right side support from file '%s'\n" , |
231 | rcFile.GetUnicode())); |
232 | } |
233 | #endif |
234 | |
235 | #ifdef RSCONTRACTS |
236 | // alloc a TLS slot |
237 | DbgRSThread::s_TlsSlot = TlsAlloc(); |
238 | _ASSERTE(DbgRSThread::s_TlsSlot != TLS_OUT_OF_INDEXES); |
239 | #endif |
240 | |
241 | #if defined(FEATURE_DBGIPC_TRANSPORT_DI) |
242 | g_pDbgTransportTarget = new (nothrow) DbgTransportTarget(); |
243 | if (g_pDbgTransportTarget == NULL) |
244 | return FALSE; |
245 | |
246 | if (FAILED(g_pDbgTransportTarget->Init())) |
247 | return FALSE; |
248 | #endif // FEATURE_DBGIPC_TRANSPORT_DI |
249 | } |
250 | break; |
251 | |
252 | case DLL_THREAD_DETACH: |
253 | { |
254 | #ifdef STRESS_LOG |
255 | StressLog::ThreadDetach((ThreadStressLog*) ClrFlsGetValue(TlsIdx_StressLog)); |
256 | #endif |
257 | |
258 | #ifdef RSCONTRACTS |
259 | // DbgRSThread are lazily created when we call GetThread(), |
260 | // So we don't need to do anything in DLL_THREAD_ATTACH, |
261 | // But this is our only chance to destroy the thread object. |
262 | DbgRSThread * p = DbgRSThread::GetThread(); |
263 | |
264 | p->Destroy(); |
265 | #endif |
266 | } |
267 | break; |
268 | |
269 | case DLL_PROCESS_DETACH: |
270 | { |
271 | #if defined(FEATURE_DBGIPC_TRANSPORT_DI) |
272 | if (g_pDbgTransportTarget != NULL) |
273 | { |
274 | g_pDbgTransportTarget->Shutdown(); |
275 | delete g_pDbgTransportTarget; |
276 | g_pDbgTransportTarget = NULL; |
277 | } |
278 | #endif // FEATURE_DBGIPC_TRANSPORT_DI |
279 | |
280 | #ifdef RSCONTRACTS |
281 | TlsFree(DbgRSThread::s_TlsSlot); |
282 | DbgRSThread::s_TlsSlot = TLS_OUT_OF_INDEXES; |
283 | #endif |
284 | } |
285 | break; |
286 | } |
287 | |
288 | return TRUE; |
289 | } |
290 | |
291 | |
292 | // The obsolete v1 CLSID - see comment above for details. |
293 | static const GUID CLSID_CorDebug_V1 = {0x6fef44d0,0x39e7,0x4c77, { 0xbe,0x8e,0xc9,0xf8,0xcf,0x98,0x86,0x30}}; |
294 | |
295 | #if defined(FEATURE_DBGIPC_TRANSPORT_DI) |
296 | |
297 | // GUID for pipe-based debugging (Unix platforms) |
298 | const GUID CLSID_CorDebug_Telesto = {0x8bd1daae, 0x188e, 0x42f4, {0xb0, 0x09, 0x08, 0xfa, 0xfd, 0x17, 0x81, 0x3b}}; |
299 | |
300 | // The debug engine needs to implement an internal Visual Studio debugger interface (defined by the CPDE) |
301 | // which augments launch and attach requests so that we can obtain information from the port supplier (the |
302 | // network address of the target in our case). See RSPriv.h for the definition of the interface. (We have to |
303 | // hard code the IID and interface definition because VS does not export it, but it's not much of an issue |
304 | // since COM interfaces are completely immutable). |
305 | const GUID IID_IDebugRemoteCorDebug = {0x83C91210, 0xA34F, 0x427c, {0xB3, 0x5F, 0x79, 0xC3, 0x99, 0x5B, 0x3C, 0x14}}; |
306 | #endif // FEATURE_DBGIPC_TRANSPORT_DI |
307 | |
308 | //***************************************************************************** |
309 | // Called by COM to get a class factory for a given CLSID. If it is one we |
310 | // support, instantiate a class factory object and prepare for create instance. |
311 | //***************************************************************************** |
312 | STDAPI DllGetClassObjectInternal( // Return code. |
313 | REFCLSID rclsid, // The class to desired. |
314 | REFIID riid, // Interface wanted on class factory. |
315 | LPVOID FAR *ppv) // Return interface pointer here. |
316 | { |
317 | HRESULT hr; |
318 | CClassFactory *pClassFactory; // To create class factory object. |
319 | PFN_CREATE_OBJ pfnCreateObject = NULL; |
320 | |
321 | |
322 | #if defined(FEATURE_DBG_PUBLISH) |
323 | if (rclsid == CLSID_CorpubPublish) |
324 | { |
325 | pfnCreateObject = CorpubPublish::CreateObject; |
326 | } |
327 | else |
328 | #endif |
329 | #if defined(FEATURE_DBGIPC_TRANSPORT_DI) |
330 | if (rclsid == CLSID_CorDebug_Telesto) |
331 | { |
332 | pfnCreateObject = Cordb::CreateObjectTelesto; |
333 | } |
334 | #else // !FEATURE_DBGIPC_TRANSPORT_DI |
335 | if(rclsid == CLSID_CorDebug_V1) |
336 | { |
337 | if (0) // if (IsSingleCLR()) |
338 | { |
339 | // Don't allow creating backwards objects until we ensure that the v2.0 Right-side |
340 | // is backwards compat. This may involve using CordbProcess::SupportsVersion to conditionally |
341 | // emulate old behavior. |
342 | // If emulating V1.0, QIs for V2.0 interfaces should fail. |
343 | _ASSERTE(!"Ensure that V2.0 RS is backwards compat" ); |
344 | pfnCreateObject = Cordb::CreateObjectV1; |
345 | } |
346 | } |
347 | #endif // FEATURE_DBGIPC_TRANSPORT_DI |
348 | |
349 | if (pfnCreateObject == NULL) |
350 | return (CLASS_E_CLASSNOTAVAILABLE); |
351 | |
352 | // Allocate the new factory object. The ref count is set to 1 in the constructor. |
353 | pClassFactory = new (nothrow) CClassFactory(pfnCreateObject); |
354 | if (!pClassFactory) |
355 | return (E_OUTOFMEMORY); |
356 | |
357 | // Pick the v-table based on the caller's request. |
358 | hr = pClassFactory->QueryInterface(riid, ppv); |
359 | |
360 | // Always release the local reference, if QI failed it will be |
361 | // the only one and the object gets freed. |
362 | pClassFactory->Release(); |
363 | |
364 | return hr; |
365 | } |
366 | |
367 | #if defined(FEATURE_DBGIPC_TRANSPORT_DI) |
368 | // In V2 we started hiding DllGetClassObject because activation was no longer performed through COM directly |
369 | // (we went through the shim). CoreCLR doesn't have a shim and we go back to the COM model so we re-expose |
370 | // DllGetClassObject to make that work. |
371 | |
372 | STDAPI DllGetClassObject( // Return code. |
373 | REFCLSID rclsid, // The class to desired. |
374 | REFIID riid, // Interface wanted on class factory. |
375 | LPVOID FAR *ppv) // Return interface pointer here. |
376 | { |
377 | return DllGetClassObjectInternal(rclsid, riid, ppv); |
378 | } |
379 | #endif // FEATURE_DBGIPC_TRANSPORT_DI |
380 | |
381 | |
382 | //***************************************************************************** |
383 | // |
384 | //********** Class factory code. |
385 | // |
386 | //***************************************************************************** |
387 | |
388 | |
389 | //***************************************************************************** |
390 | // QueryInterface is called to pick a v-table on the co-class. |
391 | //***************************************************************************** |
392 | HRESULT STDMETHODCALLTYPE CClassFactory::QueryInterface( |
393 | REFIID riid, |
394 | void **ppvObject) |
395 | { |
396 | HRESULT hr; |
397 | |
398 | // Avoid confusion. |
399 | *ppvObject = NULL; |
400 | |
401 | // Pick the right v-table based on the IID passed in. |
402 | if (riid == IID_IUnknown) |
403 | *ppvObject = (IUnknown *) this; |
404 | else if (riid == IID_IClassFactory) |
405 | *ppvObject = (IClassFactory *) this; |
406 | |
407 | // If successful, add a reference for out pointer and return. |
408 | if (*ppvObject) |
409 | { |
410 | hr = S_OK; |
411 | AddRef(); |
412 | } |
413 | else |
414 | hr = E_NOINTERFACE; |
415 | return (hr); |
416 | } |
417 | |
418 | |
419 | //***************************************************************************** |
420 | // CreateInstance is called to create a new instance of the coclass for which |
421 | // this class was created in the first place. The returned pointer is the |
422 | // v-table matching the IID if there. |
423 | //***************************************************************************** |
424 | HRESULT STDMETHODCALLTYPE CClassFactory::CreateInstance( |
425 | IUnknown *pUnkOuter, |
426 | REFIID riid, |
427 | void **ppvObject) |
428 | { |
429 | HRESULT hr; |
430 | |
431 | // Avoid confusion. |
432 | *ppvObject = NULL; |
433 | _ASSERTE(m_pfnCreateObject); |
434 | |
435 | // Aggregation is not supported by these objects. |
436 | if (pUnkOuter) |
437 | return (CLASS_E_NOAGGREGATION); |
438 | |
439 | // Ask the object to create an instance of itself, and check the iid. |
440 | hr = (*m_pfnCreateObject)(riid, ppvObject); |
441 | return (hr); |
442 | } |
443 | |
444 | |
445 | HRESULT STDMETHODCALLTYPE CClassFactory::LockServer( |
446 | BOOL fLock) |
447 | { |
448 | //<TODO>@todo: hook up lock server logic.</TODO> |
449 | return (S_OK); |
450 | } |
451 | |
452 | |
453 | //***************************************************************************** |
454 | // This helper provides access to the instance handle of the loaded image. |
455 | //***************************************************************************** |
456 | #ifndef FEATURE_PAL |
457 | HINSTANCE GetModuleInst() |
458 | { |
459 | return g_hInst; |
460 | } |
461 | #endif |
462 | |
463 | |
464 | //----------------------------------------------------------------------------- |
465 | // Substitute for mscoree |
466 | // |
467 | // Notes: |
468 | // Mscordbi does not link with mscoree, provide a stub implementation. |
469 | // Callers are in dead-code paths, but we still need to provide a stub. Ideally, we'd factor |
470 | // out the callers too and then we wouldn't need an E_NOTIMPL stub. |
471 | STDAPI GetRequestedRuntimeInfo(LPCWSTR pExe, |
472 | LPCWSTR pwszVersion, |
473 | LPCWSTR pConfigurationFile, |
474 | DWORD startupFlags, |
475 | DWORD runtimeInfoFlags, |
476 | __out_ecount_opt(dwDirectory) LPWSTR pDirectory, |
477 | DWORD dwDirectory, |
478 | DWORD *dwDirectoryLength, |
479 | __out_ecount_opt(cchBuffer) LPWSTR pVersion, |
480 | DWORD cchBuffer, |
481 | DWORD* dwlength) |
482 | { |
483 | _ASSERTE(!"GetRequestedRuntimeInfo not impl" ); |
484 | return E_NOTIMPL; |
485 | } |
486 | |
487 | //----------------------------------------------------------------------------- |
488 | // Replacement for legacy shim API GetCORRequiredVersion(...) used in linked libraries. |
489 | // Used in code:TiggerStorage::GetDefaultVersion#CallTo_CLRRuntimeHostInternal_GetImageVersionString. |
490 | // |
491 | // Notes: |
492 | // Mscordbi does not statically link to mscoree.dll. |
493 | // This is used in EnC for IMetadataEmit2::GetSaveSize to computer size of header. |
494 | // see code:TiggerStorage::GetDefaultVersion. |
495 | // |
496 | // Implemented by returning the version we're built for. Mscordbi.dll has a tight coupling with |
497 | // the CLR version, so this will match exactly the build version we're debugging. |
498 | // One potential caveat is that the build version doesn't necessarily match the install string |
499 | // (eg. we may install as "v4.0.x86chk" but that's not captured in the build version). But this should |
500 | // be internal scenarios only, and shouldn't actually matter here. If it did, we could instead get |
501 | // the last components of the directory name the current mscordbi.dll is located in. |
502 | // |
503 | HRESULT |
504 | CLRRuntimeHostInternal_GetImageVersionString( |
505 | __out_ecount_part(*pcchBuffer, *pcchBuffer) LPWSTR wszBuffer, |
506 | DWORD *pcchBuffer) |
507 | { |
508 | // Construct the cannoncial version string we're built as - eg. "v4.0.1234" |
509 | const WCHAR k_wszBuiltFor[] = W("v" ) VER_PRODUCTVERSION_NO_QFE_STR_L; |
510 | |
511 | // Copy our buffer in |
512 | HRESULT hr = HRESULT_FROM_WIN32(wcscpy_s(wszBuffer, *pcchBuffer, k_wszBuiltFor)); |
513 | |
514 | // Hand out length regardless of success - like GetCORRequiredVersion |
515 | *pcchBuffer = _countof(k_wszBuiltFor); |
516 | |
517 | return hr; |
518 | } // CLRRuntimeHostInternal_GetImageVersionString |
519 | |
520 | |
521 | #ifdef _TARGET_ARM_ |
522 | BOOL |
523 | DbiGetThreadContext(HANDLE hThread, |
524 | DT_CONTEXT *lpContext) |
525 | { |
526 | // if we aren't local debugging this isn't going to work |
527 | #if !defined(_ARM_) || defined(FEATURE_DBGIPC_TRANSPORT_DI) || !SUPPORT_LOCAL_DEBUGGING |
528 | _ASSERTE(!"Can't use local GetThreadContext remotely, this needed to go to datatarget" ); |
529 | return FALSE; |
530 | #else |
531 | BOOL res = FALSE; |
532 | if (((ULONG)lpContext) & ~0x10) |
533 | { |
534 | CONTEXT *ctx = (CONTEXT*)_aligned_malloc(sizeof(CONTEXT), 16); |
535 | if (ctx) |
536 | { |
537 | ctx->ContextFlags = lpContext->ContextFlags; |
538 | if (::GetThreadContext(hThread, ctx)) |
539 | { |
540 | *lpContext = *(DT_CONTEXT*)ctx; |
541 | res = TRUE; |
542 | } |
543 | |
544 | _aligned_free(ctx); |
545 | } |
546 | else |
547 | { |
548 | // malloc does not set the last error, but the caller of GetThreadContext |
549 | // will expect it to be set on failure. |
550 | SetLastError(ERROR_OUTOFMEMORY); |
551 | } |
552 | } |
553 | else |
554 | { |
555 | res = ::GetThreadContext(hThread, (CONTEXT*)lpContext); |
556 | } |
557 | |
558 | return res; |
559 | #endif |
560 | } |
561 | |
562 | BOOL |
563 | DbiSetThreadContext(HANDLE hThread, |
564 | const DT_CONTEXT *lpContext) |
565 | { |
566 | #if !defined(_ARM_) || defined(FEATURE_DBGIPC_TRANSPORT_DI) || !SUPPORT_LOCAL_DEBUGGING |
567 | _ASSERTE(!"Can't use local GetThreadContext remotely, this needed to go to datatarget" ); |
568 | return FALSE; |
569 | #else |
570 | BOOL res = FALSE; |
571 | if (((ULONG)lpContext) & ~0x10) |
572 | { |
573 | CONTEXT *ctx = (CONTEXT*)_aligned_malloc(sizeof(CONTEXT), 16); |
574 | if (ctx) |
575 | { |
576 | *ctx = *(CONTEXT*)lpContext; |
577 | res = ::SetThreadContext(hThread, ctx); |
578 | _aligned_free(ctx); |
579 | } |
580 | else |
581 | { |
582 | // malloc does not set the last error, but the caller of SetThreadContext |
583 | // will expect it to be set on failure. |
584 | SetLastError(ERROR_OUTOFMEMORY); |
585 | } |
586 | } |
587 | else |
588 | { |
589 | res = ::SetThreadContext(hThread, (CONTEXT*)lpContext); |
590 | } |
591 | |
592 | return res; |
593 | #endif |
594 | } |
595 | #endif |
596 | |