## Whispering Pixels: Exploiting Uninitialized Register Accesses in Modern GPUs

Frederik Dermot Pustelnik, Xhani Marvin Sass, Jean-Pierre Seifert

Technische Universität Berlin - SecT

Berlin, Germany

{f.pustelnik, sass, jean-pierre.seifert}@tu-berlin.de

Abstract-Graphic Processing Units (GPUs) have transcended their traditional use-case of rendering graphics and nowadays also serve as a powerful platform for accelerating ubiquitous, non-graphical rendering tasks. One prominent task is inference of neural networks, which process vast amounts of personal data, such as audio, text or images. Thus, GPUs became integral components for handling vast amounts of potentially confidential data, which has awakened the interest of security researchers. This lead to the discovery of various vulnerabilities in GPUs in recent years. In this paper, we uncover yet another vulnerability class in GPUs: We found that some GPU implementations lack proper register initialization routines before shader execution, leading to unintended register content leakage of previously executed shader kernels. We showcase the existence of the aforementioned vulnerability on products of 3 major vendors - Apple, NVIDIA and Qualcomm. The vulnerability poses unique challenges to an adversary due to opaque scheduling and register remapping algorithms present in the GPU firmware, complicating the reconstruction of leaked data. In order to illustrate the real-world impact of this flaw, we showcase how these challenges can be solved for attacking various workloads on the GPU. First, we showcase how uninitialized registers leak arbitrary pixel data processed by fragment shaders. We further implement information leakage attacks on intermediate data of Convolutional Neural Networks (CNNs) and present the attack's capability to leak and reconstruct the output of Large Language Models (LLMs).

## 1. Introduction

In recent years, the ubiqity of Graphic Processing Units (GPUs) has brought unprecedented computational power into the hands of end-users. Growing beyond their traditional use cases of rendering graphics only, GPUs have become a popular choice for parallelizing heavy workloads. Advancements in GPU architectures also fueled advancements in the field of artificial intelligence (AI), where the availability of increased processing power lead also to enhanced AI models. Typically, AI is used for the evaluation of large datasets, from processing medical data to automated evaluation of surveilance feeds. Today, AI suitable for everyday use has arrived at the hands of the end-user in the form of Large Language Models (LLMs) [41], [42]. The recent popularity of such LLMs is primarily thanks the ability to provide meaningful answers and the ability to reason, a skill which has been mostly absent in previous research on natural language processing

(NLP). Hence, a lot of business cases for the utilization of LLMs have emerged, ranging from automatically handling customer service requests to tasks such as text content generation. Datacenter providers have also noticed this increasing trend of AI advancements and offer GPU-asa-Service (GaaS), where users can rent GPUs on demand and pay via a usage model. Customers can enjoy a high degree of flexibility, renting GPU computational power on demand. Providers also aim to reduce underutilized resources as much as possible. Allocating a complete GPU to a single container or virtual machine (VM) can be considered a waste of resources if either only a fraction or only a limited time of processing power is needed. The ability to share GPUs gives the provider a cost-effective measure to adapt to varying workloads. As industrial examples, Google's Kubernetis Engine allows to share a GPU between up to 48 tenants, while Microsoft build GPU Paravirtualization into the Hyper-V Hypervisor, allowing VMs to share a single GPU [1], [3].

In a landscape also defined by connectivity and mobility, neural networks have also found their way into customer's end-devices. Smartphone GPU vendors offer the ability to leverage on-board graphics for acceleration of computational tasks. Real-time object recognition, facial tracking, video feed enhancement, audio classification and even voice cloning are just some examples on how users interact with neural networks on a daily basis.

Vulnerabilities within GPUs can thus lead to adversaries gaining unauthorized access to private user data.

The primary feature of GPUs on end-user devices is rendering graphical user interfaces, which display potentially confidential information. protect To against information leakage attacks, including pixel information leakage, modern operating systems deploy tight sandboxing and access management in order to enforce hard boundaries between applications. Thus, GPUs are components which process confidential user data in different ways, which moreover has attracted the interest of security researchers. Researchers have presented multiple vulnerabilities in modern GPUs to bypass application boundary mechanisms, but mostly evaluated their attacks for specifically leaking pixel data by applying filters on webbrowser iframes [40], [44]. Other recently presented GPU side-channel attacks range from frequency-based attacks to leakage via uninitialized memory [11]. Naghibijouybari et al. presented a performance counter attack in order to leak typed letters on the on-screen keyboard [27]. Cache side-channels have also been replicated on GPUs [18], [19].

In this paper, we present multiple register-based vulnerabilities found across different GPU implementations. For one, we demonstrate how uninitialized registers can be abused in order to gain access to computational results of previously executed shader kernels. Furthermore, we briefly discuss pixel leakage attacks and thoroughly evaluate the found vulnerabilities on state-of-the art neural networks in order to highlight the criticality of the presented flaws. Our attack displays a much higher data leakage count than other, similar previously discovered leakage attacks.

Our main contributions are:

- We describe uninitialized GPU register access as a new side-channel technique which affects multiple vendors.
- We evaluate our attack for multiple data exfiltration attacks against different types neural networks, such image-processing neural networks and LLMs. Further, we showcase the attack's capability for leaking pixels across application boundaries
- We showcase how uninitialized structures can be used to build a covert channel, which is, to the best of our knowledge, the fastest GPU-based covert channel.
- We propose multiple countermeasures against the aforementioned attacks and weigh the advantages and downsides of each approach.

**Outline** We structure the paper as follows: First, we describe the fundamentals of GPU architectures and the GPU pipeline in section 2. Next, in section 3 we provide the considered threat model. Afterwards, in section 4, we describe the attacks and give insight into the building blocks required for successful exploitation. In section 5, we evaluate the performance and the exploitability of our attack for various different scenarios. Next, in section 6, we propose multiple countermeasures and further discuss limitations and other aspects of our work in section 7. Related work is presented in section 8. Lastly, we conclude our paper in section 9.

#### 2. Background

In this section, we furnish insight into the GPU pipeline, the differences and similarities of different GPU architectures and General Purpose Computation for GPUs, which is the main building block for executing nongraphical tasks, such as accelerated artificial intelligence applications.

#### 2.1. GPU Architectures

Many different GPU architectures by different vendors exist, however, they all share some similarities. In this section, we provide an overview over the generalized architecture as depicted in Figure 1 and will provide an overview over differences of various vendor implementations later.

**Threading Model**: The base of all GPUs is formed by a *Core*, representing an execution unit which contains one or multiple SIMD units which in turn consist of multiple



Figure 1: Generalized GPU architecture. A GPU typically contains one or more cores, where each core can contain multiple execution units, which can execute e.g. 32 threads as a wavefront in lockstep.

ALUs. Dispatched work is divided into multiple chunks which are executed in parallel on a SIMD unit.

**Registers**: All architectures have a set of vector registers, which are accessible for general-purpose usage in the shader.

GPU Programming: An application has to use vendorprovided libraries in order to translate a shader from its high-level source code to an architecture-dependent binary code. Vendors provide these libraries for a variety of high-level languages. As such, most vendors build their efforts on the LLVM compiler framework [31], [43], due to its extensibility and flexibility. This holds for opensource solutions, as the one provided by AMD, but also closed-source solutions, i.e. by NVIDIA and Qualcomm. To the application, the same programming interface is exposed (sometimes with some vendor-specific functions), such that shader source code can be compiled to any architecture with ease. In the rest of this paper, we will refer to the high-level source code, such as the OpenGL Shading Language (GLSL), as shader source code and the architecture-dependent compilation result as shader binary. The executed shader can also be referred as shader kernel.

Memory Hierarchy: Each core has access to a dedicated cache, as well as a shared cache. GPUs further include dedicated memories for different kind of data, such as Texture Memory, Constant Memory and Global Memory for sharing data between kernels.

In order to support the parallel execution of multiple threads on a SIMD units, memory accesses have to be optimized. As a solution, GPUs support memory coalescing, where multiple memory accesses are bundled into a single large one, drastically improving throughput. Coalescing is one of the main reasons why vendors state that branching

should be avoided in shader code.

## 2.2. GPU Rendering Pipeline

The simplified GPU rendering pipeline consists of three main stages:

**Geometry**: The geometry stage performs the transformation from the world coordinate system to screen coordinates for vertices. Perspective and clipping handling are also included in this step. Generally, the geometry stage is handled via a *vertex shader*.

