From 45009f85b209be09e5385534c3d94442c42c6a5a Mon Sep 17 00:00:00 2001 From: vsadov <8218165+VSadov@users.noreply.github.com> Date: Sun, 15 Sep 2024 16:07:14 -0700 Subject: [PATCH 1/3] Add a test --- .../coreclr/GitHub_107800/test107800.cs | 25 +++++++++++++++++++ .../coreclr/GitHub_107800/test107800.csproj | 11 ++++++++ 2 files changed, 36 insertions(+) create mode 100644 src/tests/Regressions/coreclr/GitHub_107800/test107800.cs create mode 100644 src/tests/Regressions/coreclr/GitHub_107800/test107800.csproj diff --git a/src/tests/Regressions/coreclr/GitHub_107800/test107800.cs b/src/tests/Regressions/coreclr/GitHub_107800/test107800.cs new file mode 100644 index 0000000000000..dc956f4984273 --- /dev/null +++ b/src/tests/Regressions/coreclr/GitHub_107800/test107800.cs @@ -0,0 +1,25 @@ +// Licensed to the .NET Foundation under one or more agreements. +// The .NET Foundation licenses this file to you under the MIT license. + +using System; +using System.Threading; +using System.Threading.Tasks; + +public class Program +{ + public static int Main() + { + Task.Run(() => + { + for (; ; ) + { + GC.Collect(); + } + }); + + Thread.Sleep(10); + Environment.Exit(100); + + throw new Exception("UNREACHABLE"); + } +} diff --git a/src/tests/Regressions/coreclr/GitHub_107800/test107800.csproj b/src/tests/Regressions/coreclr/GitHub_107800/test107800.csproj new file mode 100644 index 0000000000000..13af8e767b04c --- /dev/null +++ b/src/tests/Regressions/coreclr/GitHub_107800/test107800.csproj @@ -0,0 +1,11 @@ + + + + true + 0 + Exe + + + + + From 61bdcfaff569ad310df9ee46babf138a98468cc5 Mon Sep 17 00:00:00 2001 From: Jan Kotas Date: Mon, 1 Jul 2024 17:13:07 -0700 Subject: [PATCH 2/3] Minimal part of "Use RtlDllShutdownInProgress to detect process shutdown on Windows (#103877)" Fixes:107800 * Use RtlDllShutdownInProgress to detect process shutdown on Windows Switching to cooperative mode is not safe during process shutdown on Windows. Process shutdown can terminate a thread in the middle of the GC. The shutdown thread deadlocks if it tries to switch to cooperative mode and wait for the GC to finish in this situation. Use RtlDllShutdownInProgress Windows API to detect process shutdown to avoid waiting for GC completion when that may lead to deadlocks. --- src/coreclr/vm/ceemain.cpp | 12 +++++++++++- src/coreclr/vm/vars.hpp | 14 +++++++++++++- 2 files changed, 24 insertions(+), 2 deletions(-) diff --git a/src/coreclr/vm/ceemain.cpp b/src/coreclr/vm/ceemain.cpp index 680a9a2fb2491..c5dff66b823a4 100644 --- a/src/coreclr/vm/ceemain.cpp +++ b/src/coreclr/vm/ceemain.cpp @@ -392,6 +392,12 @@ BOOL g_singleVersionHosting; typedef BOOL(WINAPI* PINITIALIZECONTEXT2)(PVOID Buffer, DWORD ContextFlags, PCONTEXT* Context, PDWORD ContextLength, ULONG64 XStateCompactionMask); PINITIALIZECONTEXT2 g_pfnInitializeContext2 = NULL; +static BOOLEAN WINAPI RtlDllShutdownInProgressFallback() +{ + return g_fProcessDetach; +} +PRTLDLLSHUTDOWNINPROGRESS g_pfnRtlDllShutdownInProgress = &RtlDllShutdownInProgressFallback; + #ifdef TARGET_X86 typedef VOID(__cdecl* PRTLRESTORECONTEXT)(PCONTEXT ContextRecord, struct _EXCEPTION_RECORD* ExceptionRecord); PRTLRESTORECONTEXT g_pfnRtlRestoreContext = NULL; @@ -402,8 +408,12 @@ void InitializeOptionalWindowsAPIPointers() HMODULE hm = GetModuleHandleW(_T("kernel32.dll")); g_pfnInitializeContext2 = (PINITIALIZECONTEXT2)GetProcAddress(hm, "InitializeContext2"); -#ifdef TARGET_X86 hm = GetModuleHandleW(_T("ntdll.dll")); + PRTLDLLSHUTDOWNINPROGRESS pfn = (PRTLDLLSHUTDOWNINPROGRESS)GetProcAddress(hm, "RtlDllShutdownInProgress"); + if (pfn != NULL) + g_pfnRtlDllShutdownInProgress = pfn; + +#ifdef TARGET_X86 g_pfnRtlRestoreContext = (PRTLRESTORECONTEXT)GetProcAddress(hm, "RtlRestoreContext"); #endif //TARGET_X86 } diff --git a/src/coreclr/vm/vars.hpp b/src/coreclr/vm/vars.hpp index 652c8cb7fd50a..5f9984b4673fd 100644 --- a/src/coreclr/vm/vars.hpp +++ b/src/coreclr/vm/vars.hpp @@ -478,12 +478,24 @@ GVAL_DECL(bool, g_metadataUpdatesApplied); #endif EXTERN bool g_fManagedAttach; +#ifdef HOST_WINDOWS +typedef BOOLEAN (WINAPI* PRTLDLLSHUTDOWNINPROGRESS)(); +EXTERN PRTLDLLSHUTDOWNINPROGRESS g_pfnRtlDllShutdownInProgress; +#endif + // Indicates whether we're executing shut down as a result of DllMain // (DLL_PROCESS_DETACH). See comments at code:EEShutDown for details. -inline BOOL IsAtProcessExit() +inline bool IsAtProcessExit() { SUPPORTS_DAC; +#if defined(DACCESS_COMPILE) || !defined(HOST_WINDOWS) return g_fProcessDetach; +#else + // RtlDllShutdownInProgress provides more accurace information about whether the process is shutting down. + // Use it if it is available to avoid shutdown deadlocks. + // https://learn.microsoft.com/windows/win32/devnotes/rtldllshutdowninprogress + return g_pfnRtlDllShutdownInProgress(); +#endif } enum FWStatus From 1e671dae599c68b5a7a4b4248a357236eeedcc97 Mon Sep 17 00:00:00 2001 From: Vladimir Sadov Date: Tue, 17 Sep 2024 11:06:54 -0700 Subject: [PATCH 3/3] Update src/coreclr/vm/vars.hpp Co-authored-by: Manish Godse <61718172+mangod9@users.noreply.github.com> --- src/coreclr/vm/vars.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/coreclr/vm/vars.hpp b/src/coreclr/vm/vars.hpp index 5f9984b4673fd..b34ec9426621d 100644 --- a/src/coreclr/vm/vars.hpp +++ b/src/coreclr/vm/vars.hpp @@ -491,7 +491,7 @@ inline bool IsAtProcessExit() #if defined(DACCESS_COMPILE) || !defined(HOST_WINDOWS) return g_fProcessDetach; #else - // RtlDllShutdownInProgress provides more accurace information about whether the process is shutting down. + // RtlDllShutdownInProgress provides more accurate information about whether the process is shutting down. // Use it if it is available to avoid shutdown deadlocks. // https://learn.microsoft.com/windows/win32/devnotes/rtldllshutdowninprogress return g_pfnRtlDllShutdownInProgress();