Microsoft MVP성태의 닷넷 이야기
디버깅 기술: 153. C# - PEB를 조작해 로드된 DLL을 숨기는 방법 [링크 복사], [링크+제목 복사],
조회: 18942
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)
(시리즈 글이 4개 있습니다.)
디버깅 기술: 115. windbg - 덤프 파일로부터 PID와 환경변수 등의 정보를 구하는 방법
; https://www.sysnet.pe.kr/2/0/11478

.NET Framework: 876. C# - PEB(Process Environment Block)를 통해 로드된 모듈 목록 열람
; https://www.sysnet.pe.kr/2/0/12101

디버깅 기술: 153. C# - PEB를 조작해 로드된 DLL을 숨기는 방법
; https://www.sysnet.pe.kr/2/0/12105

디버깅 기술: 157. C# - PEB.ProcessHeap을 이용해 디버깅 중인지 확인하는 방법
; https://www.sysnet.pe.kr/2/0/12115




C# - PEB를 조작해 로드된 DLL을 숨기는 방법

지난 글에서,

C# - PEB(Process Environment Block)를 통해 로드된 모듈 목록 열람
; https://www.sysnet.pe.kr/2/0/12101

로드된 DLL들이 PEB.Ldr.InMemoryOrderModuleList에 이중 연결 리스트로 보관되어 있다고 설명했었습니다. 그렇다면, 당연히 그 연결을 끊으면 로드된 DLL을 숨기는 것도 가능합니다.

간단하게 구현해 볼까요? ^^ 우선 _PEB, _PEB_LDR_DATA를 가져온 후,

// Install-Package KernelStructOffset

_PEB peb = EnvironmentBlockInfo.GetPeb();
_PEB_LDR_DATA ldrData = _PEB_LDR_DATA.Create(peb.Ldr);

해당 모듈의 연결 리스트 항목(_LDR_DATA_TABLE_ENTRY)을 구해,

_LDR_DATA_TABLE_ENTRY dllLink = ldrData.Find("ole32.dll");

흔하게 알려진 이중 링크드 리스트의 끊는 방법을 _LIST_ENTRY에 적용하시면 됩니다.

_LIST_ENTRY* pNext = (_LIST_ENTRY*)dllLink.Flink.ToPointer();
_LIST_ENTRY* pPrev = (_LIST_ENTRY*)dllLink.Blink.ToPointer();

IntPtr thisLink = pNext->Blink;
_LIST_ENTRY* thisItem = (_LIST_ENTRY*)thisLink.ToPointer();
thisItem->Blink = IntPtr.Zero;
thisItem->Flink = IntPtr.Zero;

pNext->Blink = new IntPtr(pPrev);
pPrev->Flink = new IntPtr(pNext);

끝입니다. ^^ 실제로 끊기 전과 후에 Process.GetCurrentProcess().Modules을 열람해 보면,

foreach (ProcessModule pm in Process.GetCurrentProcess().Modules)
{
    if (pm.FileName.EndsWith("ole32.dll", StringComparison.OrdinalIgnoreCase) == true)
    {
        // ... 끊기 전에는 실행되지만, 끊은 후에는 실행되지 않음.
    }
}

결과를 확인할 수 있습니다.




참고로, tasklist 같은 명령행 도구들도 PEB를 이용해 모듈을 열람한다는 것을 이번 실습을 통해 확인할 수 있습니다. 일례로 ole32.dll에 대한 연결을 끊기 전과 후의 출력을 비교할 수 있습니다.

// 끊기 전

C:\temp>tasklist /FI "PID eq 19328" /M

Image Name                     PID Modules
========================= ======== ============================================
HideModule.exe               17628 ntdll.dll, MSCOREE.DLL, KERNEL32.dll,
                                   KERNELBASE.dll, ADVAPI32.dll, msvcrt.dll,
                                   sechost.dll, RPCRT4.dll, mscoreei.dll,
                                   SHLWAPI.dll, combase.dll, ucrtbase.dll,
                                   bcryptPrimitives.dll, GDI32.dll,
                                   win32u.dll, gdi32full.dll, msvcp_win.dll,
                                   USER32.dll, IMM32.DLL, kernel.appcore.dll,
                                   VERSION.dll, clr.dll, ucrtbase_clr0400.dll,
                                   VCRUNTIME140_CLR0400.dll, psapi.dll,
                                   mscorlib.ni.dll, ole32.dll, clrjit.dll,
                                   System.ni.dll


// 끊은 후

C:\temp>tasklist /FI "PID eq 19328" /M

Image Name                     PID Modules
========================= ======== ============================================
HideModule.exe               19328 ntdll.dll, MSCOREE.DLL, KERNEL32.dll,
                                   KERNELBASE.dll, ADVAPI32.dll, msvcrt.dll,
                                   sechost.dll, RPCRT4.dll, mscoreei.dll,
                                   SHLWAPI.dll, combase.dll, ucrtbase.dll,
                                   bcryptPrimitives.dll, GDI32.dll,
                                   win32u.dll, gdi32full.dll, msvcp_win.dll,
                                   USER32.dll, IMM32.DLL, kernel.appcore.dll,
                                   VERSION.dll, clr.dll,
                                   VCRUNTIME140_CLR0400.dll,
                                   ucrtbase_clr0400.dll, psapi.dll,
                                   mscorlib.ni.dll, clrjit.dll, System.ni.dll

반면 Sysinternals 등의 도구(listdlls.exe나 process explorer.exe)에서는 PEB의 InMemoryOrderModuleList를 사용하지 않으므로 모든 DLL들이 열거됩니다.