**Rasterization**: During rasterization, the GPU determines which pixels correspond to a given primitive. In many rendering pipelines, the rasterization process cannot be determined by the programmer via a shader. Based on the rasterization information, the next pipeline stage is only run on pixels belonging to a specific primitive.

**Fragmentation**: In this pipeline stage, the color, shading and texturing of all primitives are determined. Each fragment, which is associated with one pixel on the screen, is the result of a *fragment shader* calculation. A fragment contains all data to calculate the pixel value, which may include the depth, interpolated color or texture values and the pixel position.

With the continuous advancements of graphic rendering techniques, such as ray-tracing, the rendering pipeline outgrew the traditional designs and is thus more complex for advanced 3D renderers. These steps are not important to understand the rest of the paper and hence have been omitted.

# **2.3.** General Purpose Computation on GPU (GPGPU)

Since GPUs contains much more cores than the typical main processor of the system, they are well suited for highly parallelizable workloads. With *General Purpose Computation for GPUs* (GPGPU), the usage of GPUs beyond their initial work scope has been proposed. First commercially introduced in 2006 by NVIDIA with their proprietary CUDA framework and later, in 2009, with the publication of first widely supported open-source GPGPU specification OpenCL, the GPGPU concept has been adopted by every major vendor. Frameworks, which have primarily been focused on graphics, such as DirectX and OpenGL, have also added support for their variants of compute shaders. Thus, every GPU in high-performance systems as well as mobile systems now supports GPGPU programming.

The compute GPU pipeline is much simpler than its graphics counterpart, since data passed to the GPU is normally returned to the calling application directly after shader execution finished instead of being passed through a complex multi-stage pipeline.

This new programming model opened the door for many opportunities: For one, GPUs have been extensively been used for physics simulations and other scientific applications relying on parallelized work. However, another specific application use case which has emerged are neural networks. GPUs allow to parallelize big matrix operations which are fundamental for neural networks. Open-source frameworks such as PyTorch and Tensorflow [17] are the

predominant cornerstone of modern AI, but also vendors themselves tend to provide their own optimized libraries [2]. All major frameworks have strong support for GPUs, making those the predominant accelerators for neural network workloads.

#### 2.4. Computing Workloads

GPGPU APIs give the developer the most control over the underlying hardware in contrast to graphical programming interfaces. Computing programs can dispatch workloads to the GPU in the form of workitems, where each workitem is mapped to a GPU thread. All workloads are organized in a hierarchy for better scheduling and higher throughput. These hierarchies are closely related to the hardware architecture, needing the programmer to consider the underlying architecture during development. For one, leads to a higher effort needed when porting GPU programs to a newer architecture, however, this also allows for full control and maximum optimization of GPU programs on the side of the developer.

For computational tasks, all GPGPU APIs expose the hardware in a similar fashion. Every API provides access to multiple thread groups, where each thread group typically consists of up to 1024 threads. One thread group is scheduled to a single core, where thread groups are then further divided in waves, which are a set of threads that are executed concurrently in hardware on a single SIMD unit. GPU cores consist of one or more of said SIMD execution units, which can execute one sub-thread group at once. Programming interfaces allow to index a thread in a thread group via a 3-dimensional index and to further arrange thread groups in a 3-dimensional grid. These organizational scheme is present in all aforementioned interfaces.

#### 2.5. Vendor Differences and Terminology

Despite sharing a lot of similarities, custom vendor implementations still have a lot of differences, which we will highlight in this section. Especially terminology is different between vendors, with specific terms shown in Table 1. NVIDIA provides the most complete documentation about their hardware, where Apple and Qualcomm document only essential information. Cores in NVIDIA GPUs contain multiple different specialized ALUs and cores for different worksets, named Raytracing Cores (RTs), Single Precision Units (SPs), Double Precision Units (DPs) and Special Function Units (SFUs). This is a contrast to the *unified shader architecture*, which is utilized in Apple and Adreno GPUs, where the same execution units are responsible for all aforementioned tasks.

Because different vendor terminologies, we need to use generalized terminology:

- Thread: Smallest execution unit, which corresponds to a single workitem.
- Wave: A subgroup of threads which are executed in parallel on a SIMD unit.
- **Thread group**: A collection of threads which is assigned to a single core.
- **SIMD Unit**: Execution unit in a GPU core which executes 32 threads in parallel.

TABLE 1: Vendor-specific terminology according to documentation.

|                         | Vendor                     |                     |                                    |  |
|-------------------------|----------------------------|---------------------|------------------------------------|--|
| Term                    | NVIDIA                     | Qualcomm Adreno     | Apple AGX                          |  |
| Smallest Execution Unit | SP/DP/SFU                  | ALU                 | ALU                                |  |
| Thread                  | Thread                     | Fiber               | Thread                             |  |
| Thread Model            | Cluster/Block/Thread group | Workgroup/Wave      | Grid/Thread group                  |  |
| Core Execution Unit     | 32 Threads per Warp        | 32 Threads per Wave | 32 Threads per SIMD/Warp/Wavefront |  |

• **GPU Core**: Contains multiple SIMD Units, a local register file and a local scheduler.

Both Qualcomm and NVIDIA specify in their documentation that they allow the execution of half-waves, where the SIMD execution unit is split up into two parts.

Most architectures only support a single program counter (PC) per SIMD unit and control execution via a execution mask. Thus, all threads in a wave execute the same instruction at the same time, only with different data. NVIDIA incorporated to ability to accommodate multiple PCs in a single SIMD unit, thus reducing the latency of diverging threads.

## 3. Threat Model

Our attacks require the attacker to execute a malicious user application. Beyond this, no special privileges for the attacker application are needed. The attacker needs to be able to execute a custom shader binary on the GPU. Modern systems provide a kernel command interface, for which an abstraction layer, such as a graphics or GPGPU library exist. The communication with the interface can either be replicated or hooked by the attacker. Modern operating systems do not consider graphic and rendering library access to be a privileged or sensitive, allowing an unprivileged attacker to execute arbitrary shader binaries. In our work, we consider that both the victim and the adversary can execute arbitrary code on the same machine. On a customer device, we consider a victim application and a malicious application, where latter either aims to leak pixel or neural network data of the victim application. For cloud applications, we consider a shared GPU between multiple cloud tenants, over which the adversary aims to leak information.

Another assumption we make is that the adversary has knowledge about the targeted GPU application. For neural network-based attacks, we assume that the adversary at least has knowledge about what the targeted model is and has access to the model itself. This is necessary in order to correctly reconstruct the leaked data. We later see in section 5, that this assumption can relaxed depending on the model, where sometimes only partial information about network parameters is necessary.

#### 4. Uninitialized Register Access Attacks

In this section, we present uninitialized register accesses as an information leakage mechanism targeting GPUs, allowing to leak data processed by programs executed on the GPU. First, we give a detailed insight into how the presented vulnerability works. Next, we give an overview of how to determine what data exactly is leaked by a previously executed shader, based on the

knowledge of a specific shader. Lastly, we discuss several challenges that might arise when an adversary tries to leak specific data and show potential solutions on how to handle various cases.

Our attack consists of 3 steps:

Step 1. Ensure scheduling of the leakage shader after the targeted shader.

**Step 2.** Run the leakage shader on the GPU and copy stale register values to memory.

**Step 3.** Copy form GPU to host memory and reconstruct the leaked data based on its structure.

In the following, we describe how the found vulnerability behaves on different platforms, since it affects how and what kind of information is leaked. The hand-crafted kernels for general exploitation of the vulnerability are shown in Figure 3 for all three platforms.

Adreno and AGX. Both the underlying vulnerability as well as the exploitation process are nearly identical on Adreno and AGX. On those GPUs, registers are not cleared after shader execution, leaving possible sensitive data behind. On Adreno GPUs, up to 64 registers are available to a kernel, and each register consists of four subregisters, which can be accessed via a postfix, resulting in a total number of 256 32-bit wide registers. On AGX, the shader can claim up to 128 32-bit wide registers for computation. However, both platforms are quite unlikely to utilize all available registers in practice.

**NVIDIA.** On NVIDIA GPUs, stale register values can only be leaked if said register value has previously been written to memory via a memory store instruction. On these architectures, the memory coalescing operation is implemented on half-wave level, where the memory store is first executed for the half of a wave and for the second wave afterwards, resulting in two consecutive memory bus accesses. Since our experiments suggest that we can only leak the last written memory values, we can only leak outputs of the second half wave memory access.

## 4.1. Building Blocks and Challenges

