# The Anatomy of a Triton Attention Kernel

How to achieve cross-platform state-of-the-art LLM attention using only Triton

Burkhard Ringlein, Jan van Lunteren, Radu Stoica, Thomas Parnell IBM Research Zurich, Switzerland ngl, jvl, rst, tpa@zurich.ibm.com

### **Abstract**

A long-standing goal in both industry and academia is to develop an LLM inference platform that is portable across hardware architectures, eliminates the need for low-level hand-tuning, and still delivers best-in-class efficiency. In this work, we demonstrate that portable, efficient cross-platform LLM inference is indeed possible and share our experience. We develop a state-of-the-art paged attention kernel, the core performance-critical component of many LLM deployments, that builds exclusively on the domain-specific just-in-time compiled language Triton to achieve state-of-the-art performance on both NVIDIA and AMD GPUs. We describe our high-level approach, the key algorithmic and systemlevel improvements, the parameter auto-tuning required to unlock efficiency, and the integrations into a popular inference server that are necessary to bring the performance of a generic Triton attention kernel from 19.7% of the stateof-the-art to 105.9%. Our results highlight how open-source domain-specific languages can be leveraged to unlock model portability across different GPU vendors.

CCS Concepts: • Computing methodologies  $\rightarrow$  Artificial intelligence; • Theory of computation  $\rightarrow$  Design and analysis of algorithms; • Software and its engineering  $\rightarrow$  Parallel programming languages; Just-in-time compilers; • Computer systems organization  $\rightarrow$  Cloud computing; • General and reference  $\rightarrow$  Performance; Measurement.

*Keywords:* Language Models, Portability, Domain-specific Languages, Performance of Systems, Code tuning

#### 1 Introduction

Large Language Modelss (LLMs) have evolved dramatically in the past years. Besides the improvement in model architectures and training procedures, there have been many innovations in optimizing LLM applications for modern hardware [7, 36, 21, 44, 8].

However, the race in features and performance leads to a "hardware lottery" [14] for new Artificial Intelligence (AI) or machine learning (ML) paradigms and to a gravity slope around the most dominant hardware platform. The tight interconnect between AI algorithms and AI hardware leads to limitations on the deployment and application scenario of AI, since most features are only supported for a narrow set of hardware or input problem sizes [14]. Consequently, the number and the size of libraries required to deploy LLMs



**Figure 1.** Architecture of vLLM v1 and the question of cross-vendor performance portability.

with state-of-the-art (SoTA) performance have grown dramatically.

While we welcome this Cambrian explosion of specialized libraries, this development adds an additional risk to democratizing AI [4], since it creates a hard-to-navigate jungle of dependencies on frequently closed-source proprietary tools (e.g., [44]). The dependency on third-party libraries significantly hinders the adoption of new hardware and AI applications. Additionally, writing tens of thousands of Lines of Code (LoC) to port a one-line kernel [41] slows down research and complicates deployment of new ML methods unnecessarily.

A long-standing goal in both industry and academia is to develop a fully platform-independent software stack [4]. At the inference server level, there is a tremendous momentum towards vLLM [21, 42], an open-source framework that has the aspiration to be the de facto engine of LLM inference. vLLM abstracts away many of the complicated details of deploying modern LLMs, while being an active and vibrant open-source community. However, vLLM still depends on many proprietary or closed-source dependencies to compile and run LLM kernels, which are platform- and vendor-specific. This dilemma is also depicted in Figure 1. vLLM supports many platforms, however, for the performance-critical kernels, it still depends on external libraries wrapped as backends in vLLM. This is especially true for the "core" of

1

many LLMs, the so-called (self-) attention layer [41], which is often the most performance-critical kernel of an LLM [7, 36, 21, 44, 8]. These platform-specific attention libraries have frequently tens of thousands of LoC [33, 36].

This dilemma raises the important question, as also illustrated in the lower part of Figure 1, if it is possible to design a fully platform-independent LLM implementation that can achieve state-of-the-art performance on multiple platforms? If so, what are the key steps required to develop a performance-critical, platform-independent kernel? The aim of our research is to answer these questions.

Recent efforts [33, 11] indicate that it is possible to converge back to single-source open-source libraries, while combining all SoTA LLM kernel innovations. In this work, we continue this line of research and describe and demonstrate the development of a production-ready, open-source, crossplatform, state-of-the-art LLM attention kernel. The approach we take is to build upon the OpenAI Triton domain-specific language (DSL) and show how the key steps required to make a cross-platform performance-critical kernel with SoTA performance.

We summarize our contributions as follows:

- 1. We demonstrate a feature-complete cross-platform paged attention kernel with SoTA performance.
- 2. We offer a comprehensive summary of the key lessons learned, including algorithmic optimizations, the programming and memory models adopted, and system-level trade-offs that are broadly applicable.
- 3. We open-source our kernels and micro-benchmark suite (ibm.biz/vllm-ibm-triton-lib) and integrated the kernels into the one of the most widely used inference framework vLLM (vllm.ai), where it has been adopted as the default attention kernel for AMD GPUs.

# 2 Background and Related Work

#### 2.1 vLLM: An Inference Server

Typically, LLMs are deployed via serving or inference frameworks, such as vLLM [21, 42], which abstract many details of the model deployment and request scheduling.

Figure 1 shows an illustrative example of vLLM and its internal architecture, as of version 1 ("VI"). An inference server is necessary for reducing cost by allowing multiple users to use the same LLM in parallel and, therefore, are a critical part of the AI stack. Additionally, inference servers can decrease the latency and improve the throughput of the overall application, since the initial startup costs, such as loading model weights, are only incurred once. As shown at the top of Figure 1, the user typically only provides the LLM to deploy, optional configuration regarding quantization, and the model weights, in case these are not publicly available in public repositories such as Hugging Face [15]. Today, vLLM is the de facto industry standard for LLM serving. vLLM is increasingly being adopted in production and can run on NVIDIA GPUs, AMD GPUs, as well as custom

```
instance_id = t1.program_id(axis=0)
my_block_start = instance_id * BLOCK_SIZE
offsets = my_block_start + t1.arange(0, BLOCK_SIZE)
mem_mask = offsets < n_elements
x = t1.load(x_ptr + offsets, mask=mem_mask)
y = t1.load(y_ptr + offsets, mask=mem_mask)
result = x + y
t1.store(output_ptr + offsets, result, mask=mem_mask)</pre>
```

**Listing 1.** A simple vector add program in Triton, BLOCK\_SIZE is the global tiling size, determined manually or by autotuning.

accelerators like AWS Inferentia [1], Google's TPU [39], or IBM's Spyre [17, 43].

To allow such flexibility, vLLM has a complex internal structure to separate the functionality of scheduling, model pre- and post-processing, and runtime execution. To achieve state-of-the-art performance, the vLLM runtime execution is split into two major parts, as shown in the lower half of Figure 1. The simpler layers of the deployed LLM, for example, normalization- or projection-layers, are written in platform-independent PyTorch functions and compiled and optimized automatically by using torch.compile [29]. However, the complex and most performance-critical attention layers are encapsulated in an abstraction called the attention backend. In vLLM, there are multiple backends, usually wrappers around manually optimized libraries, such as flash\_attn [36] or flashinfer [44], as shown in the middle of Figure 1. Consequently, many backends depend on external libraries and vLLM cannot be used without them.

# 2.2 Triton: A tiling DSL

The Domain-specific Language (DSL) Triton [38, 40] has recently become popular as a promising open-source alternative to writing custom CUDA kernels. Triton (sometimes called *OpenAI Triton*) enables writing and debugging kernels using high-level Python code, which can be compiled and executed on various GPU architectures. Triton kernels have been shown that they can be both highly performant and portable across different GPU platforms. For this reason, Triton is growing in popularity; it is used for many LLM stacks and is an integral part of pytorch.compile.

Triton leverages a Just-in-Time (JIT) compiler and builds on the idea of *hierarchical tiles* to automate memory coalescing, shared memory allocation, and synchronization between threads [38]. Listing 1 shows a one-dimensional parallelized vector addition in Triton. Triton kernels can be fine-tuned for different workload sizes or target architectures using hyperparameters, also called *kernel configurations*. For example, in Listing 1, BLOCK\_SIZE is a configuration parameter that influences the scheduling across the GPU cores. However, in practice, Triton kernels also require hand optimizations for specific workloads and do not perform equally well across GPU platforms. To counter this, Triton kernels and their (compiler-) parameters can be autotuned.

