Microsoft MVP성태의 닷넷 이야기
개발 환경 구성: 386. .NET Framework Native compiler 프리뷰 버전 사용법 [링크 복사], [링크+제목 복사],
조회: 19442
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 

(시리즈 글이 7개 있습니다.)
개발 환경 구성: 386. .NET Framework Native compiler 프리뷰 버전 사용법
; https://www.sysnet.pe.kr/2/0/11563

.NET Framework: 2069. .NET 7 - AOT(ahead-of-time) 컴파일
; https://www.sysnet.pe.kr/2/0/13162

닷넷: 2175. C# - DllImport 메서드의 AOT 지원을 위한 LibraryImport 옵션
; https://www.sysnet.pe.kr/2/0/13466

닷넷: 2184. C# - 하나의 resource 파일을 여러 프로그램에서 (AOT 시에도) 사용하는 방법
; https://www.sysnet.pe.kr/2/0/13483

개발 환경 구성: 696. C# - 리눅스용 AOT 빌드를 docker에서 수행
; https://www.sysnet.pe.kr/2/0/13487

닷넷: 2202. C# - PublishAot의 glibc에 대한 정적 링킹하는 방법
; https://www.sysnet.pe.kr/2/0/13529

오류 유형: 913. C# - AOT StaticExecutable 정적 링킹 시 빌드 오류
; https://www.sysnet.pe.kr/2/0/13669




.NET Framework Native compiler 프리뷰 버전 사용법

MVP 내부 그룹에 Native compiler가 preview 상태로 이야기가 오고 가길래 아직 NDA인줄 알았더니 다음과 같이 NuGet에 이미 배포된 버전입니다. ^^

Microsoft.DotNet.Framework.NativeImageCompiler
; https://www.nuget.org/packages/Microsoft.DotNet.Framework.NativeImageCompiler

게다가 문서까지 자세하게 공개되어 있습니다.

Optimize your .NET Desktop apps with native images
; https://go.microsoft.com/fwlink/?linkid=2002310
; https://learn.microsoft.com/en-us/windows/uwp/porting/desktop-to-uwp-r2r

자, 그럼 간단하게 테스트를 해볼까요? ^^

우선, 콘솔 프로젝트를 하나 만들고 다음과 같이 NuGet에서 Native compiler 패키지를 추가해 줍니다.

Install-Package Microsoft.DotNet.Framework.NativeImageCompiler -Version 0.0.1-prerelease-00002 

그다음, 아래의 조건에 맞게 프로젝트의 설정을 변경해 줍니다.

  • .NET Framework 버전은 최소 4.6.2 이상
  • (AnyCPU가 아닌) 명시적으로 x86 또는 x64로 지정
  • Release 모드로 빌드

위와 같은 조건을 만족하면 EXE 파일이 아래의 2개 폴더에 각각 생성됩니다. (유의할 것은, \bin 폴더 하위가 아닙니다.)

\obj\Release\ConsoleApp1.exe
\obj\Release\R2R\ConsoleApp1.exe

자세히 보면 \Release 폴더에 생긴 exe 파일보다 \R2R 폴더에 생긴 EXE 파일의 크기가 살짝 큰 것을 볼 수 있습니다. 왜냐하면, 해당 EXE 파일은 IL 코드 뿐만 아니라 Native 코드까지 함께 포함하고 있기 때문이라고 합니다. 여차하면 IL로 실행하겠다는 의지가 ^^ 엿보이는군요. (따라서, .NET Reflector 같은 역어셈블 도구로 보면 코드가 여전히 잘 보입니다.)

그나저나 Native compiler로 빌드하는 의미가 뭘까요? 단순히, 기계어 컴파일이 되었다는 사실에서 기존 IL 기반의 닷넷 실행 파일보다 "실행 성능"이 빠를 거라고 기대하시면 안 됩니다. 닷넷 프로그램은 Native compiler 처리를 거쳤다고 해서 관리 힙을 쓰지 않는 것은 아닙니다. 즉, 기존의 닷넷 프로그램과 실행 면에서 보면 다를 바가 없습니다. 단지, "JIT Compile" 과정만 생략된 것에 불과합니다.