Even though the presented vulnerability forms a powerful side-channel attack, there still arise multiple challenges which have to be solved. In the following, we will provide an overview over building blocks which are incorporated in order to form a reliable exploitable side-channel. Furthermore, since we leak stale data by *every* running shader, one has to be able to determine which data to consider as noise and to identify only data of interest.



Figure 2: Overview of the attack procedure. The adversary provides hand-crafted shader kernels to the GPU interface, while normal user applications use vendor-provided interfaces. In order to achieve co-location with the targeted shaders, the attacker needs to dispatch his kernel in the execution queue of every core.

```
// Get thread number
     mov.f32f32 r0.y, r51.w
(rpt2)nop
                                                                           // Get thread number
(nop3) shl.b r0.y, r0.y, 3
                                     // Get thread number
                                                                          MOV R1, c[0][28h]
                                                                          S2UR UR6, SR_CTAID.X
(nop3) add.s r0.x, r0.x, r0.y
                                     get_sr r1,
     mov.f32f32 r0.y, r0.x
                                            sr80(thread_pos_in_grid.x)
                                                                          UMOV UR7, 4
(rpt5) nop
                                      // Store stale register r0
                                                                          ULDC.64
                                                                                      UR4, c[0][160h]
  Store stale register r2.x
                                     device_store 0, i32, single,
                                                                          UIMAD.WIDE UR4, UR6, UR7, UR4
      stib.b.untyped.1d.u32.1.imm
                                                  r0, u2_u3, r1,
                                                                           // Store stale register R8
                                                                          STG.E.SYS [UR4], R8
           r2.x
                                                  unsigned, 0
      mov.f32f32 r0.z, c8.x
                                     stop
                                                                          EXIT
                                              (b) AGX kernel.
                                                                                  (c) NVIDIA kernel.
        (a) Adreno kernel.
```

Figure 3: Attacker kernels for exploiting stale register reads.

Attacker Shader Creation. The vulnerabilities we present are not necessarily exploitable from high-level code, such as Metal Compute or OpenCL. Hence, the attacker needs to craft a malicious shader binary manually. NVIDIA provides the nvcc compiler, which is able to compile host code and GPU CUDA C++ code into one single binary. For the compiled CUDA code, nvcc is able to either emit or compile parallel Thread Execution IR (PTX), which is compiled later during runtime, or to emit and compile raw GPU shader code. We use the later feature in order to maliciously modify the shader.

Adreno and Apple both use a different approach: Shader source code is compiled during program runtime, where the host application provides the high-level code to a vendor-provided shadered library, which in turn compiles the shader during application runtime. Therefore, the attacker first has to compile a placeholder shader and patches the binary code in memory at runtime. On Adreno GPUs, the shader binary code is held in a memory map of the /dev/kgsl device. The Adreno kernel driver exposes multiple memory mappings via this device, for either data, commands and code binaries passed to the GPU. This memory map are a shared memory regions with the GPU, where binary modifications are seen immediately and no further copy operations are necessary. By iterating over the memory maps, the one containing the shader binary can be found without difficulty by checking for valid disassembly. We decided to rely on OpenGL ES Compute because it is implemented on all devices with OpenGL ES 3.1 support and does not require additional libraries like OpenCL, where support is not guaranteed for every device.

For AGX, we employ a similiar approach: Both iOS and macOS use a shader cache file for each executed application. The shader cache is contained in a file named functions.data and is memory mapped when the Metal framework is loaded. Even though AGX code is placed on a shared memory region, like Adreno, we choose to modify the shader cache file since it allows for better interoperability of the exploitation technique between both OS versions. We utilize the Metal framework for shader management and modify a placeholder Metal shader.

Scheduling. Our goal is to leak stale register values which have been left behind by a victim shader. To achieve this, the attacker shader has to be scheduled on the same core as the victim immediately after it finished execution. However, the scheduling algorithms running in the firmware can be considered to be a blackbox to the programmer, since there is no programming interface to assign a shader to a specific GPU core. The placement of both shaders determines what and how much data can effectively be leaked. In order to understand how to effectively leak data, we need to gain a deeper insight into how the scheduling algorithms work for all platforms. Documentation for vendors state that a thread group is

assigned to a single core, and that a thread group cannot

be assigned to multiple cores at once. However, a core can execute multiple thread groups, even at once. Specifically for the Fermi architecture generation of NVIDIA GPUs, researchers found that shaders are assigned to cores in a round-robin fashioned manner [26]. We confirmed this to be the case through experiments for all GPU architectures mentioned in this paper. Consequently, the highest probability to achieve scheduling on the same core is by utilizing all cores and SIMD units available, which ensures queuing of the attacker code after the victim executed with high confidence.

Register Remapping. GPUs manage a large register file, where registers are allocated to running shaders by the firmware. Resultingly, the register mapping can be considered as a black-box to the adversary. While investigating register mapping on different GPUs, we found that AGX has the most agressive remapping scheme, with no observable pattern in how registers are allocated for a new shader. When two registers appear to be consecutive in one shader (e.g. r0 and r1), those two registers are unlikely to appear as consecutive registers again in another shader. In constrast, Adreno has a far more predictable remapping scheme. For example, if an adversary wishes to leak the value of register r44.y in the victim shader, he himself needs to read from register r44.y, and the target value does not seem to appear in other registers. Thus, the attacker can selectively leak data of victim shaders, leading to very precise attacks on Adreno GPUs. On NVIDIA GPUs, we observed that reading from any uninitialized registers reveals nearly all data written to GPU memory.

**Data order.** Since there are more workitems in the work queue than GPU cores available, the question arises on how the structure of data is preserved during leakage. Work is scheduled in thread groups, where all threads in a single group execute on the same core. Commonly, thread groups with a size of up to 1024 threads are typically encountered. All of these threads are split in smaller groups of the size of a single SIMD unit, which is 32 in many products, and get dispatched in parallel on a SIMD unit. We observe that we can leak data of a SIMD unit in order and further make the observation, that the waves are dispatched in order. However, since we have no knowledge of how victim thread groups are mapped to cores by the scheduler, we cannot determine the data order across thread groups.

**Determining what is leaked.** The adversary first has to determine what data is leaked from the targeted victim shaders. On NVIDIA GPUs, since only the last written values of a register during execution are leaked, intermediate values cannot be leaked. On contrary to this, AGX and Adreno GPUs leak all their register values, which conveniently can also be used to leak intermediate values of GPU calculations.

Moreover, the Adreno shader compiler uses high registers, such as r51.w often for passing runtime information to the shader In practice however, we encountered no shader which uses such high registers for any calculations.

Even though only register contents which are written to memory can be leaked, the intermediate layer outputs

| Work                           | Architecture            | Bandwidth                          | Method                   |
|--------------------------------|-------------------------|------------------------------------|--------------------------|
| Dutta et al. [12]              | Intel iGPU              | 120kb/s,<br>400kb                  | Last-Level<br>Cache      |
| Naghibijouybari<br>et al. [26] | NVIDIA                  | 4Mb/s                              | Contention (ALU, memory) |
| Dutta et al. [13]              | NVIDIA                  | 4Mb/s                              | Interconnect             |
| Naghibijouybari<br>et al. [25] | NVIDIA                  | 400kb/s                            | L1, L2 contention        |
| Ahn et al. [6]                 | NVIDIA                  | 24MB/s                             | Interconnect contention  |
| Almusaddar<br>et al. [7]       | Intel iGPU              | 1.65Kb/s<br>4.41Kb/s               | Parallel memory writes   |
| Nayak et al. [28]              | NVIDIA                  | 81Kb/s                             | TLB                      |
| This work                      | NVIDIA<br>AGX<br>Adreno | 240.7Mb/s<br>413.3Mb/s<br>19.2Mb/s | Uninitialized registers  |

TABLE 2: Performance of different GPU-based Covert Channel Implementations.

of neural network frameworks are still prone to data leakage. We examined two popular frameworks (PyTorch [17] and tinygrad [4]) and noticed that every layer is executed as a separate kernel. Hence, data between different layers can only be passed via memory, and all layers need to write their outputs to shared memory instead of keeping values in registers. This allows an adversary to leak intermediate layer data without much effort despite certain shortcomings of the exploitation primitive.

**Distinguishing Data from Noise.** Given the fact that GPUs process and handle extensive amounts of data per second, locating the targeted data to leak within system noise poses a unique and challenging problem. The adversary can take advantage of the structure of the data processed, depending if pixel data or different types of neural networks are targeted. Especially different neural networks are prone to leakage at different locations due to their internals. We will further investigate this challenge as well as solutions in our evaluation in section 5.

## 5. Evaluation

