# A<sup>3</sup>PIM: An Automated, Analytic and Accurate Processing-in-Memory Offloader

Qingcai Jiang\*, Shaojie Tan\*, Junshi Chen and Hong An

School of Computer Science and Technology, University of Science and Technology of China, Hefei, China Email: {jqc, shaojiemike}@mail.ustc.edu.cn, {cjuns, han}@ustc.edu.cn

Abstract—The performance gap between memory and processor has grown rapidly. Consequently, the energy and wall-clock time costs associated with moving data between the CPU and main memory predominate the overall computational cost. The Processing-in-Memory (PIM) paradigm emerges as a promising architecture that mitigates the need for extensive data movements by strategically positioning computing units proximate to the memory. Despite the abundant efforts devoted to building a robust and highly-available PIM system, identifying PIM-friendly segments of applications poses significant challenges due to the lack of a comprehensive tool to evaluate the intrinsic memory access pattern of the segment.

To tackle this challenge, we propose A<sup>3</sup>PIM<sup>1</sup>: an Automated, Analytic and Accurate Processing-in-Memory offloader. We systematically consider the cross-segment data movement and the intrinsic memory access pattern of each code segment via static code analyzer. We evaluate A<sup>3</sup>PIM across a wide range of real-world workloads including GAP and PrIM benchmarks and achieve an average speedup of 2.63x and 4.45x (up to 7.14x and 10.64x) when compared to CPU-only and PIM-only executions, respectively.

Index Terms—Processing-in-Memory, Static Analysis, Workload Offloading

#### I. INTRODUCTION

The widening discrepancy between the speeds of the CPU and main memory system has emerged as a major issue in modern computer systems, colloquially termed as the "memory wall". In demonstrating this, the time required for one offchip memory access from the CPU is approximately 100 times that of a double-precision operation, with energy consumption being about 1000 times greater. Simultaneously, recent years have seen the rise of data-intensive applications such as graph processing and machine learning inference, which expend a significant amount of time and energy transferring data between the CPU and main memory. As a result, this phenomenon of data movement has become a principal concern in contemporary computing systems.

Fortunately, the advent of 3D-stacked memory technologies has heralded the dawn of Processing-in-Memory (PIM) as a promising solution to counter the data movement problem. These PIM architectures integrate computational units within the logic layer of 3D-stacked memory, strategically positioning these units close to the data to reduce data movements. Fig. 1 depicts an typical PIM architecture, where computing units are integrated close the memory side, such as the logic die of Micron's hybrid memory cube (HMC) [7]. The computing units are able to perform logic operations with ultra-high memory access bandwidth. In such a CPU-PIM system, there are two different types of processing units that are suitable for different kinds of workloads: PIM units specialize in workloads with intensive and irregular memory access patterns, while CPUs excel at handling workloads with regular memory access patterns that can be effectively cached.



Fig. 1. High-level PIM architecture

To extract the benefits of such CPU-PIM system, an intelligent offloader must be developed to choose the best-fitting core for each segment of a workload. However, exploiting such an offloader remains a challenge as multiple factors must be jointly taken into account: 1. The compatibility of the segment to the PIM and CPU core in terms of the memory access pattern, as previously discussed. 2. The inter-segment switching cost which involves data movement between PIM and CPU because of data dependency, and the context-switching cost between PIM and CPU threads. ALP [9] tackle this challenge by introducing a novel prefetcher to alleviate data movement overhead. PIMProf [21] answer this question by implementating an automated PIM profiling and offloading tool. Additionally, many previous works aim to offload predetermined segments that have potential to be accelerated by PIM, such as special instructions [2], [12] and function kernels [1]. Despite this, a significant shortcoming of these research efforts is the absence of a holistic tool to assess the intrinsic memory access pattern of each code segment, which is a critical determinant in deciding the offloading policy of the CPU-PIM system.

