Microsoft MVP성태의 닷넷 이야기
VC++: 94. MPX(Memory Protection Extensions) 테스트 [링크 복사], [링크+제목 복사]
조회: 11994
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

Visual C++ - MPX(Memory Protection Extensions) 테스트

Visual C++ Team Blog에 재미있는 글이 실렸군요. ^^

Visual Studio 2015 Update 1: New Experimental Feature - MPX
; http://blogs.msdn.com/b/vcblog/archive/2016/01/20/visual-studio-2015-update-1-new-experimental-feature-mpx.aspx

MPX기능은 Skylake CPU 모델 중에서도 지난 2015년 10월부터 출시되는 제품부터 지원한다고 인텔은 발표했습니다. 애석하게도 국내의 경우, CPU 재고 소진 상황으로 인해 2015년도에 Skylake가 탑재된 컴퓨터를 구매하신 경우에는 거의 MPX 기능이 없다고 보시면 됩니다. (저도 1월 중순이 되어서야 국내에서 구할 수 있었습니다. ^^)

MPX 기능은 별도의 CPU 명령어 셋이 제공되기 때문에 해당 명령어들을 출력하려면 컴파일러 수준에서 제공되어야 합니다. Visual C++의 경우 2015 Update 1이후부티 /d2MPX 컴파일러 옵션으로 이를 제어할 수 있게 되었는데요... 간단하게 테스트를 해볼까요? ^^

다음의 소스코드는 "Visual Studio 2015 Update 1: New Experimental Feature - MPX" 글에 실린 내용을 그대로 복사해 온 것입니다.

#include "stdafx.h"
#include <Windows.h>

const int OUTBUFSIZE = 42;
wchar_t out[OUTBUFSIZE];

void copyUpper(wchar_t *str, size_t size)
{
    __try
    {
        for (unsigned int i = 0; i < size; i++)
        {
            out[i] = towupper(str[i]);
        }
    }
    __except (GetExceptionCode() == STATUS_ARRAY_BOUNDS_EXCEEDED)
    {
        wprintf(L"Caught array bounds exceeded exception\n");
    }
}

int main()
{
    wchar_t str[] = L"the quick brown fox jumps over the lazy dog";
    memset(out, 0, OUTBUFSIZE);
    copyUpper(str, wcsnlen_s(str, 255));
    wprintf(L"%s\n", out);
    return 0;
}

소스 코드를 보시면 out 문자열 배열은 42개만 잡혀 있지만 str 문자열은 43이기 때문에 안정된 버퍼 영역을 지나 (wchar_t) 2바이트를 더 쓰게 되는 오류가 있습니다. 일반적으로 이런 프로그램이 작성된 경우 out 버퍼 다음에 어떤 영역이 있는지에 따라 문제의 심각성이 달라집니다. (그래서, 오류 잡기가 더 힘들죠~~~ ^^;)

위의 소스코드를 컴파일할 때 /d2MPX 옵션을 주면 Visual C++ (2015 Update 1) 컴파일러는 MPX 관련 CPU 코드를 삽입해 줍니다. 이는 디버깅 시에 Disassembly 창을 통해 다음과 같이 확인할 수 있습니다.

    __try
    {
        for (unsigned int i = 0; i < size; i++)
00007FF7EA9C1064  mov         dword ptr [rbp+4],0  
00007FF7EA9C106B  jmp         copyUpper+45h (07FF7EA9C1075h)  
00007FF7EA9C106D  mov         eax,dword ptr [rbp+4]  
00007FF7EA9C1070  inc         eax  
00007FF7EA9C1072  mov         dword ptr [rbp+4],eax  
00007FF7EA9C1075  mov         eax,dword ptr [rbp+4]  
00007FF7EA9C1078  cmp         rax,qword ptr [size]  
00007FF7EA9C107F  bnd jae     copyUpper+0C3h (07FF7EA9C10F3h)  
        {
            out[i] = towupper(str[i]);
00007FF7EA9C1082  mov         eax,dword ptr [rbp+4]  
00007FF7EA9C1085  mov         rcx,qword ptr [str]  
00007FF7EA9C108C  movzx       ecx,word ptr [rcx+rax*2]  
00007FF7EA9C1090  bnd call    qword ptr [__imp_towupper (07FF7EA9C5220h)]  
00007FF7EA9C1097  mov         ecx,dword ptr [rbp+4]  
00007FF7EA9C109A  shl         rcx,1  
00007FF7EA9C109D  lea         rdx,[out (07FF7EA9C7050h)]  
00007FF7EA9C10A4  bndmk       bnd0,qword ptr [rdx+53h]  
00007FF7EA9C10A9  bndmov      oword ptr [rbp+148h],bnd0  
00007FF7EA9C10B1  lea         rdx,[out (07FF7EA9C7050h)]  
00007FF7EA9C10B8  add         rdx,rcx  
00007FF7EA9C10BB  mov         qword ptr [rbp+1D8h],rdx  
00007FF7EA9C10C2  mov         rcx,qword ptr [rbp+1D8h]  
00007FF7EA9C10C9  inc         rcx  
00007FF7EA9C10CC  bndmov      bnd0,oword ptr [rbp+148h]  
00007FF7EA9C10D4  mov         rdx,qword ptr [rbp+1D8h]  
00007FF7EA9C10DB  bndcu       bnd0,rcx  
00007FF7EA9C10DF  bndcl       bnd0,rdx  
00007FF7EA9C10E3  mov         rcx,qword ptr [rbp+1D8h]  
00007FF7EA9C10EA  mov         word ptr [rcx],ax  
        }
00007FF7EA9C10ED  bnd jmp     copyUpper+3Dh (07FF7EA9C106Dh)  
    }