#### 2.3 Portability and Kernel Parameter Autotuning

Autotuning is a complementary technique to compilation and can further increase performance-portability. It helps the compiler find an optimal, or nearly optimal, set of kernel configuration parameters through trial and error by leveraging microbenchmarks during or ahead of compilation. The main benefit of autotuning is that it avoids manually writing tens of thousands of highly optimized lines of code, as seen in the libraries in subsection 2.4. It is especially helpful when porting or deploying applications on new hardware. Autotuned kernels augment compilation with empirical performance tuning, generating and benchmarking a wide range of kernel variants to select the best-performing configuration for the target hardware and scenario. Autotuning reduces the parameter space a compiler needs to consider for a specific kernel compilation. This method can explore significantly more of the optimization space - often an order of magnitude more variants [10, 24, 6] — leading to improved performance and better code specialization. Due to this, autotuning offers better portability to compile-time trade-offs compared to purely compiler-based approaches, which can lead to very long compilation times [31] or even be impossible, due to the complexity of the problem [33].

In our previous work, we have shown that autotuning enables portability of Triton kernels for LLMs on GPUs [33]. There, we compared a Triton implementation of Flash Attention v2 [7] with the flash\_attn library and the ROCM flash attention implementation and demonstrated that Triton can achieve comparable performance to the vendor-specific SoTA libraries on both an NVIDIA A100 and an AMD MI250, using the same kernel code.

Outside of academic literature, autotuning for LLM applications is leveraged, for example, by Pytorch Inductor. PyTorch Inductor is the tuning frontend for torch.compile, which is also a JIT compiler that can utilize Triton as its backend. Pytorch Inductor selects different algorithms by simply trying all sequentially, and recently added support for Triton kernels [28].

However, autotuning Triton kernels to increase performanceportability still is barely or sub-optimally used in practice, primarily due to its large overhead. In section 5, we present our solution to overcome these limitations.

#### 2.4 Attention Kernels and Libraries

Flash Attention (or flash\_attn) is a popular open-source library for the attention algorithm [7, 36, 8, 9]. It contains more than 70 000 LoC, mostly CUDA. FlashAttention is known for pioneering many algorithmic innovations, such as the so called "flash trick" to improve the memory locality of the attention algorithm [8]. Flash Attention is mainly optimized to run on the latest NVIDIA GPUs. There exists also RocmAttention [35], a fork of FlashAttention containing AMD-specific optimizations and cross-compiling the CUDA code using hipify.

However, the first versions of flash attention led to high memory fragmentation as, for every request, sufficient memory must be reserved to store the maximum possible number of tokens that can be generated. Hence, "Paged Attention" [21] developed a paged version of flash attention, to leverage the concept of paging to reduce GPU memory consumption. The idea is only to reserve a small amount of memory, e.g., 16 tokens for new requests, in a data structure called a page. If the request generates more than 16 tokens, a new page is allocated. Paged attention improves the efficiency of inference servers and is a core feature of inference platforms like vLLM. Many other attention libraries like flash\_attn or flashinfer followed suit and added paged versions of their algorithms.

Flashinfer is a partly open-source library containing 51 000 LoC and depends on many proprietary binary artifacts [44]. It is written in CUDA and can only be deployed on recent NVIDIA hardware.

Flex Attention [11] tries to reduce the number of attention libraries and to cover a broader range of attention varieties. It offers an optimized attention implementation with a customizable scoring method. That way, they can support different sliding windows, paged or not paged, changes to soft-capping, or different attention masks.

Aiter [34] is a recent AMD-specific inference library with the goal of being a full wrapper for the different kernels optimized for the latest AMD GPUs. Within Aiter, kernels are written in various languages, including Triton, HIP, CK, and even assembly. There exists the option to use Aiter within vLLM for deployments on AMD GPUs.

#### 3 Solution Overview

An overview of our attention solution and its integration into vLLM is given in Figure 2. The figure shows all key steps and components that are relevant for achieving best-possible performance: The scheduler ① and gpu\_model\_runner ② as vLLM core components, as well as our triton\_attn backend ③ comprising three kernels ③a. Additionally, our backend includes configuration heuristics for these kernels ③b to improve performance portability. The Triton Backend also takes into account on which GPU ⑤ it is deployed when consulting these heuristics. In contrast to the manual optimized attention backends, most other layers in vLLM are written in native pytorch code and compiled to high-performance kernels using torch.compile ④.

A typical vLLM deployment records the CUDA-graphs or HIP-graphs at startup time (a), before the inference server reports itself as ready. To do this, pseudo metadata are created (b) and feed to the model, including the attention backend. At inference time, these graphs are then "replayed" and the actual code is no longer executed. vLLM differentiates between two modes: "Partial CUDA-graphs" and "Full CUDA-graphs". In the first mode, all the layers except attention and mamba layers are executed as CUDA graphs. In the latter



Figure 2. Overview of the triton\_attn backend in vLLM.

mode, all layers are included in one CUDA graph recording and execution.

In section 4, we describe iteratively the development process of the ③a kernels. After, we describe how to tune the kernels and ensure best portability between vendors also using heuristics ③b, in section 5. Finally, we will describe the advantages and disadvantages of CUDA/HIP-graphs (⑥a and ⑥b) and the implications of their usage in section 6.

# 4 Design and Implementation of Attention4.1 Core Concepts

Attention, as introduced in the seminal paper by Vaswani et al. [41], is computed according to:

Attention
$$(Q, K, V) = \operatorname{softmax} \left( \frac{QK^{\top}}{\sqrt{d_k}} \right) V$$
 (1)

with  $Q,K,V\in\mathbb{R}^{n\times d_k}$  being the query, key, and value matrices, respectively, derived through linear projections from the input embeddings, with n denoting the sequence length in tokens and  $d_k$  is the dimensionality of the key vectors. The intermediate matrix product  $QK^{\top}\in\mathbb{R}^{n\times n}$  contains the attentions scores between all token pairs, while  $\sqrt{d_k}$  is used as a scaling factor for stabilizing the softmax operation. A naive implementation of Equation 1 would incur a  $O(n^2)$  computation and memory complexity. To accelerate the performance of attention computation on GPUs, various optimization techniques have been developed. These are briefly discussed below and are also incorporated into our attention kernels, as discussed in this section.

**Tiled Softmax.** The softmax function is calculated independently for each row in  $QK^{\top}$ , where each row corresponds to one query token, to produce a probability distribution by normalizing the attention scores across all keys. It is computed in a numerically stable way as follows:

$$softmax(s_i) = \frac{e^{s_i - \max_j s_j}}{\sum_{k=1}^n e^{s_k - \max_j s_j}}$$
(2)

with  $s_i$  denoting the *i*-th element of the score vector s (i.e., a row in  $QK^{\top}$ ), and  $max_js_j$  representing the maximum value within that score vector. The latter is subtracted from each

element value to improve numerical stability by preventing large positive values from causing overflow and large negative values from prematurely underflowing to zero during exponentiation.

The softmax can be computed efficiently using a tiled approach (also denoted as online softmax) in which each row is partitioned into smaller tiles that are processed incrementally, rather than processing the entire row in a single pass. Tiled softmax delays the division by the sum of exponentials to the end of the computation. It maintains the maximum row value  $(max_js_j)$  and sum of exponentials  $(\sum_{k=1}^m e^{s_k-max_js_j})$  and updates these after each tile is processed. This may involve a rescaling of the intermediate results if the maximum changes.

The smaller tile sizes enables the kernel to use the fast shared memory and registers on the GPU most (or all) of the time, instead of falling back to the slower global memory. This results in substantial performance improvements. In practice, the tiled softmax is fused with the tiled matrix multiplications in Equation 1. In the following, we will refer to this fused implementation as tiled softmax. Tiled softmax is one of the key optimizations in FlashAttention [8].

**KV Cache.** The computation of the attention layers of an LLM is accelerated by caching the K and V matrices for previously processed input tokens and reusing these cached matrices for subsequent attention operations. The KV cache is initialized during prefill, when K and V must be calculated for all tokens in the prompt. During decode, however, K and V need only be calculated for each newly generated token.

Grouped and Multi Query Attention. State-of-the-art Transformer architectures are based on multi-head attention, where multiple attention heads operate in parallel in each layer, enabling the model to capture a broader range of relationships. However, this comes at the cost of repeated attention computation across heads, each having its own Q, K and V projection matrices. Grouped Query Attention (GQA) addresses this by reducing the number of key and value heads, allowing multiple query heads to share the same K and V projections. This requires fewer K and V matrices to be computed and results in a smaller KV cache. Multi Query Attention (MQA) takes this to the extreme of using a single KV head for all queries.