In this work, we design A<sup>3</sup>PIM, an automated, analytic and accurate PIM offloader for CPU-PIM system. The contributions of this work are the following:

• We identify and characterize the three key factors in the

<sup>\*</sup>These authors contributed equally to this work.

<sup>&</sup>lt;sup>1</sup>The code and benchmarks of this work are opened-sourced in: https://github.com/ACSA-PIM/A3PIM

offloading of PIM segments: 1. The compatibility of the segments to PIM and CPU core, 2. The inter-segment data movement overhead between the PIM and CPU cores when the application is partitioned between them.and 3. The context-switching time between PIM and CPU threads.

- We design a novel basic clock offloader on CPU-PIM system, A<sup>3</sup>PIM, which leverage static code analyzer to evaluate the intrinsic memory access pattern of each basic block and design an novel algorithm that jointly consider the compatibility of the segment to the PIM and CPU core and data movement overhead.
- We evaluate the effictiveness of A<sup>3</sup>PIM on a wide range of workloads: GAP [5] and PrIM [10]. The nummerical results show that the CPU-PIM hybrid execution by A<sup>3</sup>PIM achieve an average speedup of 2.63x and 4.45x (up to 7.14x and 10.64x) when compared to CPU-only and PIM-only executions, respectively.

## II. BACKGROUND AND MOTIVATION

# A. Related Work

Prior workds on PIM offloading exhibits two significant limitations. Firstly, some existing works necessitate additional hardware implementations for runtime scheduling [9], [21], resulting in time-consuming investigations and added hardware design complexity. Secondly, previous studies predominantly concentrate on specific segment properties, such as data locality [2] or cache hit rates [8] during runtime measurements. However, these approaches lack static intrinsic metrics for scheduling units, which could unveil segment affinity for allocation to either the CPU or PIM.

#### B. Static Code Analyzer

A static code analyzer [14], [16] is a tool designed to investigate the intrinsic properties of a given **basic block**, such as the estimated execution time, memory access patterns, and dependencies between each instruction. Previous works [15], [18], [19] about static code analyzers have unveiled the intrinsic properties of code basic blocks, including memory port pressure. This analysis has shed light on the performance preferences of code basic blocks when executed on either PIM or CPU cores, which can serve as a theoretical foundation for decision-making.

## C. Data Movement Overhead

During the offload process, offloading an application's code into segments on PIM or the CPU can lead to a substantial amount of inter-segment data movement overhead, i.e., data generated in one segment and used in other segments. Previous research [20] has demonstrated that the instructions responsible for generating inter-segment data remain consistent across various executions of a program with different input sets. This consistency ensures that we can make offloading decisions statically, without the need for dynamic offloading.

# III. COST MODELING OF A<sup>3</sup>PIM

## A. Componets of execution time in CPU-PIM system

In order to provide good offloading decisions for a program, it is necessary to understand the cost model of different program executions for minimizing the total cost. We identify **two major sources of cost**: 1. The execution cost which is the execution time of the code region on CPU or PIM and 2. The switching cost which is the overhead for maintaining the consistency of data and the program context when switching between CPU and PIM.

1) Execution cost: In the offload decision process, a key concern is that each segment of a program varies in execution time when running on either a CPU or a PIM core. This numerical difference arises from architectural disparities between the PIM and CPU core, such as memory access latency, which can be statically evaluated using the inherent characteristics obtained from a static code analyzer.

2) Switching cost: The switching cost comes from two sources: data dependency and context switch.

The **first** source comes from the data dependency between code regions that are placed on different processing units, e.g., one region on CPU and another on PIM. In this scheme, the data must be transferred from one core's cache to another's. We succinctly refer to this process as cache line data movement (CL-DM).



Fig. 2. Example of (a) Cache Line Data Movement (CL-DM) and Corresponding (b) Context Switch Graph in Continuous ARM64 Assembly Code.

