

# Can Asymmetric Tile Buffering Be Beneficial?

Chengyue Wang<sup>1</sup>, Wesley Pang<sup>2</sup>, Xinrui Wu<sup>1</sup>, Gregory Jun<sup>2</sup>, Luis Romero<sup>3</sup>, Endri Taka<sup>4</sup>, Diana Marculescu<sup>4</sup>, Tony Nowatzki<sup>1</sup>, Pranathi Vasireddy<sup>5</sup>, Joseph Melber<sup>5</sup>, Deming Chen<sup>2</sup>, Jason Cong<sup>1</sup>

<sup>1</sup>UCLA <sup>2</sup>UIUC <sup>3</sup>EPFL <sup>4</sup>UT Austin <sup>5</sup>AMD

chengyue@ucla.edu, qpang2@illinois.edu, cong@cs.ucla.edu

## Abstract

General matrix multiplication (GEMM) is the computational backbone of modern AI workloads, and its efficiency is critically dependent on effective tiling strategies. Conventional approaches employ *symmetric tile buffering*, where the buffered tile size of the input  $A$  along the dimension  $M$  matches the output tile size of  $C$ .

In this paper, we introduce *asymmetric tile buffering* (ATB), a simple but powerful technique that decouples the buffered tile dimensions of the input and output operands. We show, for the first time, that ATB is both practical and highly beneficial. To explain this effect, we develop a performance model that incorporates both the benefits of ATB (higher arithmetic intensity) and its overheads (higher kernel switching costs), providing insight into how to select effective ATB tiling factors. As a case study, we apply ATB to AMD's latest XDNA2™ AI Engine (AIE), achieving up to a 4.54× speedup, from 4.8 to 24.6 TFLOPS on mixed-precision BFP16–BF16 GEMM, establishing a new performance record for XDNA2™ AIE.

## 1 Introduction

The rapid progress of artificial intelligence (AI) has placed unprecedented demands on efficient GEMM computations. In GEMM, we perform matrix multiplication  $C = A \times B$ , where  $A \in \mathbb{R}^{M \times K}$ ,  $B \in \mathbb{R}^{K \times N}$ , and  $C \in \mathbb{R}^{M \times N}$ , with  $M$ ,  $N$ , and  $K$  denoting the number of rows of  $A$ , the number of columns of  $B$ , and the reduction dimension, respectively. GEMM efficiency relies critically on effective tiling strategies, which increase arithmetic intensity by reusing data to reduce total data movement to and from memory. As a result, tiling has become a standard practice across CPUs, GPUs, and domain-specific accelerators, enabling high performance while respecting tight memory-bandwidth constraints.

Conventional practice is to employ *symmetric tile buffering*, in which the buffered tile size of input  $A$  along the  $M$  dimension and input  $B$  along the  $N$  dimension are the same as the output tile size of  $C$ . This uniform mapping, defined by the three parameters  $(T_M, T_K, T_N)$ , simplifies scheduling and implementation and is widely adopted in CPU, GPU, and NPU software libraries, as well as hardware accelerator designs [19–21, 23]. Despite its ubiquity, symmetric buffering increases buffer requirements and leaves arithmetic intensity below the theoretical maximum (Fig. 1, right), a limitation that is particularly restrictive when targeting narrow memories such as L1 caches or scratchpads.

However, we show that this symmetry is unnecessary for tiling. In principle, the buffer lifetime of input  $A$  is only as long as it takes to accumulate a single row of  $C$ ; once that row is complete, the corresponding row of  $A$  can be safely evicted. This observation implies that the input and output tile buffers do not need to share the same  $M$  dimension, opening the opportunity to reduce the buffering requirements for  $A$  and allocate more capacity to  $B$  and



Figure 1: Architecture and Roofline: AMD AIE XDNA2™ spatial architecture (left) and the roofline view illustrating the performance impact of kernel and tiling strategies (right).

C. This insight raises the central question of this work: *When and how can asymmetric tile buffering be beneficial?* To answer it:

- We introduce ATB, a general tiling strategy that decouples the buffered  $M$  dimensions of  $A$  and  $C$ , reducing input buffer pressure and increasing arithmetic intensity.
- We develop an analytical performance model that captures both the benefits of ATB (higher reuse and arithmetic intensity) and its costs (more kernel switching overheads), providing principled guidance for selecting tiling factors.
- We present the first case study applying ATB to AMD NPU, achieving a speedup of up to 4.54× over the state of the art.
- We analyze ATB's performance on AIE and show that it improves GEMM throughput by up to 40% even over highly optimized symmetric kernels.

## 2 Background

### 2.1 Reconfigurable XDNA2™ Architecture

Fig. 1 (left) shows the latest AMD XDNA2™ NPU architecture, integrated in Strix [3] and Kraken [2] Point laptops. It features an AIE array consisting of 32 compute cores, eight shim cores, and eight memory cores. Those cores are reconfigurable blocks connected with FPGA-like adaptable interconnects [1]. Each compute core integrates a VLIW processor, 64 KB of local L1 memory, DMA engines, register files, and two input and two output stream channels. Inter-core connections are reconfigurable through stream channels, enabling flexible communication patterns such as broadcasting from a single producer core to multiple consumer cores. Each memory core provides 512 KB of L2 memory along with six input and six output stream channels. To access the flexible compute and memory hierarchy, toolkits such as IRON [14] provide explicit control over data movement, allowing developers to configure stream interfaces

and data movement across cores in Python while linking C++ AIE kernels for programming the VLIW processor.

## 2.2 Challenges of Data Movement and Tiling

Arithmetic intensity measures the ratio of computation to data movement and is critical for overcoming memory bottlenecks (Fig. 1, right) [13, 22]. For GEMM, arithmetic intensity increases with tile size. Each AIE compute core provides only two input streams, typically used for operands  $A$  and  $B$ , leaving no channel to bring partial sums back for further accumulation. Moreover, memory cores lack accumulator units, so partial sums cannot be stored in L2 or reloaded to the compute cores. As a result, each output tile of  $C$  must be fully accumulated in L1 before being written out, and the L2 output tile size exactly equals the aggregate of L1 tiles across the used compute cores. For example, if each core computes a  $64 \times 64$  BF16 tile (8 KB), then 32 cores collectively produce 256 KB of output in L1, so the L2 output tile size is also 256 KB. The L1 output tile size is therefore critical: it directly determines the L2 output tile size, which directly governs the reuse of off-chip input data.

