Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)
(시리즈 글이 6개 있습니다.)
VC++: 64. x64 Visual C++에서 TEB 주소 구하는 방법
; https://www.sysnet.pe.kr/2/0/1387

.NET Framework: 348. .NET x64 응용 프로그램에서 Teb 주소를 구하는 방법
; https://www.sysnet.pe.kr/2/0/1388

.NET Framework: 349. .NET Thread 인스턴스로부터 COM Apartment 유형 확인하는 방법
; https://www.sysnet.pe.kr/2/0/1389

VC++: 91. 자식 스레드에 자동 상속되는 TEB의 SubProcessTag 필드
; https://www.sysnet.pe.kr/2/0/10797

디버깅 기술: 150. windbg - Wow64, x86, x64에서의 커널 구조체(예: TEB) 구조체 확인
; https://www.sysnet.pe.kr/2/0/12097

디버깅 기술: 205. Windbg - KPCR, KPRCB
; https://www.sysnet.pe.kr/2/0/13842




.NET x64 응용 프로그램에서 Teb 주소를 구하는 방법

예전에 x86에서 TEB(Thread Environment Block) 주소를 구해오는 방법을 아래의 글에서 설명했었습니다.

.NET System.Threading.Thread 개체에서 Native Thread Id를 구할 수 있을까?
; https://www.sysnet.pe.kr/2/0/1244

그리고 지난번 이야기에서 Visual C++ x64에서 TEB 주소를 구하는 것에 대해 설명했는데요.

x64 Visual C++에서 TEB 주소 구하는 방법
; https://www.sysnet.pe.kr/2/0/1387

따라서, x64 C# 응용 프로그램에서는 NtCurrentTeb를 DllImport로 P/Invoke 호출이 불가능하기 때문에 남은 방법은 Thread 타입의 DONT_USE_InternalThread 필드를 이용하는 수밖에 없습니다.

그래도 다른 방법이 있지 않을까요? ^^ TEB 주소가 결국 gs:[30h]에 있는 값이라는 사실에서 이야기를 진행해 보겠습니다.

문제는 Visual C++에서 사용된 __readgsqword가 DLL에서 export된 함수가 아니라 Visual C++의 intrinsics 함수에 불과하다는 점입니다. 따라서 아래의 코드를 C#에서 P/Invoke로 불러올 수는 없습니다.

int _tmain(int argc, _TCHAR* argv[])
{
    unsigned __int64 fsReg = __readgsqword(0x30);
    return 0;
}

하지만, 여러분들 중에 아래의 글을 기억하는 분이라면 방법을 알 수 있을 것입니다.

C++의 inline asm 사용을 .NET으로 포팅하는 방법
; https://www.sysnet.pe.kr/2/0/1267

따라서 위의 소스 코드를 아래와 같이 변경하고,

unsigned __int64 GetTEB()
{
    return __readgsqword(0x30);
}

int _tmain(int argc, _TCHAR* argv[])
{
    unsigned __int64 fsReg = GetTEB();
    printf("%I64x\n", fsReg);
    return 0;
}

__readgsqword 코드에 BP(Break Point)를 건 후 Debug 모드로 진입한 다음, 마우스 오른쪽 버튼을 눌러 "Go To Disassembly" 메뉴를 선택하면 다음의 화면을 볼 수 있습니다.

ntcurrentteb_api_1.png

아하... 답이 나왔군요. ^^

위의 바이트를 그대로 배열에 저장하고,

private readonly static byte[] x64TebBytes =
    {
        0x40, 0x57, // push rdi
        
        0x65, 0x48, 0x8B, 0x04, 0x25, 0x30, 0x00, 0x00, 0x00, // mov rax, qword ptr gs:[30h]

        0x5F, // pop rdi
        0xC3, // ret
    };

이를 delegate로 만들어 주면 됩니다.

static IntPtr _codePointer;
static GetTebDelegate _getTebDelg;

static Program()
{
    byte[] codeBytes = x64TebBytes;

    if (IntPtr.Size == 4)
    {
        throw new NotSupportedException();
    }

    _codePointer = VirtualAlloc(IntPtr.Zero, new UIntPtr((uint)codeBytes.Length),
        AllocationType.COMMIT | AllocationType.RESERVE,
        MemoryProtection.EXECUTE_READWRITE
    );

    Marshal.Copy(codeBytes, 0, _codePointer, codeBytes.Length);

    _getTebDelg = (GetTebDelegate)Marshal.GetDelegateForFunctionPointer(
        _codePointer, typeof(GetTebDelegate));
}

static long GetTebAddress()
{
    if (_getTebDelg == null)
    {
        throw new ObjectDisposedException("GetTebAddress");
    }

    return _getTebDelg();
}

이제부터는 언제든지 GetTebAddress를 호출해 주면 TEB 주소를 구할 수 있습니다. ^^

static void Main(string[] args)
{
    Console.WriteLine(GetTebAddress().ToString("x"));
}

이렇게 출력된 값이 정확히 TEB 주소를 가리키는지 확인하는 방법은 지난번 글에서 설명했으므로 생략합니다. ^^

참고로, x64에서 DONT_USE_InternalThread 필드를 이용하여 Thread 개체로부터 TEB 주소를 구하는 방법은 다음과 같습니다.

Thread currrentThread = Thread.CurrentThread;
FieldInfo fieldInfo = typeof(Thread).GetField("DONT_USE_InternalThread", 
            BindingFlags.NonPublic | BindingFlags.Instance);
IntPtr objValue = (IntPtr)fieldInfo.GetValue(currrentThread);