**CL-DM.** We analyze the data dependency between basic blocks at cache line granularity, as memory transfer is cacheline-grained. When the same cache line of data is shared by two program regions that execute in different places (PIM and CPU), we model a single data transfer as the total cost of one cache line flush issued by the source, plus one cache line fetch issued by the destination. Figure 2 shows a code example with multiple basic blocks. Assume that we execute *basic block* 1, *basic block* 3 on CPU and *basic block* 2 on PIM, then when we switch between them, the memory locations of %xmm0 and %rcx each incurs a data dependency cost. The number of data dependency instances increases if there are multiple shared locations. To compute the total data dependency cost of a certain offloading decision, a naive way is to go over all memory accesses, and increase the cost wherever a data transfer happens.

The **second** source comes from the context switch, which mainly includes the overhead of saving and restoring the process or thread states, TLB information etc. Different from the data dependency cost, context movement has a more or less constant cost, which is determined by the operating system. We refer to this cost as light-weight context switch data movement. Previous research has either overestimated the cost of a context switch to 2000 cycles [21] or underestimated it to 0 cycles [9]. In our work, we estimate the context switch cost for CPU-PIM to be around 800 cycles.

This estimated cost is derived from the following factors:

- 1) Given the current programming models, the context switch between PIM and CPU takes place at the thread level, not the process level. As a result, the context switch between threads on PIM and CPU is anticipated to have a more streamlined context, thereby reducing the overhead.
- 2) PIM cores generally make use of Arm cores with L1 caches that adhere to the Virtually Indexed, Physically Tagged (VIPT) scheme [4]. During thread switching, there is no need to alter the virtual address mapping, thus eliminating the need to flush the L1 cache. Consequently, this further lowers the overhead associated with context switching.

To further validate our estimation, we conduct a thread context switching cost test on a real 64-core Armv8 server processor [11] following the experimental steps outlined in [17]. The result approximates to around 800 cycles. Thus, based on these considerations, we adopt an assumed context switch cost of 800 cycles for the CPU-PIM system.

**Context Switch Cost.** The context switch cost arises when two neighboring basic blocks are executed in different cores. Typically, the context switch cost is constant and depends on the operating system. To compute this cost,  $A^3PIM$  keeps track of the program's execution as it crosses the boundary between basic blocks. This is achieved by constructing a weighted directed graph, where each edge's weight represents the number of times the execution transitions from one region to another. Figure 2 shows an example of the context switch graph.

To elaborate on the cost modeling in the CPU-PIM system, we design an experiment to analyze the proportion of each time cost component across a range of applications, as illustrated in Table I. The corresponding simulation methodologies will be detailed in V. We make a key observation that when scheduling is performed at the granularity of basic blocks, the cost of context switch is significantly larger than the cost of CL-DM. This observation provides valuable guidance for our design of data movement overhead elimination, emphasizing the importance of addressing context switch to achieve efficient execution in the system.

 TABLE I

 Comparison of different applications' execution time and data

 movement time using Architecture-Suitability/Greedy offload

 strategy

| Application | Execution Time | Data Movement Time |                       |
|-------------|----------------|--------------------|-----------------------|
|             |                | CL-DM              | <b>Context Switch</b> |
| bc          | 31.37%         | 14.17%             | 54.46%                |
| sssp        | 1.56%          | 1.57%              | 96.86%                |
| bfs         | 49.59%         | 2.21%              | 48.2%                 |
| pr          | 71.74%         | 0.01%              | 28.24%                |
| select      | 8.82%          | 0.0%               | 91.18%                |
| unique      | 10.62%         | 0.0%               | 89.37%                |
| AVÊRAGE     | 28.95%         | 3.0%               | 68.05%                |

While measuring the cost of CL-DM necessitates dynamic tracking of memory traces and cache line usage [9], [21], our observation regarding the relationship between context switch cost and CL-DM suggests that we can achieve CL-DM elimination as a byproduct of context switch cost elimination. This revelation allows us to develop a comprehensive set of static methods for eliminating both context switch cost and CL-DM which will be detailed introduced in Section IV-B.