**Batching.** Finally, and not limited to attention computation, batching multiple sequences increases parallelism, which typically results in more efficient utilization of GPU resources.

### 4.2 Terminology

We use the following terminology, which relates to a single sequence and is also used by vLLM:

 Context Length: The number of past tokens in the sequence whose K and V matrices are stored in the KV cache.



Figure 3. Q Block and KV Tiles.

- Query Length: The number of new tokens that are currently being processed, for which attention is calculated against the cached context.
- **Sequence Length:** The total number of tokens in the sequence, defined as the sum of the context length and the query length.

For prefill attention, the context length equals zero, and the query length equals the prompt size. For decode attention, the query length equals one.

In addition, we use **Prefix Length** to denote the number of tokens preceding a given token in a sequence. This can include both previously processed tokens and new tokens that appear before the current token (e.g., in a prompt).

#### 4.3 Baseline Triton Attention Kernel

For our first implementation of paged attention in Triton, we followed the original algorithm for paged attention [21]. Our implementation assumes that Q, K, and V have already been computed before the kernel launch and stored in the KV cache in blocks, i.e. "paged memory". The KV cache in vLLM is accessed through a block table (analogous to a page table), and the parameter BLOCK\_SIZE defines the maximum number of tokens stored in a single KV cache block.

Although we use the very same kernel for prefill and decode, we launch it with different launch grids for each phase. For the prefill phase, we launch tokens\_in\_batch × query\_heads instances of the kernel, i.e., a two-dimensional launch grid. For the decode phase, we noticed that sequences\_in\_batch × query\_heads is a better launch grid.

We provide in Appendix A a detailed description of the implementation, including Listing 3.

#### 4.4 Prefill and GQA Optimization

The baseline attention kernel described in the previous section processes only a single pair of query token and query head per program instance. We now introduce an optimization that increases the number of combinations handled within each program instance by including:

- Multiple successive tokens from the same prompt, in case of prefill attention.
- Query heads that share the same KV head.

These combinations collectively form what we refer to as a Q Block, as illustrated in Figure 3, covering a total of BLOCK\_M combinations of BLOCK\_Q query tokens and  $\frac{BLOCK\_M}{BLOCK\_Q}$  query heads. By setting  $\frac{BLOCK\_M}{BLOCK\_Q} = \frac{num\_query\_heads}{num\_kv\_heads}$ , each Q Block cover all query heads that map to a single KV head.

Structurally, the Q Block is a three-dimensional block, with dimensions corresponding to the number of query tokens, query heads, and the head size. However, for implementation efficiency, the Q Block is represented as a two-dimensional tensor with a shape of BLOCK\_M × HEAD\_SIZE. This flattening simplifies memory access patterns and aligns better with Triton's programming model.

For a given sequence, a total of  $\left\lceil \frac{\text{query\_length}}{\text{BLOCK\_Q}} \right\rceil$  Q Blocks are required to process all query tokens. In the case of decode attention, where only a single query token is processed at a time (i.e., BLOCK\_Q = 1), this results in one Q Block per sequence.

The Q Block structure enables the kernel to process multiple attention computations in parallel, which improves efficiency. For prefill attention, many query tokens attend to the same preceding tokens, allowing efficient reuse of the K and V matrices. Similarly, in Grouped Query Attention (GQA), multiple query heads correspond to the same KV head, which only needs to be loaded once per Q Block. All of this reduces memory bandwidth and improves computational efficiency by increasing the arithmetic density.

The structure of K and V tiles processed for each Q Block using the tiled softmax approach remains unchanged from the baseline attention kernel discussed in Section 4.3, as also illustrated in Figure 3. These tiles correspond to the KV head to which the query heads in the Q Block are mapped, and span the tokens preceding those in the Q Block, up to the maximum prefix length of any token in the Q Block. An example code for the Q-Block-based optimized attention kernel is described in Appendix B and Listing 4.

#### 4.5 Parallel Tiled Softmax

When using the kernels discussed in the previous sections for performing decode attention, the first dimension of the launch grid corresponds to the number of sequences in the batch. As a result, small batch sizes lead to a limited number of program instances being launched, which can underutilize available GPU resources and result in degraded performance. This limitation does not apply to prefill attention, because prompts typically contain many tokens, resulting in a sufficiently high number of Q Blocks to saturate the available compute resources on the GPU.

To extract enough parallelism during decode attention, the iterative processing of tiled softmax can be parallelized across multiple program instances. This approach is particularly advantageous for small batches of long sequences, where a large number of tiles must be processed and the GPU tends to be underutilized. By distributing the processing of these tiles across multiple program instances executed in parallel, rather than executing them sequentially within one program instance, significant speedups can be achieved.



Figure 4. Parallel Tiled Softmax.

To describe our implementation of parallel tiled softmax, we introduce the following terminology. A tile refers to the granularity at which the tiled softmax operations process Q against *K* and *V* in an iterative fashion. A *segment* comprises the tiles that are executed together within a single program instance. This is illustrated by an example in Figure 4. Figure 4 (a) shows an attention computation partitioned into 32 tiles, which are processed iteratively while maintaining and updating intermediate results, maximum, and sum of exponentials, as discussed in Section 4.1. Figure 4 (b) shows how these 32 tiles are distributed over 4 segments, each comprising 8 tiles. The attention computation within the segments is performed in the usual iterative manner, but in separate program instances. To produce the final result, a reduction step is required to merge the outputs from all segments. For this purpose, the program instances executing the segments stores the intermediate results in memory. Finally, to obtain the final attention output, the intermediary results are retrieved, combined, and rescaled in a manner analogous to the iterative tiled softmax approach. An example implementation is presented in Appendix C.

The kernel implementing the parallel tiled softmax is only launched for decode attention on small batches involving longer sequences. This behavior is implemented by the heuristics in our Triton backend (3b) in Figure 2). To determine whether to use this kernel or the original attention kernel, we apply a heuristic-based selection strategy.

# 4.6 Adjustable Tile Sizes

In the kernels discussed in the previous sections, the tile size used in the tiled softmax was constrained to match the number of tokens contained in each KV cache block, defined by one parameter called BLOCK\_SIZE. We have extended our kernel implementation to decouple the tile size from BLOCK\_SIZE, enabling it to be configured independently: smaller, equal, or larger. A key advantage is that this allows tuning the tile size independently of the block size and separately for prefill and decode for best performance. Another advantage is that it facilitates support for hybrid models that combine conventional Transformer attention layers with state-space model (SSM) layers (e.g., Mamba [12]). In such architectures, attention layers often utilize large, non-power-of-two block sizes to achieve proper page alignment between the attention and SSM layers.

#### 4.7 Static Launch Grid

As the last optimization step, we modify the kernels to use a static launch grid, which consistently launches the same number of program instances. This was achieved by adapting the number of Q blocks processed by each instance. The primary benefit of this approach is improved compatibility with full CUDA or HIP graphs, as will be discussed in subsection 6.2.

# 5 Autotuning and Portability

# 5.1 Disadvantages of Current Autotuning

Recent works have shown that single-source Triton kernels can be competitive on multiple platforms, but may need autotuning (c.f. subsection 2.3). However, in practice, autotuning of Triton kernels comes with a large overhead in tuning time [33, 30, 22] which renders it unusable for many use cases, e.g., the use in vLLM (c.f. [20]). For example, tuning flash attention v2 extensively to achieve best-possible performance took nearly 24 hours for each GPU type [33].

This overhead can be reduced in some situations if the autotuning results are cached so they can be reused between deployments [33, 30, 22, 27, 16]. These caches of the Triton autotuner contain the results of an autotuning run with a simple mapping of a scenario to the autotuning results in that scenario. A scenario is a specific combination of arguments passed to the Triton kernels, such as tensor pointers, their shapes, strides, and other scalar arguments. In vLLM attention backends, some of these kernel arguments change, e.g., with the sequence length. However, this caching of autotuning states helps only if the exact same scenario occurs again, in which case the autotuning phase is skipped and the cached result is used instead. For example, after tuning for 32 tokens, a request might arrive again with 32 tokens; in this case, the autotuning of the attention kernel could be skipped. But if a request with 33 tokens arrives, then the kernel needs to be autotuned again.