Given this, asymmetric tile buffering in L1 memory offers a promising way to increase arithmetic intensity and improve off-chip data reuse. However, prior AIE studies [10, 16–18, 23, 24] have adopted only symmetric tile buffering at both L1 and L2, with the tiling space defined solely by the three parameters ( $T_M$ ,  $T_K$ ,  $T_N$ ) at each memory level. Although simulation-based design space exploration studies [11, 15] have explored the idea of asymmetric tile buffering, they neither analyze its overheads on real hardware nor its impact on kernel performance. Moreover, to the best of our knowledge, CPU and GPU implementations also rarely adopt such strategies, due to both hardware constraints and the complexity of managing data movement across multiple memory levels. For example, widely used GEMM frameworks such as BLIS [20] and CUTLASS [19] also rely on symmetric tile buffering in L1 caches.

## 2.3 Challenges of Kernel Design

As shown in Fig. 1 (right), an efficient kernel design drives the bound of attainable performance. Inside a kernel, the fundamental unit of computation is the *microkernel*: a fully unrolled inner loop that performs fused multiply–accumulate (MAC) operations on small tiles of data held in registers. Running on a 7-way VLIW processor, AIE microkernel performance depends on how efficiently the VMAC instruction is scheduled at compile time.

While most prior AIE studies [10, 16–18, 23, 24] focus primarily on data movement between L1, L2, and off-chip memory, they generally lack a systematic discussion of AIE microkernel optimization. Additionally, although microkernels can be optimized with compiler pragmas such as chess::unroll\_loop in the Chess compiler for AIE, analytical modeling of the underlying execution is essential to guide programmers in their effective use of pragmas. Furthermore, mixed precision complicates instruction pipelines and makes it difficult for the compiler to optimize without manual code refactoring. This paper presents an analytical model to guide manual kernel optimization.

Between kernels, additional performance challenges arise from execution gaps introduced during kernel switching. These gaps result from inherent overheads such as kernel initialization, pipeline



Figure 2: Comparison of symmetric (left) and asymmetric (right) tile buffering. Asymmetric tiles ( $T_{M_A}$ ,  $T_{M_C}$ ,  $T_K$ ,  $T_N$ ) increase arithmetic intensity but also switching overhead.

resets, and synchronization delays. They accumulate across the many kernels invoked in a tiled GEMM, widening the difference between microkernel-level attainable performance and core-level attainable performance.

## 2.4 BFP16 Data Type

XDNA2™ introduces native support for the Block Floating Point (BFP16) format [6], a quantized precision that balances efficiency and accuracy. Unlike BF16, where each value carries its own exponent, BFP16 shares one 8-bit exponent across a block of 8 values, each storing only an 8-bit mantissa. This packs 8 elements into 9 bytes, represented in AIE as bfp16ebs8 [4]. Each compute core executes BFP16 on a dedicated  $8 \times 8 \times 8$  MAC unit, sustaining 512 MACs per cycle. A VMAC instruction multiplies two  $8 \times 8$  BFP16 vector registers and accumulates into an  $8 \times 8$  FP32 accumulator register. At 1.8 GHz [5], this delivers 1.84 TFLOPS per core and 58.8 TFLOPS across the whole 32-core array. However, the current best mixed-precision BF16-BFP16 GEMM result on an AMD Strix Point laptop is only 4.8 TFLOPS [9].

## 3 Modeling Arithmetic Intensity

In GEMM, symmetric tile buffering uses  $(T_M, T_K, T_N)$  to define tile sizes for  $A$ ,  $B$ , and  $C$ , where  $A_{\text{tile}} \in \mathbb{R}^{T_M \times T_K}$ ,  $B_{\text{tile}} \in \mathbb{R}^{T_K \times T_N}$ , and  $C_{\text{tile}} \in \mathbb{R}^{T_M \times T_N}$ . Within each tile, consider the canonical loop order for  $m$  in  $T_M$  { for  $n$  in  $T_N$  { for  $k$  in  $T_K$  {} }}. For a fixed  $m$ , the row  $A_{\text{tile}}[m, :]$  is reused across all  $n$  until the corresponding row  $C_{\text{tile}}[m, :]$  has been fully accumulated, after which that row of  $A$  is no longer needed. In contrast, each row of  $C_{\text{tile}}$  must remain live for the entire reduction over  $K$  (full problem or next-level tile size). Because the lifetimes of  $A$  and  $C$  differ along the dimension  $M$ , their buffered extents do not need to be equal. This enables *asymmetric tile buffering*, parameterized by  $(T_{M_A}, T_{M_C}, T_K, T_N)$ , where  $T_{M_A}$  buffers rows of  $A$  and  $T_{M_C}$  buffers rows of  $C$  (Fig. 2). In this scheme,  $A_{\text{tile}} \in \mathbb{R}^{T_{M_A} \times T_K}$ ,  $B_{\text{tile}} \in \mathbb{R}^{T_K \times T_N}$ , and  $C_{\text{tile}} \in \mathbb{R}^{T_{M_C} \times T_N}$  ( $T_{M_C} \geq T_{M_A}$ ).

### 3.1 Arithmetic Intensity with ATB

Let  $a, b, c > 0$  be per-element byte costs (e.g., 2 for BF16 and 1.25 for BFP16) for  $A$ ,  $B$ , and  $C$ , and define the asymmetry ratio  $\rho = T_{M_C}/T_{M_A}$  ( $\rho \geq 1$ ). We consider an output-stationary schedule: each output tile is fully iterated  $N_{\text{iterk}} = K/T_K$  times (reading  $A$ ,  $B$  each time) and written once at the end, where  $K$  is the problem size

along the reduction dimension of the GEMM. For a tile of size  $(T_{MA}, T_{MC}, T_K, T_N)$ , the arithmetic intensity is

