Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 5개 있습니다.)
(시리즈 글이 3개 있습니다.)
.NET Framework: 299. 해당 어셈블리가 Debug 빌드인지, Release 빌드인지 알아내는 방법
; https://www.sysnet.pe.kr/2/0/1227

.NET Framework: 315. 해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법
; https://www.sysnet.pe.kr/2/0/1279

.NET Framework: 328. 해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법 - 두 번째 이야기
; https://www.sysnet.pe.kr/2/0/1296




해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법 - 두 번째 이야기


예전에 다음과 같은 글을 쓴 적이 있었는데요. ^^

해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법
; https://www.sysnet.pe.kr/2/0/1279

그때 당시에, 좀 더 개선될 수 있는 것으로 PE 파일 포맷 분석하는 방법을 남겨놓았었는데요. 혹시나, ^^ 어떤 분이 그에 관해서 글을 쓰시지 않을까 하는... 기대를 갖고 있었는데, 오늘 기준으로 아무도 안 쓰시네요. ^^; 별로 관심 사항은 아닌가 봅니다.

어쨌든, 이번에는 PE 파일을 직접 살펴보고 Managed/Unmanaged 판별을 해낼 텐데요.

사실, 그다지 어려운 문제는 아닙니다. 게다가 PE 파일 포맷에 대해 잘 설명된 책도 있고요.

Windows 시스템 실행파일의 구조와 원리
; http://www.yes24.com/24/goods/1493278?scode=029

그럼, 우선 IMAGE_DOS_HEADER 먼저 구조체를 채워보는 것부터 시작하겠습니다.

책에 설명된 데로, 해당 구조체는 WinNT.h에 정의되어 있지만 여기서는 C# 에서 구현할 것이기 때문에 적절한 Interop 구조체가 필요합니다. 역시 ^^ pinvoke.net이 그 해답이겠지요.

IMAGE_DOS_HEADER
; http://www.pinvoke.net/default.aspx/Structures/IMAGE_DOS_HEADER.html

위의 정의를 이용하면 간단하게 다음과 같이 IMAGE_DOS_HEADER의 내용을 채울 수 있습니다.

using (FileStream fs = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.ReadWrite))
{
    // IMAGE_DOS_HEADER를 읽어들이고,
    IMAGE_DOS_HEADER dosHeader = new IMAGE_DOS_HEADER();
    {
        int dosHeaderSize = Marshal.SizeOf(dosHeader);
        byte[] dosHeaderBytes = new byte[dosHeaderSize];
        fs.Read(dosHeaderBytes, 0, dosHeaderSize);

        fixed (byte* p = dosHeaderBytes)
        {
            IntPtr ptr = new IntPtr(p);
            object objDosHeader = Marshal.PtrToStructure(ptr, typeof(IMAGE_DOS_HEADER));
            dosHeader = (IMAGE_DOS_HEADER)objDosHeader;
        }

        if (dosHeader.isValid == false) // _e_magic == "MZ"
        {
            return false;
        }
    }
}

(이후의 PE 포맷에 관련된 헤더들도 모두 기본적으로는 위와 같은 구조를 따릅니다.)

IMAGE_DOS_HEADER 이후에는 IMAGE_NT_HEADERS 구조체를 읽어들여야 하는데, 아쉽게도 IMAGE_NT_HEADERS 내부에 포함하고 있는 IMAGE_OPTIONAL_HEADER가 32비트/64비트 모듈에 따라 달라지므로 이 2가지를 한꺼번에 지원하기 위해 이에 대한 판단을 먼저 해야 합니다.

그러려면, 우선 'PE Signature' 바이트를 읽어서 넘어가고,

// IMAGE_NT_HEADERS - signature 를 읽어들이고,
{
    fs.Position = dosHeader.e_lfanew;
    byte [] ntSignature = new byte[4];
    fs.Read(ntSignature, 0, 4);

    // PE 헤더임을 확인 (IMAGE_NT_SIGNATURE == 0x00004550)
    if (ntSignature[0] != 80 || ntSignature[1] != 69 || ntSignature[2] != 0 || ntSignature[3] != 0)
    {
        return false;
    }
}

다행히 IMAGE_NT_HEADERS 내에 IMAGE_FILE_HEADER 구조체까지 이어서 읽어들이면 32비트/64비트 구별을 할 수 있게 됩니다.