Besides the tuning overhead, autotuning in inference servers causes additional disadvantages: First, even if all possible scenarios are tuned ahead of time, the lookup of the optimal configuration to be used in a concrete scenario usually adds around tens of micro-seconds to the Triton kernel launch time, consuming the latency improvements of the tuned configuration for short workloads. Second, and more important, if using CUDA or HIP graphs, there is no way to look up the optimal configuration for each kernel launch, since HIP/CUDA graphs are recorded once and then just replayed without another consultation of the Triton autotuner (c.f. section 3 and subsection 6.2).

#### 5.2 Usage of autotuning in the vllm-triton-backend

To circumvent the tuning time overhead and the inflexibility of the Triton autotuner, we decided to follow a two-step approach, as depicted in Figure 5: First, we created a microbenchmark framework to perform kernel tuning outside of



**Figure 5.** Workflow of tuning in vllm-triton-backend.

the vLLM runtime components. This way, we avoid adding additional latency to the vLLM startup time or increasing the cost of Continous Integration (CI) pipelines, where autotuning would be executed every time. The micro-benchmarks are designed to call the same kernel code as the kernels in vLLM and simulate specific request patterns and LLM architectures. Additionally, the microbenchmarks help us better understand some performance artifacts of end-to-end results. For example, some kernels in the field are written for batches that always contain the same amount of tokens in every request, e.g., a batch of 8 prompts with 1000 tokens each. However, in reality, this is very unlikely for the most use cases of LLM inference serving. In our micro-benchmark framework, we can simulate varying context lengths, prompt lengths, and batch sizes and adjust the kernels accordingly.

After the kernel tuning using micro-benchmarks is completed, we analyze the results of the autotuning and export them as heuristics — simple if-else "decision trees" in this case — as a second step, as shown on the right-hand side of Figure 5. One example of such a heuristic is given in Listing 2. In contrast to the static reuse of autotuning caches [33, 30, 22, 2, 32], decision trees have the advantage of providing optimized configurations for scenarios that are not in the autotuning cache, i.e., that were not part of the tuning. Consequently, using simple decision trees has benefits at runtime and at tuning time: It also reduces the amount of tuning necessary, because tuning just for the average and corner use-cases will produce decision trees applicable to most scenarios.

# 6 Inference Server and System Integration

Despite optimizing the kernels themselves to achieve SoTA performance, our attention kernels need to be integrated into the vLLM. For this, we needed to adapt (1) the creation and computation of the attention metadata and (2) balance the trade-off of using HIP/CUDA graphs.

#### 6.1 Computation of Metadata

In vLLM, after the scheduler decides which (partial) requests are included in the next batch to be processed, the data is copied to the GPU (if not already there), fitted into the paged data structures, and the corresponding metadata is computed. This attention metadata in vLLM contain, e.g., the tensors with the list of request lengths in the batch. For some backends, the batch is also sorted to start with decode or prefill requests.

For the Triton backend, we needed to do the following adaptation: First, we count the number of decodes in the batch, so that we can decide whether the parallel tiled softmax version should be used or not. Second, we construct a tensor that stores the accumulated number of Q Blocks for the sequences in the batch. In each launched program instance, this tensor is used to perform a binary search and determine the sequence index corresponding to the Q Block index for that instance, as described at the end of Section 4.4. The total number of Q Blocks required for processing the batch is also derived from this tensor.

# 6.2 Triton Launch Overhead and CUDA/HIP Graphs

One major issue that must to be addressed for the integration of the Triton Backend is the use of CUDA [5] or HIP [13] graphs. CUDA/HIP graphs are helpful to remove the software overhead of a forward pass through the model, if each forward pass invokes exactly the same computations. Triton kernels could benefit from individual configurations per batch shape, i.e., the use of different Triton configurations depending on the number of requests in the batch, their duration, etc. (c.f. section 5). However, this means that the Triton Attention Backend changes its behavior in every forward pass and cannot be executed with HIP/CUDA graphs. In this approach, without CUDA/HIP graphs, the software overhead of Triton kernel launches is in the order of 100 μs to 300 µs, according to our profiling. This launch overhead dominates the kernel execution time for sequences below roughly 1000 tokens (c.f. subsection 7.2). This time excludes the JIT compilation time that occurs during the first kernel execution. Instead, it refers to the software overhead created at every kernel launch by Triton, e.g., by checking if another invocation of the JIT is required. Even if we circumvent some of these checks by caching them [18], we still have a launch overhead of around 80 µs. To avoid this, we had to use CUDA/HIP graphs, but this creates a different set of trade-offs: For every CUDA/HIP graph, the arguments to all kernels in this graph are frozen [37, 5, 13]. This includes the pointers of tensors. Consequently, if the numbers of recorded CUDA/HIP graphs rise, the GPU memory gets filled by reserved memory for the CUDA/HIP graphs. Additionally, the CUDA/HIP graphs require some memory themselves. Therefore, vLLM decided to limit the number of graphs to one per batch size, and even only power-of-two batch sizes are considered [37].

Hence, at vLLM startup time, all power-of-two batch sizes up to 128, usually, are recorded with one dummy request each. Since the required GPU memory for the kernels is allocated during this recording run, the recording run needs to occur with the maximum possible model sequence lengths. In other words, if using CUDA/HIP graphs, all kernels are always invoked as if the batch contains only requests of the maximum model length (or, depending on the implementation, maximum batched tokens). For our Triton kernels, this created another penalty, as we determine the number

```
BLOCK_M = 64 if max_seqlen_q > 1 and avg_seqlen_q >= 4096 \
and is_nvidia_gpu() \
else 16
BLOCK_N = 32 if max_seqlen_k <= 64 or avg_seqlen_q <= 4096 \
or is_amd_gpu() \
else 64
```

**Listing 2.** Decision tree as heuristic for Triton (tuned).

of kernel instances to launch based on the batch metadata, which is a common practice for Triton kernels. However, when using CUDA/HIP graphs, the launch grid is also fixed after the initial graph was recorded. Therefore, if we use CUDA/HIP graphs for our backend, we always launch as many Triton kernels as we would need for the longest request possible. If shorter requests are part of the batch, which is usually the case, the excess kernel instances will exit immediately; therefore, the computation of the kernel is always correct. But the launch of the excess instances still causes the GPU scheduler to schedule too many GPU "waves", i.e., rounds of kernel executions for all GPU cores (i.e., Streaming Multiprocessors).

Our evaluation revealed that the resulting additional runtime latency of the excess instances outweighs the saving of the launch overhead in nearly all cases. Therefore, we changed our kernel implementations to always have a static launch grid, close but smaller than the number of available GPU cores.

#### 7 Performance Evaluation

We evaluate the kernels using a two-track approach: Measuring incremental performance changes with a micro-benchmark suite, followed by comprehensive end-to-end testing. Our evaluation aims to answer the following research questions:

- 1. How big is the performance impact of the individual optimization steps (c.f. section 4)?
- 2. Can autotuning, in the form of simple heuristics, improve the performance further (c.f. section 5)?
- 3. What is the end-to-end performance of the whole inference server?
- 4. What are the effects of CUDA/HIP graphs vs. dynamic JIT compilation?

#### 7.1 Methodology

We run our evaluation on two GPUs, the NVIDIA H100-80GB and the AMD MI250-128GB. We selected these two GPUs because they utilize the same technology nodes (5 nm manufactured by TSMC), represent the two major HW vendors, and because of their popularity and availability.

For the micro-benchmarks, we base our kernel parameters on the Llama3-8B LLM architecture [23] (128 head size, 32 query heads, and 8 KV heads) and vary sequence lengths and batch sizes based on real-world samples. The sequences contained within a batch have variable lengths, as is often the case in real-world online inference scenarios. For every measurement in the micro-benchmarks, the kernel is

warmed up with 20 iterations, and we take the mean of the 100 following iterations as the result.

We run end-to-end experiments using the benchmark suite included with vLLM [42], on both NVIDIA H100 and AMD MI300 GPUs. For these measurements, we disable prefix caching of vllm, since we want to evaluate kernel improvements and not prompt redundancies. We use random data and ignore the end-of-sequence token of the model. The number of warmup and measurement iterations is kept at their default values of 10 and 30, respectively.

# 7.2 Evaluation of the Optimization Steps