$$\begin{aligned} AI_\rho &= \frac{\text{Operations}}{\text{Accesses}} = \frac{\text{Operations per output tile}}{\text{Accesses per output tile}} \\ &= \frac{2 T_{MC} T_K T_N N_{iterk}}{a T_{MC} T_K N_{iterk} + b T_K T_N N_{iterk} + c T_{MC} T_N} \\ &= \frac{2 K T_{MC} T_N}{a T_{MC} K + b K T_N + c T_{MC} T_N} = \frac{2}{\frac{a}{T_N} + \frac{b}{T_{MC}} + \frac{c}{K}}, \end{aligned}$$

which is bounded by the available buffer capacity  $S$ , assuming double buffering of  $A$ ,  $B$  and single buffering of  $C$ :

$$\frac{2a}{\rho} T_{MC} T_K + 2b T_K T_N + c T_{MC} T_N \leq S. \quad (1)$$

As shown above,  $AI_\rho$  increases monotonically with  $T_{MC}$  and  $T_N$ . Importantly,  $AI_\rho$  does not depend explicitly on  $T_K$  or  $\rho$ ; their effects appear only through the buffer constraint (Eq. 1). The smaller  $T_K$  and the larger  $\rho$  reduce the buffer cost associated with  $T_{MC}$  and  $T_N$ . This frees more buffer space, allowing larger  $(T_{MC}, T_N)$  values to be used, which in turn increases  $AI_\rho$ . The optimal strategy to maximize  $AI_\rho$  is to choose  $T_K$  as small as possible and  $\rho$  as large as possible. In the idealized case, this corresponds to  $T_K = 1$  and  $T_{MA} = 1$ , with  $\rho = T_{MC}$ .

### 3.2 From Single Core to Whole Array

In XDNA™, computation is organized in three levels of tiling: from off-chip memory to L2, from L2 to L1, and from L1 to registers. As discussed in Section 2.2, the L2 output tile is the aggregation of the L1 output tiles produced by all compute cores. From L2 to L1, one input tile in a memory core can be broadcast to multiple compute cores; specifically, L2  $A$  tiles are broadcast across columns, and L2  $B$  tiles are broadcast across rows. We do not apply asymmetric tiling at the L2 level, since the L2 tile size is limited by the L1 tile size and the available L2 memory capacity is sufficient to hold symmetric tiles. For an array of  $N_{aie\_row} = 4$  rows and  $N_{aie\_col} = 8$  columns, the effective L2 tile sizes are

$$T_M^{L2} = N_{aie\_row} \cdot T_{MC}^{L1} = 4 T_{MC}^{L1}, \quad T_N^{L2} = N_{aie\_col} \cdot T_N^{L1} = 8 T_N^{L1}, \quad T_K^{L2} = T_K^{L1}.$$

The modeling of arithmetic intensity for data moved from off-chip for the whole array is therefore:

$$AI_{array} = \frac{2}{\frac{a}{T_N^{L2}} + \frac{b}{T_M^{L2}} + \frac{c}{K}} = \frac{2}{\frac{a}{8 T_N^{L1}} + \frac{b}{4 T_{MC}^{L1}} + \frac{c}{K}}, \quad (2)$$

which is constrained by the L1 capacity  $S_{L1} = 63$  KB per core:

$$\frac{2a}{\rho} T_{MC}^{L1} T_K^{L1} + 2b T_K^{L1} T_N^{L1} + c T_{MC}^{L1} T_N^{L1} \leq S_{L1} = 63 \text{ KB}. \quad (3)$$

Similar to the analysis of  $AI_\rho$ , the optimal strategy to maximize  $AI_{array}$  is to minimize  $T_K^{L1}$  and maximize  $\rho$ .

### 3.3 Limits of AI-Only Modeling

The arithmetic intensity model predicts that performance is maximized by minimizing  $T_K$  and maximizing  $\rho$ . In practice, this overlooks microkernel execution: a very small  $T_K$  introduces bubbles in the compute pipeline and inflates prolog/epilog costs. To capture



**Figure 3: Example schedule of an accumulation chain.** Assume 3-cycle latency per instruction, and loading one input takes 1 instruction, accumulator load/store takes 2.

these effects, we must incorporate *compute core efficiency*, which itself depends on the tiling factors. We refine this in the next section 5, specifically focusing on AIE's VLIW cores as a case study.

## 4 Modeling Instruction-Level Parallelism

A key characteristic of the VLIW architecture is its deterministic instruction schedule, which is fixed statically at compile time. This property enables performance to be estimated directly from the instruction sequence itself. While runtime hazards such as memory stalls or lock acquisition can introduce variability, the static instruction schedule provides a clear representation of the kernel's peak potential for our analysis. The performance of a compute core is evaluated by how closely its achieved throughput approaches the theoretical maximum of the hardware, which can be quantified by:

$$\text{Efficiency} = \frac{\text{Achieved Throughput (MACs/cycle)}}{\text{Peak Throughput (MACs/cycle)}}. \quad (4)$$

The key challenge in kernel design is to exploit instruction-level parallelism (ILP) to hide *data movement latency between L1 memory and registers* and maximize the *issue rate of VMAC instructions*. For a GEMM kernel, the most critical VLIW slots are:

- **1 Vector Arithmetic Unit:** Executes the VMAC instruction.
- **2 Vector Load Units:** Load vector operand from L1 memory.
- **1 Vector Store Unit:** Writes the result block back to memory.

From an instruction-scheduling perspective, maximizing efficiency corresponds to maximizing utilization of the vector arithmetic unit. Thus, we evaluate kernel schedules using the ratio of issued VMAC instructions to the total number of cycles in the code region of interest, typically the microkernel.

### 4.1 Accumulation Chain in Microkernel

In general, instruction scheduling of arbitrary DAGs under resource constraints is NP-hard [12] and admits no closed-form solution. This section narrows the scope to the GEMM microkernel. Within this setting, the instruction schedule can be well approximated using simple analytical bounds to guide microkernel optimization.

The basic unit in our model is the *accumulation chain*, which represents the sequence of instructions required to update one accumulator operand of a VMAC instruction without spilling the accumulator register. All instructions are pipelined with a throughput of one per cycle, but may have varying latency. We decompose the instruction scheduling into three phases:

- **Prolog:** The period before the first VMAC instruction can be issued, during which operand registers are loaded (e.g., loads of  $A$ ,  $B$ , and  $C$  in Fig. 3). Assume there are  $M$  instruction types, each with

latency  $\ell_i$  and a count  $n_i$ , to be issued on  $U_{LD}$  vector load slots. The minimum time for all operands to become ready is