#### B. Cost modeling formula

Based on the above observations and discussions, we propose the following equation to model the time overhead of scheduling in a CPU-PIM system.

$$Time \ Overhead = \sum_{i \in PIM} PIM_i + \sum_{j \in CPU} CPU_j + \sum_{i \in PIM} \sum_{j \in CPU} (CL_DM(i, j) + CXT(i, j))$$

Here,  $CL_DM(i, j)$  represents the cost of cache line data movement generated when program region i is scheduled on PIM and program region j is scheduled on the CPU. Similarly, CXT(i,j) represents the cost of context switch data movement incurred when unit i is scheduled on PIM and unit j is scheduled on the CPU. Additionally,  $PIM_i$  denotes the execution time of program region i when scheduled on PIM, and  $CPU_j$  denotes the execution time of program region j when scheduled on the CPU. Both  $PIM_i$  and  $CPU_j$  can be collectively referred to as the execution time. Furthermore,  $CL_DM(i, j)$  and CXT(i, j) can be commonly referred to as data movement overhead. This formulation effectively captures the intricate interplay of data movement costs and execution times in the CPU-PIM system, providing a comprehensive model for analyzing scheduling overhead.

In previous works, there have been two main approaches to minimize the overall cost in the above formula. Some works [12], [13] focused on measuring partial values, such as optimizing for overall memory bandwidth savings, where the actual data movement overhead was measured for scheduling purposes. However, these approaches overlooked the benefits in terms of execution time. Other works [21] considered both the execution time of scheduling units and the data movement overhead between them. They designed heuristic



Fig. 3. A<sup>3</sup>PIM overview

algorithms to reduce the time complexity of traversing all possible scheduling combinations, as exhaustively exploring all combinations would have a time complexity of  $2^N$  for N scheduling units.

We make a key observation that the data movement overhead is significantly larger than the execution time. Inspired by this, we propose a two-stage minimization algorithm. The first stage involves clustering to eliminate data movement overhead, and the second stage leverages the intrinsic characteristics of scheduling units to find the scheduling choices with the smallest execution time. By doing so, we obtain near-optimal scheduling results without relying on dynamic measurement information.

This approach effectively addresses the challenge of minimizing the overall cost in the system without dynamically measuring information, taking into account the significant impact of data movement overhead compared to execution time.

# IV. A<sup>3</sup>PIM DESIGN

Based on above observations, we present  $A^3PIM$ , an novel end-to-end compile-time scheduling framework designed for efficient task distribution between the CPU and PIM architectures. The overall workflow of  $A^3PIM$  is shown in Figure 3.

### A. Program Instrumentation

A<sup>3</sup>PIM statically instruments the program using an LLVM [16] compiler pass. It divides the entire program into small regions to enable a fine-grained profiling by inserting lightweight marker functions, and the program regions are used as the basic unit for profiling and offloading. The granularity of program regions is configurable so that applications that vary in size, parallelism and data dependency patterns may all benefit from offloading. In this work, we compare the different granularities of regions within a program, specifically basic blocks [3] and functions.

#### B. Data Movement Alleviation

In the first step of  $A^3$ PIM algorithm, based on automatically-instrumented region boundaries (e.g., basic block or function), we aim to cluster related regions in order to reduce the impact of high inter-segment data movement. Taking into account the two types of data movement, we employ a comprehensive clustering method that considers various factors to mitigate the cost of data movement.

Based on previous research, we integrate two methods: one based on memory access temporal locality metrics [22] and the other on register data dependency between basic blocks [9], aiming to eliminate data movement. Combining these methods, we propose the following formula for connectivity:

$$Connectivity = \frac{\alpha \cdot Memory\_Reuse + (1 - \alpha) \cdot Register\_Reuse}{Instruction\_Count}$$

