Microsoft MVP성태의 닷넷 이야기
.NET Framework: 167. 다른 스레드의 호출 스택 덤프 구하는 방법 [링크 복사], [링크+제목 복사],
조회: 30648
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)
(시리즈 글이 7개 있습니다.)
.NET Framework: 167. 다른 스레드의 호출 스택 덤프 구하는 방법
; https://www.sysnet.pe.kr/2/0/802

.NET Framework: 260. .NET 스레드 콜 스택 덤프 (2) - Managed Stack Explorer 소스 코드를 이용한 스택 덤프 구하는 방법
; https://www.sysnet.pe.kr/2/0/1162

.NET Framework: 261. .NET 스레드 콜 스택 덤프 (3) - MSE 소스 코드 개선
; https://www.sysnet.pe.kr/2/0/1163

.NET Framework: 262. .NET 스레드 콜 스택 덤프 (4) - .NET 4.0을 지원하지 않는 MSE 응용 프로그램 원인 분석
; https://www.sysnet.pe.kr/2/0/1164

.NET Framework: 311. .NET 스레드 콜 스택 덤프 (5) - ICorDebug 인터페이스 사용법
; https://www.sysnet.pe.kr/2/0/1249

.NET Framework: 392. .NET 스레드 콜 스택 덤프 (6) - MDbg를 이용한 방법
; https://www.sysnet.pe.kr/2/0/1534

.NET Framework: 606. .NET 스레드 콜 스택 덤프 (7) - ClrMD(Microsoft.Diagnostics.Runtime)를 이용한 방법
; https://www.sysnet.pe.kr/2/0/11043





다른 스레드의 호출 스택 덤프 구하는 방법


처음 이 기능이 필요했을 때 매우 쉬울 것이라고 생각했습니다. 왜냐하면, 이미 System.Diagnostics.StackTrace 타입에서 충분한 기능을 제공하기 때문입니다.

문제는 "다른 스레드"의 콜 스택을 얻을 때 발생했습니다. StackTrace의 생성자 목록에는 이에 대한 배려도 되어 있는데, 실제로 사용해 보면... 결과는 다소 당혹스럽습니다.

static void Main(string[] args)
{
    Thread newThread = new Thread(Run);
    newThread.Start();

    System.Diagnostics.StackTrace trace = new System.Diagnostics.StackTrace(newThread, false);
}

private static void Run()
{
    Thread.Sleep(1000 * 10);
}

위의 코드를 실행해 보면, 다음과 같은 예외를 만나게 됩니다.


Unhandled Exception: System.Threading.ThreadStateException: Thread in invalid state.
at System.Diagnostics.StackTrace.GetStackFramesInternal(StackFrameHelper sfh, Int32 iSkip, Exception e)
at System.Diagnostics.StackTrace.CaptureStackTrace(Int32 iSkip, Boolean fNeedFileInfo, Thread targetThread, Exception e)
at System.Diagnostics.StackTrace..ctor(Thread targetThread, Boolean needFileInfo)
at ConsoleApplication1.Program.Main(String[] args) in C:\temp\...\Program.cs:line 21



닷넷 소스 코드 디버깅을 이용해 보면, 다음의 라인에서 오류가 나는 것을 확인할 수 있습니다.

[MethodImplAttribute(MethodImplOptions.InternalCall)]
internal static extern void GetStackFramesInternal(StackFrameHelper sfh, int iSkip, Exception e);

private void CaptureStackTrace(int iSkip, bool fNeedFileInfo, Thread targetThread,
                               Exception e)
{
    m_iMethodsToSkip += iSkip; 
    StackFrameHelper StackF = new StackFrameHelper(fNeedFileInfo, targetThread); 

    GetStackFramesInternal (StackF, 0, e);  // 예외 발생

	// ... [생략]...
}

MethodImplOptions.InternalCall로 정의된 메서드라서 더 이상 추적해 볼 여지가 없습니다. 이제 ^^ 구글로 넘어가 봐야죠.

다행히, 이와 관련해서 "Rotor" 소스 코드를 찾을 수가 있었습니다.