$$T_{\text{prolog}} \geq T_{\text{load}} = \max_{i=1,\dots,M} \left( \ell_{(i)} + \left\lceil \frac{S_{(i)}}{U_{LD}} \right\rceil - 1 \right), \quad (5)$$

where instruction types are sorted in descending order of latency, i.e.,  $\ell_{(1)} \geq \ell_{(2)} \geq \dots$ , and  $S_{(i)} = \sum_{k=1}^i n_{(k)}$  is the cumulative number of loads from the first  $i$  operand types. Intuitively, the formula assumes all loads are independent and can be reordered arbitrarily. The longest-latency loads are issued first so their delay is hidden as much as possible, while shorter-latency loads can be issued later and overlap with them. The final time is then determined by whichever group of loads (long or short) finishes last. In Fig. 3, with 3 instruction types of latency  $\ell = 3$ , counts  $n_C = 2$ ,  $n_A = n_B = 1$ , and  $U_{LD} = 2$ , we have  $S_{(1)} = 2$ ,  $S_{(2)} = 3$ ,  $S_{(3)} = 4$ . Substituting gives  $T_{\text{load}} = \max(3 + \lceil 2/2 \rceil - 1, 3 + \lceil 3/2 \rceil - 1, 3 + \lceil 4/2 \rceil - 1) = \max(3, 4, 4) = 4$  cycles.

- **Steady State:** The region where VMAC instructions are interleaved with operand loads and scheduled periodically with an *initiation interval* of  $II$  cycles. For a single accumulation chain,  $II$  is constrained by the RAW dependency distance on the accumulator operand and the load bandwidth, giving

$$II_{\text{single}} = \max \left( P, \left\lceil \frac{R_{\text{load}}}{U_{LD}} \right\rceil \right), \quad (6)$$

where  $P$  is the VMAC pipeline depth, and  $R_{\text{load}}$  is the number of operand loads per VMAC. To hide the latency  $P$  and improve VMAC throughput, a microkernel typically employs multiple parallel accumulation chains interleaved in a software-pipelined schedule. With  $C$  parallel chains interleaved, the effective  $II$  is:

$$II_{\text{parallel}} = \frac{\max(P+1-C, \left\lceil \frac{R_{\text{load}}}{U_{LD}} \right\rceil)}{C}. \quad (7)$$

The steady-state cost is then lower-bounded by:

$$T_{\text{steady}} \geq II_{\text{parallel}} \cdot (N_{\text{accum}} - C), \quad (8)$$

where  $N_{\text{accum}}$  is the number of accumulator updates (i.e., the length of the accumulation chain).

- **Epilog:** The region after the last operand load where the final VMAC instructions complete and the accumulator results are written back to memory. For  $C$  accumulation chains, it consists of  $C$  compute-store pairs:

$$T_{\text{epilog}} \geq (\ell_{\text{vmac} \rightarrow \text{vst}} + \ell_{\text{store}} + n_{\text{store}} - 1) + (C - 1), \quad (9)$$

where  $\ell_{\text{vmac} \rightarrow \text{vst}}$  is the latency from the last VMAC to when its store can be issued,  $\ell_{\text{store}}$  is the store instruction latency, and  $n_{\text{store}}$  is the number of store instructions required per accumulator.

## 4.2 Modeling of Total Latency

*Double Buffering of Input Registers.* In our steady-state modeling, we considered only the load bandwidth and assumed that sufficient registers are available to hold parallel loads, fully hiding the load latency. This assumption is reasonable in practice, as compilers often perform register renaming and allocate disjoint registers to achieve this effect automatically. When the compiler cannot apply such optimizations effectively, AIE programmers can manually pin variables to registers using the `chess_storage(register)` attribute. In this case, ping-pong buffering of input registers becomes an important technique. By alternating between two disjoint register sets (*Ping/Pong*), the microkernel can preload operands in one set



Figure 4: Comparison of single-buffered (left) vs. double-buffered (right) input register allocation.



Figure 5: GEMM microkernel optimizations: top-left, input sharing across a  $2 \times 2$  chain cluster; bottom-left, prolog-epilog overlap across clusters; right, corresponding code.

while the other is consumed by VMAC instructions. Fig. 4 illustrates this: the left schedule repeatedly reloads into the same registers, creating RAW dependencies between loads and compute, while the right employs double buffering so that loads overlap with compute.

*Sharing Input Across Chain Cluster.* To improve the microkernel performance, it is important to employ multiple parallel accumulation chains so that the effective  $II$  in the steady state approaches 1, as modeled in Eq. (7). In practice, these chains are not fully independent: they often share common input operands from matrix  $A$  or  $B$ , which we refer to as a *chain cluster*. When the microkernel shares input registers across a chain cluster, the number of loads per VMAC is reduced, thus alleviating pressure on the load bandwidth and lowering  $R_{\text{load}}$  in Eq. (7). This reuse also reduces input register usage, creating more opportunities for double buffering to hide load latency. The cluster size is bounded by the number of accumulator operands that can be held simultaneously in the accumulator register file. Fig. 5 (top-left) illustrates this optimization, where shared inputs feed multiple chains simultaneously; for example, a  $2 \times 2$  chain cluster reduces the load count from 8 to 4.

*Overlapping Successive Chain Clusters.* Even if the optimal  $II = 1$  is achieved in steady state using a chain cluster, the total latency still includes the prolog and epilog phases. If successive chain clusters run sequentially, with each starting only after the previous

completes, the total latency is bounded by

$$L_{\text{total}} \geq (T_{\text{prolog}} + T_{\text{steady}} + T_{\text{epilog}}) \cdot N_{\text{clusters}}. \quad (10)$$

However, the chain length is algorithmically constrained by the  $K$  dimension of the tile in L1 memory. When  $K$  is small, the prolog and epilog overheads are significant relative to  $T_{\text{steady}}$ . To address this, successive clusters can overlap in a software-pipelined schedule: while one cluster is draining its epilog, the next cluster can begin its prolog. If the overlap ideally hides the boundary costs, the latency of  $N_{\text{clusters}}$  chain clusters, each containing  $C$  chains, is bounded by