To evaluate the individual optimization steps, we used microbenchmarks to ensure precise measurements of this single component. The Figure 6a shows four different kernel implementations of paged attention evaluated on an H100. The flash\_attn refers to the state-of-the-art library Flash Attention 3 [36], while the other three implementations are our Triton implementations. Triton (naive) is our baseline implementation from subsection 4.3, Triton (GQA opt.) is the version described in subsection 4.4, and Triton (parallel tiled) refers to the optimization explained in subsection 4.5. The very same implementations are shown Figure 6b, for the AMD MI300 GPU. However, for AMD GPU there is no competitive paged attention implementation besides ours, hence there are only three Triton implementations depicted. In the subfigures Figure 6a and Figure 6b, the individual plots represent batches with the maximum sequence length as denoted on the top, and the batch size is shown on the x-axis. The y-axis shows the latency. The leftmost value of the baselines is used to normalize all latency values of the micro-benchmarks.

If we examine the orange curves, which represents the naive implementation, we see that they are nearly an order of magnitude slower than FlashAttention across all sequence lengths. The optimization for better data locality, especially with GQA models (shown as the purple curve), shows improvements over the naive implementation for small sequence lengths and small batch sizes. Sometimes, it is even faster than FlashAttention. But, it does not bring significant benefits for sequence lengths larger than 500-1000 tokens. The optimization of parallel tiled softmax appears to increase the performance even less.

However, if we plot the very same data not by sequence lengths, but by the different compositions of the batch, as done in subfigures 6c and 6d, we see a totally different behavior. For these figures, we aggregated the batch size times the sequence lengths in the batch on the x-axis and divided the plots by the percentage of decode-only requests in the batch: 0%, 50%, and 100%. Since vLLM is always prioritizing decode requests, batches with a high share of decode requests are common. As shown in subfigures 6c and 6d, the GQA optimized kernel version has its strength primarily in handling prefill-heavy batches, which is expected, since prefill is compute-bound. The memory-bound decode phase



Figure 6. Comparing performance of different kernel optimizations with the baselines.



Figure 7. Comparing performance of the adjustable tile size optimization (c.f. subsection 4.6)



Figure 8. Comparing performance of the optimized QGA kernel with and without tuning for prefill sizes.

requires the greater parallelization of the parallel tiled softmax implementation. As depicted by the right most subplot of Figure 6c, the blue curve exhibits nearly the same performance as FlashAttention. For very long decodes, the parallel tiled softmax also outperforms the GQA optimized kernel variant, as shown in subsection 7.4.

In Figure 7 we compare the flexible block approach described in subsection 4.6. The two best approaches of Figure 6 are also presented in this figure for comparison. Since it has proven more insightful, we show here only the plots sorted by decode share. As can be seen and is expected, the flex block versions both outperform their respective comparable implementations of the previous figure.

#### 7.3 Evaluation of our Autotuning Approach

Next, we evaluate our approach to autotuning. Our preliminary tests showed that autotuning mainly affects the prefill phase; therefore, Figure 8 shows only evaluations for prefill-heavy batches, again plotted by sequence length. For the tuned version in Figure 8, we followed the flow described in section 5 and decided to settle on a very simple decision tree, based on the individual autotuning results. The simple decision tree is shown in Listing 2. As can be seen, this limited autotuning approach further reduces the kernel latency for short (up to 9.8× on H100) and medium prompts (up to 75%) on both platforms. To not exceed the scope of this paper, we do not evaluate more complex heuristics since it

would require more complex overhead and trade-off analyses. However, already this limited heuristics demonstrated to be beneficial and is also used in the end-to-end evaluation.

#### 7.4 End-to-End Evaluations

Last, we evaluate our kernel and system improvements for a popular model and large requests end-to-end. Figure 9 presents the results of the latency benchmark for the metallama/Llama-3.1-8B-Instruct model [23], using a batch size of 1 and a prompt length of 500 tokens, evaluated across varying numbers of generated output tokens. This test configuration enables a detailed analysis of the impact of the various optimizations discussed in Section 4 and excludes scheduling decisions from affecting the measurements.

The Triton Static Launch Grid-based kernel integrates the Q-Block and Parallel Tiled Softmax optimizations found in the other kernels, and additionally incorporates a static launch grid (see subsection 4.7), along with being executed with full CUDA/HIP-graphs enabled (see also subsection 6.2). All other experiments were run with partial HIP/CUDA-graphs. Furthermore, the Triton Static Launch Grid kernel includes heuristics for tile- and segment-sizes found during autotuning (c.f. subsection 7.3).

The benchmark results obtained on the H100, shown in Figure 9a, demonstrate a consistent improvement across the successively optimized kernels. In particular, the Parallel Tiled Softmax optimization shows a clear advantage for longer sequences, reducing the observed end-to-end latency by more than a factor of two compared to the kernel that only implements the Q-Block optimization, for an output length of 12,800 tokens. The Static Launch Grid-based kernel, executed with full CUDA Graphs enabled, further reduces latency by approximately 6% for the same output length. These results suggest that larger reductions in latency can be expected as sequence lengths increase. For reference, Figure 9 also includes results from the same experiments conducted using FlashAttention V3. From the comparison, it is evident that in this setup, the performance of the Static Launch Gridbased kernel is comparable to that of FlashAttention V3. As Figure 9 shows, the baseline implementation achieved only 19.7% of the performance of FlashAttention3. Our first optimized Triton kernel (c.f. Appendix B) increases this by  $2.1 \times$ to 49%, while the last optimization — the static launch grid, c.f. subsection 4.7 — achieves even 98.6% — 105.9% versus FlashAttention3.

The results for the MI300, shown on the right-hand side of Figure 9, reveal a higher impact of the launch overhead on performance compared to the H100. The Triton Static Launch Grid with heuristics and full-HIP-graphs is up to 1.99× faster than the Q Block and Par Ts version. This is evident from the higher latency of the Parallel Tiled Softmax kernel relative to the Q-Block kernel for shorter output lengths, as the former includes an additional launch of a reduction kernel (subsection 4.5). For longer sequences, however, the launch overhead becomes negligible v.s. the actual

processing time, allowing the Parallel Tiled Softmax optimization to take effect. The impact of the launch overhead is even more apparent from the substantial performance improvement achieved by enabling full HIP Graphs (a key solution for mitigating launch overhead) in the Triton Static Launch Grid-based kernel, which reduces latency by about a factor of two across all output lengths compared to the Parallel Tiled Softmax kernel. Combined, the three optimizations shown in Figure 9b exhibit a speedup of 5.9×.

# 8 Insights and Future Work

Throughout the presented work, we describe how to develop a Triton-only state-of-the-art attention kernel. However, we also discovered insights that extend beyond our specific use case. For example, these lesson learned are applied to improve the kernels for mamba/SSM layers in vLLM [19]. We discuss these insights in the following section and also mention future work.

Triton kernels need to be specific. Throughout our research, we experimented with kernel versions that would merge the three different kernels described in section 4. In particular, we had kernels that would detect if it is a decodeonly in the beginning and then would branch to either the prefill kernel (cf. subsection 4.4) or a version of decodingoptimized kernel (similar to the baseline described in subsection 4.3). The goal of these experiments was to reduce the number of kernels launched, thereby minimizing the control and software overhead associated with these kernel launches. However, we observed that the performance of these kernels drops by at least 2x, by far outweighing the saved launch overhead latency (in the order of 150 µs). Subsequent analysis of the Triton Intermediate Representations (IRs) revealed that the software pipelining did not really produce useful pipelines in these fused kernels. Hence, we concluded that Triton kernels always need to be written around one specific problem with a strong data-dependency within this problem, and we would rather "pay" for multiple kernel launches otherwise.

**Usage of tl.dot.** The multiplication  $QK^{\top}$ , which is part of the attention computation in Equation 1, can be implemented in Triton using either of the following approaches:

- Element-wise multiplication between K and a broadcasted version of Q, followed by a summation (reduction): t1.sum(K \* Q[:, None], axis=0).
- A matrix multiplication instruction: tl.dot(Q, K).

This approach also extends to multiplying the softmax output by K.

The second approach, using t1.dot, requires that the dimensions of the input matrices be multiples of specific MMA (Matrix Multiply-Accumulate) tile sizes (e.g., 8, 16, 32). This often necessitates padding to satisfy these requirements. Despite this, it is generally preferred, as it almost always results in better performance. This is because the compiler will map it directly to the MMA units, such as NVIDIA's Tensor Cores.





Figure 9. vLLM latency benchmark results for Llama-3.1-8B with batch size 1 and input length of 500 tokens.

In contrast, the element-wise multiplication followed by summation is often not recognized as matrix multiplication by the compiler.