The connectivity metric calculates the tightness of data movement between two basic blocks, serving as the basis for clustering.

Here,  $\alpha$  is a parameter that balances the contributions of memory access temporal locality and register data dependency to data movement overhead. *Memory\_Reuse* represents the number of times two basic blocks access the same memory address, while *Register\_Reuse* counts the occurrences of accessing the same register between two basic blocks. *Instruction\_Count* denotes the maximum number of instructions among the two basic blocks. Introducing this parameter is essential because, in situations with identical *Memory\_Reuse* and *Register\_Reuse*, a higher number of instructions is more likely to hide data movement latency, and greater instruction count may exceed the cost of scheduling to different locations.

This formula allows us to flexibly adjust the relative importance of both methods, enabling us to achieve optimal data movement elimination results under various scenarios.

A connectivity value of 0 indicates no data reuse, while a value close to 1 suggests a very high data movement cost (i.e., a value near 1 implies that the instructions in basic blocks almost exclusively contain reused memory addresses or registers).

After clustering based on the connectivity value, all the regions of the program are grouped together with their tightlyconnected neighboring regions. This clustering effectively reduces the data movement between these blocks, leading to significant alleviation of data movement overhead.

## C. Preferences based on intrinsic characteristics of clusters

After the clustering step, A<sup>3</sup>PIM collects static statistics for each cluster using a static code analyzer. These statistics include the predicted number of execution cycles, the loadstore port pressure for each cluster, and the quantity and type of instructions in each cluster. We utilize the flollowing static parameters to obtain the optimal offload decision for each cluster:

- Load-store port pressure is a outcome of a static code analyzer, indicating the level of congestion on the loadstore ports during the execution of instructions within a basic block. This information provides crucial insights into the characteristics and resource requirements of each region.
- Arithmetic intensity (AI) is defined as the ratio of computational instructions to memory access instructions within a scheduling unit. This metric quantifies the amount of computation performed per memory request.

Different from previous works, we do not utilize the lastlevel CPU cache misses per Kilo-instruction (MPKI) metric since we consider it would introduce hardware overhead to performance monitor counter and runtime scheduling overhead. Load-store port pressure is capable of fully replacing its functionality.

Algorithm 1 represents the key takeaways we obtain from our memory bottleneck classification. Based on three key metrics, we classify workloads into those that prefer execution on PIM and those that prefer execution on the CPU.

Algorithm 1 Offloading Clusters With the instrinsic characteristics of clusters Input: the basic blocks of the cluster Output: The mapping of each cluster to PIM or CPU 1: if The cluster shows high parallelism then MAP(cluster, PIM); 2: 3: else if The cluster suffer in load-store port pressure then 4: MAP(cluster, PIM); 5: else 6: if The cluster shows high memory intensity then 7: MAP(cluster, PIM); 8: else 9: MAP(cluster, CPU); 10: end if 11: end if 12: 13: end if

In summary, our work shows that employing a static code analyzer can provide sufficient support for PIM scheduling based on static information. By combining cost clustering of data movement in advance, we can effectively eliminate a significant portion of data movement overhead.

# V. EVALUATION METHODOLOGY

## A. Evaluation Configurations

We model a CPU-PIM system on the Sniper simulator [6]. The configuration of this system is listed in Table II. The baseline configuration comprises out-of-order CPU cores, while the PIM configuration includes atom-like in-order cores [1].

TABLE II SYSTEM CONFIGURATION

| Out-of-Order CPU (baseline)                         |  |  |
|-----------------------------------------------------|--|--|
| 1 General purpose processors                        |  |  |
| 3GHz, 4-way superscalar                             |  |  |
| 32kB L1I, 32kB L1D, 256kB L2, 2MB L3                |  |  |
| General-purpose in-order (PIM)                      |  |  |
| 32 general purpose cores                            |  |  |
| 32kB L1I, 32kB L1D                                  |  |  |
| Data movement Cost                                  |  |  |
| Cache line fetch/flush on CPU: 60 ns, on PIM: 30 ns |  |  |
| Register data movement: 2 cache line fetch & flush  |  |  |