$$L_{\text{total}} \geq T_{\text{prolog}} + (T_{\text{steady}} + II_{\text{parallel}} \cdot C) \cdot N_{\text{clusters}} + T_{\text{epilog}}. \quad (11)$$

Fig. 5 (bottom-left) illustrates this effect, where overlapping reduces the gaps between chain clusters for higher utilization.

### 4.3 Towards BFP16 and Mixed Precision

Since public documentation does not provide precise register behavior, we approximate parameters using microbenchmarks with small VMAC kernels and explicit register binding. For the bfp16ebs8 VMAC instruction, we observe five accumulator registers, each holding  $8 \times 8$  FP32 elements, with a pipeline distance of  $P = 3$  cycles. Operand loads have a latency of 8 cycles; because bfp16ebs8 is unaligned, a VLDA .POP precedes the load whenever the access is to a non-consecutive address. The latency from VMAC to store availability is  $\ell_{\text{vmac} \rightarrow \text{vst}} = 6$  cycles, and writing back one accumulator register requires two store instructions with latency  $\ell_{\text{store}} = 2$  cycles.

Architectural constraints can reduce BFP16 kernel performance below the analytical bounds. The instruction pipeline favors sequential L1 access, because unaligned BFP16 addresses require extra instructions and may introduce bubbles. In addition, conversions between BFP16 and BF16 occupy accumulator registers, and the small accumulator file further limits concurrent load-store operations, which creates false dependencies between successive clusters. Our model helps designers reason about whether performance loss stems from hardware limitations or compiler scheduling, and guides manual optimizations such as unrolling, register-level tiling, register binding, and double buffering.

## 5 Tiling Tradeoffs Towards Final Performance

The final performance of the array is jointly bounded by *data movement* and *compute capability*:

$$\text{Perf}_{\text{array}} = \min(\text{AI}_{\text{array}} \cdot BW_{\text{offchip}}, \text{Eff}_{\text{core}} \cdot \text{Perf}_{\text{core}}^{\text{peak}} \cdot N_{\text{core}}), \quad (12)$$

where  $BW_{\text{offchip}}$  is the available off-chip bandwidth, which is around 65 GB/s for the Strix Point laptop.  $N_{\text{core}} = 32$  is the number of compute cores,  $\text{Perf}_{\text{core}}^{\text{peak}} = 1.84$  TFLOPS is the per-core peak performance, and  $\text{Eff}_{\text{core}}$  is the compute core efficiency.

*Impact of  $T_K^{L_1}$ .* A larger  $T_K^{L_1}$  increases the accumulation chain length  $N_{\text{accum}}$  in the microkernel, allowing a longer steady state in Eq. 11 or Eq. 10. We use Eq. 10 for conservativeness. Combining it with Eq. 8, we model the microkernel efficiency  $\text{Eff}_{\text{micro}}$  as

$$\text{Eff}_{\text{micro}} = \frac{N_{\text{clusters}} \cdot N_{\text{accum}}}{(T_{\text{prolog}} + II \cdot (N_{\text{accum}} - C) + T_{\text{epilog}}) \cdot N_{\text{clusters}}}.$$

Since  $N_{\text{accum}}$  scales with  $T_K^{L_1}$ , we can approximate

$$\text{Eff}_{\text{micro}}(T_K^{L_1}) = \frac{\eta T_K^{L_1}}{\epsilon + \eta II T_K^{L_1}}. \quad (13)$$

**Table 1: Microkernel-level and core-level performance under different  $T_K$  and  $\rho$ .**

| $T_K$ | $\rho$ | Perf <sub>micro</sub> (Tflops) | Eff <sub>micro</sub> | Perf <sub>core</sub> (Tflops) | Eff <sub>core</sub> |
|-------|--------|--------------------------------|----------------------|-------------------------------|---------------------|
| 8     | 1      | 0.36                           | 0.2                  | 0.290                         | 0.156               |
| 8     | 2      | 0.36                           | 0.2                  | 0.277                         | 0.149               |
| 8     | 4      | 0.36                           | 0.2                  | 0.250                         | 0.134               |
| 8     | 8      | 0.36                           | 0.2                  | 0.216                         | 0.116               |
| 16    | 1      | 0.66                           | 0.36                 | 0.529                         | 0.284               |
| 16    | 2      | 0.66                           | 0.36                 | 0.501                         | 0.272               |
| 16    | 4      | 0.66                           | 0.36                 | 0.474                         | 0.257               |
| 16    | 8      | 0.66                           | 0.36                 | 0.359                         | 0.195               |
| 32    | 2      | 0.76                           | 0.41                 | 0.615                         | 0.331               |
| 32    | 4      | 0.75                           | 0.41                 | 0.574                         | 0.312               |
| 32    | 8      | 0.77                           | 0.42                 | 0.562                         | 0.306               |
| 64    | 4      | 1.16                           | 0.63                 | 0.920                         | 0.511               |
| 64    | 8      | 1.16                           | 0.63                 | 0.867                         | 0.482               |

This expression captures diminishing returns:  $\text{Eff}_{\text{micro}}$  grows linearly for  $T_K^{L_1} \ll \frac{\epsilon}{\eta II}$ , but saturates to  $\frac{1}{II}$  as  $T_K^{L_1}$  increases.

*Impact of asymmetry.* Between consecutive microkernels, we observe a fixed overhead for switching microkernels, measured as  $\delta \approx 50$  cycles in tracing. For a core to finish an output tile ( $T_{M_C}^{L_1} T_N^{L_1}$ ), in the symmetric case it requires  $\frac{K}{T_K^{L_1}}$  switches, and  $\rho \cdot \frac{K}{T_K^{L_1}}$  switches when using asymmetric tile buffering. Its latency is

$$T_{\text{asym}} = \frac{2 T_{M_C}^{L_1} K T_N^{L_1}}{\text{Perf}_{\text{core}}^{\text{peak}} \cdot \text{Eff}_{\text{micro}}} + \delta \cdot \rho \cdot \frac{K}{T_K^{L_1}}. \quad (14)$$

*Combined efficiency.* The resulting effective core efficiency is

$$\text{Eff}_{\text{core}} = \frac{2 T_{M_C}^{L_1} K T_N^{L_1}}{T_{\text{asym}}} = \frac{1}{\frac{1}{\text{Eff}_{\text{micro}}} + \frac{\delta \rho \text{Perf}_{\text{core}}^{\text{peak}}}{2 T_{M_C}^{L_1} T_N^{L_1} T_K^{L_1}}}. \quad (15)$$