00007FF7EA9C10F3  jmp         copyUpper+102h (07FF7EA9C1132h)  
    __except (GetExceptionCode() == STATUS_ARRAY_BOUNDS_EXCEEDED)
    {
        wprintf(L"Caught array bounds exceeded exception\n");
00007FF7EA9C10F5  lea         rax,[string L"Caught array bounds "... (07FF7EA9C5370h)]  
00007FF7EA9C10FC  bndmk       bnd0,qword ptr [rax+4Fh]  
00007FF7EA9C1101  bndmov      oword ptr [rbp+178h],bnd0  
00007FF7EA9C1109  lea         rax,[string L"Caught array bounds "... (07FF7EA9C5370h)]  
00007FF7EA9C1110  bndmov      bnd0,oword ptr [rbp+178h]  
00007FF7EA9C1118  bndstx      qword ptr [rsp+rax],bnd0  
00007FF7EA9C111C  lea         rcx,[string L"Caught array bounds "... (07FF7EA9C5370h)]  
00007FF7EA9C1123  bndmov      bnd0,oword ptr [rbp+178h]  
00007FF7EA9C112B  bnd call    wprintf (07FF7EA9C1430h)  
00007FF7EA9C1131  nop  
    }

MPX 관련해서 128비트 길이를 가진 4개의 경계 레지스터(bounds registers)가 BND0 ~ BND3까지 제공되는데, 상위 경계 8바이트와 하위 경계 8바이트를 하나의 레지스터에서 담게 되는 구조입니다.

위의 명령어들 중에서 특히 bndmk를 거치면 MPX 관련 CPU 레지스터에 유효 주소 범위값이 설정됩니다.

mpx_test_sample_1.png

그리고 그 범위 밖의 메모리 주소를 접근하면 예외가 발생하게 되고 __except 핸들러 내부의 코드가 실행됩니다. (MPX가 지원되는 CPU에서만 __except 내로 진입합니다.)

(첨부한 파일은 위의 코드를 포함하는 간단한 Visual C++ 프로젝트입니다.)




좀 더 자세한 정보는 인텔이 배포하고 있는 문서를 보시면 됩니다.

Intel_MPX_EnablingGuide.pdf
; https://software.intel.com/sites/default/files/managed/9d/f6/Intel_MPX_EnablingGuide.pdf

물론 MPX 명령어가 지원되는 CPU들이 일반화되려면 시간이 한참 지나야 할 것입니다. 그래도 미리 이 명령어가 추가되도록 /d2MPX 옵션을 줘도 상관없는데요, 왜냐하면 MPX 지원이 안되는 CPU에서 이 명령어들이 실행되면 CPU 입장에서 그냥 NOP 명령어를 수행하는 것과 동일하게 처리하므로 결국 아무런 영향도 주지 않게 됩니다. 따라서, CPU 종류를 개발자 입장에서 신경쓸 필요도 없다는 이야기입니다.

단지, MPX가 지원되지 않는 컴퓨터에서 "Intel MPX Runtime Driver"는 설치될 수 있는데요. 이런 환경에서 MPX 추가된 응용 프로그램을 실행하면 다음과 같은 메시지를 동반하는 블루스크린을 볼 수 있습니다.

Your PC ran into a problem and needs to restart. We're just collecting some error info, and then we'll restart for you. (..% complete)

If you'd like to know more, you can search online later for this error: KMODE_EXCEPTION_NOT_HANDLED (MpxRuntime.sys)

