Microsoft MVP성태의 닷넷 이야기
개발 환경 구성: 386. .NET Framework Native compiler 프리뷰 버전 사용법 [링크 복사], [링크+제목 복사],
조회: 19400
글쓴 사람
정성태 (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가 맞습니다. 저도 이때 이후로 제대로 테스트를 못했는데... 진우 님께서 한번 해보시고 결과 좀 공유해 주세요. ^^
정성태

... 121  122  123  124  125  126  127  128  129  130  [131]  132  133  134  135  ...
NoWriterDateCnt.TitleFile(s)
1780정성태10/15/201424243오류 유형: 249. The application-specific permission settings do not grant Local Activation permission for the COM Server application with CLSID
1779정성태10/15/201419761오류 유형: 248. Active Directory에서 OU가 지워지지 않는 경우
1778정성태10/10/201418228오류 유형: 247. The Netlogon service could not create server share C:\Windows\SYSVOL\sysvol\[도메인명]\SCRIPTS.
1777정성태10/10/201421328오류 유형: 246. The processing of Group Policy failed. Windows attempted to read the file \\[도메인]\sysvol\[도메인]\Policies\{...GUID...}\gpt.ini
1776정성태10/10/201418341오류 유형: 245. 이벤트 로그 - Name resolution for the name _ldap._tcp.dc._msdcs.[도메인명]. timed out after none of the configured DNS servers responded.
1775정성태10/9/201419461오류 유형: 244. Visual Studio 디버깅 (2) - Unable to break execution. This process is not currently executing the type of code that you selected to debug.
1774정성태10/9/201426656개발 환경 구성: 246. IIS 작업자 프로세스의 20분 자동 재생(Recycle)을 끄는 방법
1773정성태10/8/201429817.NET Framework: 471. 웹 브라우저로 다운로드가 되는 파일을 왜 C# 코드로 하면 안되는 걸까요? [1]
1772정성태10/3/201418616.NET Framework: 470. C# 3.0의 기본 인자(default parameter)가 .NET 1.1/2.0에서도 실행될까? [3]
1771정성태10/2/201428119개발 환경 구성: 245. 실행된 프로세스(EXE)의 명령행 인자를 확인하고 싶다면 - Sysmon [4]
1770정성태10/2/201421718개발 환경 구성: 244. 매크로 정의를 이용해 파일 하나로 C++과 C#에서 공유하는 방법 [1]파일 다운로드1
1769정성태10/1/201424140개발 환경 구성: 243. Scala 개발 환경 구성(JVM, 닷넷) [1]
1768정성태10/1/201419560개발 환경 구성: 242. 배치 파일에서 Thread.Sleep 효과를 주는 방법 [5]
1767정성태10/1/201424669VS.NET IDE: 94. Visual Studio 2012/2013에서의 매크로 구현 - Visual Commander [2]
1766정성태10/1/201422522개발 환경 구성: 241. 책 "프로그래밍 클로저: Lisp"을 읽고 나서. [1]
1765정성태9/30/201426066.NET Framework: 469. Unity3d에서 transform을 변수에 할당해 사용하는 특별한 이유가 있을까요?
1764정성태9/30/201422312오류 유형: 243. 파일 삭제가 안 되는 경우 - The action can't be comleted because the file is open in System
1763정성태9/30/201423880.NET Framework: 468. PDB 파일을 연동해 소스 코드 라인 정보를 알아내는 방법파일 다운로드1
1762정성태9/30/201424561.NET Framework: 467. 닷넷에서 EIP/RIP 레지스터 값을 구하는 방법 [1]파일 다운로드1
1761정성태9/29/201421625.NET Framework: 466. 윈도우 운영체제의 보안 그룹 이름 및 설명 문자열을 바꾸는 방법파일 다운로드1
1760정성태9/28/201419904.NET Framework: 465. ICorProfilerInfo::GetILToNativeMapping 메서드가 0x80131358을 반환하는 경우
1759정성태9/27/201431006개발 환경 구성: 240. Visual C++ / x64 환경에서 inline-assembly를 매크로 어셈블리로 대체하는 방법파일 다운로드1
1758정성태9/23/201437904개발 환경 구성: 239. 원격 데스크톱 접속(RDP)을 기존의 콘솔 모드처럼 사용하는 방법 [1]
1757정성태9/23/201418456오류 유형: 242. Lync로 모임 참여 시 소리만 들리지 않는 경우 - 두 번째 이야기
1756정성태9/23/201427483기타: 48. NVidia 제품의 과다한 디스크 사용 [2]
1755정성태9/22/201434273오류 유형: 241. Unity Web Player를 설치해도 여전히 설치하라는 화면이 나오는 경우 [4]
... 121  122  123  124  125  126  127  128  129  130  [131]  132  133  134  135  ...