// IMAGE_NT_HEADERS - IMAGE_FILE_HEADER를 읽어들임
IMAGE_FILE_HEADER ntFileHeader = new IMAGE_FILE_HEADER();
bool is64BitHeader = false;
{
    int ntFileHeaderSize = Marshal.SizeOf(ntFileHeader);
    byte[] ntFileHeaderBytes = new byte[ntFileHeaderSize];
    fs.Read(ntFileHeaderBytes, 0, ntFileHeaderSize);

    fixed (byte* p = ntFileHeaderBytes)
    {
        IntPtr ptr = new IntPtr(p);
        object objNtFileHeader = Marshal.PtrToStructure(ptr, typeof(IMAGE_FILE_HEADER));
        ntFileHeader = (IMAGE_FILE_HEADER)objNtFileHeader;
    }

    if (ntFileHeader.Machine == IMAGE_FILE_MACHINE_I386)
    {
        is64BitHeader = false;
    }
    else if (ntFileHeader.Machine == IMAGE_FILE_MACHINE_AMD64 ||
        ntFileHeader.Machine == IMAGE_FILE_MACHINE_IA64)
    {
        is64BitHeader = true;
    }
    else
    {
        return false;
    }
}

이걸로 모든 준비는 끝났습니다. .NET DLL/EXE의 경우, IMAGE_OPTIONAL_HEADER 내부의 DataDirectory 구조체 중의 하나인 CLRRuntimeHeader 값을 포함하고 있기 때문에 그 값이 유효한지를 판단해 내면 됩니다. 다음은 32비트인 경우에 사용될 수 있는 코드입니다.

ushort optionalHeaderSize = ntFileHeader.SizeOfOptionalHeader;
if (is64BitHeader == false)
{
    IMAGE_OPTIONAL_HEADER32 optionalHeader32 = new IMAGE_OPTIONAL_HEADER32();

    byte[] optionalHeaderBytes = new byte[optionalHeaderSize];
    fs.Read(optionalHeaderBytes, 0, optionalHeaderSize);

    fixed (byte* p = optionalHeaderBytes)
    {
        IntPtr ptr = new IntPtr(p);
        object objOptionalHeader = Marshal.PtrToStructure(ptr, typeof(IMAGE_OPTIONAL_HEADER32));
        optionalHeader32 = (IMAGE_OPTIONAL_HEADER32)objOptionalHeader;

        if (optionalHeader32.Magic != MagicType.IMAGE_NT_OPTIONAL_HDR32_MAGIC
            || optionalHeader32.NumberOfRvaAndSizes != 16)
        {
            return false;
        }
    }

    return optionalHeader32.CLRRuntimeHeader.VirtualAddress != 0;
}
else
{
    ...[생략: 32비트와 유사]...
}

원래 DataDirectory는 16개의 배열로 정의되는 것이 보통인데, 위의 코드에서는 C# Interop 코드에서 명시적으로 15번째 COM descriptor 요소를 CLRRuntimeHeader 속성으로 제공하고 있어 쉽게 접근하고 있습니다. 따라서, 그 값이 0이면 Native DLL이고, 그렇지 않으면 .NET DLL이 됩니다.

첨부된 파일은 지난번 글에 포함된 소스 코드에다 이번 글의 IsManagedFromPE 메서드를 추가 구현한 것을 포함하고 있습니다.




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 7/10/2021]

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

비밀번호

댓글 작성자
 



2012-06-08 07시01분
[윤성민] 저 같은 경우는 닷넷의 Version클래스로.. dll의 버전을 읽었을때 가져오면.. 그냥 managed판별하고 0으로 돌려주면 Unmanaged로 판별했던것으로 기억합니다만
[guest]
2012-06-08 02시48분
윤성민 님의 경우 AssemblyName.GetAssemblyName(assemblyPath).Version으로 알아낸다는 건가요? 그렇게 되면 결국 지난번 글에서의 문제점과 같은 현상이 발생합니다.

해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법
; http://www.sysnet.pe.kr/2/0/1279

위의 글 중간에 설명을 했지만, 경우에 따라서 System.BadImageFormatException이 발생할 수 있습니다. ^^
정성태
2015-01-05 04시04분
정성태
2021-03-03 01시23분
정성태