### B. Evaluation Workloads

For evaluation, We use two widelyused benchmark suites that contain a variety of workloads to demonstrate the flexibility of  $A^{3}PIM$ : (1) graph benchmark suite (GAP) [5] — high memory intensity and parallelism (bc, sssp, cc, bfs and pr). (2) PrIM benchmarks [10] - the first benchmark suite for a real-world PIM architecture (gemv, select, unique, hashjoin and mlp).

## VI. EVALUATION

## A. System Configurations

In our evaluation, we consider five distinct system configurations as the baseline models for comparison against A<sup>3</sup>PIM.

- **CPU-only** refers to offload all regions to CPU.
- PIM-only refers to offload all regions to PIM.
- MPKI-based refers to offload the regions to PIM if the MPKI of the program region exceeds the threshold.
- Architecture-Suitability/Greedy refers to offload to the min execution-cost location without considering the data dependency cost.
- **Theoretical Upper Bound (TUB)** refers to the minimum value obtained by exploring all possible permutations of scheduling combinations.
- A<sup>3</sup>PIM: Data-movement-aware & intrinsiccharacteristics-aware: A<sup>3</sup>PIM will undergo evaluation with two distinct offloading granularities: *function-level scheduling granularity* (A<sup>3</sup>PIM-func) and *basic-blocklevel scheduling granularity* (A<sup>3</sup>PIM-bbls).

# B. Performance

Figure 4 depicts the execution time breakdown for each application using six different offloading strategies.

We make four key oberservations from these results: **First**, not all applications benefit from complete offloading to PIM. Graph applications and memory-intensive applications such as gemv, select, and unique demonstrate significant improvements, while hashjoin and mlp underperform, resulting in



Fig. 4. Execution time breakdown of GAP and PrIM workloads using different offloading decisions

a speed decrease in PIM-ONLY strategy on average. Second, both the traditional MPKI and Greedy methods suffer from substantial data movement overhead, primarily attributed to Context Switch costs rather than CL-DM. However, the Greedy method proves suitable for specific applications like cc, gemv, and mlp, which consist of multiple independent kernels. Third, A<sup>3</sup>PIM-bbls is performing better than A<sup>3</sup>PIMfunc. A<sup>3</sup>PIM-func provides a 1.25× speedup over CPU-only and a 2.11× speedup over PIM-only, while A<sup>3</sup>PIM-bbls offers a 2.63× speedup over CPU-only and a 4.45× speedup over PIM-only. This performance difference arises from the coarse partitioning introduced by function-level granularity, leading to a loss in the potential performance gains achievable through finer-grained scheduling. And last, A<sup>3</sup>PIM-bbls effectively offloads all application cases, including PIM-friendly graph applications, CPU-friendly applications like hashjoin, and even uncovers the PIM benefit potential in specified applications such as mlp, where the PIM-ONLY method leads to performance degradation. Consequently, A<sup>3</sup>PIM-bbls approaches the theoretical peak attainable performance speedup of 4.56x over PIM-ONLY.

# VII. CONCLUSION

In this work, we proposes A<sup>3</sup>PIM, an automated, analytic and accurate offloading framework for CPU-PIM systems. By leveraging static code analysis to model the intrinsic characteristics of code segments, A<sup>3</sup>PIM is able to make informed task partitioning and scheduling decisions to minimize data movement overheads. Our evaluations on real-world benchmarks demonstrate that A<sup>3</sup>PIM can effectively extract the performance benefits of the PIM architecture and achieve average of 2.63x and 4.45x speedups compared to CPU-only and PIM-only executions. The proposed techniques provide a promising direction to enable performance-aware scheduling for heterogeneous CPU-PIM systems without costly runtime profiling.

#### ACKNOWLEDGMENT

