Microsoft MVP성태의 닷넷 이야기
.NET Framework: 2069. .NET 7 - AOT(ahead-of-time) 컴파일 [링크 복사], [링크+제목 복사],
조회: 15829
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 4개 있습니다.)
(시리즈 글이 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 7 - AOT(ahead-of-time) 컴파일

여기서 잠깐, 닷넷 AOT 역사에 대해 잠깐 훑어볼까요? ^^

Conversation about crossgen2
; https://devblogs.microsoft.com/dotnet/conversation-about-crossgen2/

가장 처음에, ".NET Framework"의 NGen이 있었습니다.

Ngen.exe (Native Image Generator)
; https://learn.microsoft.com/en-us/dotnet/framework/tools/ngen-exe-native-image-generator

NGen은 윈도우 플랫폼 종속적이었고, 비록 native 코드를 생성하긴 했지만 그래도 managed 모듈을 필요로 했습니다.

이후 "Windows Phone"이 나와 크로스-플랫폼을 지원해야 하면서 별도로 AOT 코드 생성기가 개발됩니다. 얼마 안 돼 윈도우 폰은 사라졌지만, 다행히 그 역사를 .NET Core의 CrossGen으로 세대를 이어갑니다. 이후 닷넷 6에 포함된 CrossGen2는 이전 CrossGen을 (몇몇 기능 향상과 함께) C# 언어로 포팅한 것에 해당합니다.

하지만, NGen부터 CrossGen2까지의 공통점이 있다면, 바로 native + managed 모듈이 함께 필요하다는 점입니다.




자, 그리고 이제 .NET 7에서 진정한 네이티브 코드 생성기인 AOT 컴파일 기능이 공식적으로 포함됐습니다.

Native AOT Deployment
; https://learn.microsoft.com/en-us/dotnet/core/deploying/native-aot/

실제로 테스트를 해볼까요? ^^

간단한 닷넷 7 콘솔 애플리케이션을 만들고, csproj에 PublishAot 옵션을 추가합니다.

<Project Sdk="Microsoft.NET.Sdk">

    <PropertyGroup>
        <OutputType>Exe</OutputType>
        <TargetFramework>net7.0</TargetFramework>
        <ImplicitUsings>enable</ImplicitUsings>
        <Nullable>enable</Nullable>

        <PublishAot>true</PublishAot>

        <IsAotCompatible>true</IsAotCompatible> 
        <!-- IsTrimmable, EnableTrimAnalyzer, EnableSingleFileAnalyzer, EnableAotAnalyzer 4개의 값을 true로 설정한 효과 (.NET 8부터 지원) ---->
        
    </PropertyGroup>

</Project>

이후 "dotnet publish" 해주면 ".\bin\Debug\net7.0\win-x64\publish" 디렉터리에 실행 파일과 PDB 파일이 생성됩니다.

// 디버그 빌드 기준
net7_pubone_sample.exe, 약 4.69MB
net7_pubone_sample.pdb, 약 15.9MB

// 릴리스 빌드 기준
net7_pubone_sample.exe, 약 2.85MB
net7_pubone_sample.pdb, 약 12.1MB

당연히, net7_pubone_sample.exe만 (닷넷 런타임이 설치되지 않은) 다른 PC에 복사해 실행할 수 있습니다.




이렇게 완전히 AOT로 빌드된 모듈은 역어셈블이 가능할까요?

C# - PublishSingleFile로 배포한 이미지의 역어셈블 가능 여부 (난독화 필요성)
; https://www.sysnet.pe.kr/2/0/13161

위의 글에서 살펴본 .NET 6와는 달리, 이제 Windbg는 mscordaccore.dll을 로드하려고 해도,

// c:\temp\mscordaccore.dll이 있는 경우,

0:004> .cordll -lp c:\temp
CLR DLL status: No load attempts

실패합니다. 즉, 이제는 대상 프로세스를 .NET 프로세스로 볼 수 없는 것입니다. 비록 관리 Heap을 사용해 GC는 발생하겠지만 그 외의 모든 면에서 일반 native 프로세스와 동일한데요, 달리 말하면, 그냥 여러분들이 C/C++로 GC 기능을 구현한 다음 그것을 활용해 코딩한 것과 같은 것입니다.

따라서, AOT로 빌드한 경우라면 이제 native 모드로 디버깅을 해야 하는데,

0:005> ~
   0  Id: 2df8.1424 Suspend: 1 Teb: 00000036`a2077000 Unfrozen
   1  Id: 2df8.2144 Suspend: 1 Teb: 00000036`a2079000 Unfrozen
   2  Id: 2df8.21c8 Suspend: 1 Teb: 00000036`a207b000 Unfrozen
   3  Id: 2df8.2210 Suspend: 1 Teb: 00000036`a207d000 Unfrozen
   4  Id: 2df8.2e44 Suspend: 1 Teb: 00000036`a207f000 Unfrozen
.  5  Id: 2df8.1ddc Suspend: 1 Teb: 00000036`a2081000 Unfrozen

0:005> ~0s
rax=0000000000000006 rbx=00000036a22ff748 rcx=0000000000000050
rdx=0000000000000000 rsi=0000000000000050 rdi=0000000000000000
rip=00007ffabe40d184 rsp=00000036a22ff478 rbp=00000036a22ff5c0
 r8=0000000000000020  r9=0000000000000001 r10=0000000000000000
r11=00000188b1aeffe0 r12=0000000000000000 r13=0000000000000000
r14=0000000000000000 r15=0000000000000000
iopl=0         nv up ei pl zr na po nc
cs=0033  ss=002b  ds=002b  es=002b  fs=0053  gs=002b             efl=00000244
ntdll!NtReadFile+0x14:
00007ffa`be40d184 c3              ret

0:000> k
 # Child-SP          RetAddr               Call Site
00 00000036`a22ff478 00007ffa`bbaf5593     ntdll!NtReadFile+0x14
01 00000036`a22ff480 00007ff6`5f259a0d     KERNELBASE!ReadFile+0x73
02 00000036`a22ff500 00007ff6`5f25953f     net7_pubone_sample!System_Console_Interop_Kernel32___ReadFile_g____PInvoke_43_0+0x7d
03 00000036`a22ff5d0 00007ff6`5f25ad36     net7_pubone_sample!System_Console_Interop_Kernel32__ReadFile+0x5f [/_/src/libraries/System.Console/src/Microsoft.Interop.LibraryImportGenerator/Microsoft.Interop.LibraryImportGenerator/LibraryImports.g.cs @ 412] 
04 00000036`a22ff640 00007ff6`5f25ab6b     net7_pubone_sample!System_Console_System_ConsolePal_WindowsConsoleStream__ReadFileNative+0xc6 [/_/src/libraries/System.Console/src/System/ConsolePal.Windows.cs @ 1156] 
05 00000036`a22ff6f0 00007ff6`5f25e86c     net7_pubone_sample!System_Console_System_ConsolePal_WindowsConsoleStream__Read+0x6b [/_/src/libraries/System.Console/src/System/ConsolePal.Windows.cs @ 1114] 
06 00000036`a22ff770 00007ff6`5f1dcbef     net7_pubone_sample!System_Console_System_IO_ConsoleStream__Read+0x8c [/_/src/libraries/System.Console/src/System/IO/ConsoleStream.cs @ 76] 
07 00000036`a22ff7e0 00007ff6`5f1dcd6d     net7_pubone_sample!S_P_CoreLib_System_IO_StreamReader__ReadBuffer+0x1af [/_/src/libraries/System.Private.CoreLib/src/System/IO/StreamReader.cs @ 616] 
08 00000036`a22ff8a0 00007ff6`5f25ee63     net7_pubone_sample!S_P_CoreLib_System_IO_StreamReader__ReadLine+0x7d [/_/src/libraries/System.Private.CoreLib/src/System/IO/StreamReader.cs @ 785] 
09 00000036`a22ff960 00007ff6`5f25a1f5     net7_pubone_sample!System_Console_System_IO_SyncTextReader__ReadLine+0x63 [/_/src/libraries/System.Console/src/System/IO/SyncTextReader.cs @ 77] 
0a 00000036`a22ff9c0 00007ff6`5f25f688     net7_pubone_sample!System_Console_System_Console__ReadLine+0x35 [/_/src/libraries/System.Console/src/System/Console.cs @ 722] 
0b 00000036`a22ffa10 00007ff6`5f2ceaa7     net7_pubone_sample!net7_pubone_sample_net7_pubone_sample_Program__Main+0x28 [E:\net7_pubone_sample\net7_pubone_sample\Program.cs @ 8] 
0c 00000036`a22ffa50 00007ff6`5f2ceb3a     net7_pubone_sample!net7_pubone_sample__Module___MainMethodWrapper+0x17
0d 00000036`a22ffa80 00007ff6`5f0ceb6e     net7_pubone_sample!net7_pubone_sample__Module___StartupCodeMain+0x8a
0e 00000036`a22ffae0 00007ff6`5f0bdf54     net7_pubone_sample!wmain+0xae [D:\a\_work\1\s\src\coreclr\nativeaot\Bootstrap\main.cpp @ 205] 
0f (Inline Function) --------`--------     net7_pubone_sample!invoke_main+0x22 [D:\a\_work\1\s\src\vctools\crt\vcstartup\src\startup\exe_common.inl @ 90] 
10 00000036`a22ffb30 00007ffa`bcba74b4     net7_pubone_sample!__scrt_common_main_seh+0x10c [D:\a\_work\1\s\src\vctools\crt\vcstartup\src\startup\exe_common.inl @ 288] 
11 00000036`a22ffb70 00007ffa`be3c26a1     KERNEL32!BaseThreadInitThunk+0x14
12 00000036`a22ffba0 00000000`00000000     ntdll!RtlUserThreadStart+0x21

sos 확장의 도움 없이 닷넷 프로그램을 문제 분석하는 경험은... 그다지 유쾌하진 않을 것입니다. ^^ 참고로, dotnet-dump 역시 Native AOT로 빌드된 프로세스는 인식하지 못합니다.

c:\temp> dotnet-dump ps
No supported .NET processes were found

(그래도, GC Heap 관련 정보는 디버깅할 수 있는 방법을 제공해줘야 하지 않을까요? ^^;)




문서에 나오지만,

Limitations of Native AOT deployment

  • No dynamic loading (for example, Assembly.LoadFile)
  • No runtime code generation (for example, System.Reflection.Emit)
  • No C++/CLI
  • No built-in COM (only applies to Windows)
  • Requires trimming, which has limitations
  • Implies compilation into a single file, which has known incompatibilities
  • Apps include required runtime libraries (just like self-contained apps, increasing their size, as compared to framework-dependent apps)

당연히 .NET Reflection과 같은 기능들은 AOT 지원이 안 됩니다. 그리고 .NET 7의 BCL 중에서도 위와 같은 조건에 걸리는 타입들이 많기 때문에 아직은 AOT가 WPF/WinForm 등의 응용 프로그램은 지원하지 못합니다. 사실 저런 조건들이 AOT 입장에서 제약이지, WPF/WinForm 등의 개발 프레임워크를 만들 때는 유용하게 쓸 수 있는 기술들이어서 제법 광범위하게 사용했던 것입니다. 그런 이유에서, Native AOT가 과연 "아직" WPF/WinForm을 지원하지 못한 것인지, "영원히" 지원하지 못할 것인지는 두고 봐야 알 수 있을 것입니다. ^^

결국 현재의 닷넷 7 AOT는 "Console Application"을 대상으로 한 프로젝트만 지원하고 WPF/WinForm 등의 프로젝트에서 PublishAot 옵션을 주면 "error NETSDK1168: WPF is not supported or recommended with trimming enabled. Please go to https://aka.ms/dotnet-illink/wpf for more details." 빌드 오류가 발생합니다.




참고로, AOT 빌드는 UnmanagedCallersOnlyAttribute 특성과 어우러져 Native API(윈도우의 경우 Win32 DLL)로 동작하는 라이브러리를 만들기에는 아주 이상적인 환경을 제공합니다.

Build native libraries
; https://learn.microsoft.com/en-us/dotnet/core/deploying/native-aot/#build-native-libraries

관련 예제 코드
; https://github.com/dotnet/samples/tree/main/core/nativeaot/NativeLibrary




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 12/22/2023]

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