CUDA/HIP graphs do not always help. We observe a common trend in the community to address the overhead of Tritons JIT- and launch-overhead (or also the overhead of other JIT frameworks, such as CUTLASS): If in doubt, use HIP/CUDA graphs. As discussed in subsection 6.2, CUDA/HIP graphs come with trade-offs. However, using such recorded graphs at the kernel/binary level forces the application to select exactly one binary version of the Triton kernel, thereby sabotaging the philosophy behind a JIT language. Triton specializes kernels not only based on constants, but also on memory access patterns (e.g., if a stride is divisible by 16). Autotuning (cf. section 5) further adds to this diversity. Hence, we observed that in scenarios where a single kernel runtime is equal to the average launch overhead of Triton (in the order of 200 µs), HIP/CUDA graphs do not help in decreasing overall latency, unless a kernel is developed for this purpose, as described in subsection 6.2.

Future Work: Consider GPU-specific-parameters in kernel tuning. Besides the platform-agnostic parameters like num\_warps (related to the number of threads) and num\_stages (related to the depths of the software pipeline), newer Triton versions introduced GPU-specific parameters. For example, num\_consumer\_groups and num\_buffers\_warp\_spec for a concept called "Warp specialization" for Hopper and Blackwell GPUs [25, 3]. Another example is waves\_per\_eu for AMD GPUs [26]. The parameters can be explored using the approach described in section 5, but remain outside the scope of this paper. However, in early experiments using these parameters in tuning showed an improvement of up to 80% for some kernels. Related work in literature reports even improvements above 100%, for example, by using warp specialization in flash\_attn3 [36].

Future Work: Combining Autotuning with CUDA/HIP graphs. CUDA/HIP graphs limit the flexibility of kernel-variant selection at runtime, as discussed above. Hence, it is also not possible to adapt a kernel block size and subsequently the launch grid of this kernel to the concrete number of tokens in a batch if using CUDA/HIP graphs.

Consequently, to still be able to leverage the flexibility of autotuning, we have to move the decision trees, e.g., deciding the block size *inside* a Triton kernel. Hence, the Triton binary remains the same, but it can adjust its loops accordingly, with some tweaking. However, Triton loops do not support early exits using break or return statements. Hence, we have to insert instructions that do not have an impact (i.e., NOPs) to be able to realize the required masking (cf. 5). In our experiments, this type of tuning yielded lesser performance improvements than transitioning to a static launch grid; therefore, this optimization remains future work.

### 9 Conclusions

The democratization of AI depends on both the widespread use of LLM and AI models, but likewise on the flexibility and portability of AI deployments. Therefore, implementations of LLM solutions must allow for the selection of different hardware platforms. In this work, we argue that open-source solutions, such as Triton and vLLM, are key enablers for achieving this goal. We demonstrate that our Triton implementations of the paged attention kernel achieve state-ofthe-art performance on NVIDIA and AMD GPUs, using the same source code. During our research, we learned lessons regarding the handling of the Triton memory hierarchy, improving data locality, auto-tuning to improve performanceportability, and balancing the integration of Just-in-Time compiled binaries in dataflow-only graphs, such as CUDA and HIP graphs. These investigated optimizations exhibit a total speedup of up to 589% and we contributed our kernels and frameworks as open-source (ibm.biz/vllm-ibm-tritonlib). Finally, our resulting highly optimized kernels are the default attention backend in vLLM for AMD deployments.

# Acknowledgments

We would like to thank our colleagues Chih-Chieh Yang and Sara Kokkila Schumacher for their always helpful discussions of Triton behavior and feedback to portable designs.

#### References

- [1] Amazon Web Services, Inc. AWS Inferentia. 2021.
- [2] Anyscale, director. How IBM Research Achieved vLLM Platform Portability with Triton Autotuning | Ray Summit 2024. Oct. 18, 2024. URL: https://www.youtube.com/watch?v=GG1qi82]8Hg (visited on 08/19/2025).
- [3] Automatic Warp Specialization Optimization by Manman-Ren · Pull Request #6289 · Triton-Lang/Triton. GitHub. URL: https://github.com/triton-lang/triton/pull/6289 (visited on 08/19/2025).
- [4] Albert Cohen, Xipeng Shen, Josep Torrellas, James Tuck, Yuanyuan Zhou, Sarita Adve, Ismail Akturk, Saurabh Bagchi, Rajeev Balasubramonian, Rajkishore Barik, Micah Beck, Ras Bodik, Ali Butt, Luis Ceze, Haibo Chen, Yiran Chen, Trishul Chilimbi, Mihai Christodorescu, John Criswell, Chen Ding, Yufei Ding, Sandhya Dwarkadas, Erik Elmroth, Phil Gibbons, Xiaochen Guo, Rajesh Gupta, Gernot Heiser, Hank Hoffman, Jian Huang, Hillery Hunter, John Kim, Sam King, James Larus, Chen Liu, Shan Lu, Brandon Lucia, Saeed Maleki, Somnath Mazumdar, Iulian Neamtiu, Keshav Pingali, Paolo Rech, Michael Scott, Yan Solihin, Dawn Song, Jakub Szefer, Dan Tsafrir, Bhuvan Urgaonkar, Marilyn Wolf, Yuan Xie, Jishen Zhao, Lin Zhong, and Yuhao Zhu. "Inter-Disciplinary Research Challenges in Computer Systems for the 2020s". In: (September 2018). URL: https://dl.acm.org/citation.cfm?id=3297279{&} picked=prox.
- [5] CUDA Graph Management. URL: https://docs.nvidia.com/cuda/cuda-runtime-api/index.html (visited on 08/21/2025).
- [6] Junio Cezar Ribeiro Da Silva, Lorena Leão, Vinicius Petrucci, Abdoulaye Gamatié, and Fernando Magno Quintão Pereira. "Mapping Computations in Heterogeneous Multicore Systems with Statistical Regression on Program Inputs". In: ACM Trans. Embed. Comput. Syst. (Oct. 2021). ISSN: 1539-9087. DOI: 10.1145/3478288. URL: https://doi.org/10.1145/3478288.
- [7] Tri Dao. FlashAttention-2: Faster Attention with Better Parallelism and Work Partitioning. July 17, 2023. arXiv: 2307.08691 [cs]. url: http://arxiv.org/abs/2307.08691 (visited on 03/20/2024). Pre-published.
- [8] Tri Dao, Daniel Y. Fu, Stefano Ermon, Atri Rudra, and Christopher Ré. FlashAttention: Fast and Memory-Efficient Exact Attention with IO-Awareness. June 23, 2022. DOI: 10.48550/arXiv.2205.14135. arXiv: 2205.14135 [cs]. URL: http://arxiv.org/abs/2205.14135 (visited on 01/24/2024). Pre-published.
- [9] *Dao-AILab/Flash-Attention*. Dao AI Lab. url: https://github.com/Dao-AILab/flash-attention (visited on 08/20/2025).
- [10] D. Diamantopoulos, B. Ringlein, M. Purandare, G. Singh, and C. Hagleitner. "Agile Autotuning of a Transprecision Tensor Accelerator Overlay for TVM Compiler Stack". In: *Proceedings of the 30*{{{\cdots}}}\text{textsuperscript}{th}}

- IEEE International Conference on Field-Programmable Logic and Applications (FPL). Gothenburg, Sweden: IEEE, Aug. 31–Sept. 4, 2020. ISBN: 978-1-7281-9902-3. DOI: 10.1109/FPL50879.2020.00058.
- [11] Juechu Dong, Boyuan Feng, Driss Guessous, Yanbo Liang, and Horace He. Flex Attention: A Programming Model for Generating Optimized Attention Kernels. Dec. 7, 2024. DOI: 10.48550/arXiv.2412.05496. arXiv: 2412.05496 [cs]. URL: http://arxiv.org/abs/2412.05496 (visited on 08/16/2025). Pre-published.
- [12] Albert Gu and Tri Dao. *Mamba: Linear-Time Sequence Modeling with Selective State Spaces*. May 31, 2024. DOI: 10.48550/arXiv.2312.00752. arXiv: 2312.00752 [cs]. URL: http://arxiv.org/abs/2312.00752 (visited on 09/24/2025). Pre-published.
- [13] HIP Runtime API Reference: Graph Management HIP 6.4.43484 Documentation. URL: https://rocm.docs.amd.com/projects/HIP/en/latest/doxygen/html/group\_\_\_graph.html (visited on 08/21/2025).
- [14] Sara Hooker. "The Hardware Lottery". In: *Communications of The Acm* (Nov. 2021). ISSN: 0001-0782. DOI: 10.1145/3467017. URL: https://doi.org/10.1145/3467017.
- [15] Hugging Face, Inc. *HuggingFace Repositories*. URL: https://huggingface.co/docs/hub/en/repositories (visited on 11/13/2024).
- [16] *IBM/Triton-Dejavu*. International Business Machines, Aug. 5, 2024. URL: https://github.com/IBM/triton-dejavu (visited on 08/16/2025).
- [17] Introducing the IBM Spyre AI Accelerator Chip. IBM Research. Feb. 9, 2021. URL: https://research.ibm.com/blog/spyre-for-z (visited on 09/24/2025).
- [18] [Kernel] Adding Basic Triton JitCache for Triton\_attn by Bringlein · Pull Request #16606 · Vllm-Project/Vllm.

  URL: https://github.com/vllm-project/vllm/pull/16606
  (visited on 09/24/2025).
