TorchInductor GPU Profiling¶
This section lists useful commands and workflows that can help you dive into a model’s performance in TorchInductor. When a model is not running as fast as expected, you may want to check individual kernels of the model. Usually, those kernels taking the majority of the GPU time are the most interesting ones. After that, you may also want to run individual kernels directly and inspect its perf. PyTorch provides tools to cover everything mentioned above.
Relevant Environment Variables¶
You can use the following environment variables in your analysis:
TORCHINDUCTOR_UNIQUE_KERNEL_NAMES
By default, TorchInductor names a Triton kernel as
‘triton\_’
. When this environmental variable is enabled, inductor generates a more meaningful kernel name in the trace, for example,triton_poi_fused_cat_155
which contains the kernel category (poi
for pointwise) and original ATen operator. This config is disabled by default to improve the chance of compilation cache hit.
TORCHINDUCTOR_BENCHMARK_KERNEL
Enabling this will make inductor codegen harness to benchmark individual triton kernels.
TORCHINDUCTOR_MAX_AUTOTUNE
Inductor autotuner will benchmark more
triton.Configs
and pick the one with the best performance results. This will increase compilation time with the hope to improve performance.
Breakdown Model GPU Time¶
Below are the steps to breakdown execution time of a model into
individual kernels. We take mixnet_l
as an example.
Run the benchmark script for the model:
TORCHINDUCTOR_UNIQUE_KERNEL_NAMES=1 TORCHINDUCTOR_BENCHMARK_KERNEL=1 python -u benchmarks/dynamo/timm_models.py –backend inductor –amp –performance –dashboard –only mixnet_l –disable-cudagraphs –training
Note
The tool relies on kernel name to decide its category. Enabling
TORCHINDUCTOR_UNIQUE_KERNEL_NAMES
is crucial for that.In the output log, look for lines:
**Compiled module path: /tmp/torchinductor_shunting/qz/cqz7hvhood7y3psp7fy6msjxsxyli7qiwiybizdwtjw6ffyq5wwd.py**
We have one line for each compiled module. If there are no extra graph breaks, we would see 2 such lines in the log, one for the forward graph and one for the backward graph.
For our example command, we get the following compiled module for the forward and backward graphs respectively:
https://gist.github.com/shunting314/c2a4d8a28b00fcb5586d0e9d9bf77f9f
https://gist.github.com/shunting314/48efc83b12ec3ead950052e4a0220b10
Now we can dive into the perf for each individual compiled module. Let’s pick the one for the forward graph for illustration purposes. I’ll name it
fwd.py
for convenience. Run it directly with the-p
argument:**> python fwd.py -p**
See the full output log in this example gist.
In the output, you can notice the following:
We write a chrome trace file for the profile so we can load the trace and interact with it. In the log, look for lines as follows to find the path of the trace file.
Chrome trace for the profile is written to /tmp/compiled_module_profile.json
Loading the trace into Chrome (visit chrome://tracing in the chrome browser and load the file as the UI suggested) will show UI as follows:
You can zoom in and out to check the profile.
We report the percent of GPU time regarding to the wall time by log line like:
Percent of time when GPU is busy: 102.88%
Sometimes you may see a value larger than 100%. The reason is because PyTorch uses the kernel execution time with profiling enabled while using wall time with profiling disabled. Profiling may distort the kernel execution time a bit. But overall it should not be a big deal.
If we run the model like
densenet121
with a small batch size, we would see low percent of time when GPU is busy:(Forward graph) Percent of time when GPU is busy: 32.69%
This means the model has a lot of CPU overhead. This is consistent with the fact that enabling cudagraphs improve densenet121’s perf a lot.
We can break down the GPU time to different categories of kernels. In the
mixnet_l
example, we seepointwise kernel takes 28.58%
reduction kernel takes 13.85%
persistent reduction kernel takes 3.89%
the rest are cutlass/cudnn kernels for mm/conv which takes 56.57%
This information can be found in the summary line (last line) of the report for each kernel category.
We also call zoom into a certain category of kernels. For example, let’s check reduction kernels:
We can see an ordered table of execution time for each individual reduction kernel. We also see how many times a kernel is executed. This is helpful for a few reasons:
If a kernel only takes a tiny amount of time, for example, 0.1%, improving it will at most bring 0.1% overall gain. It is not worth spending a lot of effort on it.
Ff a kernel takes 2% of time, improving it by 2x will bring in 1% overall gain which justifies the effort.
Benchmark Individual Triton Kernel¶
Let’s say we want to take a closer look at
triton_red_fused\__native_batch_norm_legit_functional_16
which is the
most expensive reduction kernel and takes 2.19% of overall wall time for
the forward graph.
We can lookup the kernel name in the fwd.py
, and find comment like:
# kernel path: /tmp/torchinductor_shunting/jk/cjk2vm3446xrk7rth7hr6pun7xxo3dnzubwcn6ydrpifal4eykrz.py
I’ll rename it k.py for convenience. Here is a paste for this file.
k.py
is a standalone Python module containing the kernel code and its
benchmark.
Run k.py
directly will report its execution time and bandwidth:
We can check if max-autotune helps this kernel, by running:
**TORCHINDUCTOR_MAX_AUTOTUNE=1 python /tmp/k.py**
We may also temporarily add more reduction heuristics and run the script again to check how that helps with the kernel.