... 181  182  183  184  185  186  187  188  189  190  191  192  193  [194]  195  ...
NoWriterDateCnt.TitleFile(s)
91정성태11/14/200518725VC++: 12. VS.NET 2005 VC++ Debug: Expression: ( (state != ST_INVALID ) )
90정성태1/27/200519549.NET Framework: 22. Debug: The underlying connection was closed: Unable to connect to the remote server.
89정성태1/26/200524015VC++: 11. Delay Loaded DLL
87정성태1/23/200517614VS.NET IDE: 18. VS.NET 2005 Beta 1 - VC++ 프로젝트에서 Connection Point 구현시 버그
88정성태1/23/200517327    답변글 VS.NET IDE: 18.1. VS.NET 2003 : VC++ 프로젝트에서 Connection Point 추가시에도 버그
86정성태1/23/200523037.NET Framework: 21. Code Snippet - Enum과 관련된 다양한 형변환 [1]
85정성태1/23/200521210스크립트: 4. Windows 2003에서 BHO(Browser Helper Objects) 동작 안하는 현상 [1]
83정성태1/18/200526355.NET Framework: 20. System.AccessViolationException 예외가 발생한 한 예.
82정성태1/3/200519820VS.NET IDE: 17. Windows 운영 - 특정 사용자 또는 그룹에 대해서 파일 공유 접근 금지
79정성태1/20/200527765기타: 8. DELL Latitude D800 노트북 컴퓨터의 PC Beep 소음(!) 문제.
78정성태12/27/200420121VS.NET IDE: 16. MS 제품 관련 사용되는 TCP/IP 포트 열거파일 다운로드1
77정성태12/27/200420378VS.NET IDE: 15. Virtual CD-ROM Control Panel - ISO 이미지를 CD-ROM 드라이브처럼 접근하게 해주는 EXE 프로그램 [1]파일 다운로드1
76정성태12/27/200421459VS.NET IDE: 14. VPN 접속시 IP를 고정적으로 할당받는 방법 [1]
75정성태12/27/200417674VS.NET IDE: 13. VS.NET 2005 Beta 1 - Portfolio Explorer 에 등록된 Team Server 항목 삭제 방법
84정성태1/19/200518573    답변글 VS.NET IDE: 13.1. VS.NET 2005 Beta 1 : Team Server 에 등록된 포트폴리오 프로젝트 삭제 방법
74정성태12/26/200419171VS.NET IDE: 12. [시나리오] VS.NET 2005 Team Foundation Server을 Virtual Server에 설치 [1]
80정성태12/31/200418488    답변글 VS.NET IDE: 12.1. Client Tier, 즉 VS.NET 2005가 설치된 컴퓨터도 ActiveDirectory에 참여를 해야 합니다.
81정성태12/31/200420362    답변글 VS.NET IDE: 12.2. Tier 컴퓨터를 모두 영문으로 재구성
109정성태3/4/200515563    답변글 VS.NET IDE: 12.3. [보완] MS 공식 아티클 - Installing the December CTP Release of Visual Studio Team System
73정성태11/14/200517392.NET Framework: 19. VS.NET 2005 Team Foundation Server 설치오류 - 26204 예외
72정성태12/26/200418870.NET Framework: 18. .NET Framework 2.0 Beta 설치 후에 Windows SharePoint Service 오류 [1]
136정성태3/31/200518736    답변글 .NET Framework: 18.1. Windows Sharepoint Services 를 설치한 이후 ASP.NET 오류 문제
71정성태12/26/200417099VS.NET IDE: 11. SQL Server 2005 Beta 2 를 네트워크 드라이브로부터 설치시 오류
70정성태12/26/200419922VS.NET IDE: 10. WSS 설치 후 localhost 접근 보안 오류
69정성태12/5/200416999VS.NET IDE: 9. 다른 컴퓨터(방화벽 설치)에 설치된 SQL Server에 통합 인증을 할 때 필요한 포트
68정성태10/31/200421975.NET Framework: 17. Win32_NTLogEvent를 c#에서 wmi 쿼리할 때..에러..
... 181  182  183  184  185  186  187  188  189  190  191  192  193  [194]  195  ...