- [19] [Kernel] Chunk-aligned Mamba2 by Tdoublep · Pull Request #24683 · Vllm-Project/Vllm. GitHub. URL: https://github.com/vllm-project/vllm/pull/24683 (visited on 09/24/2025).
- [20] [Kernel][ROCM] Upstream Prefix Prefill Speed up for vLLM V1 by Maleksan85 · Pull Request #13305 · Vllm-Project/Vllm. GitHub. URL: https://github.com/vllm-project/vllm/pull/13305 (visited on 08/19/2025).
- [21] Woosuk Kwon, Zhuohan Li, Siyuan Zhuang, Ying Sheng, Lianmin Zheng, Cody Hao Yu, Joseph E. Gonzalez, Hao Zhang, and Ion Stoica. *Efficient Memory Management for Large Language Model Serving with PagedAttention*. Sept. 12, 2023. arXiv: 2309.06180 [cs]. URL: http://arxiv.org/abs/2309.06180 (visited on 01/24/2024). Pre-published.
- [22] Bert Maher. *Cache Autotune Timings to Disk*. github.com/triton-lang/triton. Mar. 20, 2025. URL: https://github.com/triton-lang/triton/pull/6261 (visited on 03/20/2025).

- [23] Meta-Llama/Llama-3.1-8B-Instruct · Hugging Face. Dec. 6, 2024. URL: https://huggingface.co/meta-llama/Llama-3.1-8B-Instruct (visited on 08/21/2025).
- [24] Thierry Moreau, Tianqi Chen, Luis Vega, Jared Roesch, Eddie Yan, Lianmin Zheng, Josh Fromm, Ziheng Jiang, Luis Ceze, Carlos Guestrin, and Arvind Krishnamurthy. "A Hardware–Software Blueprint for Flexible Deep Learning Specialization". In: *IEEE Micro* (Sept. 2019). ISSN: 1937-4143. DOI: 10.1109/MM.2019.2928962. arXiv: 1807.04188v3 [cs.LG].
- [25] NVIDIA Corporation. *NVIDIA H100 Tensor Core GPU Architecture Overview*. NVIDIA. 2023. URL: https://resources.nvidia.com/en-us-tensor-core (visited on 11/13/2024).
- [26] Optimizing Triton Kernels ROCm Documentation. URL: https://rocm.docs.amd.com/en/docs-6.1.1/how-to/llm-fine-tuning-optimization/optimizing-triton-kernel.html#auto-tunable-kernel-configurations-and-environment-variables (visited on 08/19/2025).
- [27] PyTorch Community. *PersistentCache*. 2025. URL: https://github.com/pytorch/pytorch/blob/main/torch/\_inductor/codecache.p (visited on 03/10/2025).
- [28] PyTorch Community. *PyTorch Inductor (Algorithm Selection)*. 2025. URL: https://github.com/pytorch/pytorch/tree/main/torch/\_inductor/select\_algorithm. py (visited on 10/03/2025).
- [29] PyTorch community. *Torch.Compile*. URL: https://docs.pytorch.org/docs/stable/generated/torch.compile. html,/generated/torch.compile.html (visited on 08/16/2025).
- [30] Burkhard Ringlein. [RFC] "Autotuner Deja-vu" Save and Restore Autotuner Cache Persistently. github.com/triton-lang/triton. May 28, 2024. URL: https://github.com/triton-lang/triton/issues/4020 (visited on 03/10/2025).
- [31] Burkhard Ringlein, Francois Abel, Dionysios Diamantopoulos, Beat Weiss, Christoph Hagleitner, and Dietmar Fey. "Advancing Compilation of DNNs for FPGAs Using Operation Set Architectures". In: IEEE Computer Architecture Letters (Jan. 2023). ISSN: 1556-6064. DOI: 10.1109/LCA.2022.3227643. URL: https://ieeexplore.ieee.org/document/9984183/.
- [32] Burkhard Ringlein and Thomas Parnell. "Achieving Platform Portability for vLLM by Using Triton Autotuning and Remembering It". In: Ray Summit. Sept. 30, 2024. URL: https://research.ibm.com/publications/achieving-platform-portability-for-vllm-by-using-triton-autotuning-and-remembering-it (visited on 08/19/2025).
- [33] Burkhard Ringlein, Thomas Parnell, and Radu Stoica. GPU Performance Portability Needs Autotuning. July 17, 2025. DOI: 10.48550/arXiv.2505.03780. arXiv: 2505.03780 [cs]. URL: http://arxiv.org/abs/2505.03780 (visited on 08/12/2025). Pre-published.
- [34] ROCm/Aiter. AMD ROCm<sup>TM</sup> Software. URL: https://github.com/ROCm/aiter (visited on 08/16/2025).

- [35] ROCm/Flash-Attention. AMD ROCm<sup>™</sup> Software. URL: https://github.com/ROCm/flash-attention (visited on 08/20/2025).
- [36] Jay Shah, Ganesh Bikshandi, Ying Zhang, Vijay Thakkar, Pradeep Ramani, and Tri Dao. FlashAttention-3: Fast and Accurate Attention with Asynchrony and Low-precision. July 12, 2024. DOI: 10.48550/arXiv.2407.08608. arXiv: 2407.08608. URL: http://arxiv.org/abs/2407.08608 (visited on 10/17/2024). Pre-published.
- [37] vLLM Team. *vLLM V1: A Major Upgrade to vLLM's Core Architecture*. vLLM Blog. url: https://blog.vllm. ai/2025/01/27/v1-alpha-release.html (visited on 08/21/2025).
- [38] Philippe Tillet, H. T. Kung, and David Cox. "Triton: An Intermediate Language and Compiler for Tiled Neural Network Computations". In: *Proceedings of the 3rd ACM SIGPLAN International Workshop on Machine Learning and Programming Languages.* PLDI '19: 40th ACM SIGPLAN Conference on Programming Language Design and Implementation. Phoenix AZ USA: ACM, June 22, 2019. ISBN: 978-1-4503-6719-6. DOI: 10.1145/3315508.3329973. URL: https://dl.acm.org/doi/10.1145/3315508.3329973 (visited on 01/18/2024).
- [39] *TPU Architecture*. Google Cloud. URL: https://cloud.google.com/tpu/docs/system-architecture-tpu-vm (visited on 09/24/2025).
- [40] Triton Community. *Triton*. URL: https://github.com/triton-lang/triton.
- [41] Ashish Vaswani, Noam Shazeer, Niki Parmar, Jakob Uszkoreit, Llion Jones, Aidan N Gomez, Łukasz Kaiser, and Illia Polosukhin. "Attention Is All You Need". In: Advances in Neural Information Processing Systems.

  Curran Associates, Inc., 2017. URL: https://proceedings.
  neurips.cc/paper/2017/hash/3f5ee243547dee91fbd053c1c4a845aa-Abstract.html (visited on 01/24/2024).
- [42] vLLM Community. vLLM. URL: https://github.com/vllm-project/vllm.
- [43] *Vllm-Project/Vllm-Spyre*. vLLM. url: https://github.com/vllm-project/vllm-spyre (visited on 09/24/2025).
- [44] Zihao Ye, Lequn Chen, Ruihang Lai, Wuwei Lin, Yineng Zhang, Stephanie Wang, Tianqi Chen, Baris Kasikci, Vinod Grover, Arvind Krishnamurthy, and Luis Ceze. FlashInfer: Efficient and Customizable Attention Engine for LLM Inference Serving. Jan. 2, 2025. DOI: 10.48550/arXiv.2501.01005. arXiv: 2501.01005 [cs]. URL: http://arxiv.org/abs/2501.01005 (visited on 01/20/2025). Pre-published.