The optimal strategy to maximize  $\text{Eff}_{\text{core}}$  is to choose  $T_K$  as large as possible and  $\rho$  as small as possible, which is the opposite of the strategy that maximizes  $\text{AI}_{\rho}$ . The final performance (Eq. 12) therefore emerges from a *tradeoff between maximizing arithmetic intensity and sustaining high compute efficiency*.

## 6 Evaluation

### 6.1 Experimental Setup

Our experiments use AMD Ryzen™ AI 9 HX 370 (Strix Point NPU). We program the NPU using the IRON API and the MLIR-AIE toolchain [14] to measure performance under three precision configurations: **Config 1:** BF16–BFP16 mixed-precision. Input A and output C are in BF16, weights B are in BFP16, and accumulation is performed in BF16. **Config 2:** A, B, and C are all stored in BFP16, with accumulation also performed in BFP16. **Config 3:** A, B, and C are all stored in BFP16, with accumulation performed in BF16.

### 6.2 Kernel Performance

Table 1 summarizes both microkernel-level performance ( $\text{Perf}_{\text{micro}}$ ,  $\text{Eff}_{\text{micro}}$ ) and full core-level performance ( $\text{Perf}_{\text{core}}$ ,  $\text{Eff}_{\text{core}}$ ) across different combinations of  $T_K$  and  $\rho$ , evaluated under precision Config 1 with L1 tile sizes  $T_{M_A} = 128$  and  $T_N = 128$ . For  $T_K = 64$ , the

**Table 2: Whole-array BF16/BFP16 GEMM speedup across configurations.**

| Precision | Source   | Problem size (M×K×N) | L1 tile size ( $T_{MC}^{L_1} \times T_K^{L_1} \times T_N^{L_1}$ ) | $\rho$ | Used Buffer (KB) | Buffer if $\rho = 1$ | Perf <sub>core</sub> (Tflops) | Compute-bound (Tflops) | AI <sub>array</sub> (op/B) | Memory-bound (Tflops) | Perf <sub>array</sub> (Tflops) |
|-----------|----------|----------------------|-------------------------------------------------------------------|--------|------------------|----------------------|-------------------------------|------------------------|----------------------------|-----------------------|--------------------------------|
| BF16-INT4 | Ryzen AI | 2048×4096×4096       | Unknown                                                           | 1      | Unk              | Unk                  | Unk                           | Unk                    | Unk                        | Unk                   | 10.4                           |
| Config 1  | MLIR-AIE | 8192×4224×4096       | 64×88×64                                                          | 1      | 50.4             | 50.4                 | 0.32                          | <b>10.2</b>            | 216                        | 14.1                  | 4.8 (1.00×)                    |
|           | Ours     | 2048×4096×2048       | 64×64×128                                                         | 1      | 54.5             | 54.5                 | 0.95                          | 30.5                   | 273                        | <b>17.8</b>           | 17.3 (3.61×)                   |
|           | Ours     | 2048×4480×2048       | 64×224×64                                                         | 4      | 57.4             | 111                  | 0.96                          | 30.7                   | 217                        | <b>14.1</b>           | 12.3 (2.56×)                   |
|           | Ours     | 4096×4096×2048       | 128×64×128                                                        | 4      | 60.3             | 91                   | 0.95                          | 30.4                   | 410                        | <b>26.6</b>           | 24.3 (4.54×)                   |
| Config 2  | MLIR-AIE | 7680×4096×8192       | 96×128×64                                                         | 1      | 58.5             | 58.5                 | 0.38                          | <b>12.2</b>            | 333                        | 21.7                  | 10.2 (1.00×)                   |
|           | Ours     | 4096×4096×2048       | 128×64×128                                                        | 1      | 54               | 54                   | 0.84                          | <b>26.9</b>            | 504                        | 32.8                  | 25.9 (2.54×)                   |
|           | Ours     | 4096×4096×2048       | 256×64×128                                                        | 8      | 58.5             | 90                   | 0.73                          | <b>23.3</b>            | 728                        | 47.3                  | 22.3 (2.19×)                   |
| Config 3  | Ours     | 3072×4096×1536       | 192×128×96                                                        | 6      | 56.3             | 101                  | 1.06                          | <b>33.9</b>            | 562                        | 36.5                  | 31.3 (3.07×)                   |
|           | Ours     | 3072×4096×2048       | 96×64×128                                                         | 1      | 57               | 57                   | 1.00                          | <b>32</b>              | 418                        | <b>27.2</b>           | 27.0 (2.65×)                   |
|           | Ours     | 4096×4096×2048       | 128×64×128                                                        | 4      | 56.5             | 70                   | 0.90                          | <b>28.8</b>            | 504                        | 32.8                  | 28.5 (2.79×)                   |

configurations  $\rho = 1$  and  $\rho = 2$  exceed the available 63 KB L1 memory capacity, and for  $T_K = 32$ ,  $\rho = 1$  also exceeds it.

Overall, the measured results closely follow the analytical trends predicted in Section 5. As expected, microkernel efficiency  $\text{Eff}_{\text{micro}}$  depends only on  $T_K$  and is almost invariant across  $\rho$ . For a fixed  $T_K$ , increasing  $\rho$  consistently reduces core efficiency  $\text{Eff}_{\text{core}}$  because larger asymmetry increases the number of microkernel switches and amplifies launch overheads. For example, at  $T_K = 8$ ,  $\text{Eff}_{\text{core}}$  decreases by 25.6% (from 0.156 to 0.116) as  $\rho$  increases from 1 to 8. At  $T_K = 64$ , the reduction is much smaller, only 5.8% (from 0.511 to 0.482), because a larger  $T_K$  yields longer kernel execution and therefore better amortization of the switching cost. In contrast, for a fixed  $\rho$ , a larger  $T_K$  improves both the efficiency of the microkernel and the core by extending the steady state and reducing the relative impact of the prolog/epilog cycles. At  $\rho = 4$ ,  $\text{Eff}_{\text{core}}$  increases from 0.134 at  $T_K = 8$  to 0.511 at  $T_K = 64$ , a 3.8× improvement. These results match the theoretical prediction that core efficiency improves with larger  $T_K$  while increasing  $\rho$  reduces efficiency.

