Microsoft MVP성태의 닷넷 이야기
디버깅 기술: 153. C# - PEB를 조작해 로드된 DLL을 숨기는 방법 [링크 복사], [링크+제목 복사],
조회: 10998
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

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

비밀번호

댓글 작성자
 




... 61  62  63  64  65  66  67  68  [69]  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
11922정성태5/29/201911567.NET Framework: 840. ML.NET 데이터 정규화파일 다운로드1
11921정성태5/28/201916496Math: 55. C# - 다항식을 위한 최소 자승법(Least Squares Method)파일 다운로드1
11920정성태5/28/201910060.NET Framework: 839. C# - PLplot 색상 제어
11919정성태5/27/201913205Math: 54. C# - 최소 자승법의 1차 함수에 대한 매개변수를 단순 for 문으로 구하는 방법 [1]파일 다운로드1
11918정성태5/25/201914380Math: 53. C# - 행렬식을 이용한 최소 자승법(LSM: Least Square Method)파일 다운로드1
11917정성태5/24/201914525Math: 52. MathNet을 이용한 간단한 통계 정보 처리 - 분산/표준편차파일 다운로드1
11916정성태5/24/201912514Math: 51. MathNET + OxyPlot을 이용한 간단한 통계 정보 처리 - Histogram파일 다운로드1
11915정성태5/24/201914807Linux: 11. 리눅스의 환경 변수 관련 함수 정리 - putenv, setenv, unsetenv
11914정성태5/24/201914504Linux: 10. 윈도우의 GetTickCount와 리눅스의 clock_gettime파일 다운로드1
11913정성태5/23/201912142.NET Framework: 838. C# - 숫자형 타입의 bit(2진) 문자열, 16진수 문자열 구하는 방법파일 다운로드1
11912정성태5/23/201911790VS.NET IDE: 137. Visual Studio 2019 버전 16.1부터 리눅스 C/C++ 프로젝트에 추가된 WSL 지원
11911정성태5/23/201910867VS.NET IDE: 136. Visual Studio 2019 - 리눅스 C/C++ 프로젝트에 인텔리센스가 동작하지 않는 경우
11910정성태5/23/201919529Math: 50. C# - MathNet.Numerics의 Matrix(행렬) 연산 [1]파일 다운로드1
11909정성태5/22/201913950.NET Framework: 837. C# - PLplot 사용 예제 [1]파일 다운로드1
11908정성태5/22/201912336.NET Framework: 836. C# - Python range 함수 구현파일 다운로드1
11907정성태5/22/201910126오류 유형: 541. msbuild - MSB4024 The imported project file "...targets" could not be loaded
11906정성태5/21/201910083.NET Framework: 835. .NET Core/C# - 리눅스 syslog에 로그 남기는 방법
11905정성태5/21/201910756.NET Framework: 834. C# - 폴더 경로 문자열에서 "..", "." 표기를 고려한 최종 문자열을 얻는 방법 - 두 번째 이야기
11904정성태5/21/201917027.NET Framework: 833. C# - Open Hardware Monitor를 이용한 CPU 온도 정보 [1]파일 다운로드1
11903정성태5/21/201911979오류 유형: 540. .NET Core - System.PlatformNotSupportedException: The named version of this synchronization primitive is not supported on this platform.
11902정성태5/21/201911132오류 유형: 539. mstest 실행 시 "The directory name is invalid." 오류 발생
11901정성태5/21/201912281오류 유형: 538. msbuild 오류 - Could not find a part of the path '%LOCALAPPDATA%\Temp\2\.NETFramework,Version=v4.0.AssemblyAttributes.cs'
11900정성태5/18/201911553오류 유형: 537. "sfc /scannow" 실행 중 시스템이 부팅되는 현상
11899정성태5/17/201912583Linux: 9. Linux에서 윈도우의 OutputDebugString 대신 사용할 수 있는 syslog [1]
11898정성태5/16/201913966VC++: 130. C++ string의 c_str과 data 함수의 차이점 [3]
11897정성태5/16/201920635오류 유형: 536. Visual Studio - "Developer Pack"을 설치했는데도 "대상 프레임워크" 목록에 나오지 않는 경우 [2]
... 61  62  63  64  65  66  67  68  [69]  70  71  72  73  74  75  ...