Rotor Source code
; http://www.123aspx.com/rotor/RotorSrc.aspx?rot=42047
; https://github.com/SSCLI/sscli20_20060311/blob/master/clr/src/bcl/system/diagnostics/stacktrace.cs
; https://github.com/dotnet/coreclr/blob/release/2.2/src/vm/debugdebugger.cpp#L327-L800

중간에 보면, 다음과 같은 코드가 확인됩니다.

public StackTrace(Thread targetThread, bool needFileInfo)
{    
    m_iNumOfFrames = 0;
    m_iMethodsToSkip = 0;

    if (targetThread != null)
    {
        if (targetThread != Thread.CurrentThread)
        {
            if (((targetThread.ThreadState & System.Threading.ThreadState.Suspended) != 0)
             && ((targetThread.ThreadState & System.Threading.ThreadState.SuspendRequested) != 0)
             && (targetThread.ThreadState != System.Threading.ThreadState.Stopped)
             && (targetThread.ThreadState != System.Threading.ThreadState.Unstarted))
            {
                throw new ThreadStateException (
                            Environment.GetResourceString("ThreadState_NeedSuspended"));                        
            }
        }
        else
            targetThread = null;
    }

    CaptureStackTrace (METHODS_TO_SKIP, needFileInfo, targetThread, null);

}

오호... 그러니까, "실행 상태"에 있다면 콜 스택을 얻을 수 없다는 것이군요. 그래서, 최초의 StackTrace 코드를 다음과 같이 수정해 보았습니다.

static void Main(string[] args)
{
    Thread newThread = new Thread(Run);
    newThread.Start();

    newThread.Suspend();
    System.Diagnostics.StackTrace trace = new System.Diagnostics.StackTrace(newThread, false);
    newThread.Resume();
    Console.WriteLine(trace.ToString());
}

와~~~ ^^ 이제 예외가 발생하지 않습니다. 하지만! 출력 결과가 다소 실망스럽습니다.


at ConsoleApplication1.Program.Run()
at System.Threading.ThreadHelper.ThreadStart_Context(Object state)
at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state)
at System.Threading.ThreadHelper.ThreadStart()



달랑, Run 메서드 프레임 하나.
이상하다 싶어서 중간에 newThread.Sleep(3000);을 한 번 더 주고 나니 정상적으로 아래와 같이 콜 스택이 출력되었습니다. (즉, 위의 콜 스택 출력 결과는 너무 빨리 시도를 했기 때문에 발생한 것으로 그것 자체도 정상적인 출력이었습니다.)


at System.Threading.Thread.SleepInternal(Int32 millisecondsTimeout)
at System.Threading.Thread.Sleep(Int32 millisecondsTimeout)

at ConsoleApplication1.Program.Run()
at System.Threading.ThreadHelper.ThreadStart_Context(Object state)
at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state)
at System.Threading.ThreadHelper.ThreadStart()




이 외에, StackTrace.ToString의 결과를 사용자 정의하고 싶다면 다음의 코드를 참조하시면 도움이 되겠습니다.

How do I make a thread dump in .NET ? (a la JVM thread dumps)
; http://stackoverflow.com/questions/190236/how-do-i-make-a-thread-dump-in-net-a-la-jvm-thread-dumps

코드가 아닌, 툴을 이용하고 싶다면 다음의 방법이 좋겠고.

.NET production debugging 101
; http://www.tomergabel.com/NETProductionDebugging101.aspx

혹시나, StackTrace를 이용하여 부모 메서드에 따라 동작을 다르게 만들고 싶다면 다음과 같은 함정을 주의하시기 바랍니다.

Caveats about System.Diagnostics.StackTrace
; https://learn.microsoft.com/en-us/archive/blogs/jmstall/caveats-about-system-diagnostics-stacktrace

첨부된 파일은 위의 코드를 포함하고 있는 (VS 2010 beta2)프로젝트입니다.

끝!