In this section, we evaluate the performance of the uninitialized register reading attack under different conditions. We first present a covert channel implementation using the vulnerability, and leverage it for performance evaluation. Afterwards, we thoroughly evaluate the real-world impact of the vulnerability and how different workloads can be attacked, ranging graphics to neural network information leakage. All experiments in this section were conducted on the hardware as listed in Table 3.

#### 5.1. Covert Channel

We first introduce how the uninitialized register read exploit can be utilized for creating a covert channel. On Android and iOS, where only a given API can be used for







Figure 4: Covert Channel Performance evaluation for varying numbers of thread groups for sender and receiver.

| Device             | GPU        | OS version                  | Display       |
|--------------------|------------|-----------------------------|---------------|
| Macbook Air        | M1 AGX     | macOS 13.4.1<br>Asahi Linux | 4k            |
| iPhone X           | A11 AGX    | iOS 13.6.1                  | 1125×<br>2436 |
| Samsung Galaxy A52 | Adreno 618 | Android 13                  | 1080×<br>2400 |
| Custom             | RTX 2070   | Ubuntu 22.04                | 4k            |

TABLE 3: Tested GPUs.

Inter-Process Communication (IPC), attackers can aim to use covert channels in order to evade either restrictions of IPC usage or evade being detected. Since GPU access is unprivileged on both operating systems, stale GPU register values are an ideal building block for covert channels. As we can read stale data, the bandwidth and the noise resistance of the covert channel is high. On Adreno GPUs, mostly registers r0 and r1 are used, even though up to 64 general-purpose registers are available. Thus, the usage of higher registers for a covert channel is advised, as they are less likely to be used and overwritten in our observations. To covertly transmit data, the sender executes a simple shader, which writes the passed data to send to a register of choice. To uniquely identify the sender data, a format as magic\_value | counter | data is used. If the magic value and counter are e.g. 4 bytes in size, 124 bytes (on AGX and Adreno) or 60 bytes (on NVIDIA) can be transmitted respectively. Based on the magic value and the counter, the receiver of the covert channel can easily recover the original message. Performance evaluation results for the covert channel implementation compared to raw data leakage speeds are shown in Table 4. Our covert channel can achieve up to 413.3Mb/s on AGX GPUs, 240.7Mb/s on NVIDIA, and 19.2Mb/s on Adreno, which is drastically more than other proposed GPU-based covert channel mechanisms. Most GPU-based covert channels unveiled by researchers are able to leak data in ranges of Kb/s to Mb/s. A detailed comparison with other research is presented in table 2. To the best of our knowledge, stale register reads is the first GPU-based covert channel which achieves a bandwidth in the range of hundreds of Mb/s.

#### 5.2. Performance

By building a covert channel, we developed a good base for estimating an upper bound for how much information can be leaked when targeting one specific application. Since the covert channel setup allows to measure the transfer rate between two applications without any system noise, we can measure the maximum rate at which data of one victim application can be extracted.

The raw leakage rate is much larger on all systems, mainly due to the GPU also rendering the display contents. In order to also evaluate the possibility of contention, we further showcase the leakage performance in detail with varying amounts of workgroups dispatched to the GPU, as depicted in Figure 4.

For both AGX and NVIDIA, the configuration where maximum throughput is achieved always happens when the number of sender cores matches the number of receiver cores. On NVIDIA, the highest throughput is reached when both the receiver and sender use 32 thread groups, whereas on AGX the maximum is achieved with 64 thread groups for both. Using more sender then receiver thread groups and vice versa leads to under-utilization of the covert channel. An interesting situation happens on NVIDIA GPUs, if 64 sender groups, but 8 or 16 receiver groups are utilized: We observed that the black-box scheduler does not schedule the sender and receiver in such a way, that any leakage can be observed.

| Vendor | Sender ×<br>Receiver    | CC                  | Raw                | Ratio           |
|--------|-------------------------|---------------------|--------------------|-----------------|
| NVIDIA | 32×32                   | 240Mb/s             | 4.4Gb/s            | 0.053           |
|        | 32×64                   | 7.8Mb/s             | 4.06Gb/2           | 0.0019          |
| Adreno | $8\times4$ $32\times16$ | 19.2Mb/s<br>1.2Mb/s | 246Mb/s<br>459Mb/s | 0.078<br>0.0026 |
| AGX    | $64 \times 64$          | 413.4Mb/s           | 82.2Gb/s           | 0.0049          |
|        | $2 \times 64$           | 30.3Mb/s            | 2.81Gb/s           | 0.0105          |

TABLE 4: Covert Channel performance in comparison to raw data leakage performance. Best-case with worst-case configurations are compared.

Out of all vendors, we observed the slowest leakage on Adreno. We attribute this to the usage of OpenGL Compute, a very rarely used GPGPU implementation which is nonetheless widely supported. We identified the bottleneck to be the shader creation, not its execution time. As such, we accomplished covert channel speeds of only up to 19.2Mb/s. Nevertheless, our implementation on Adreno is still faster than many other covert channel implementations. We also observed on Adreno that the usage of many cores quickly leads to a bottleneck on the device. The fastest configuration uses 4 sender and 8 receiver cores.

This performance evaluation also provides an understanding of how many cores an adversary should utilize if he wishes to leak data with a high rate from a victim shader. On both NVIDIA and AGX, he needs to dispatch the same number of thread groups as the victim to achieve the best results.

#### 5.3. Leaking Fragment Shader Data

Modern operating systems employ a strict permission system, which manages app access to sensitive resources. One sensitive resource is the permission to record the screen, whereby the OS requests explicit user consent before an application can record the user's screen. In this section, we showcase how a malicious app can circumvent this protection mechanism by targeting fragment shaders.

The fragment shader is the one render pipeline step responsible for producing the color data of rendered primitives. If an adversary wishes to leak pixel information on the target system, he therefore has to target fragmentation shader outputs. Mobile GPUs often employ a rendering strategy called *tiled rendering*. Because of limited memory space and bandwidth, the screen is partitioned into multiple tiles, and one tile is rendered by a single core. Afterwards, the results are stored in a local tile memory instead of directly being written to the framebuffer, which resides in the bandwidth-constrained main memory of the system. As such, the tile size is primarily determined by the tile memory size.

```
// texture sampling
                 0, r11_r1h_r21, ts0, ss0, tex_2d
texture_sample
wait
                 0
fadd32
                 r0, r11.discard, -0.0 // red
                 r1, r1h.discard, -0.0 // green
fadd32
                 r2, r21.discard, -0.0 // blue
fadd32
mov_imm
                 r3, 0x3f800000
                                        // alpha
writeout
                 12, 0
                 r0_r1_r2_r3, quad
st tile
  Write RGBA to output
```

Listing 1: Simplified fragment shader disassembly for AGX GPUs. After texture samping, registers r0-r3 contain pixel information which is stored in tile memory afterwards.

**Reverse Engineering Tiling.** Before pixel data can effectively be leaked from registers, we first have to gain an insight into how tiled rendering is implemented on modern GPUs. Tiles are rendered on a per-application



Figure 5: Fragment shader leaking approach. Leaked fragments are reconstructed via a genetic puzzle solving algorithm, which is able to reconstruct the screen data.

basis, meaning that only the window contents of the rendering applications are divided into tiles instead of the whole screen. Thus, if we are able to leak tile data, the leaked tile will only contain pixel data of one application only. First, we need to understand how screen pixels are mapped to tiles. In order to understand the hardware pixel mapping process, we first develop a fragment shader which embedds pixel coordinates in the pixel color. Leaking this data, we can extract the pixel to SIMD execution unit mapping, allowing us to later correctly reconstruct leaked tiles. Moreover, we also made another observation during the reverse engineering process: Not only is pixel color data represented as a floating point between 0.0 and 1.0, but additionally the other parts of the rendering pipeline work on data in that range. As an example, the vertex on-screen positions are also in the same value range, which results in another challenge for pixel reconstruction and requires further attention.

Leaking Multiple Tiles. When leaking multiple tiles, the original input image is not trivial to reconstruct due to the vast amount of leaked small tiles. Jigsaw puzzle solving algorithms are a subclass of computational imaging algorithms, which aim to solve such image puzzle problems for arbitrary input images [34]. In our work, we employ the algorithm proposed by Sholomon et al [38] due to its ability to recover image puzzles with up to 30.000 pieces. The presented algorithm is based on a Genetic Algorithm (GA) approach, which crosses over two parent inputs. Each of the parent inputs is an unfinished arrangement of the tiles, which get processed by a crossover operator. The presented operator is able to convey arrangement information from the parents to the child, if subparts of the image are already solved, thus only keeping suitable information of the parents and rejecting false tile arrangements. Figure 5 depictures the adversary's procedure for leaking data. First, the victim