Additionally, MLIR-AIE [9] achieves only 0.32 TFLOPS on a single core, while our best design reaches 0.92 TFLOPS—a 2.88× improvement. Their approach relies primarily on compiler-driven optimization, which is ineffective for BFP16, while our kernel design applies manual scheduling techniques such as unrolling, register-level tiling, explicit register binding, and double buffering.

**Table 3: Performance of whole-array GEMM under varying matrix problem sizes.**

| M    | K    | N    | Performance (Tflops) |
|------|------|------|----------------------|
| 1024 | 2048 | 2048 | 19.2                 |
| 1024 | 4096 | 2048 | 21.8                 |
| 1024 | 8192 | 2048 | 23.3                 |
| 2048 | 2048 | 2048 | 21.8                 |
| 2048 | 4096 | 2048 | 23.3                 |
| 2048 | 8192 | 2048 | 24.0                 |
| 4096 | 2048 | 2048 | 23.2                 |
| 4096 | 4096 | 2048 | 24.3                 |
| 4096 | 8192 | 2048 | 24.6                 |

### 6.3 Whole Array Performance

Table 2 reports full-array GEMM throughput across three configurations, comparing our kernels against both the AMD MLIR-AIE [9]

and the BF16-INT4 backend used in AMD Ryzen AI [8] (DynamicDispatch [7]). For each problem size, the table lists the measured single-core throughput, the compute-bound limit ( $\text{Perf}_{\text{core}} \times 32$  cores), the arithmetic intensity, and the corresponding memory-bound limit ( $\text{AI}_{\text{array}} \times \text{off-chip bandwidth}$ ), as defined in Eq. 12.

Across all configurations, our kernels substantially outperform the state-of-the-art MLIR-AIE and Ryzen AI baselines. In Config 1, our best design reaches 24.3 TFLOPS, a 4.54× speedup over MLIR-AIE’s 4.8 TFLOPS. Config 2 shows similar gains, with throughput up to 31.3 TFLOPS (3.07× over the 10.2 TFLOPS baseline). Config 3 achieves 28.5 TFLOPS, exceeding the MLIR-AIE Config 2 by 2.79× while simultaneously offering higher accumulation precision. Compared to the Ryzen AI BF16-INT4 GEMM (10.4 TFLOPS), our Config 1 BF16-BFP16 GEMM delivers 24.3 TFLOPS, providing a 2.4× improvement while using higher precision.

We can observe the impact of ATB in Table 2. In Config 1, all three configurations are memory-bound. ATB enables the 128×64×128 tile, which would require 91 KB and be infeasible under symmetric buffering. With ATB, this configuration becomes valid and improves throughput from 17.3 to 24.3 TFLOPS—a 40% gain. Fixing  $\rho = 4$  also highlights the effect of  $T_K$ : the 64×224×64 kernel achieves only 56% of the throughput of the 128×64×128 tile, since larger  $T_K$  forces smaller ( $T_{MC}, T_N$ ) and reduces AI. These observations match the theoretical prediction that memory-bound kernels benefit from smaller  $T_K$  and larger  $\rho$ , which enable larger ( $T_{MC}, T_N$ ).

Config 2 is mostly compute-bound and shows the trade-off between AI and core efficiency. Although 256×64×128 has higher AI, its lower core efficiency limits performance; the 192×128×96 tile, with larger  $T_K$ , reaches 31.3 TFLOPS. ATB is again essential: without it, this configuration would require 101 KB and would not fit in L1, and ATB provides a 21% throughput gain over the symmetric case. Config 3 further shows that ATB can move GEMM from memory-bound to compute-bound. The final performance does not increase noticeably because the kernel is already compute-bound. In summary, when memory-bound, ATB helps increase arithmetic intensity through larger ( $T_{MC}, T_N$ ); when compute-bound, ATB helps improve core efficiency through larger  $T_K$ .

Finally, Table 3 reports the performance of our Config 1 ATB kernel ( $\rho = 4$ , 128×64×128 L1 tile) across different problem dimensions. Performance scales smoothly with matrix size, achieving up

to 24.6 TFLOPS on large GEMMs and remaining above 19.2 TFLOPS for smaller problems.

## 7 Conclusion

In this work, we introduced asymmetric tile buffering (ATB), a general tiling strategy that reduces buffer pressure and improves arithmetic intensity beyond conventional symmetric buffering on NPUs. On AMD XDNA2™ AIE, ATB improves GEMM throughput by up to 40% even over highly optimized symmetric kernels. We also present kernel design strategies that deliver a 4.54× speedup over the state-of-the-art GEMM implementation. In the future, we plan to explore automated search and scheduling to make ATB-based kernel generation fully automatic.

## References