This work is partly supported by Strategic Priority Research Program of the Chinese Academy of Sciences (Grant No. XDB0500102) and the National Natural Science Foundation of China (Grant No. 62102389).

#### REFERENCES

- Junwhan Ahn *et al.*, "A scalable processing-in-memory accelerator for parallel graph processing", in *ISCA*, 2015.
   Junwhan Ahn *et al.*, "PIM-enabled instructions: A low-overhead,
- [2] Junwhan Ahn et al., "PIM-enabled instructions: A low-overhead, locality-aware processing-in-memory architecture", in ACM SIGARCH Computer Architecture News, 2015.
- [3] Frances E Allen. "Control flow analysis", in ACM Sigplan Notices, 1970.
- [4] ARM. ARM Cortex-A Series Programmer's Guide for ARMv7-A, URL: https://developer.arm.com/documentation/den0013/d/Caches/Cachearchitecture/Virtual-and-physical-tags-and-indexes.
- [5] S. Beamer et al., "The GAP benchmark suite", CoRR, 2015. [Online]. Available: http://arxiv.org/abs/1508.03619
- [6] Trevor E Carlson et al., "An evaluation of high-level mechanistic core models", in TACO, 2014.
- [7] Hybrid Memory Cube Consortium, "HMC Specification 2.0", 2014.
- [8] Mingyu Gao, Grant Ayers, and Christos Kozyrakis. "Practical near-data processing for in-memory analytics frameworks", in *PACT*, 2015.
- [9] Nika Mansouri Ghiasi et al., "ALP: Alleviating CPU-Memory Data Movement Overheads in Memory-Centric Systems", in TETC, 2022.
- [10] Juan Gómez-Luna *et al.*, "Benchmarking Memory-centric Computing Systems: Analysis of Real Processing-in-Memory Hardware", in *IGSC*, 2021.
- [11] HiSilicon, "Kunpeng 920 Chipset", https://www.hisilicon.com/en/products /Kunpeng/Huawei-Kunpeng/Huawei-Kunpeng-920, 2021.
- [12] Kevin Hsieh et al., "Transparent offloading and mapping (TOM) enabling programmer-transparent near-data processing in GPU systems", in ACM SIGARCH Computer Architecture News, 2016.
- [13] Nai, Lifeng, *et al.*, "Graphpim: Enabling instruction-level pim offloading in graph computing frameworks", in *HPCA*, 2017.
- [14] Intel, "Intel Architecture Code Analyzer User's Guide", https://software. intel.com/content/dam/develop/external/us/en/documents/intelarchitecture-code-analyzer-3-0-users-guide-157552.pdf, 2017.
- [15] Qingcai Jiang *et al.*, "Quantifying Throughput of Basic Blocks on ARM Microarchitectures by Static Code Analyzers: A Case Study on Kunpeng 920", in *HPCC*, 2022.
- [16] Chris Lattner and Vikram Adve., "LLVM: A compilation framework for lifelong program analysis & transformation", in CGO, 2004.
- [17] James Litton *et al.*, "Light-Weight Contexts: An OS Abstraction for Safety and Performance", in OSDI, 2016.
- [18] Charith Mendis *et al.*, "Ithemal: Accurate, portable and fast basic block throughput estimation using deep neural networks", in *International Conference on machine learning*, 2019.
- [19] Tan Shaojie *et al.*, "Uncovering the performance bottleneck of modern HPC processor with static code analyzer: a case study on Kunpeng 920", in *CCF Trans. HPC*, 2023.
- [20] M Aater Suleman *et al.*, "Data marshaling for multi-core architectures", in *ISCA*, 2010.
- [21] Yizhou Wei et al., "PIMProf: an automated program profiler for processing-in-memory offloading decisions", in DATE, 2022.
- [22] J. Weinberg *et al.*, "Quantifying Locality In The Memory Access Patterns of HPC Applications", in *SC*, 2005.