application has to render its contents on screen in order to leave stale fragment data in registers. Afterwards, the adversary has to read out all of this fragment data. This step requires to properly separate system noise from target data. As shown in the example fragment shader disassembly in Listing 1, register r3 sets the alpha value of the rendered texture to 1.0. Fragments can thus be properly identified if there are uniform distributed leaked values of 1.0 in the leaked data, which further occur around three times as much as other seen values, since registers r0-r2 contain color information. Empirical evaluation shows that this is a reliable approach to identify rendered fragments. Due to register remapping, other leaked values cannot be properly assigned to a single color. We thus choose to reconstruct the data in grayscale color mode, by eliminating fragments with a high similarity, resulting in a single channel of color.

# **5.4.** Leaking Data from Convolutional Neural Networks

Convolutional Neural Networks (CNNs) are networks, whose architecture and connectivity is inspired by the animal visual cortex. By using convolutions, specialized layers apply filters to the input, in order to automatically learn features of images. Albeit being primarily focused on visual tasks, CNNs can also be used for processing other structured data, and have been adapted to text and audio.

CNNs find application in a variety of image-related tasks, such as object detection, classification and feature extraction. Typical building blocks for CNNs are convolutional, pooling and fully connected layers. Convolutional layers act as automatically learned image filters, extract high-level features of an input image. Pooling layers down-sample the image as it passes through the network. Different pooling-functions, such as the maximum or average pooling, multiple pixels of the layer input are mapped to a single pixel on the output, thus reducing the image size. Pooling layers fulfil multiple purposes, from retaining only the most important features in an area, to reducing the computational cost. Typically, the last layer of a CNN is a fully connected layer, with as many neurons as classes to detect.

When attacking CNNs, we specifically target the output of the first convolutional layer. An example output of said layer is shown in Figure 6a. Since CNNs apply multiple filter kernels to the input image, the first output therefore contains the image multiple times. Henceforth, the convolution outputs can give a very strong indication of what the original input values were. Our implementation, which is based on the original authors design [33], consists of four layers: Convolutional, RELU, max pooling and a fully connected layer. Our implementation works on input images of size  $28 \times 28$ . The targeted convolutional layer processes its inputs in 8 thread groups with 576 threads each, and each filter kernel is applied to the input image in parallel in a separate thread group. When attacking the CNN, we consider a realistic scenario, where the CNN in executed only once.



(a) Original convolutional layer output.



(c) Leak pattern on NVIDIA. White pixels indicate that these CNN pixel outputs can be successfully leaked.



(b) Consecutively leaked data on Adreno.



(d) Heuristically reconstructed data on NVIDIA.

Figure 6: CNNs data leakage.

Leaking Data on NVIDIA. On NVIDIA GPUs, we can leak 16 consecutive pixels per single kernel invocation due to memory coalescing. Further examination revealed that we can only leak the first 16 outputs of a warp of size 32, resulting in a leakage pattern as depictured in Figure 6c. One attacker thread group has shown to be capable of recovering one half-wavefront, resulting in the challenge to reconstruct the leaked image from the leaked data. Thus, we implemented a heuristic approach to reconstruct the leaked output: As depicted in the leak pattern, the outputs of leaked half-wavefronts overlap by 8 pixels. Image reconstruction is thus be done by arranging the leaked data such that the pixel difference between two half-wavefronts is minimal at the overlapped region. The resulting reconstruction of the leaked image is shown in Figure 6d.

Leaking Data on Adreno. Experiments revealed that the probability of leaking consecutive data is higher on Adreno due to a less aggressive register remapping strategy. By setting the thread group size to the same one as of the victim, we can leak 256 consecutive pixel values without any need of reordering or any image reconstruction algorithms. Further, we observed that leaked data is always led by a fixed number of zero bytes and registers are less aggressively remapped, resulting in straightforward identification of the leaked image. A single result of a leaked value is depictured in Figure 6b. Our attacker targets an intermediate filter result, which is held in register r2.x, which results in the slightly different image. Parts of the original first layer output are clearly visible. Since CNNs mostly process images, even a distorted image contains relevant information. In a realistic scenario where the victim CNN has only been executed once, an attacker is able to capture up to 44.4% of one convolution first layer output. Another insightful implication of this is that data on Adreno can be leaked across multiple thread groups in order.



Figure 7: Simplified GPT-2 architecture. The attacker targets the output of the positional embedding of the input token embeddings.

#### 5.5. Attacking Large Language Models

Recently, Large Language Models (LLMs) rapidly gained a lot of popularity. LLMs are capable of a variety of language-based tasks, ranging from more simple tasks like sentiment analysis, summarization and translation to complex chain-of-thought tasks [45]. Their versatility leads to their usage for a variety of use-cases, and companies offer convenient programming interfaces for developers to integrate the power of LLMs into their own services for low costs. Besides handling user support tasks such automatically handling customer requests, LLMs are being used as a supportive tool at the workplace. LLM vendors even begin to offer enterprise editions of their products, which promise to not use enterprise customer's data for LLM learning purposes. As such, LLMs handle a huge variety of potentially confidential and highly personal information.

In this section, we present how an attacker can extract information from a GPT-2 LLM running on the same GPU. In this setup, we target a NVIDIA GPU, since the model parameter size explosion of LLMs can only be handled by thoses datacenter-class accelerators.

**5.5.1. GPT-2 Architecture.** The recent rise of LLMs has been thanks to the Transformer architecture, a neural network architecture presented by Google researchers, which is depictured in Figure 7. We specifically target the second version of the Generative Pre-trained Transformer by OpenAI, GPT-2, which is the open-source predecessor to the proprietary model used in ChatGPT [32], [36].

Like other text-processing neural networks, the input is first mapped to an embedding space, which is a vector representation of tokens. As such, semantically similar tokens are spatially near each other in this vector space. When giving a human-readable input, a tokenizer first has to convert this input to a vector in embedding space. GPT-2 particularity uses Byte Pair Encoding (BPE). BPE works on the byte-level instead of a given character encoding, such as ASCII.

The key component to this architecture is the attention mechanism, which allows the network to weigh the importance of different parts of the sequence when generating words. For each word in the sequence, three vectors are provided: query (q), key (k) and value (v). When generating a word, the attention mechanism calculates similarity scores between different parts of the sequence using these vectors. A token can advertise its features via the value and search for matching tokens using the query and key parameters. As a result, the LLM can learn which words are of relevance for each word, enabling it to process and generate long sequences of data. This is a huge advantage in contrast to other, preestablished text processing networks, such as Recurrent Neural Networks (RNNs), which are shown to have difficulties processing long input sequences due to vanishing gradients during learning. The GPT-2 architecture further involves multiple attention mechanisms in parallel - this allows for learning multiple features of word dependence. Not only are semantic connections between words captured, but attention applied in parallel allows for capturing different grammar rules in different heads.

The last important layer in the Transformer is a Feed-Forward Neural Network, which is a 2-layer fully connected network. This adds further trainable parameters to the network, which allow each token to process the previously seen similarities.

**5.5.2. Information Extraction.** Our attack primarily focuses on extracting information from the first calculation of the GPT-2 architecture, where embeddings and positional information are encoded. Embeddings allow to encode tokens as vectors, allowing a neural network to leverage additional information encoded in these vectors. The embedding space is represented by a matrix of size  $50257 \times 768$ , containing a total number of 50257 possible tokens, where each one represented by a vector with a dimension of 768. GPT also employs positional encodings, which, contain information about the position of a token. Since the order of words is an important aspect of language, LLMs leverage this way of encoding the order of words for learning how to process words in the correct order.

In this work, we focus on the implementation of GPT-2 provided by the tinygrad framework [4], because it contains minimal boilerplate code and allows for rapid changes to the framework code, which allowed us to easily debug internals of the GPT-2 implementation. Our attack targets the first step of GPT-2, where tokens are positionally encoded and feed into the Transformer architecture. Since tokens are encoded using a fixed space, each embedding can only be encoded to a fixed number of positions. GPT-2 has a limited context length of 1024, resulting in  $50257 \cdot 1024 = 51463168$  possible vectors, which the result of the encoding can be, thus limiting the search space.

The primary advantage of targeting the embedding layer is that the embedding layer is independent of the

