Skip to content

Profiling .NET MAUI Apps

Shane Neuville edited this page Dec 10, 2024 · 20 revisions

.NET Performance Investigation

https://youtu.be/hIYwz5dsZ6M

Feature XYZ was faster in Xamarin.Forms

First, verify you are comparing Release builds of both the Xamarin.Forms and .NET MAUI applications. The default settings in Debug builds will results in drastically different performance characteristics.

  • .NET MAUI, Debug: has UseInterpreter=true by default. This enables C# hot reload, which is a very useful feature in Debug mode.

  • Xamarin.Forms, Debug: does not have UseInterpreter=true by default.

When comparing Debug builds, you are comparing a JIT-compiled Xamarin.Forms application versus an interpreted .NET MAUI application. If you are not using C# hot reload, and it is impacting the development experience of your app, you could consider disabling the interpeter in Debug mode:

<PropertyGroup>
  <UseInterpreter>false</UseInterpreter>
</PropertyGroup>

This could be useful when debugging math-intensive code, for example.

If you are still experiencing poor performance with a .NET MAUI application in Release mode:

  • Record a .nettrace or .speedscope file using the instructions below.

  • File an issue: sharing either the trace or a sample application.

Profiling on Mono Platforms

For details about using dotnet trace on specific platforms, see:

Windows & PerfView

PerfView is probably the simplest way to profile a .NET MAUI application running on Windows. We recommend using a Release build with ReadyToRun enabled. See our Windows Publishing Documentation for details about building a Release version of your app.

Begin by selecting the Collect > Collect menu. For a project named hellomaui, you can filter on hellomaui.exe:

PerfView Collect Screen

Click Start Collection and manually launch your app.

Click Stop Collection when your app has launched and reached the point you are finished profiling.

Open CPU Stacks to view timing information of methods inside your app:

PerfView CPU Stacks

Use the Flame Graph tab for a graphical view:

PerfView Flame Graph

You can also do File > Save View As to save the file in SpeedScope format. This allows you to open the file in https://speedscope.app/ as we recommend for iOS/Android projects.

Windows & dotnet-trace

You can get a .nettrace or .speedscope file by publishing your app "unpackaged":

dotnet publish -f net6.0-windows10.0.19041.0 -c Release -p:PublishReadyToRun=true -p:WindowsPackageType=None

Then run dotnet-trace against the published .exe:

dotnet trace collect -- bin\Release\net6.0-windows10.0.19041.0\win10-x64\publish\YourApp.exe

This will place a .nettrace file in the current directory. You can also use --format speedscope to open the file in https://speedscope.app/.

Controls.Sample.Profiling Sample

To build the profiling app from source, use the local dotnet instance:

./bin/dotnet/dotnet build src/Controls/samples/Controls.Sample.Profiling -f net6.0-android -t:run -c Release -p:IsEmulator=true

This builds the app and runs it on the device/emulator.

The IsEmulator property can be used to control the environment variables of the app. The main reason is that it uses different ports on the Android device depending on whether it is a local emulator or a connected device.

You may have to build the build tasks for the first run:

./bin/dotnet/dotnet build Microsoft.Maui.BuildTasks-net6.slnf

Physical Devices

If a physical device is to be used, then there needs to be a port opened on the Android device via adb:

adb reverse tcp:9000 tcp:9001

This commands forwards port 9000 from the device to port 9001 on the host machine.

The same run command can be used for emulators, except that the IsEmulator property should be set to false.

Measuring Startup Times

Android

We have a few different useful links & documentation on this subject.

iOS, Mac, etc.

Install Tooling

Running dotnet-dsrouter on the Host

If you run dotnet-dsrouter without any args it will give you all available options.

dotnet-dsrouter ios-sim
  • Make a note of the PID that is output from running this command.

Launching application

You'll need to locate the most recent version of mlaunch

cd /usr/local/share/dotnet/packs/
ls Microsoft.iOS.Sdk.net*
# alias the most recent version
alias mlaunch=/usr/local/share/dotnet/packs/Microsoft.iOS.Sdk.net9.0_18.0/18.0.9617/tools/bin/mlaunch
# use `mlaunch --listsim=simulators.xml` to get a list of all the simulators on your system
mlaunch --launchsim=YourApplication.app --device :v2:runtime=com.apple.CoreSimulator.SimRuntime.iOS-18-0,devicetype=com.apple.CoreSimulator.SimDeviceType.iPhone-XS --wait-for-exit --stdout=$(tty) --stderr=$(tty) --argument --connection-mode --argument none '--setenv:DOTNET_DiagnosticPorts=127.0.0.1:9000,nosuspend,listen'
  • Use suspend if you want to block application startup, so you can actually dotnet-trace startup times of the application. If you aren't tracing startup then nosuspend is fine.

Creating the Speed Scope File

Locate the process id for the dsrouter

dotnet-trace collect -p 38604 --format speedscope

The --format argument is optional and it defaults to nettrace. However, nettrace files can be viewed only with Perfview or Visual Studio on Windows, while the speedscope JSON files can be viewed "on" Unix by uploading them to https://speedscope.app/.

Windows

One approach is to log a message on startup while measuring the time. See the measure-startup sample for an example of this.

We can also use PerfView to capture specific ETW events related to startup in WindowsAppSdk (and .NET MAUI) applications on Windows. We recommend using a Release build with ReadyToRun enabled. See our Windows Publishing Documentation for details about building a Release version of your app.

Using Collect > Collect menu, as mentioned above, we also need the Advanced Options:

PerfView Collect Startup

Only record two event types, for measuring the fewest events needed:

  • Kernel Base
  • Microsoft-Windows-XAML:0x44:Informational in Additional Properties

Each additional event measured will slow down the startup time of your app -- resulting in less-accurate timing.

When collecting, manually launch and close your app 3-5 times before stopping the collection session.

Next, open the Events report, and we'll need to do some math!

PerfView Events

Review the Windows Kernel/Process/Start events and start a Microsoft Excel document. Save the Time MSec and process ID values. for the next step:

PerfView Process Start

Review the Microsoft-Windows-XAML/Frame/Stop events, and find the first instance of each matching each process ID from before:

PerfView XAML Frame Stop

Here I filtered on an app named bar (instead of hellomaui), which matches my project name.

Subtract the value of the first Microsoft-Windows-XAML/Frame/Stop event with a matching process ID of a Windows Kernel/Process/Start event.

So, for example:

  • 8,773.041: bar (53164)
  • 7,696.075: bar (53164)

8,773.041 - 7,696.075 = 1,076.966 is roughly a 1 second 76 millisecond startup time.

Do this calculation for each run (which is why Excel is nice!) and get an overall average. You might consider throwing out the first time from the average as well.

Clone this wiki locally