[이 토픽에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 8/21/2023]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 61  [62]  63  64  65  66  67  68  69  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12424정성태11/24/202019560VC++: 140. C++의 연산자 동의어(operator synonyms), 대체 토큰 [1]
12423정성태11/22/202019562.NET Framework: 974. C# 9.0 - (16) 제약 조건이 없는 형식 매개변수 주석(Unconstrained type parameter annotations)파일 다운로드1
12422정성태11/21/202017029.NET Framework: 973. .NET 5, .NET Framework에서만 허용하는 UnmanagedCallersOnly 사용예파일 다운로드1
12421정성태11/19/202016145.NET Framework: 972. DNNE가 출력한 NE DLL을 직접 생성하는 방법파일 다운로드1
12420정성태11/19/202016740오류 유형: 684. Visual C++ - MSIL .netmodule or module compiled with /GL found; restarting link with /LTCG; add /LTCG to the link command line to improve linker performance
12419정성태11/19/202017010VC++: 139. Visual C++ - .NET Core의 nethost.lib와 정적 링크파일 다운로드1
12418정성태11/19/202019167오류 유형: 683. Visual C++ - error LNK2038: mismatch detected for 'RuntimeLibrary': value 'MT_StaticRelease' doesn't match value 'MDd_DynamicDebug'파일 다운로드1
12417정성태11/19/202017618오류 유형: 682. Visual C++ - warning LNK4099: PDB '...pdb' was not found with '...lib(pch.obj)' or at '...pdb'; linking object as if no debug info
12416정성태11/19/202018756오류 유형: 681. Visual C++ - error LNK2001: unresolved external symbol _CrtDbgReport
12415정성태11/18/202018284.NET Framework: 971. UnmanagedCallersOnly 특성과 DNNE 사용파일 다운로드1
12414정성태11/18/202021057VC++: 138. x64 빌드에서 extern "C"가 아닌 경우 ___cdecl name mangling 적용 [4]파일 다운로드1
12413정성태11/17/202019723.NET Framework: 970. .NET 5 / .NET Core - UnmanagedCallersOnly 특성을 사용한 함수 내보내기파일 다운로드1
12412정성태11/16/202021727.NET Framework: 969. .NET Framework 및 .NET 5 - UnmanagedCallersOnly 특성 사용파일 다운로드1
12411정성태11/12/202018297오류 유형: 680. C# 9.0 - Error CS8889 The target runtime doesn't support extensible or runtime-environment default calling conventions.
12410정성태11/12/202019160디버깅 기술: 174. windbg - System.TypeLoadException 예외 분석 사례
12409정성태11/12/202020802.NET Framework: 968. C# 9.0의 Function pointer를 이용한 함수 주소 구하는 방법파일 다운로드1
12408정성태11/9/202035849도서: 시작하세요! C# 9.0 프로그래밍 [8]
12407정성태11/9/202020897.NET Framework: 967. "clr!JIT_DbgIsJustMyCode" 호출이 뭘까요?
12406정성태11/8/202021846.NET Framework: 966. C# 9.0 - (15) 최상위 문(Top-level statements) [5]파일 다운로드1
12405정성태11/8/202020362.NET Framework: 965. C# 9.0 - (14) 부분 메서드에 대한 새로운 기능(New features for partial methods)파일 다운로드1
12404정성태11/7/202020954.NET Framework: 964. C# 9.0 - (13) 모듈 이니셜라이저(Module initializers)파일 다운로드1
12403정성태11/7/202019043.NET Framework: 963. C# 9.0 - (12) foreach 루프에 대한 GetEnumerator 확장 메서드 지원(Extension GetEnumerator)파일 다운로드1
12402정성태11/7/202021267.NET Framework: 962. C# 9.0 - (11) 공변 반환 형식(Covariant return types) [1]파일 다운로드1
12401정성태11/5/202020591VS.NET IDE: 153. 닷넷 응용 프로그램에서의 "My Code" 범위와 "Enable Just My Code"의 역할 [1]
12400정성태11/5/202016579오류 유형: 679. Visual Studio - "Source Not Found" 창에 "Decompile source code" 링크가 없는 경우
12399정성태11/5/202019704.NET Framework: 961. C# 9.0 - (10) 대상으로 형식화된 조건식(Target-typed conditional expressions)파일 다운로드1
... 61  [62]  63  64  65  66  67  68  69  70  71  72  73  74  75  ...