model size. This allows an adversary to target even other LLMs, with billions of parameters, since all LLMs use a variant of an embedding scheme, which allows to scale the attack even to the biggest models available.

**Efficient Reconstruction.** For reconstructing the leaked input/output of GPT-2, we implemented an efficient reconstruction algorithm, which is a necessity since a brute-force search approach exceeds the memory limits on our machine. First, we calculate all possible permutations of embeddings and positional encodings (Line 12). We use the values of all created vectors as an index for look-up-table, where each entry determines whether a value is contained in the matrix and if yes, for which tokens and positions (Line 17). We use the binary integer representation of the floating point values for the table index. We afterwards iterate over all leaked data in chunks of size 16 (Line 22). Now, for all elements in this chunk, we determine if they are part of a embedding/position permutation an record which we encountered how often (Line 28). We record the index of the currently computed value in an embedding. We do this in order to check if we found 16 consecutive values in an embedding vector (Line 39). If that is the case, we can determine that the current chunk is an embedding at a certain position. Our experiments revealed that the reconstruction accuracy is 100%, and we never encountered falsely reconstructed or missing data.

```
data: float[]
2
    max_pos: int
    embeddings: float[50257,
                                    7261
    pos_embeddings: float[50257, 726]
    begin
       lut ← []
       11m_outputs \leftarrow []
10
       // Build LUT
11
       for pos = 0 : max_pos:
12
         for embedding : embeddings:
token_at_pos \leftarrow embedding +
13
15
                             pos_embeddings[pos]
16
17
          for v in token_at_pos:
            // Use integer representation as index
lut[(int)v] ← (pos, embedding)
18
19
       for idx = 0 : length(data); idx+=16:
22
          chunk \leftarrow data[idx:idx+16]
23
          for embedding in embeddings:
24
25
            ctr \leftarrow 0
26
27
            emb\_indices \leftarrow []
            for \ v \ in \ chunk:
29
               // Use integer representation as index
               emb, pos \leftarrow lut [(int)v]
30
31
               if emb, pos \neq (Ø,Ø):
32
                 ctr++
                  // Determine at which index value v
33
34
                  // is in the embedding
35
                 idx_{v\_in\_emb} \leftarrow \text{get\_idx\_in\_emb}(v, \text{emb}, \text{pos})
                 emb_indices ← [
36
37
                      emb_indices, idx_{v\_in\_emb}]
38
39
            if ctr > 16 \land
40
                ascending (embedding_v_indices):
41
               11m_outputs.append(embedding)
42
```

Listing 2: Efficient LLM output reconstruction algorithm.

On our machine (8-core AMD Ryzen 7 3800x, 32GB RAM), the LUT table consumes around 20GB memory and token reconstruction speed is 22.5 seconds/token. Calculating the LUT is a one-time effort, since it can conveniently be stored on disk after creation. Calculating the LUT for up to 30 token position takes 8 minutes.

#### 6. Countermeasures

In this section, we discuss possible countermeasures to thwart exploitation of the presented vulnerability. At first, we discuss a userleve-based countermeasure and two countermeasure approaches on the operating system level next. Afterwards, we discuss the firmware-based approach.

Cleanup after Execution. The self-protective approach, where a kernel tries to protect itself by inserting additional instructions at the end to zero used does only work on AGX and Adreno GPUs. Since shader compilation is only accessible via an abstract interface, this would require either modification of vendor-provided libraries or a runtime-based protection mechanism, which builds upon the shader modification techniques as presented in section 4. This gives application developers which want to confidentially process data on the GPU a simple way of implementing a countermeasure against the shown attacks. Therefore, this approach is a good alternative to operating system-based mitigations, which might not be available on end-user devices, since they require modification and updates of low-level components.

Compartmentalized Compilation. The vulnerability presented in this paper crosses application boundaries to applications by executing a maliciously crafted shader binary on the GPU. As such, a solution is to compartmentalize the shader source compilation in a privileged process with a tightly controlled and abstract interface. Instead of executing the compilation process inside the user application, we propose a privileged system service, a privileged shader compiler, which handles the compilation process. This privileged compiler passes the resulting binary to the kernel and returns a handle to the calling application. Special care needs to be taken to apply proper security and authentication measures to the handle, such that applications cannot execute the compiled shaders of other applications. However, this approach suffers by bloating the Trusted Computing Base (TCB) of the underlying system, by inserting a whole compiler toolchain.

OS Kernel-based Sanitizer. Another approach is to sanitize all compiled shaders before execution on the kernel side. We propose a kernel-based sanitization module, which disassembles every shader prior to its execution. Afterwards, the sanitizer iterates over all disassembled instructions and checks if any uninitialized registers are read. If this is the case, the the shader is rejected for execution.

We argue that this approach is far more lightweight than the compartmented shader compilation technique. However, it cannot be built from existing codebases and needs to be developed from the ground up with the kernel environment in mind. Because most available shader compilers and disassemblers, both open-source and proprietary, use LLVM internally, porting is unfeasible. For one, the size of the LLVM framework would blow up the kernel's size tremendously and might introduce a larger attack surface for software-based attacks against the kernel just as for the compartmentalized compilation approach. For another, LLVM-based disassemblers are not designed with speed in mind, which poses a crucial bottleneck for system performance. Handwritten disassemblers are mostly small in size and therefore the best solution for a software-based approach.

Firmware Modification. Lastly, this type of attack can also be mitigated on the GPU firmware side. As the most trivial solution, the GPU firmware can be modified to clear the contents of a newly allocated register window, mitigating most of the attacks presented in the paper. However, this approach requires development effort on the vendor's side. GPUs typically include secure boot mechanisms for their management firmware, which prevents loading unauthenticated firmware images. As such, this approach cannot be implemented by any other entities than the GPU vendor.

#### 7. Discussion

In this section, we discuss various aspects of our attack and evaluation and present possible future work afterwards.

Bandwidth. Since reading stale register values in a shader is not compute-intensive, the data bandwidth for data leakage is constrained by memory speed. On Adreno however, we detected that the leakage is mostly constrained by the vendor shader dispatch code due to our usage of OpenGL Compute. Leakage bandwidth We aim for our attacks to be as stealthy as possible in order to avoid detection by users on the system. An adversary must be aware of resource contention occurring on the system and might need to lower the interval of kernel executions. For all data leakage attacks, we first read stale registers and simply return the leaked register contents via a shared memory location to the host. We later resume any analysis on the host side, because data analysis on the shader side leads to long shader runtimes, which effectively influences GPU performance much more, making our attacks noticeable to system users. We aim for our attacks to be stealthy and not noticeable for the user. For example, Adreno documentation directly states a maximum shader runtime of 10ms, which should not be exceeded in order to not influence Android GUI rendering, which occurs every 30ms. Estimating an upper bound of data which can be leaked gives us an estimation on how practical these attacks are. A faster shader execution time also gives us the advantage that more shader executions can be dispatched, thus occupying more entries in the scheduler pipelines and thus increasing the chance that the attacker shader is executed diretly after the targeted victim shader.

Limitations. One limitation, as mentioned previously, is only data which is processed within a single SIMD

unit can be leaked in order. Despite this limitation, we presented solutions for various workloads. However, when leaking unstructured data, the problem still remains. Another limitation is the exploitataility from High-Level Code. Our investigation shows that binary shader access is needed in all scenarios for enforcing reading stale register values. All shader compilers on the investigated platforms did not emit stale register reads when e.g. reading an uninitialized variable in high-level shader code. LLVM is the main framework used for shader compilation and supports default variable initialization for many years. We explored Javascript WebGL implementations of three major browsers (Chrome, Firefox and Safari) and found that those implementations initialize registers to zero before usage.

Dedicated Neural Accelerators. Apple devices contain a dedicated Neural Engine (ANE) since the introduction of the iPhone X. Frameworks provided by Apple, which abstract the underlying devices (such as the VisionFramework), often use CoreML under the hood. CoreML can automatically decide with which accelerator backend a neural network should be executed and chooses between CPU, GPU and ANE. This decision process is only partially documented, for example, Apple states that networks needing 32 bit floaing-point numbers are executed only on the CPU and GPU. CoreML chooses to execute the VisionFramework on the GPU instead of the available ANE for the iPhone X, in contrary to the tested MacBook Air, which uses the dedicated accelerator. As such, the frameworks affected by our attack vary from device to device.

**Future Work.** Future work in GPU-related vulnerability research can focus on detecting similar vulnerabilities in other implementations of other vendors. Such vulnerabilities of uninitialized interrupt structures have been found in CPUs [9]. If GPU vendors failed to secure register access, there might also be other architectural structures which are prone to leakage. We further encourage research on leakage of other neural network types, to broaden the scope of the attack.