Console.WriteLine("DONT_USE_InternalThread: " + objValue.ToString("x"));
IntPtr teb = new IntPtr(Marshal.ReadInt64(objValue, 16 * 6));
Console.WriteLine("teb: " + teb.ToString("x"));

하는 김에 Native Thread Id도 구해볼까요? 우선 _TEB 구조체로부터 RealClientId 필드의 옵셋을 알아내고 _CLIENT_ID 구조체 값을 확인하면 답이 나옵니다.

0:024> dt _TEB
ntdll!_TEB
    ...[생략]...
   +0x2f0 GdiTebBatch      : _GDI_TEB_BATCH
   +0x7d8 RealClientId     : _CLIENT_ID
   +0x7e8 GdiCachedProcessHandle : Ptr64 Void
   +0x7f0 GdiClientPID     : Uint4B
   +0x7f4 GdiClientTID     : Uint4B
   ...[생략]...

0:024> dt _CLIENT_ID
ntdll!_CLIENT_ID
   +0x000 UniqueProcess    : Ptr64 Void
   +0x008 UniqueThread     : Ptr64 Void

따라서 0x7d8 옵셋을 기준으로 계산해 주면 됩니다.

long clientPid = Marshal.ReadInt64(teb, 0x7d8);
long clientTid = Marshal.ReadInt64(teb, 0x7d8 + 8);

Console.WriteLine("Process Id: " + clientPid.ToString("x"));
Console.WriteLine("Native Thread Id: " + clientTid.ToString("x"));

첨부된 파일은 위에 설명한 전체 C# 코드를 담은 프로젝트입니다.




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 6/27/2021]

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

비밀번호

댓글 작성자
 




... 151  152  153  154  155  156  157  158  159  160  161  [162]  163  164  165  ...
NoWriterDateCnt.TitleFile(s)
1029정성태4/28/201142234개발 환경 구성: 117. XCopy에서 파일/디렉터리 확인 질문 없애기 [2]
1028정성태4/27/201139639오류 유형: 119. Visual Studio 2010 SP1 설치 후 Windows Phone 개발자 도구로 인한 재설치 문제 [3]
1027정성태4/25/201128796디버깅 기술: 40. 상황별 GetFunctionPointer 반환값 정리 - x86파일 다운로드1
1026정성태4/25/201147454디버깅 기술: 39. DebugDiag 1.1을 사용한 덤프 분석 [7]
1025정성태4/24/201129258개발 환경 구성: 116. IIS 7 관리자 - Active Directory Certification Authority로부터 SSL 사이트 인증서 받는 방법 [2]
1024정성태4/22/201130606오류 유형: 118. Windows 2008 서버에서 Event Viewer / PowerShell 실행 시 비정상 종료되는 문제 [1]
1023정성태4/20/201131446.NET Framework: 210. Windbg 환경에서 확인해 본 .NET 메서드 JIT 컴파일 전과 후 [1]
1022정성태4/19/201126775디버깅 기술: 38. .NET Disassembly 창에서의 F11(Step-into) 키 동작파일 다운로드1
1021정성태4/18/201129146디버깅 기술: 37. .NET 4.0 응용 프로그램의 Main 함수에 BreakPoint 걸기
1020정성태4/18/201129910오류 유형: 117. Failed to find runtime DLL (mscorwks.dll), 0x80004005
1019정성태4/17/201130607디버깅 기술: 36. Visual Studio의 .NET Disassembly 창의 call 호출에 사용되는 주소의 의미는? [1]파일 다운로드1
1018정성태4/16/201134491오류 유형: 116. 윈도우 업데이트 오류 - 0x8020000E
1017정성태4/14/201129040개발 환경 구성: 115. MSBuild - x86/x64, .NET 2/4, debug/release 빌드에 대한 배치 처리파일 다운로드1
1016정성태4/13/201145193개발 환경 구성: 114. Windows Thin PC 설치 [2]
1015정성태4/9/201130484.NET Framework: 209. AutoReset, ManualReset, Monitor.Wait의 차이파일 다운로드1
1014정성태4/7/2011107961오류 유형: 115. ORA-12516: TNS:listener could not find available handler with matching protocol stack [2]
1013정성태4/7/201125726Team Foundation Server: 45. SharePoint 2010 + TFS 2010 환경에서 ProcessGuidance.html 파일 다운로드 문제
1012정성태4/6/201134503.NET Framework: 208. WCF - 접속된 클라이언트의 IP 주소 알아내는 방법 [1]
1011정성태3/31/201136855오류 유형: 114. 인증서 갱신 오류 - The request contains no certificate template information.
1010정성태3/30/201127657개발 환경 구성: 113. 응용 프로그램 디자인 스케치 도구 - SketchFlow [4]
1009정성태3/29/201139937개발 환경 구성: 112. Visual Studio 2010 - .NET Framework 소스 코드 디버깅 [4]
1008정성태3/27/201132265.NET Framework: 207. C# - Right operand가 음수인 Shift 연산 결과 [2]
1007정성태3/16/201133207개발 환경 구성: 111. Excel - XML 파일 연동 [5]파일 다운로드1
1006정성태3/15/201126838.NET Framework: 206. XML/XSD - 외래키처럼 참조 제한 거는 방법파일 다운로드1
1005정성태3/11/201136729개발 환경 구성: 110. 엑셀 매크로 함수 관련 오류 [2]
1004정성태3/3/201125963개발 환경 구성: 109. SharePoint Health Analyzer 디스크 부족 경고 제어
... 151  152  153  154  155  156  157  158  159  160  161  [162]  163  164  165  ...