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분
정성태

... 61  62  63  64  65  66  67  68  [69]  70  71  72  73  74  75  ...
NoWriterDateCnt.TitleFile(s)
12211정성태4/27/202019279개발 환경 구성: 486. WSL에서 Makefile로 공개된 리눅스 환경의 C/C++ 소스 코드 빌드
12210정성태4/20/202020735.NET Framework: 903. .NET Framework의 Strong-named 어셈블리 바인딩 (1) - app.config을 이용한 바인딩 리디렉션 [1]파일 다운로드1
12209정성태4/13/202017427오류 유형: 614. 리눅스 환경에서 C/C++ 프로그램이 Segmentation fault 에러가 발생한 경우 (2)
12208정성태4/12/202016003Linux: 29. 리눅스 환경에서 C/C++ 프로그램이 Segmentation fault 에러가 발생한 경우
12207정성태4/2/202015857스크립트: 19. Windows PowerShell의 NonInteractive 모드
12206정성태4/2/202018451오류 유형: 613. 파일 잠금이 바로 안 풀린다면? - The process cannot access the file '...' because it is being used by another process.
12205정성태4/2/202015116스크립트: 18. Powershell에서는 cmd.exe의 명령어를 지원하진 않습니다.
12204정성태4/1/202015136스크립트: 17. Powershell 명령어에 ';' (semi-colon) 문자가 포함된 경우
12203정성태3/18/202017967오류 유형: 612. warning: 'C:\ProgramData/Git/config' has a dubious owner: '...'.
12202정성태3/18/202021216개발 환경 구성: 486. .NET Framework 프로젝트를 위한 GitLab CI/CD Runner 구성
12201정성태3/18/202018460오류 유형: 611. git-credential-manager.exe: Using credentials for username "Personal Access Token". [1]
12200정성태3/18/202018550VS.NET IDE: 145. NuGet + Github 라이브러리 디버깅 관련 옵션 3가지 - "Enable Just My Code" / "Enable Source Link support" / "Suppress JIT optimization on module load (Managed only)"
12199정성태3/17/202016183오류 유형: 610. C# - CodeDomProvider 사용 시 Unhandled Exception: System.IO.DirectoryNotFoundException: Could not find a part of the path '...\f2_6uod0.tmp'.
12198정성태3/17/202019542오류 유형: 609. SQL 서버 접속 시 "Cannot open user default database. Login failed."
12197정성태3/17/202018849VS.NET IDE: 144. .NET Core 콘솔 응용 프로그램을 배포(publish) 시 docker image 자동 생성 - 두 번째 이야기 [1]
12196정성태3/17/202015972오류 유형: 608. The ServicedComponent being invoked is not correctly configured (Use regsvcs to re-register).
12195정성태3/16/202018285.NET Framework: 902. C# - 프로세스의 모든 핸들을 열람 - 세 번째 이야기
12194정성태3/16/202021007오류 유형: 607. PostgreSQL - Npgsql.NpgsqlException: sorry, too many clients already
12193정성태3/16/202017957개발 환경 구성: 485. docker - SAP Adaptive Server Enterprise 컨테이너 실행 [1]
12192정성태3/14/202019995개발 환경 구성: 484. docker - Sybase Anywhere 16 컨테이너 실행
12191정성태3/14/202021073개발 환경 구성: 483. docker - OracleXE 컨테이너 실행 [1]
12190정성태3/14/202015656오류 유형: 606. Docker Desktop 업그레이드 시 "The process cannot access the file 'C:\Program Files\Docker\Docker\resources\dockerd.exe' because it is being used by another process."
12189정성태3/13/202021254개발 환경 구성: 482. Facebook OAuth 처리 시 상태 정보 전달 방법과 "유효한 OAuth 리디렉션 URI" 설정 규칙
12188정성태3/13/202026043Windows: 169. 부팅 시점에 실행되는 chkdsk 결과를 확인하는 방법
12187정성태3/12/202015633오류 유형: 605. NtpClient was unable to set a manual peer to use as a time source because of duplicate error on '...'.
12186정성태3/12/202017416오류 유형: 604. The SysVol Permissions for one or more GPOs on this domain controller and not in sync with the permissions for the GPOs on the Baseline domain controller.
... 61  62  63  64  65  66  67  68  [69]  70  71  72  73  74  75  ...