## 8. Related Work

In the following, we provide a summary of existing GPU-based attacks and classify them into different categories, and further present existing research on countermeasures.

Architectural side-channel Attacks. Like CPUs, GPUs have also been shown to suffer from cache side-channel attacks [5], [13], [44]. Just like their CPU counterparts, GPUs also heavily rely on caches in order to improve system performance. Researchers have previously been targeting the memory model on NVIDIA GPUs, showing that it is possible to leak data from a victim shader by abusing uninitialized memory [11] [24]. These attacks also were able to partially leak some register contents, however, these rely on register spilling to memory. Register spilling is only applied if the executed shader kernel is too big and the number of live variables cannot be hold in

the register set anymore. However, the probability for variables to be spilled to memory is quite low, since most GPUs support between 128 and 256 registers per kernel, making register spilling quite unlikely. Just like their CPU counterparts, GPUs are also vulnerable against performance counter attacks [27]. Performance counters have also been used in a countermeasure context as workload classification method, in order to detect malicious attempts of cryptomining by malware on a user's system [35]. To combat advanced attacks on GPUs, there has been academic effort to secure traditional GPUs in loud scenarios [16], [22], however, they often implement their efforts only on AMD GPUs, because of their mostly open-source software stack.

Software-based Frequency Attacks. Taneja et al. presented a frequency and power side-channel attack targeting GPUs in ARM systems [40]. The researchers showed that the power consuption and the reaction of the GPU's dynamic voltage and frequency scaling (DVFS) mechanisms also relies on the workload and specific data which is processed. This enabled the leaking of pixel values of other applications, as well as profiling the currently visited websites of a user. Further, because the attack relies on frequency measurements, it can even be executed from Javascript code running in the browser. In comparison to our approach, this presented attack has a lower leakage bandwidth, however, it also allows rogue websites to launch the attack. A similar approach has been introduced as a covert channel by González-Gómez et al [15]. Leveraging the emitted temperature, up to 8.75 bps could be covertly transmitted.

An attack leveraging the overclocking capabilities of GPUs has been presented by Sabbagh et al. [37]. The overclocking capabilities expose the frequency to set among other tweakable parameters. By scaling the frequency up to a point where some calculations fault and return compromised results, but not crashing the GPU, the researchers were able to break an GPGPU version of AES via differential fault analysis. Qiu et al. leveraged DVFS for attacking deep neural network inference [39].

Physical Attacks. Beyond the realm of software, researchers have introduced multiple types of physical attacks targeting GPUs. Researchers presented a power-side-channel on a NVIDIA TESLA GPU by measuring the current consumtion at the ATX power suppy [21]. Maia et al showed that a physical attacker can extract information about neural networks running on the GPU via a magnetic inductance probe [23]. More specifically, the researchers were able to extract exact information about the network topology and hyperparameters, but did not leak data processed by these.

Research on resistance against physical manipulation attacks on GPUs has been mostly done in context of reliability, often not security. [14], [20].

**Countermeasures.** Due to previously discovered vulnerabilities in GPUs, various countermeasures have been proposed which cover a wide range of attack vectors. Researchers aimed to apply the concepts of Trusted Execution Environments (TEEs) also on GPUs,

developing multiple GPU TEE implementation in recent years [10], [16], [22], [46]. Due to their popularity for artificial intelligence, some academic work also focuses on providing confidentiality solely for neural network inference and training via GPUs [8], [29].

Not only academic researchers, but also industry GPU vendors seem to slowly adapt security concepts known from CPUs. As such, enterprise solutions are now capable of GPU-based virtualization solutions, and NVIDIA announced TEE compatibility for their H100 flagship lineup, which tightly integrates the CPU TEE implementations AMD SEV, Intel TDX and ARM CCA [30]. However, published documents does not seem to hint at mitigations for register-based attacks being in place.

### 9. Conclusion

In this paper, we introduced uninitialized register reads as a powerful GPU-based exploit vector, which is capable of leaking vast amounts of processed data. During the attack implementation, multiple obstacles emerged, such as the need to reconstruct the leaked data due to black-box thread group scheduling mechanisms. We presented multiple mechanisms to deal with various types of workloads in our evaluation, thereby giving an adversary the capability reconstruct real-world workloads. We have shown, that such reconstruction mechanisms are able to reconstruct pixel information through stale fragment shader registers, and that such methods can also be applied to complex and large neural networks. Finally, we discussed multiple possible countermeasures against the aforementioned attack to thwart uninitialized register read attacks.

#### **Responsible Disclosure**

We disclosed our findings to Apple and NVIDIA in July and to Qualcomm in August of 2023. All vendors acknowledged the vulnerabilities found in the paper. At the time of writing, all three vendors were able to reproduce the issue and have stated to be working on fixing the vulnerabilities.

#### References

- [1] Gpu accelerated ml training. https://learn.microsoft.com/en-us/windows/ai/directml/gpu-accelerated-training, 2023.
- [2] Qualcomm neural processing sdk. https://developer.qualcomm. com/software/qualcomm-neural-processing-sdk, 2023.
- [3] Time-sharing gpus on gke. https://cloud.google.com/ kubernetes-engine/docs/concepts/timesharing-gpus, 2023.
- [4] tinygrad: For something between pytorch and karpathy/micrograd. https://github.com/tinygrad/tinygrad, 2023.
- [5] Jaeguk Ahn, Cheolgyu Jin, Jiho Kim, Minsoo Rhu, Yunsi Fei, David Kaeli, and John Kim. Trident: A hybrid correlation-collision gpu cache timing attack for aes key recovery. In 2021 IEEE International Symposium on High-Performance Computer Architecture (HPCA), pages 332–344. IEEE, 2021.
- 6] Jaeguk Ahn, Jiho Kim, Hans Kasan, Leila Delshadtehrani, Wonjun Song, Ajay Joshi, and John Kim. Network-on-chip microarchitecture-based covert channel in gpus. In MICRO-54: 54th Annual IEEE/ACM International Symposium on Microarchitecture, pages 565–577, 2021.