따라서 JIT 과정이 없어짐으로 인해 초기 로딩 속도가 빨라지는 것이지 "실행 성능" 자체가 좋아지는 것은 아닙니다. 간단하게 예를 들어, 기존에는 소수 판정 프로그램이 1초 걸렸다면 Native compiler로 처리했어도 여전히 1초가 걸리는 것에는 변함이 없습니다. 물론, "최초 소수 판정" 코드의 실행 속도는 빠릅니다. 단지 이후의 동일한 소수 판정 코드가 빨라지는 것은 아닙니다.




참고로, NuGet에서 다운로드 한 Native compiler 실행 파일(NgenR2R.exe)은 다음의 폴더에 위치해 있습니다.

%USERPROFILE%\.nuget\packages\runtime.win10-x64.microsoft.dotnet.framework.nativeimagecompiler\0.0.1-prerelease-00002\build\net462\..\..\tools

그런데, 환경에 따라 비주얼 스튜디오에서 빌드 시 다음과 같은 오류가 발생할 수 있습니다.

Error The command "C:\Users\Quick Tester\.nuget\packages\runtime.win10-x64.microsoft.dotnet.framework.nativeimagecompiler\0.0.1-prerelease-00002\build\net462\..\..\tools\NgenR2R.exe "@obj\Release\\R2R\ConsoleApp1.rsp"" exited with code 9009.

"exited with code 9009" 오류는,

You receive a build error message when you use the Assembly Registration tool (Regasm.exe) in Visual C#
; https://support.microsoft.com/ha-latn-ng/help/908268/you-receive-a-build-error-message-when-you-use-the-assembly-registrati

대개 실행 파일을 찾을 수 없어 발생하는 것 같습니다. 실제로 비주얼 스튜디오가 실행한 하위 프로세스의 명령행을 보면 동일한 이유가 발생했음을 알 수 있습니다.

C:\Users\Quick Tester\.nuget\packages\runtime.win10-x64.microsoft.dotnet.framework.nativeimagecompiler\0.0.1-prerelease-00002\build\net462\..\..\tools\NgenR2R.exe "@obj\Release\\R2R\ConsoleApp1.rsp"

짐작하셨죠? ^^ 중간에 공백 문자가 있어서 나타난 문제인데 일단 이것을 수정할 방법은 없고 다음과 같이 명령행에서 수작업으로 직접 따옴표를 추가해 실행 파일 경로를 묶어 주면 됩니다.

"C:\Users\Quick Tester\.nuget\packages\runtime.win10-x64.microsoft.dotnet.framework.nativeimagecompiler\0.0.1-prerelease-00002\build\net462\..\..\tools\NgenR2R.exe" "@obj\Release\\R2R\ConsoleApp1.rsp"

어차피 ^^ preview 버전에 대한 테스트일 뿐이니 딱히 불편한 정도는 아닙니다. 어쨌든, 이렇게 하면 다음과 같이 잘 실행이 됩니다.

E:\ConsoleApp1\ConsoleApp1>"%USERPROFILE%\.nuget\packages\runtime.win10-x64.microsoft.dotnet.framework.nativeimagecompiler\0.0.1-prerelease-00002\build\net462\..\..\tools\NgenR2R.exe" "@obj\Release\\R2R\ConsoleApp1.rsp"
Microsoft (R) CLR Native Image Generator - Version 4.8.3631.0
Copyright (c) Microsoft Corporation.  All rights reserved.

Loading C:\Users\Quick Tester\.nuget\packages\runtime.win10-x64.microsoft.dotnet.framework.nativeimagecompiler\0.0.1-prerelease-00002\tools\ref\mscorlib.dll
Loading E:\ConsoleApp1\ConsoleApp1\obj\Release\ConsoleApp1.exe
Native image obj\Release\\R2R\ConsoleApp1.exe generated successfully.




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







[최초 등록일: ]
[최종 수정일: 4/4/2024]

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

비밀번호

댓글 작성자
 



2022-08-16 08시06분
[진우] 이것은 실행시에 CLR 이 필요한 NGEN 인가요?
아니면 NET Native 인가요?
[guest]
2022-08-16 11시25분
.NET Native가 맞습니다. 저도 이때 이후로 제대로 테스트를 못했는데... 진우 님께서 한번 해보시고 결과 좀 공유해 주세요. ^^
정성태

