dotnet-native-aot
Full Native AOT compilation pipeline for .NET 8+ applications: PublishAot configuration, ILLink descriptor XML for type preservation, reflection-free coding patterns, P/Invoke considerations, binary size optimization, self-contained deployment with runtime-deps base images, and diagnostic analyzers (EnableAotAnalyzer /EnableTrimAnalyzer ).
Version assumptions: .NET 8.0+ baseline. Native AOT for ASP.NET Core Minimal APIs and console apps shipped in .NET 8. .NET 9 improved trimming warnings and library compat. .NET 10 enhanced request delegate generator and expanded Minimal API AOT support.
Scope
-
PublishAot MSBuild configuration (apps vs libraries)
-
Diagnostic analyzers (EnableAotAnalyzer, EnableTrimAnalyzer)
-
ILLink descriptor XML for type preservation
-
Reflection-free coding patterns
-
P/Invoke with LibraryImport source generation
-
Binary size optimization and self-contained deployment
-
ASP.NET Core Native AOT (Minimal APIs, CreateSlimBuilder)
Out of scope
-
MAUI iOS/Mac Catalyst AOT pipeline -- see [skill:dotnet-maui-aot]
-
AOT-first design patterns (source gen, DI, serialization) -- see [skill:dotnet-aot-architecture]
-
Trim-safe library authoring -- see [skill:dotnet-trimming]
-
WASM AOT for Blazor/Uno -- see [skill:dotnet-aot-wasm]
-
Source generator authoring (Roslyn API) -- see [skill:dotnet-csharp-source-generators]
-
DI container patterns -- see [skill:dotnet-csharp-dependency-injection]
-
Serialization depth -- see [skill:dotnet-serialization]
-
Container deployment orchestration -- see [skill:dotnet-containers]
Cross-references: [skill:dotnet-aot-architecture] for AOT-first design patterns, [skill:dotnet-trimming] for trim-safe library authoring, [skill:dotnet-aot-wasm] for WebAssembly AOT, [skill:dotnet-maui-aot] for MAUI-specific AOT, [skill:dotnet-containers] for runtime-deps base images, [skill:dotnet-serialization] for AOT-safe serialization, [skill:dotnet-csharp-source-generators] for source gen as AOT enabler, [skill:dotnet-csharp-dependency-injection] for AOT-safe DI, [skill:dotnet-native-interop] for general P/Invoke patterns and cross-platform library resolution.
PublishAot Configuration
Enabling Native AOT
<!-- App .csproj --> <PropertyGroup> <PublishAot>true</PublishAot> </PropertyGroup>
Publish as Native AOT
dotnet publish -c Release -r linux-x64
Publish for specific targets
dotnet publish -c Release -r win-x64 dotnet publish -c Release -r osx-arm64
MSBuild Properties: Apps vs Libraries
Apps and libraries use different MSBuild properties. Do not mix them.
For applications (console apps, ASP.NET Core Minimal APIs):
<PropertyGroup> <!-- Enable Native AOT compilation on publish --> <PublishAot>true</PublishAot>
<!-- Enable analyzers during development (not just publish) --> <EnableAotAnalyzer>true</EnableAotAnalyzer> <EnableTrimAnalyzer>true</EnableTrimAnalyzer> </PropertyGroup>
For libraries (NuGet packages, shared class libraries):
<PropertyGroup> <!-- Declare the library is AOT-compatible (auto-enables analyzers) --> <IsAotCompatible>true</IsAotCompatible> <!-- Declare the library is trim-safe (auto-enables trim analyzer) --> <IsTrimmable>true</IsTrimmable> </PropertyGroup>
IsAotCompatible and IsTrimmable automatically enable the AOT and trim analyzers respectively. Do not also set PublishAot in library projects -- libraries are not published as standalone executables.
Diagnostic Analyzers
Enable AOT and trim analyzers during development to catch issues before publishing:
<PropertyGroup> <EnableAotAnalyzer>true</EnableAotAnalyzer> <EnableTrimAnalyzer>true</EnableTrimAnalyzer> </PropertyGroup>
Analysis Without Publishing
Run analysis during dotnet build without a full publish:
Analyze AOT compatibility without publishing
dotnet build /p:EnableAotAnalyzer=true /p:EnableTrimAnalyzer=true
See per-occurrence warnings (not grouped by assembly)
dotnet build /p:EnableAotAnalyzer=true /p:EnableTrimAnalyzer=true /p:TrimmerSingleWarn=false
This reports IL2xxx (trim) and IL3xxx (AOT) warnings without producing a native binary, enabling fast feedback during development.
Common Diagnostic Codes
Code Category Meaning
IL2026 Trim Member has [RequiresUnreferencedCode] -- may break after trimming
IL2046 Trim Trim attribute mismatch between base/derived types
IL2057-IL2072 Trim Various reflection usage that the trimmer cannot analyze
IL3050 AOT Member has [RequiresDynamicCode] -- generates code at runtime
IL3051 AOT [RequiresDynamicCode] attribute mismatch
ILLink Descriptors for Type Preservation
When code uses reflection that the trimmer cannot statically analyze, use ILLink descriptor XML to preserve types. Do not use legacy RD.xml -- it is a .NET Native/UWP format that is silently ignored by modern .NET AOT.
ILLink Descriptor XML
<!-- ILLink.Descriptors.xml --> <linker> <!-- Preserve all public members of a type --> <assembly fullname="MyApp"> <type fullname="MyApp.Models.LegacyConfig" preserve="all" /> <type fullname="MyApp.Services.PluginLoader"> <method name="LoadPlugin" /> </type> </assembly>
<!-- Preserve an entire external assembly --> <assembly fullname="IncompatibleLibrary" preserve="all" /> </linker>
<!-- Register in .csproj --> <ItemGroup> <TrimmerRootDescriptor Include="ILLink.Descriptors.xml" /> </ItemGroup>
[DynamicDependency] Attribute
For targeted preservation in code (preferred over ILLink XML for small, localized cases):
using System.Diagnostics.CodeAnalysis;
// Preserve a specific method [DynamicDependency(nameof(LegacyConfig.Initialize), typeof(LegacyConfig))] public void ConfigureApp() { /* ... */ }
// Preserve all public members [DynamicDependency(DynamicallyAccessedMemberTypes.All, typeof(PluginBase))] public void LoadPlugins() { /* ... */ }
When to Use Which
Scenario Approach
One or two methods/types [DynamicDependency] attribute
Entire assembly or many types ILLink descriptor XML
Third-party library not AOT-safe ILLink descriptor XML or <TrimmerRootAssembly>
Your own code with analyzed reflection Refactor to source generators (best long-term)
Reflection-Free Patterns
Native AOT works best with code that avoids runtime reflection entirely. Replace reflection patterns with compile-time alternatives.
Reflection Pattern AOT-Safe Replacement
Activator.CreateInstance<T>()
Factory method or explicit new T()
Type.GetProperties() for mapping Mapperly source generator or manual mapping
Assembly.GetTypes() for DI scanning Explicit services.AddScoped<T>()
JsonSerializer.Deserialize<T>(json)
JsonSerializer.Deserialize(json, Context.Default.T)
MethodInfo.Invoke() for dispatch switch on type or interface dispatch
See [skill:dotnet-aot-architecture] for comprehensive AOT-first design patterns.
P/Invoke Considerations
P/Invoke (platform invoke) calls to native libraries generally work with Native AOT, but require attention:
Direct P/Invoke (Preferred)
// Direct P/Invoke -- AOT-compatible, no runtime marshalling overhead [LibraryImport("libsqlite3", EntryPoint = "sqlite3_open")] internal static partial int Sqlite3Open( [MarshalAs(UnmanagedType.LPStr)] string filename, out nint db);
Use [LibraryImport] (.NET 7+) instead of [DllImport] -- it generates marshalling code at compile time via source generators, making it fully AOT-compatible.
DllImport vs LibraryImport
Attribute AOT Compatibility Marshalling
[DllImport]
Partial -- some marshalling requires runtime codegen Runtime marshalling
[LibraryImport]
Full -- compile-time source gen Compile-time marshalling
// Migrate from DllImport to LibraryImport // Before: [DllImport("kernel32.dll", SetLastError = true)] static extern bool CloseHandle(IntPtr hObject);
// After: [LibraryImport("kernel32.dll", SetLastError = true)] [return: MarshalAs(UnmanagedType.Bool)] internal static partial bool CloseHandle(IntPtr hObject);
Native Library Deployment
When publishing as Native AOT, native libraries (.so , .dylib , .dll ) must be alongside the binary:
<ItemGroup> <!-- Include native library in publish output --> <NativeLibrary Include="libs/libcustom.so" /> </ItemGroup>
Size Optimization
Binary Size Reduction Options
<PropertyGroup> <PublishAot>true</PublishAot>
<!-- Strip debug symbols (significant size reduction) --> <StripSymbols>true</StripSymbols>
<!-- Optimize for size over speed --> <OptimizationPreference>Size</OptimizationPreference>
<!-- Enable invariant globalization (removes ICU data) --> <InvariantGlobalization>true</InvariantGlobalization>
<!-- Remove stack trace strings (reduces size, harder debugging) --> <StackTraceSupport>false</StackTraceSupport>
<!-- Remove EventSource/EventPipe (if not using diagnostics) --> <EventSourceSupport>false</EventSourceSupport> </PropertyGroup>
Typical Binary Sizes
Configuration Console App ASP.NET Minimal API
Default AOT ~10-15 MB ~15-25 MB
-
StripSymbols ~8-12 MB ~12-20 MB
-
Size optimization ~6-10 MB ~10-18 MB
-
InvariantGlobalization ~4-8 MB ~8-15 MB
Size Analysis
Analyze what contributes to binary size
dotnet publish -c Release -r linux-x64 /p:PublishAot=true
Use sizoscope (community tool) for detailed size analysis
https://github.com/AdrianEddy/sizoscope
Self-Contained Deployment with runtime-deps
Native AOT produces self-contained binaries that include the .NET runtime. Use the runtime-deps base image for minimal container size since the runtime is already embedded in the binary.
Build stage
FROM mcr.microsoft.com/dotnet/sdk:10.0 AS build WORKDIR /src COPY . . RUN dotnet publish -c Release -r linux-x64 -o /app/publish
Runtime stage -- use runtime-deps, not aspnet or runtime
FROM mcr.microsoft.com/dotnet/runtime-deps:10.0-noble-chiseled AS runtime WORKDIR /app COPY --from=build /app/publish . ENTRYPOINT ["./MyApp"]
The runtime-deps image contains only OS-level dependencies (libc, OpenSSL, etc.) -- no .NET runtime. This is the smallest possible base image for AOT-published apps (~30 MB). See [skill:dotnet-containers] for full container patterns.
ASP.NET Core Native AOT
Minimal API Support (.NET 8+)
ASP.NET Core Minimal APIs support Native AOT. MVC controllers are not AOT-compatible (they rely on reflection for model binding, filters, and routing).
var builder = WebApplication.CreateSlimBuilder(args);
// Use source-generated JSON context builder.Services.ConfigureHttpJsonOptions(options => { options.SerializerOptions.TypeInfoResolverChain.Insert(0, AppJsonContext.Default); });
var app = builder.Build();
app.MapGet("/api/products/{id}", (int id) => Results.Ok(new Product(id, "Widget")));
app.Run();
[JsonSerializable(typeof(Product))] internal partial class AppJsonContext : JsonSerializerContext { }
record Product(int Id, string Name);
CreateSlimBuilder vs CreateBuilder
Method AOT Support Includes
WebApplication.CreateSlimBuilder()
Full Minimal services, no MVC, no Razor
WebApplication.CreateBuilder()
Partial Full feature set, some features need reflection
Use CreateSlimBuilder for Native AOT applications. It excludes features that require runtime code generation.
.NET 10 ASP.NET Core AOT Improvements
.NET 10 brings improvements across the ASP.NET Core and runtime Native AOT stack. Target net10.0 to benefit automatically.
<PropertyGroup> <TargetFramework>net10.0</TargetFramework> <PublishAot>true</PublishAot> </PropertyGroup>
Request Delegate Generator improvements: The source generator that creates request delegates for Minimal API endpoints handles more parameter binding scenarios in .NET 10, including additional TypedResults return types and complex binding patterns. This reduces the need for manual workarounds that were required in .NET 8/9 when the generator could not produce AOT-safe code for certain endpoint signatures.
Reduced linker warning surface: Many ASP.NET Core framework APIs that previously emitted trim/AOT warnings (IL2xxx/IL3xxx) have been annotated or refactored for AOT compatibility. Projects upgrading from .NET 9 to .NET 10 will see fewer false-positive linker warnings when publishing with PublishAot .
OpenAPI in the webapiaot template: The webapiaot project template now includes OpenAPI document generation via Microsoft.AspNetCore.OpenApi by default, so AOT-published APIs get auto-generated API documentation without additional setup.
Runtime NativeAOT code generation: The .NET 10 runtime improves AOT code generation for struct arguments, enhances loop inversion optimizations, and improves method devirtualization -- resulting in better throughput for AOT-published applications without code changes.
Blazor Server and SignalR: Blazor Server and SignalR remain not supported with Native AOT in .NET 10. Blazor WebAssembly AOT (client-side compilation) is a separate concern covered by [skill:dotnet-aot-wasm]. For Blazor Server apps, continue using JIT deployment.
Compatibility snapshot (.NET 10):
Feature AOT Support
gRPC Fully supported
Minimal APIs Partially supported (most scenarios work)
MVC Not supported
Blazor Server Not supported
SignalR Not supported
JWT Authentication Fully supported
CORS, HealthChecks, OutputCaching Fully supported
WebSockets, StaticFiles Fully supported
Agent Gotchas
-
Do not use PublishAot in library projects. Libraries use IsAotCompatible (which auto-enables the AOT analyzer). PublishAot is for applications that produce standalone executables.
-
Do not use legacy RD.xml for type preservation. RD.xml is a .NET Native/UWP format that is silently ignored by modern .NET AOT. Use ILLink descriptor XML files and [DynamicDependency] attributes instead.
-
Do not use [DllImport] in new AOT code. Use [LibraryImport] (.NET 7+) which generates marshalling at compile time. [DllImport] may require runtime marshalling that is not available in AOT.
-
Do not use WebApplication.CreateBuilder() for AOT APIs. Use CreateSlimBuilder() which excludes reflection-heavy features. CreateBuilder() includes MVC infrastructure that is not AOT-compatible.
-
Do not use dotnet publish --no-actual-publish for analysis. That flag does not exist. Use dotnet build /p:EnableAotAnalyzer=true /p:EnableTrimAnalyzer=true to get diagnostic warnings without publishing.
-
Do not assume MVC controllers work with Native AOT. MVC relies on reflection for model binding, action filters, and routing. Use Minimal APIs for AOT-published web applications.
References
-
Native AOT deployment
-
ASP.NET Core Native AOT
-
ILLink descriptor format
-
LibraryImport source generation
-
Optimize AOT deployments