비밀번호

댓글 작성자
 



2024-12-06 09시50분
How Much Memory Do You Need in 2024 to Run 1 Million Concurrent Tasks?
; https://hez2010.github.io/async-runtimes-benchmarks-2024/

100만 개 테스트에서 오히려 rust보다 (AOT 무관하게) C#의 메모리 사용이 더 적었다고. ^^
정성태

... 151  152  153  154  155  156  [157]  158  159  160  161  162  163  164  165  ...
NoWriterDateCnt.TitleFile(s)
1124정성태9/17/201126409.NET Framework: 240. System.Collections.ArrayList가 .NET 4.5에서 지원이 안된다??? [2]
1123정성태9/17/201165219Windows: 53. 2가지 모드의 Internet Explorer 10과 ActiveX [6]
1122정성태9/16/201132916Windows: 52. 새롭게 지원되는 WinRT 응용 프로그램 [7]
1121정성태9/12/201127665Java: 5. WTP 내에서 서블릿을 실행하는 환경
1120정성태9/11/201127593.NET Framework: 239. IHttpHandler.IsReusable 속성 이야기파일 다운로드1
1119정성태9/11/201126688Java: 4. 이클립스에 WTP SDK가 설치되지 않는다면? [2]
1118정성태9/11/201138331Java: 3. 이클립스에서 서블릿 디버깅하는 방법 [4]
1117정성태9/9/201125618제니퍼 .NET: 17. 제니퍼 닷넷 적용 사례 (2) - 웹 애플리케이션 hang의 원인을 알려주다.
1116정성태9/8/201156673Java: 2. 자바에서 "Microsoft SQL Server JDBC Driver" 사용하는 방법
1115정성태9/4/201130194Java: 1. 닷넷 개발자가 처음 실습해 본 서블릿
1114정성태9/4/201134723Math: 2. "Zhang Suen 알고리즘(세선화, Thinning/Skeletonization)"의 C# 버전 [4]파일 다운로드1
1113정성태9/2/201134283개발 환경 구성: 129. Hyper-V에 CentOS 설치하기
1112정성태9/2/201151049Linux: 1. 리눅스 <-> 윈도우 원격 접속 프로그램 사용 [3]
1111정성태8/29/201125438제니퍼 .NET: 16. 적용 사례 (1) - DB Connection Pooling을 사용하지 않았을 때의 성능 저하를 알려주다. [1]
1110정성태8/26/201126785오류 유형: 136. RDP 접속이 불연속적으로 끊기는 문제
1109정성태8/26/201129685오류 유형: 135. 어느 순간 Active Directory 접속이 안되는 문제
1108정성태8/22/201131173오류 유형: 134. OLE/COM Object Viewer - DllRegisterServer in IVIEWERS.DLL failed. [1]
1107정성태8/21/201128987디버깅 기술: 43. Windows Form의 Load 이벤트에서 발생하는 예외가 Visual Studio에서 잡히지 않는 문제
1106정성태8/20/201127279웹: 26. FailedRequestTracing 설정으로 인한 iisexpress.exe 비정상 종료 문제
1105정성태8/19/201127206.NET Framework: 238. Web Site Model 프로젝트에서 Trace.WriteLine 출력이 dbgview.exe에서 확인이 안 되는 문제파일 다운로드1
1104정성태8/19/201127410웹: 25. WebDev보다 IIS Express가 더 나은 점 - 다중 가상 디렉터리 매핑 [1]
1103정성태8/19/201133309오류 유형: 133. WCF 포트 바인딩 실패 오류 - TCP error(10013) [1]
1102정성태8/19/201131031Math: 1. 방탈출3 - Room 10의 '중복가능한 조합' 문제를 위한 C# 프로그래밍 [2]파일 다운로드1
1101정성태8/19/201129685.NET Framework: 237. WCF AJAX 서비스와 JavaScript 간의 DateTime 연동 [1]파일 다운로드1
1100정성태8/17/201128793.NET Framework: 236. SqlDbType - DateTime, DateTime2, DateTimeOffset의 차이점파일 다운로드1
1099정성태8/15/201128235오류 유형: 132. 어느 순간 갑자기 접속이 안 되는 TFS 서버
... 151  152  153  154  155  156  [157]  158  159  160  161  162  163  164  165  ...