- [7] Ghadeer Almusaddar and Hoda Naghibijouybari. Exploiting parallel memory write requests for covert channel attacks in integrated cpu-gpu systems. *arXiv preprint arXiv:2307.16123*, 2023.
- [8] Aref Asvadishirehjini, Murat Kantarcioglu, and Bradley Malin. Ginn: Fast gpu-tee based integrity for neural network training. In Proceedings of the Twelfth ACM Conference on Data and Application Security and Privacy, pages 4–15, 2022.
- [9] Pietro Borrello, Andreas Kogler, Martin Schwarzl, Moritz Lipp, Daniel Gruss, and Michael Schwarz. {ÆPIC} leak: Architecturally leaking uninitialized data from the microarchitecture. In 31st USENIX Security Symposium (USENIX Security 22), pages 3917– 3934, 2022.
- [10] Yunjie Deng, Chenxu Wang, Shunchang Yu, Shiqing Liu, Zhenyu Ning, Kevin Leach, Jin Li, Shoumeng Yan, Zhengyu He, Jiannong Cao, et al. Strongbox: A gpu tee on arm endpoints. In *Proceedings* of the 2022 ACM SIGSAC Conference on Computer and Communications Security, pages 769–783, 2022.
- [11] Roberto Di Pietro, Flavio Lombardi, and Antonio Villani. Cuda leaks: Information leakage in gpu architectures. arXiv preprint arXiv:1305.7383, 2013.
- [12] Sankha Baran Dutta, Hoda Naghibijouybari, Nael Abu-Ghazaleh, Andres Marquez, and Kevin Barker. Leaky buddies: Crosscomponent covert channels on integrated cpu-gpu systems. In 2021 ACM/IEEE 48th Annual International Symposium on Computer Architecture (ISCA), pages 972–984. IEEE, 2021.
- [13] Sankha Baran Dutta, Hoda Naghibijouybari, Arjun Gupta, Nael Abu-Ghazaleh, Andres Marquez, and Kevin Barker. Spy in the gpu-box: Covert and side channel attacks on multi-gpu systems. In Proceedings of the 50th Annual International Symposium on Computer Architecture, pages 1–13, 2023.
- [14] Marcio M Goncalves, Fabio Benevenuti, Hector Munoz, Marcelo Brandalero, Michael Hubner, Fernanda Kastensmidt, and Jose Rodrigo Azambuja. Investigating floating-point implementations in a softcore gpu under radiation-induced faults. In 2020 27th IEEE International Conference on Electronics, Circuits and Systems (ICECS), pages 1–4. IEEE, 2020.
- [15] Jeferson González-Gómez, Kevin Cordero-Zuñiga, Lars Bauer, and Jörg Henkel. The first concept and real-world deployment of a gpubased thermal covert channel: Attack and countermeasures. In 2023 Design, Automation & Test in Europe Conference & Exhibition (DATE), pages 1–6. IEEE, 2023.
- [16] Tyler Hunt, Zhipeng Jia, Vance Miller, Ariel Szekely, Yige Hu, Christopher J Rossbach, and Emmett Witchel. Telekine: Secure computing with cloud {GPUs}. In 17th USENIX Symposium on Networked Systems Design and Implementation (NSDI 20), pages 817–833, 2020.
- [17] Sagar Imambi, Kolla Bhanu Prakash, and GR Kanagachidambaresan. Pytorch. Programming with TensorFlow: Solution for Edge Computing Applications, pages 87–104, 2021.
- [18] Zhen Hang Jiang, Yunsi Fei, and David Kaeli. A complete key recovery timing attack on a gpu. In 2016 IEEE International symposium on high performance computer architecture (HPCA), pages 394–405. IEEE, 2016.
- [19] Zhen Hang Jiang, Yunsi Fei, and David Kaeli. A novel sidechannel timing attack on gpus. In *Proceedings of the on Great Lakes Symposium on VLSI 2017*, pages 167–172, 2017.
- [20] Fernanda Kastensmidt and Paolo Rech. Radiation effects and fault tolerance techniques for fpgas and gpus. In FPGAs and Parallel Architectures for Aerospace Applications: Soft Errors and Fault-Tolerant Design, pages 3–17. Springer, 2016.
- [21] Chao Luo, Yunsi Fei, Pei Luo, Saoni Mukherjee, and David Kaeli. Side-channel power analysis of a gpu aes implementation. In 2015 33rd IEEE International Conference on Computer Design (ICCD), pages 281–288. IEEE, 2015.
- [22] Haohui Mai, Jiacheng Zhao, Hongren Zheng, Yiyang Zhao, Zibin Liu, Mingyu Gao, Cong Wang, Huimin Cui, Xiaobing Feng, and Christos Kozyrakis. Honeycomb: Secure and efficient {GPU} executions via static validation. In 17th USENIX Symposium on Operating Systems Design and Implementation (OSDI 23), pages 155–172, 2023.

- [23] Henrique Teles Maia, Chang Xiao, Dingzeyu Li, Eitan Grinspun, and Changxi Zheng. Can one hear the shape of a neural network?: Snooping the gpu via magnetic side channel. arXiv preprint arXiv:2109.07395, 2021.
- [24] Clémentine Maurice, Christoph Neumann, Olivier Heen, and Aurélien Francillon. Confidentiality issues on a gpu in a virtualized environment. In Financial Cryptography and Data Security: 18th International Conference, FC 2014, Christ Church, Barbados, March 3-7, 2014, Revised Selected Papers 18, pages 119–135. Springer, 2014.
- [25] Hoda Naghibijouybari and Nael Abu-Ghazaleh. Covert channels on gpgpus. IEEE Computer Architecture Letters, 16(1):22–25, 2016.
- [26] Hoda Naghibijouybari, Khaled N Khasawneh, and Nael Abu-Ghazaleh. Constructing and characterizing covert channels on gpg-pus. In Proceedings of the 50th Annual IEEE/ACM International Symposium on Microarchitecture, pages 354–366, 2017.
- [27] Hoda Naghibijouybari, Ajaya Neupane, Zhiyun Qian, and Nael Abu-Ghazaleh. Rendered insecure: Gpu side channel attacks are practical. In *Proceedings of the 2018 ACM SIGSAC conference on computer and communications security*, pages 2139–2153, 2018.
- [28] Ajay Nayak, Vinod Ganapathy, and Arkaprava Basu. (mis) managed: A novel tlb-based covert channel on gpus. In *Proceedings of the 2021 ACM Asia Conference on Computer and Communications Security*, pages 872–885, 2021.
- [29] Lucien KL Ng, Sherman SM Chow, Anna PY Woo, Donald PH Wong, and Yongjun Zhao. Goten: Gpu-outsourcing trusted execution of neural network training. In *Proceedings of the AAAI* Conference on Artificial Intelligence, volume 35, pages 14876– 14883, 2021.
- [30] NVIDIA. Confidential compute on nvidia hopper h100 whitepaper. https://images.nvidia.com/aem-dam/en-zz/Solutions/ data-center/HCC-Whitepaper-v1.0.pdf, 2023.
- [31] NVIDIA. Cuda llvm compiler. https://developer.nvidia.com/ cuda-llvm-compiler, 2023.
- [32] OpenAI. Gpt-4 technical report, 2023.
- [33] Keiron O'Shea and Ryan Nash. An introduction to convolutional neural networks. arXiv preprint arXiv:1511.08458, 2015.
- [34] Dolev Pomeranz, Michal Shemesh, and Ohad Ben-Shahar. A fully automated greedy square jigsaw puzzle solver. In CVPR 2011, pages 9–16. IEEE, 2011.
- [35] Claudius Pott, Berk Gulmezoglu, and Thomas Eisenbarth. Overcoming the pitfalls of hpc-based cryptojacking detection in presence of gpus. In Proceedings of the Thirnth ACM Conference on Data and Application Security and Privacy, pages 177–188, 2023.
- [36] Alec Radford, Jeffrey Wu, Rewon Child, David Luan, Dario Amodei, Ilya Sutskever, et al. Language models are unsupervised multitask learners. *OpenAI blog*, 1(8):9, 2019.
- [37] Majid Sabbagh, Yunsi Fei, and David Kaeli. A novel gpu overdrive fault attack. In 2020 57th ACM/IEEE Design Automation Conference (DAC), pages 1–6. IEEE, 2020.
- [38] Dror Sholomon, Omid E David, and Nathan S Netanyahu. An automatic solver for very large jigsaw puzzles using genetic algorithms. Genetic Programming and Evolvable Machines, 17:291–313, 2016.
- [39] Rihui Sun, Pefei Qiu, Yongqiang Lyu, Donsheng Wang, Jiang Dong, and Gang Qu. Lightning: Striking the secure isolation on gpu clouds with transient hardware faults. arXiv preprint arXiv:2112.03662, 2021.
- [40] Hritvik Taneja, Jason Kim, Jie Jeff Xu, Stephan van Schaik, Daniel Genkin, and Yuval Yarom. Hot pixels: Frequency, power, and temperature attacks on gpus and arm socs. arXiv preprint arXiv:2305.12784, 2023.
- [41] Hugo Touvron, Thibaut Lavril, Gautier Izacard, Xavier Martinet, Marie-Anne Lachaux, Timothée Lacroix, Baptiste Rozière, Naman Goyal, Eric Hambro, Faisal Azhar, et al. Llama: Open and efficient foundation language models. arXiv preprint arXiv:2302.13971, 2023.
- 42] Ashish Vaswani, Noam Shazeer, Niki Parmar, Jakob Uszkoreit, Llion Jones, Aidan N Gomez, Łukasz Kaiser, and Illia Polosukhin. Attention is all you need. Advances in neural information processing systems, 30, 2017.

- [43] Frauke Gräter Vedran Miletić, Szilárd Páll. Llvm amdgpu for high performance computing: are we competitive yet? *European LLVM Developers' Meeting*, 2017.
- [44] Yingchen Wang, Riccardo Paccagnella, Zhao Gang, Willy R Vasquez, David Kohlbrenner, Hovav Shacham, and Christopher W Fletcher. Gpu.zip: On the side-channel implications of hardwarebased graphical data compression. In 2024 IEEE Symposium on Security and Privacy (SP), pages 84–84. IEEE Computer Society, 2023.
- [45] Jason Wei, Yi Tay, Rishi Bommasani, Colin Raffel, Barret Zoph, Sebastian Borgeaud, Dani Yogatama, Maarten Bosma, Denny Zhou, Donald Metzler, et al. Emergent abilities of large language models. arXiv preprint arXiv:2206.07682, 2022.
- [46] Ardhi Wiratama Baskara Yudha, Jake Meyer, Shougang Yuan, Huiyang Zhou, and Yan Solihin. Lite: a low-cost practical interoperable gpu tee. In *Proceedings of the 36th ACM International* Conference on Supercomputing, pages 1–13, 2022.