```
@triton.jit
     def kernel_attention(...):
         query_token_idx = tl.program_id(0)
         query_head_idx = tl.program_id(1)
         kv_head_idx = query_head_idx // num_queries_per_kv
         # determine sequence idx and prefix length for current token
         seq_idx = find_seq_idx(query_token_idx, ...)
         prefix_len = calc_prefix_len(seq_idx, query_token_idx, ...)
10
                                                                               10
11
         # load 0 for current query head
                                                                               11
               for current token in current sequence
12
                                                                               12
         Q = tl.load(...)
13
                                                                               13
14
                                                                               14
         # initialize tiled softmax tensors
15
                                                                               15
         res. max. expsum = ...
                                                                               16
16
17
                                                                               17
         # iterate through tiles to attend current token to all
18
                                                                               18
19
               previous tokens
         num_tiles = (prefix_len + BLOCK_SIZE - 1) // BLOCK_SIZE
20
                                                                               20
         for j in range(0, num_tiles):
             # load j-th K and V tiles from KV cache
                 for current KV head and current sequence
23
                                                                               23
             K_j = tl.load(...)
24
                                                                               24
             V_j = tl.load(...)
25
                                                                               25
26
                                                                               26
             # calculate tiled softmax
27
                                                                               27
28
                   update res, max, and expsum
                                                                               28
29
             attn\_result, \ max, \ expsum =
                                                                               29
30
                 tiled_attn(Q, K_j, V_j, attn_result, max, expsum)
                                                                               30
31
                                                                               31
32
         # store result
                                                                               32
         attn result = attn result / expsum
33
                                                                               33
         tl.store(..., attn_result)
34
                                                                               34
35
                                                                               35
36
                                                                               36
     kernel_attention[(tot_query_length, num_query_heads,)](...)
37
                                                                               37
     kernel attention (num segs. num guerv heads.) (...)
```

**Listing 3.** Baseline Triton Attention Kernel.

### **A** Triton Baseline Attention Kernel

Listing 3 presents a high-level, simplified representation of our baseline implementation of an attention kernel in Triton. It assumes that Q, K, and V have already been computed prior to the kernel launch and stored in the KV cache. The KV cache in vLLM is accessed through a block table (analogous to a page table), although this is not directly discussed here, except for references to the parameter BLOCK\_SIZE, which defines the maximum number of tokens stored in a single KV cache block.

As can be seen from lines 36 to 40, the kernel is launched separately for sequences being in prefill and decode phases. In the launch grids, num\_seqs denotes the number of batched sequences, while tot\_query\_length refers to the total number of tokens across these sequences for which attention has to be calculated (i.e., the sum of the query lengths for all batched sequences). For prefill attention, the latter value is obtained by summing the prompt lengths for all sequences, whereas for decode attention, it simply equals the number of sequences in the batch.

The launch grids show that a separate program instance is launched for each combination of a query token and a query head. The program execution begins by identifying the KV head corresponding to the given query head (line

```
@triton.jit
def kernel_attention(...):
    q_block_idx = tl.program_id(0)
    kv_{head_idx} = tl.program_id(1)
    query_head_idx = kv_head_idx * num_queries_per_kv +
        tl.arange(0, num_queries_per_kv)
    # determine sequence idx and prefix length for current token
    seq_idx = find_seq_idx(q_block_idx, ...)
    max_prefix_len = calc_prefix_len(seq_idx, q_block_idx, ...)
    # load 0 for current guery head
          for current token in current sequence
    0 = tl.load(...)
    # initialize tiled softmax tensors
    attn_result, max, expsum = ...
    # iterate through tiles to attend current token to all
          previous tokens
    num_tiles = (max_prefix_len + BLOCK_SIZE - 1) // BLOCK_SIZE
    for j in range(0, num_tiles):
    # load j-th K and V tiles from KV cache
              for current KV head and current sequence
        K_j = tl.load(...)
        V_j = tl.load(...)
        # calculate tiled softmax
              update attn_result, max, and expsum
        attn_result, max, expsum =
            tiled_attn(Q, K_j, V_j, attn_result, max, expsum)
    # store result
    attn result = attn result / expsum
    tl.store(..., attn_result)
kernel_attention[(tot_num_q_blocks, num_kv_heads,)](...)
kernel_attention[(num_seqs, num_kv_heads,)](...)
```

**Listing 4.** Attention Kernel Optimized for Prefill and QGA.

5) and retrieving the sequence index (using a binary search on a tensor storing the accumulated query lengths for all sequences in the batch) and prefix length for the given query token (lines 8 and 9). Next, the corresponding *Q* matrix is loaded (line 13). Lines 15 to 30 implement the tiled softmax computation, following the approach for tiled softmax outlined in section 4.1. The tile size equals BLOCK\_SIZE (i.e., the KV cache block size).

# B Triton Attention Kernel Optimized for Prefill and GQA

Listing 4 presents the Triton implementation of the Q-Block based optimized attention kernel, using a similar high-level and simplified representation as with the baseline kernel. A key difference lies in the launch grid which now includes the total number of Q Blocks across all sequences in the batch, combined with the number of KV heads (line 38). For decode, the total number of Q Blocks equals the number of batched sequences (line 41). Furthermore, now the query head indices are derived from the KV head (lines 5-6). The sequence index is determined in a similar way using a binary search on a tensor storing the accumulated number of Q Blocks for all sequences in the batch (line 9).

```
@triton.jit
     def kernel_attention_par_ts(...):
         q_block_idx = tl.program_id(0)
         kv_head_idx = tl.program_id(1)
         segm_idx = tl.program_id(2)
         query_head_idx = kv_head_idx * num_queries_per_kv +
             tl.arange(0, num_queries_per_kv)
         # determine sequence idx and prefix length for current token
10
11
         sea idx = find sea idx(a block idx, ...)
         max_prefix_len = calc_prefix_len(seq_idx, q_block_idx, ...)
12
13
         # load Q for current guery head
14
              for current token in current sequence
15
        Q = tl.load(...)
16
17
         # initialize tiled softmax tensors
18
19
         segm_res, segm_max, segm_expsum = ...
20
         # iterate through tiles within current segment
         num_tiles = (max_prefix_len + BLOCK_SIZE - 1) // BLOCK_SIZE
23
         for j in range(
24
                segm_idx * tiles_per_segment,
                min((segm_idx + 1) * tiles_per_segment, num_tiles),
26
             # load i-th K and V tiles from KV cache
27
                  for current KV head and current sequence
28
            K_j = tl.load(...)
29
30
            V_j = tl.load(...)
31
             # calculate tiled softmax
32
                  update segm_res, segm_max, and segm_expsum
33
             segm\_result, segm\_max, segm\_expsum =
34
                tiled_attn(Q, K_j, V_j, segm_result, segm_max, segm_expsum)
35
36
37
         # store segment results
         tl.store(..., segm_result)
         tl.store(..., segm_max)
         tl.store(..., segm_expsum)
41
43
    @triton.jit
     def reduce_segments(...):
45
         query_token_idx = tl.program_id(0)
         query_head_idx = tl.program_id(1)
46
47
48
         segm idx = tl.arange(0. num segments)
49
         segm_result = t1.load(...)
50
         segm_max = tl.load(...)
51
52
         segm_expsum = tl.load(...)
53
54
         # calculate overall result by merging and rescaling segment results
55
         attn_result = merge_segments(segm_results, segm_max, segm_expsum)
56
57
         tl.store(..., attn_result)
     reduce_segments[(num_seqs, num_query_heads)](...)
64
```

**Listing 5.** Kernel Supporting Parallel Tiled Softmax.

# C Triton Attention Kernel Using Parallel Tiled Softmax

To extract enough parallelism during decode attention, the iterative processing of tiled softmax within the body of the for loop at line 22 in Listing 4 can be parallelized. Listing 5 presents the corresponding Triton implementation. In this version, a three-dimensional launch grid is used to launch the attention kernel, with the number of segments per Q Block

and KV head combination forming the third dimension (line 61). The segment index (line 5), assigned to each program instance, determines which subset of tiles will be processed iteratively within that program instance (lines 21 to 26). Once the attention computation for a segment is completed, the intermediate results are then stored in memory (lines 37 to 40). After the first kernel finishes, the reduction kernel is launched to compute the final attention output from the segment-level results.