이것 역시 그다지 흔한 환경은 아닐 것이기 때문에 문제가 될 수준은 아니겠지요! ^^

정리하면... 그냥 MPX는 인텔(과 컴파일러 회사들)이 지원해주는 공짜 점심(Free lunch)라고 보시면 됩니다.




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







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

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

비밀번호

댓글 작성자
 




... 46  47  48  49  50  51  52  53  54  [55]  56  57  58  59  60  ...
NoWriterDateCnt.TitleFile(s)
12248정성태6/29/20208174오류 유형: 624. SQL 서버 오류 - service-specific error code 17051
12247정성태6/29/20209702.NET Framework: 918. C# - 불린 형 상수를 반환값으로 포함하는 3항 연산자 사용 시 단축 표현 권장(IDE0075) [2]파일 다운로드1
12246정성태6/29/202010505.NET Framework: 917. C# - USB 관련 ETW(Event Tracing for Windows)를 이용한 키보드 입력을 감지하는 방법
12245정성태6/24/202010985.NET Framework: 916. C# - Task.Yield 사용법 (2) [2]파일 다운로드1
12244정성태6/24/202010764.NET Framework: 915. ETW(Event Tracing for Windows)를 이용한 닷넷 프로그램의 내부 이벤트 활용 [1]파일 다운로드1
12243정성태6/23/20208379VS.NET IDE: 147. Visual C++ 프로젝트 - .NET Core EXE를 "Debugger Type"으로 지원하는 기능 추가
12242정성태6/23/20209120오류 유형: 623. AADSTS90072 - User account '...' from identity provider 'live.com' does not exist in tenant 'Microsoft Services'
12241정성태6/23/202012431.NET Framework: 914. C# - Task.Yield 사용법파일 다운로드1
12240정성태6/23/202013723오류 유형: 622. 소켓 바인딩 시 "System.Net.Sockets.SocketException: An attempt was made to access a socket in a way forbidden by its access permissions" 오류 발생
12239정성태6/21/202010160Linux: 30. (윈도우라면 DLL에 속하는) .so 파일이 텍스트로 구성된 사례 [1]
12238정성태6/21/202010096.NET Framework: 913. C# - SharpDX + DXGI를 이용한 윈도우 화면 캡처 라이브러리
12237정성태6/20/20209881.NET Framework: 912. 리눅스 환경의 .NET Core에서 "test".IndexOf("\0")가 0을 반환
12236정성태6/19/202010259오류 유형: 621. .NET Standard 대상으로 빌드 시 dynamic 예약어에서 컴파일 오류 - error CS0656: Missing compiler required member 'Microsoft.CSharp.RuntimeBinder.CSharpArgumentInfo.Create'
12235정성태6/19/20209892오류 유형: 620. Windows 10 - Inaccessible boot device 블루 스크린
12234정성태6/19/20209616개발 환경 구성: 494. NuGet - nuspec의 패키지 스키마 버전(네임스페이스) 업데이트 방법
12233정성태6/19/20209316오류 유형: 619. SQL 서버 - The transaction log for database '...' is full due to 'LOG_BACKUP'. - 두 번째 이야기
12232정성태6/19/20208271오류 유형: 618. SharePoint - StoreBusyRetryLater 오류
12231정성태6/15/202010674.NET Framework: 911. Console/Service Application을 위한 SynchronizationContext - AsyncContext
12230정성태6/15/202010060오류 유형: 617. IMetaDataImport::GetMethodProps가 반환하는 IL 코드 주소(RVA) 문제
12229정성태6/13/202011915.NET Framework: 910. USB/IP PROJECT를 이용해 C#으로 USB Keyboard + Mouse 가상 장치 만들기 [1]
12228정성태6/12/202011998.NET Framework: 909. C# - Source Generator를 적용한 XmlCodeGenerator파일 다운로드1
12227정성태6/12/202015948오류 유형: 616. Visual Studio의 느린 업데이트 속도에 대한 원인 분석 [5]
12226정성태6/11/202013280개발 환경 구성: 493. OpenVPN의 네트워크 구성 [4]파일 다운로드1
12225정성태6/11/202012236개발 환경 구성: 492. 윈도우에 OpenVPN 설치 - 클라이언트 측 구성
12224정성태6/11/202020100개발 환경 구성: 491. 윈도우에 OpenVPN 설치 - 서버 측 구성 [1]
12223정성태6/9/202014131.NET Framework: 908. C# - Source Generator 소개 [10]파일 다운로드2
... 46  47  48  49  50  51  52  53  54  [55]  56  57  58  59  60  ...