- [1] AMD. 2023. Adaptive Computing Technology Overview. <https://www.amd.com/content/dam/amd/en/documents/solutions/adaptive-computing-technology-overview.pdf>. Accessed: 2025-11-19.
- [2] AMD. 2024. AMD Ryzen AI 5 340 Processor. <https://www.amd.com/en/products/processors/laptop/ryzen/ai-300-series/amd-ryzen-ai-5-340.html> Accessed: 2025-11-18.
- [3] AMD. 2024. AMD Ryzen AI 9 HX 370 Processor. <https://www.amd.com/en/products/processors/laptop/ryzen/ai-300-series/amd-ryzen-ai-9-hx-370.html> Accessed: 2025-11-18.
- [4] AMD. 2025. AI Engine API – Matrix Multiplication (MMUL) Reference. [https://download.amd.com/docnav/aiengine/xilinx2025\\_1/aiengine\\_api/aie\\_api/doc/group\\_group\\_mmul.html](https://download.amd.com/docnav/aiengine/xilinx2025_1/aiengine_api/aie_api/doc/group_group_mmul.html). Accessed: 2025-10-01.
- [5] AMD. 2025. AMD XDNA Driver for Linux. <https://github.com/amd/xdna-driver>. Accessed: 2025-11-16.
- [6] AMD. 2025. BFP16 (Block Floating Point) Quantization Tutorial. [https://quark-docs.amd.com/latest/pytorch/tutorial\\_bfp16.html](https://quark-docs.amd.com/latest/pytorch/tutorial_bfp16.html). Accessed: 2025-10-01.
- [7] AMD. 2025. DynamicDispatch. <https://github.com/amd/DynamicDispatch>. Accessed: 2025-10-01.
- [8] AMD. 2025. Ryzen AI Software. <https://github.com/amd/RyzenAI-SW?tab=readme-ov-file>. Accessed: 2025-10-01.
- [9] AMD. 2025. Xilinx MLIR-AIE BFP16 matrix multiplication implementation. [https://github.com/Xilinx/mlir-aie/tree/main/programming\\_examples/ml\\_block\\_datatypes/matrix\\_multiplication](https://github.com/Xilinx/mlir-aie/tree/main/programming_examples/ml_block_datatypes/matrix_multiplication). Accessed: 2025-10-01.
- [10] Xiaodong Deng, Shijie Wang, Tianyi Gao, Jing Liu, Longjun Liu, and Nanning Zheng. 2024. AMA: An Analytical Approach to Maximizing the Efficiency of Deep Learning on Versal AI Engine. In *2024 34th International Conference on Field-Programmable Logic and Applications (FPL)*. 227–235. doi:10.1109/FPL64840.2024.00039
- [11] Michael Gilbert, Yannan Wu, Joel Emer, and Vivienne Sze. 2024. LoopTree: Exploring the Fused-Layer Dataflow Accelerator Design Space. *IEEE Transactions on Circuits and Systems for Artificial Intelligence PP* (09 2024), 1–15. doi:10.1109/TCASAI.2024.3461716
- [12] D. W. Gillies and J. W.-S. Liu. 1990. Scheduling Tasks with AND/OR Precedence Constraints. In *Proceedings of the Second IEEE Symposium on Parallel and Distributed Processing*. 394–401. doi:10.1109/SPDP.1990.143572
- [13] Qijing Huang, Po-An Tsai, Joel S. Emer, and Angshuman Parashar. 2024. Mind the Gap: Attainable Data Movement and Operational Intensity Bounds for Tensor Algorithms. In *2024 ACM/IEEE 51st Annual International Symposium on Computer Architecture (ISCA)*. 150–166. doi:10.1109/ISCA59077.2024.00021
- [14] Erika Hunhoff, Joseph Melber, Kristof Denolf, Andra Bisca, Samuel Bayliss, Stephen Neuendorffer, Jeff Fifield, Jack Lo, Pranathi Vasireddy, Phil James-Roxby, and Eric Keller. 2025. Efficiency, Expressivity, and Extensibility in a Close-to-Metal NPU Programming Interface. In *2025 IEEE 33rd Annual International Symposium on Field-Programmable Custom Computing Machines (FCCM)*. 85–94. doi:10.1109/FCCM62733.2025.00043
- [15] Linyan Mei, Pouya Houshmand, Vikram Jain, Sebastian Giraldo, and Marian Verhelst. 2020. ZigZag: A Memory-Centric Rapid DNN Accelerator Design Space Exploration Framework. arXiv:2007.11360 [cs.DC]. <https://arxiv.org/abs/2007.11360>
- [16] Kaustubh Mhatre, Endri Taka, and Aman Arora. 2025. GAMA: High-Performance GEMM Acceleration on AMD Versal ML-Optimized AI Engines. arXiv:2504.09688 [cs.AR]. <https://arxiv.org/abs/2504.09688>
- [17] Endri Taka, Aman Arora, Kai-Chiang Wu, and Diana Marculescu. 2023. Max-EVA: Maximizing the Efficiency of Matrix Multiplication on Versal AI Engine. arXiv:2311.04980 [cs.AR]. <https://arxiv.org/abs/2311.04980>
- [18] Endri Taka, Dimitrios Gourounas, Andreas Gerstlauer, Diana Marculescu, and Aman Arora. 2024. Efficient Approaches for GEMM Acceleration on Leading AI-Optimized FPGAs. In *2024 IEEE 32nd Annual International Symposium on Field-Programmable Custom Computing Machines (FCCM)*. 54–65. doi:10.1109/FCCM60383.2024.00015
- [19] Vijay Thakkar, Pradeep Ramani, Cris Cecka, Aniket Shivam, Honghao Lu, Ethan Yan, Jack Kosaian, Mark Hoemmen, Haicheng Wu, Andrew Kerr, Matt Nicely, Duane Merrill, Dustyn Blasig, Fengqi Qiao, Piotr Majcher, Paul Springer, Markus Hohnerbach, Jin Wang, and Manish Gupta. 2023. CUTLASS. <https://github.com/NVIDIA/cutlass>
- [20] Field G. Van Zee and Robert A. van de Geijn. 2015. BLIS: A Framework for Rapidly Instantiating BLAS Functionality. *ACM Trans. Math. Software* 41, 3 (June 2015), 14:1–14:33. <https://doi.acm.org/10.1145/2764454>
- [21] Chengyue Wang, Xiaofan Zhang, Jason Cong, and James C. Hoe. 2025. Reconfigurable Stream Network Architecture. In *Proceedings of the 52nd Annual International Symposium on Computer Architecture (ISCA '25)*. Association for Computing Machinery, New York, NY, USA, 1848–1866. doi:10.1145/3695053.3731088
- [22] Samuel Williams, Andrew Waterman, and David Patterson. 2009. Roofline: an insightful visual performance model for multicore architectures. *Commun. ACM* 52, 4 (April 2009), 65–76. doi:10.1145/1498765.1498785
- [23] Jinming Zhuang, Jason Lau, Hanchen Ye, Zhiping Yang, Yubo Du, Jack Lo, Kristof Denolf, Stephen Neuendorffer, Alex Jones, Jingtong Hu, Deming Chen, Jason Cong, and Peipei Zhou. 2023. CHARM: Composing Heterogeneous Accelerators for Matrix Multiply on Versal ACAP Architecture. arXiv:2301.02359 [cs.AR]. <https://arxiv.org/abs/2301.02359>
- [24] Jinming Zhuang, Zhiping Yang, and Peipei Zhou. 2023. AutoMM: Energy-Efficient Multi-Data-Type Matrix Multiply Design on Heterogeneous Programmable System-on-Chip. arXiv:2305.18698 [cs.AR]. <https://arxiv.org/abs/2305.18698>