... 136  137  138  139  140  141  142  143  144  145  146  147  148  149  [150]  ...
NoWriterDateCnt.TitleFile(s)
1303정성태6/26/201227397개발 환경 구성: 152. sysnet DB를 SQL Azure 데이터베이스로 마이그레이션
1302정성태6/25/201229404개발 환경 구성: 151. Azure 웹 사이트에 사용자 도메인 네임 연결하는 방법
1301정성태6/20/201225763오류 유형: 156. KB2667402 윈도우 업데이트 실패 및 마이크로소프트 Answers 웹 사이트 대응
1300정성태6/20/201231767.NET Framework: 329. C# - Rabin-Miller 소수 생성방법을 이용하여 RSACryptoServiceProvider의 개인키를 직접 채워보자 [1]파일 다운로드2
1299정성태6/18/201232880제니퍼 .NET: 21. 제니퍼 닷넷 - Ninject DI 프레임워크의 성능 분석 [2]파일 다운로드2
1298정성태6/14/201234405VS.NET IDE: 72. Visual Studio에서 pfx 파일로 서명한 경우, 암호는 어디에 저장될까? [2]
1297정성태6/12/201231048VC++: 63. 다른 프로세스에 환경 변수 설정하는 방법파일 다운로드1
1296정성태6/5/201227675.NET Framework: 328. 해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법 - 두 번째 이야기 [4]파일 다운로드1
1295정성태6/5/201225077.NET Framework: 327. RSAParameters와 System.Numerics.BigInteger 이야기파일 다운로드1
1294정성태5/27/201248524.NET Framework: 326. 유니코드와 한글 - 유니코드와 닷넷을 이용한 한글 처리 [7]파일 다운로드2
1293정성태5/24/201229773.NET Framework: 325. System.Drawing.Bitmap 데이터를 Parallel.For로 처리하는 방법 [2]파일 다운로드1
1292정성태5/24/201223754.NET Framework: 324. First-chance exception에 대해 조건에 따라 디버거가 멈추게 할 수는 없을까? [1]파일 다운로드1
1291정성태5/23/201230274VC++: 62. 배열 초기화를 위한 기계어 코드 확인 [2]
1290정성태5/18/201235078.NET Framework: 323. 관리자 권한이 필요한 작업을 COM+에 대행 [7]파일 다운로드1
1289정성태5/17/201239237.NET Framework: 322. regsvcs.exe로 어셈블리 등록 시 시스템 변경 사항 [5]파일 다운로드2
1288정성태5/17/201226461.NET Framework: 321. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (3) - Type Library파일 다운로드1
1287정성태5/17/201229297.NET Framework: 320. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (2) - .NET 4.0 + .NET 2.0 [2]
1286정성태5/17/201238217.NET Framework: 319. regasm.exe로 어셈블리 등록 시 시스템 변경 사항 (1) - .NET 2.0 + x86/x64/AnyCPU [5]
1285정성태5/16/201233264.NET Framework: 318. gacutil.exe로 어셈블리 등록 시 시스템 변경 사항파일 다운로드1
1284정성태5/15/201225692오류 유형: 155. Windows Phone 연결 상태에서 DRIVER POWER STATE FAILURE 블루 스크린 뜨는 현상
1283정성태5/12/201233307.NET Framework: 317. C# 관점에서의 Observer 패턴 구현 [1]파일 다운로드1
1282정성태5/12/201226105Phone: 6. Windows Phone 7 Silverlight에서 Google Map 사용하는 방법 [3]파일 다운로드1
1281정성태5/9/201233187.NET Framework: 316. WPF/Silverlight의 그래픽 단위와 Anti-aliasing 처리를 이해하자 [1]파일 다운로드1
1280정성태5/9/201226154오류 유형: 154. Could not load type 'System.ServiceModel.Activation.HttpModule' from assembly 'System.ServiceModel, ...'.
1279정성태5/9/201224917.NET Framework: 315. 해당 DLL이 Managed인지 / Unmanaged인지 확인하는 방법 [1]파일 다운로드1
1278정성태5/8/201226145오류 유형: 153. Visual Studio 디버깅 - Unable to break execution. This process is not currently executing the type of code that you selected to debug.
... 136  137  138  139  140  141  142  143  144  145  146  147  148  149  [150]  ...