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

(시리즈 글이 6개 있습니다.)
개발 환경 구성: 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




.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가 맞습니다. 저도 이때 이후로 제대로 테스트를 못했는데... 진우 님께서 한번 해보시고 결과 좀 공유해 주세요. ^^
정성태

1  2  3  4  5  6  7  8  9  10  11  12  13  14  [15]  ...
NoWriterDateCnt.TitleFile(s)
13248정성태2/7/20234052오류 유형: 841. 리눅스 - [사용자 계정] is not in the sudoers file. This incident will be reported.
13247정성태2/7/20234963VS.NET IDE: 180. Visual Studio - 닷넷 소스 코드 디버깅 중 "Decompile source code"가 동작하는 않는 문제
13246정성태2/6/20234089개발 환경 구성: 664. Hyper-V에 설치한 리눅스 VM의 VHD 크기 늘리는 방법 - 두 번째 이야기
13245정성태2/6/20234637.NET Framework: 2093. C# - PEM 파일을 이용한 RSA 개인키/공개키 설정 방법파일 다운로드1
13244정성태2/5/20233989VS.NET IDE: 179. Visual Studio - External Tools에 Shell 내장 명령어 등록
13243정성태2/5/20234856디버깅 기술: 190. windbg - Win32 API 호출 시점에 BP 거는 방법 [1]
13242정성태2/4/20234304디버깅 기술: 189. ASP.NET Web Application (.NET Framework) 프로젝트의 숨겨진 예외 - System.UnauthorizedAccessException
13241정성태2/3/20233827디버깅 기술: 188. ASP.NET Web Application (.NET Framework) 프로젝트의 숨겨진 예외 - System.IO.FileNotFoundException
13240정성태2/1/20233987디버깅 기술: 187. ASP.NET Web Application (.NET Framework) 프로젝트의 숨겨진 예외 - System.Web.HttpException
13239정성태2/1/20233629디버깅 기술: 186. C# - CacheDependency의 숨겨진 예외 - System.Web.HttpException
13238정성태1/31/20235640.NET Framework: 2092. IIS 웹 사이트를 TLS 1.2 또는 TLS 1.3 프로토콜로만 운영하는 방법
13237정성태1/30/20235328.NET Framework: 2091. C# - 웹 사이트가 어떤 버전의 TLS/SSL을 지원하는지 확인하는 방법
13236정성태1/29/20234972개발 환경 구성: 663. openssl을 이용해 인트라넷 IIS 사이트의 SSL 인증서 생성
13235정성태1/29/20234515개발 환경 구성: 662. openssl - 윈도우 환경의 명령행에서 SAN 적용하는 방법
13234정성태1/28/20235561개발 환경 구성: 661. dnSpy를 이용해 소스 코드가 없는 .NET 어셈블리의 코드를 변경하는 방법 [1]
13233정성태1/28/20236903오류 유형: 840. C# - WebClient로 https 호출 시 "The request was aborted: Could not create SSL/TLS secure channel" 예외 발생
13232정성태1/27/20234703스크립트: 43. uwsgi의 --processes와 --threads 옵션
13231정성태1/27/20233618오류 유형: 839. python - TypeError: '...' object is not callable
13230정성태1/26/20234031개발 환경 구성: 660. WSL 2 내부로부터 호스트 측의 네트워크로 UDP 데이터가 1개의 패킷으로만 제한되는 문제
13229정성태1/25/20234972.NET Framework: 2090. C# - UDP Datagram의 최대 크기
13228정성태1/24/20235119.NET Framework: 2089. C# - WMI 논리 디스크가 속한 물리 디스크의 정보를 얻는 방법 [2]파일 다운로드1
13227정성태1/23/20234823개발 환경 구성: 659. Windows - IP MTU 값을 바꿀 수 있을까요? [1]
13226정성태1/23/20234497.NET Framework: 2088. .NET 5부터 지원하는 GetRawSocketOption 사용 시 주의할 점
13225정성태1/21/20233749개발 환경 구성: 658. Windows에서 실행 중인 소켓 서버를 다른 PC 또는 WSL에서 접속할 수 없는 경우
13224정성태1/21/20234095Windows: 221. Windows - Private/Public/Domain이 아닌 네트워크 어댑터 단위로 방화벽을 on/off하는 방법
13223정성태1/20/20234288오류 유형: 838. RDP 연결 오류 - The two computers couldn't connect in the amount of time allotted
1  2  3  4  5  6  7  8  9  10  11  12  13  14  [15]  ...