이쯤에서 한 가지 더 설명해야 하는데 ^^ 모듈 목록은 PEB.Ldr.InMemoryOrderModuleList뿐만 아니라 PEB.Ldr.InLoadOrderModuleList에도 보관하고 있습니다. (원래는 필드 이름이 의미하는 대로 구분이 있었던 것 같지만 Windows 10에서 테스트해 보면 목록 순서가 동일하게 출력됩니다.) 따라서 좀 더 잘 숨기고 싶다면 InLoadOrderModuleList에서도 연결을 끊어주는 작업이 필요합니다.

현재 이러한 기능을 추가해 KernelStructOffset을 업데이트했고 따라서 다음과 같은 식의 코드로 링크 연결을 끊거나 재연결할 수 있습니다.

_PEB peb = EnvironmentBlockInfo.GetPeb();
_PEB_LDR_DATA ldrData = _PEB_LDR_DATA.Create(peb.Ldr);

string moduleName = "ole32.dll";
FindModules(moduleName);

DllOrderLink hiddenModuleLink = null;

try
{
    hiddenModuleLink = ldrData.HideDLL(moduleName);
    FindModules(moduleName);
}
finally
{
    if (hiddenModuleLink != null)
    {
        ldrData.UnhideDLL(hiddenModuleLink);
    }
}

또한 InLoadOrderModuleList, InMemoryOrderModuleList 필드의 목록을 다음의 코드로 열람할 수 있습니다.

foreach (var item in ldrData.EnumerateLoadOrderModules())
{
    Console.WriteLine("\t" + item.FullDllName.GetText());
}

foreach (var item in ldrData.EnumerateMemoryOrderModules())
{
    Console.WriteLine("\t" + item.FullDllName.GetText());
}

이 글의 완전한 예제 코드는 github에 올려 두었습니다.

DotNetSamples/WinConsole/PEFormat/HideModule/
; https://github.com/stjeong/DotNetSamples/tree/master/WinConsole/PEFormat/HideModule




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 12/14/2022]

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

비밀번호

댓글 작성자
 




... 91  [92]  93  94  95  96  97  98  99  100  101  102  103  104  105  ...
NoWriterDateCnt.TitleFile(s)
11666정성태8/22/201819249사물인터넷: 37. 아두이노 - 코딩으로 대신하는 오실레이터 회로의 소리 출력파일 다운로드1
11665정성태8/22/201822812사물인터넷: 36. 오실레이터 회로 동작을 아두이노의 코딩으로 구현하는 방법파일 다운로드1
11664정성태8/22/201822271개발 환경 구성: 393. 윈도우 환경에서 elasticsearch의 한글 형태소 분석기 설치 [1]
11663정성태8/22/201825089개발 환경 구성: 392. 윈도우 환경에서 curl.exe를 이용한 elasticsearch 6.x 기본 사용법
11662정성태8/21/201817887사물인터넷: 35. 병렬 회로에서의 커패시터파일 다운로드1
11661정성태8/21/201820538사물인터넷: 34. 트랜지스터 동작 - 컬렉터-이미터 간의 저항 측정파일 다운로드1
11660정성태8/19/201820015사물인터넷: 33. 세라믹 커패시터의 동작 방식파일 다운로드1
11659정성태8/19/201819950사물인터넷: 32. 9V 전압에서 테스트하는 PN2222A 트랜지스터파일 다운로드1
11658정성태8/18/201823426사물인터넷: 31. 커패시터와 RC 회로파일 다운로드3
11657정성태8/18/201821562사물인터넷: 30. 릴레이(Relay) 제어파일 다운로드3
11656정성태8/16/201816577사물인터넷: 29. 트랜지스터와 병렬로 연결한 LED파일 다운로드1
11655정성태8/16/201819225사물인터넷: 28. 저항과 병렬로 연결한 LED파일 다운로드1
11654정성태8/15/201820505사물인터넷: 27. 병렬 회로의 저항, 전압 및 전류파일 다운로드1
11653정성태8/14/201821038사물인터넷: 26. 입력 전압에 따른 LED의 전압/저항 변화 [1]파일 다운로드1
11652정성태8/14/201818437사물인터넷: 25. 컬렉터 9V, 베이스에 5V와 3.3V 전압으로 테스트하는 C1815 트랜지스터파일 다운로드1
11651정성태8/14/201824177사물인터넷: 24. 9V 전압에서 테스트하는 C1815 트랜지스터 [1]파일 다운로드3
11650정성태8/14/201818338사물인터넷: 23. 가변저항으로 분압파일 다운로드1
11649정성태8/12/201820244사물인터넷: 22. 저항에 따른 전류 테스트파일 다운로드1
11648정성태8/12/201821554사물인터넷: 21. 퓨즈를 이용한 회로 보호파일 다운로드3
11647정성태8/8/201822409오류 유형: 476. 음수의 음수는 여전히 음수가 되는 수(절대값이 음수인 수)
11646정성태8/8/201817704오류 유형: 475. gacutil.exe 실행 시 "Failure initializing gacutil" 오류 발생
11645정성태8/8/201820338오류 유형: 474. 닷넷 COM+ - Failed to load the runtime. [1]
11644정성태8/6/201823406디버깅 기술: 118. windbg - 닷넷 개발자를 위한 MEX Debugging Extension 소개
11643정성태8/6/201823147사물인터넷: 20. 아두이노 레오나르도 R3 호환 보드의 3.3v 핀의 LED 전압/전류 테스트 [1]파일 다운로드1
11642정성태8/3/201821456Graphics: 20. Unity - LightMode의 ForwardBase에 따른 _WorldSpaceLightPos0 값 변화
11641정성태8/3/201826899Graphics: 19. Unity로 실습하는 Shader (10) - 빌보드 구현 [1]파일 다운로드1
... 91  [92]  93  94  95  96  97  98  99  100  101  102  103  104  105  ...