

# ACCELERATING MATRIX MULTIPLICATION: A PERFORMANCE COMPARISON BETWEEN MULTI-CORE CPU AND GPU

MUFAKIR QAMAR ANSARI AND MUDABIR QAMAR ANSARI

**ABSTRACT.** Matrix multiplication is a foundational operation in scientific computing and machine learning, yet its computational complexity makes it a significant bottleneck for large-scale applications. The shift to parallel architectures, primarily multi-core CPUs and many-core GPUs, is the established solution, and these systems are now ubiquitous from datacenters to consumer laptops. This paper presents a direct, empirical performance analysis of matrix multiplication on a modern, consumer-grade heterogeneous platform. We implemented and benchmarked three versions of the algorithm: a baseline sequential C++ implementation, a parallel version for its multi-core CPU using OpenMP, and a massively parallel version for its discrete GPU using CUDA with shared memory optimizations. The implementations were evaluated with square matrices of varying dimensions, from 128x128 to 4096x4096. Our results show that while the parallel CPU provides a consistent speedup of 12-14x over the sequential version, the GPU's performance scales dramatically with problem size. For a 4096x4096 matrix, the GPU implementation achieved a speedup of approximately 593x over the sequential baseline and 45x over the optimized parallel CPU version. These findings quantitatively demonstrate the profound impact of many-core GPU architectures on accelerating data-parallel workloads, underscoring that significant performance gains are readily accessible even on consumer-level hardware.

## 1. Introduction

**1.1. Context and Motivation.** Matrix-matrix multiplication constitutes a foundational computational kernel, underpinning a vast spectrum of algorithms in scientific computing, data science, and artificial intelligence [1]. Its applications are pervasive, forming the computational core of linear algebra libraries (BLAS), enabling the training of deep neural networks, and driving simulations in fields from physics to finance [2]. The significance of this operation is further underscored by its central role in emerging paradigms like neuromorphic computing [3]. However, the algorithm's asymptotic complexity of  $O(n^3)$ , where  $n$  is the matrix dimension, presents a formidable computational challenge. As datasets and model sizes continue to grow exponentially, this operation frequently becomes a primary performance bottleneck, demanding architectural and algorithmic solutions that can deliver extreme-scale performance.

**1.2. The Rise of Parallelism.** The computational demands of such operations have long surpassed the capabilities of sequential processing. The era of improving performance by increasing the clock frequency of single-core processors has concluded, halted by the fundamental physical constraints of power consumption and heat dissipation—a phenomenon often termed

---

*Date:* July 29, 2025.

*2020 Mathematics Subject Classification.* Primary 68W10; Secondary 65Y05, 68M20.

*Key words and phrases.* High-Performance Computing, GPU, CUDA, OpenMP, Matrix Multiplication, Parallel Computing.

the “power wall” [4]. This architectural inflection point has forced the high-performance computing (HPC) industry to embrace parallelism as the primary path to greater computational power. This has led to the development of two dominant, yet architecturally divergent, classes of processors: the multi-core Central Processing Unit (CPU) and the many-core Graphics Processing Unit (GPU). These parallel architectures are no longer confined to supercomputers; they are now a standard component in commodity hardware, including the consumer-grade laptop used in this study.

CPUs are engineered for low-latency execution on a wide variety of tasks, employing sophisticated control logic and deep cache hierarchies to accelerate single-thread performance. In contrast, GPUs are designed as throughput-oriented engines, featuring thousands of simpler, highly-efficient cores that excel at executing the same operation on massive datasets in parallel [5]. The co-location of these distinct architectures in a single system gives rise to the field of heterogeneous computing [6], where the central challenge is to effectively map computational tasks to the hardware best suited for them.

**1.3. Research Objective.** Against this backdrop of architectural divergence, this paper provides a direct, empirical performance comparison of matrix multiplication on a modern, consumer-grade heterogeneous platform, comprising a multi-core CPU and a many-core GPU. We aim to quantify the performance differences and analyze the scaling behavior of each architecture across a range of problem sizes. To this end, we implement and benchmark a standard sequential C++ version alongside optimized parallel implementations using OpenMP for the CPU [7] and the CUDA framework for the GPU [8]. By presenting clear, reproducible data from a widely available hardware configuration, this study seeks to illuminate the practical performance characteristics of these competing hardware paradigms for a workload that is fundamental to the entire HPC ecosystem.

**1.4. Paper Structure.** The remainder of this paper is structured as follows. Section 2 reviews the existing literature on CPU-GPU performance analysis and optimization principles. Section 3 details the methodology, including the specific algorithm implementations and the experimental setup. Section 4 presents the performance results, which are then analyzed and discussed in Section 5. Finally, Section 6 offers our conclusions and suggests directions for future research.

## 2. Related Work

The performance characteristics of High-Performance Computing (HPC) architectures have been a subject of intense research, particularly with the rise of General-Purpose computing on Graphics Processing Units (GPGPU). This study builds upon a significant body of work that compares CPU and GPU capabilities, explores optimization principles, and evaluates the programming models used to harness these powerful processors.

**2.1. The Performance Landscape of CPUs and GPUs.** The relative performance of multi-core CPUs and many-core GPUs has been a subject of extensive investigation, often framed by claims of orders-of-magnitude speedups. A seminal study by Lee et al. [9] sought to debunk the “100X GPU vs. CPU myth,” demonstrating that when applications are highly optimized for both architectures—leveraging SIMD vectorization and multi-threading on the CPU and CUDA principles on the GPU—the performance gap narrows substantially to an average of 2.5x in favor of the GPU. Our work serves as a contemporary validation of this

principle, applying a similar comparative analysis to a foundational HPC kernel on modern hardware.

Subsequent research has consistently shown that the ideal architecture is highly dependent on the algorithm’s characteristics. Teodoro et al. [10] conducted a performance analysis across CPUs, GPUs, and Intel’s Many Integrated Core (MIC) architecture, finding that GPUs excel at tasks with regular, predictable data access patterns, a key feature of dense matrix multiplication. This conclusion is echoed by Huang et al. [2], who analyzed matrix multiplication specifically and noted that the GPU’s performance advantage increases dramatically with matrix size—a trend our results confirm. The broader field of heterogeneous computing, which aims to leverage the unique strengths of different processors, is thoroughly surveyed by Mittal and Vetter [6], who frame the motivation for combining, rather than competing, CPU and GPU resources.

**2.2. Optimization Principles for GPU Computing.** Achieving high performance on many-core GPUs is contingent upon a deep understanding of their underlying architecture; it is not merely a matter of offloading code. An early and influential paper by Fatahalian et al. [11] analyzed the efficiency of GPU algorithms for matrix-matrix multiplication and identified the ratio between arithmetic computation and memory bandwidth as the key limiting factor. This insight remains central to GPU optimization today.

The work by Ryoo et al. [5] provides a foundational set of optimization principles for the CUDA programming model. They established that effective management of the GPU’s memory hierarchy is paramount for performance. Their work highlights the critical importance of achieving coalesced global memory access to maximize effective bandwidth and of utilizing the on-chip shared memory to reduce latency and increase data reuse. The CUDA kernel implemented in our study directly applies these fundamental principles to minimize data movement and maximize computational throughput. More recent work has explored even more specialized optimizations for non-square or sparse matrices [12, 13], demonstrating that performance can be further enhanced by tailoring algorithms to specific data structures.

**2.3. Programming Models and the HPC Ecosystem.** This study utilizes two dominant, industry-standard programming models to harness the parallel capabilities of the target architectures: OpenMP and CUDA. The work by Dagum and Menon [7] describes OpenMP as a portable, directive-based API designed for accessible, incremental parallelization of code on shared-memory systems, which aligns with our straightforward parallelization of the CPU algorithm. For the GPU, the official NVIDIA CUDA C Programming Guide [8] serves as the definitive reference for the CUDA execution model, memory hierarchy, and programming interface that our implementation is built upon.

The choice between GPU programming models has also been explored in the literature. Karimi et al. [14] presented a performance comparison between CUDA and OpenCL, the other major GPGPU framework. They found that for NVIDIA hardware, CUDA often delivers superior performance in both data transfer and kernel execution, which they attribute to the tight vertical integration of NVIDIA’s hardware, compiler, and API. This finding helps justify our use of CUDA as a representative model for a high-performance GPU implementation. These programming models are essential tools for tackling the challenges and opportunities presented by the industry-wide shift to parallel computing, a landscape defined by Asanovic et al. [4].

### 3. Methodology

To conduct our comparative analysis, we developed and benchmarked three distinct implementations of a square matrix multiplication algorithm ( $C = A \times B$ ). These implementations were designed to represent a baseline sequential case and optimized parallel cases for both multi-core CPU and many-core GPU architectures.

#### 3.1. Implementations.

**3.1.1. Sequential CPU.** The baseline for our study is a standard, "naive" C++ implementation of matrix multiplication. It consists of three nested `for` loops, iterating through the rows of matrix A, the columns of matrix B, and the inner dimension, respectively. This version serves as the fundamental reference point against which all speedups are calculated.

**3.1.2. Parallel CPU (OpenMP).** To leverage the shared-memory parallelism of the host CPU, the sequential code was annotated with OpenMP directives. Specifically, we used the `#pragma omp parallel for collapse(2)` directive applied to the two outer loops. The `collapse(2)` clause is critical for performance, as it instructs the OpenMP runtime to merge the two nested loops into a single, larger iteration space, allowing for more effective load balancing across the CPU's 16 available threads.

**3.1.3. Parallel GPU (CUDA).** The GPU implementation was developed using the CUDA C++ framework to run on the system's dedicated NVIDIA GPU. A custom kernel was written wherein the workload is partitioned such that each thread is responsible for calculating a single element of the resulting matrix C. To mitigate the high latency of global memory access, which is a primary bottleneck in GPU computing, our kernel makes extensive use of on-chip shared memory. Before computation, threads within a thread block cooperatively load small, contiguous blocks (or tiles) of the input matrices A and B into this fast, shared memory. The subsequent matrix multiplication is then performed using data from this low-latency memory, dramatically reducing global memory traffic and increasing overall computational throughput.

**3.2. Experimental Setup.** All benchmarks were executed on a Lenovo IdeaPad Gaming 3 15ACH6 laptop to ensure a consistent hardware and software environment.

- **Hardware:** The test system was configured with the following components:
  - **CPU:** An 8-core, 16-thread AMD Ryzen 7 5800H processor with 16 MB of L3 cache.
  - **GPU:** An NVIDIA GeForce GTX 1650 Mobile GPU with 4096 MiB of dedicated VRAM. All CUDA computations were executed exclusively on this device.
  - **System Memory:** 32 GB of DDR4 RAM, operating at a configured speed of 2667 MT/s.
- **Software:**
  - **Operating System:** Ubuntu 24.04.2 LTS (Kernel 6.8.0-64-generic).
  - **Compilers and Toolchains:** The CPU and GPU codes were compiled using g++ version 12.4.0 and the NVIDIA CUDA Toolkit version 12.2, respectively. Standard -O3 optimization flags were enabled for all compilations.
  - **GPU Driver:** The system was running NVIDIA Driver version 535.247.01.
- **Benchmarking Protocol:** The experiments were conducted on square matrices with dimensions ranging from 128x128 to 4096x4096. Input matrices were populated with random 32-bit floating-point values. The execution time for each implementation was

measured using C++’s `std::chrono::high_resolution_clock`. For the GPU implementation, the measured wall-clock time is comprehensive, including the time required for memory allocation on the device, the transfer of input matrices from host memory to device memory (`cudaMemcpyHostToDevice`), kernel execution, and the transfer of the result matrix back from device memory to host (`cudaMemcpyDeviceToHost`).

**3.3. Performance Metrics.** We used two primary metrics to evaluate and compare the performance of the implementations:

- (1) **Execution Time:** The total wall-clock time, measured in milliseconds (ms), required to complete the matrix multiplication operation.
- (2) **Speedup:** A dimensionless quantity that quantifies the performance improvement of a parallel implementation relative to the sequential baseline. It is calculated as:  $S = T_{\text{sequential}}/T_{\text{parallel}}$ .

## 4. Results

This section presents the empirical data from our performance benchmarks. We first provide the raw execution times and calculated speedups in a tabular format, followed by a series of visualizations that illustrate the performance trends across the different implementations and matrix sizes.

**4.1. Performance Data.** The execution times for the sequential CPU, parallel CPU (OpenMP), and parallel GPU (CUDA) implementations were recorded across seven different square matrix dimensions, from 128x128 to 4096x4096. From these timings, we calculated the speedup of each parallel approach relative to the sequential baseline, as well as the direct speedup of the GPU over the parallel CPU. The comprehensive results are presented in Table 1.

TABLE 1. Execution Times (in milliseconds) and Speedups for Matrix Multiplication.

| Matrix Size (N x N) | Seq. CPU   | Par. CPU  | Par. GPU  | Calculated Speedups |                 |             |
|---------------------|------------|-----------|-----------|---------------------|-----------------|-------------|
|                     | Time (ms)  | Time (ms) | Time (ms) | Par. CPU vs Seq.    | GPU vs Par. CPU | GPU vs Seq. |
| 128x128             | 2.18       | 7.10      | 0.26      | 0.31x               | 27.02x          | 8.29x       |
| 256x256             | 20.70      | 2.89      | 0.40      | 7.16x               | 7.18x           | 51.43x      |
| 512x512             | 264.37     | 19.43     | 2.10      | 13.60x              | 9.25x           | 125.77x     |
| 1024x1024           | 3721.52    | 295.86    | 13.35     | 12.58x              | 22.16x          | 278.71x     |
| 2048x2048           | 44 691.46  | 3554.41   | 124.01    | 12.57x              | 28.66x          | 360.38x     |
| 3072x3072           | 171 811.07 | 11 998.88 | 332.69    | 14.32x              | 36.07x          | 516.42x     |
| 4096x4096           | 393 280.52 | 30 332.07 | 663.24    | 12.97x              | 45.73x          | 592.97x     |

**4.2. Performance Visualizations.** To better illustrate the performance characteristics and scaling trends, the data from Table 1 are visualized in the following figures.

Figure 1 plots the execution time of all three implementations as a function of matrix size. A logarithmic scale is used for the Y-axis (Execution Time) to effectively visualize the data. The execution times span several orders of magnitude, from over 390,000 ms for the largest sequential run to under 1 ms for the smallest GPU runs. A linear scale would render the performance differences for the two parallel methods at smaller matrix sizes indistinguishable, while obscuring the overall trend.

Figure 2 presents the speedup achieved by the parallel CPU and parallel GPU implementations relative to the sequential CPU baseline. This bar chart clearly illustrates the substantial



FIGURE 1. A comparison of execution times for sequential CPU, parallel CPU (OpenMP), and parallel GPU (CUDA) implementations across varying matrix sizes. The logarithmic Y-axis is used to accommodate the wide range of execution times.

performance gains offered by both parallel approaches as the problem size increases, with the GPU demonstrating a significantly higher rate of improvement.

Finally, Figure 3 provides a direct comparison of the two parallel architectures by plotting the speedup of the parallel GPU over the parallel CPU. This visualization isolates the performance advantage of the many-core GPU architecture relative to the multi-core CPU, showing a clear trend of increasing advantage as the matrix dimensions grow.

## 5. Discussion

The results presented in the previous section provide a clear quantitative measure of the performance differences between the multi-core CPU and many-core GPU architectures within a consumer-grade laptop. This section provides an interpretation of these results, contextualizes them within the broader academic literature, and acknowledges the scope and limitations of this study.

**5.1. Analysis of Results.** Our empirical data reveals three distinct performance regimes that highlight the fundamental architectural trade-offs between the AMD Ryzen 7 5800H CPU and the NVIDIA GeForce GTX 1650 Mobile GPU.

First, for small problem sizes, the cost of parallelization can outweigh its benefits. This phenomenon, known as parallel overhead, is evident in our results for the 128x128 matrix (Table 1), where the parallel OpenMP implementation is significantly slower than the sequential version (7.10 ms vs. 2.18 ms). This performance degradation is attributable to the overhead of



FIGURE 2. Speedup of the parallel CPU (OpenMP) and parallel GPU (CUDA) implementations relative to the sequential CPU baseline. Note that the Y-axis is on a logarithmic scale to clearly display the vast difference in speedup factors.

thread creation, management, and synchronization on the CPU, which for a computationally trivial workload, consumes more time than is saved by parallel execution.

Second, as the matrix dimensions increase, the parallel CPU implementation demonstrates strong and consistent scaling. For matrices of size 256x256 and larger, the OpenMP version achieves a stable speedup of approximately 12-14x over the sequential baseline. This indicates that for computationally significant workloads, the OpenMP model effectively utilizes the 8 cores and 16 threads of the Ryzen 7 processor, providing a substantial and predictable performance improvement characteristic of mature shared-memory parallel programming.

Third, the results clearly demonstrate the asymptotic dominance of the GPU for large-scale, data-parallel tasks, even in a mobile form factor. While the parallel CPU provides a consistent speedup, the GTX 1650's performance advantage grows super-linearly as the matrix size increases. For the 4096x4096 matrix, the GPU is not only  $\sim 593$ x faster than the sequential CPU but also over 45x faster than the highly-optimized parallel CPU implementation. This is a particularly striking result, showing that even a mobile, entry-level discrete GPU can dramatically outperform a powerful host CPU on a data-parallel task. This performance gain is a direct consequence of the GPU's many-core architecture, which exploits the immense parallelism inherent in matrix multiplication to a degree that a multi-core CPU, with its fewer but more complex cores, cannot match.

**5.2. Contextualizing with Existing Literature.** Our findings are in strong agreement with the conclusions of Lee et al. [9], who argued that while GPUs offer a distinct performance



FIGURE 3. The performance advantage of the parallel GPU implementation relative to the parallel CPU implementation. The plot shows that the GPU’s lead grows as the computational workload increases.

advantage, claims of 100–1000x speedups often result from comparing an optimized GPU implementation against unoptimized sequential CPU code. Our results affirm this perspective; the speedup of the GPU is a formidable  $\sim 45x$  when compared to an *optimized, parallel* CPU implementation on the same machine. This figure is both impressive and realistic, aligning with their finding that the true performance gap between fully-optimized implementations on both architectures is significant but not mythological. Our work thus serves as a contemporary validation of this principle on a foundational HPC kernel, demonstrating its applicability even on consumer-grade heterogeneous hardware.

**5.3. Limitations of the Study.** It is important to acknowledge the boundaries of this research. First, the implementations, while robust, represent one approach among many. The CUDA kernel, though effective in its use of shared memory, could be further optimized with more advanced techniques such as register tiling or dynamic block sizing, which might yield additional performance. Second, this study was conducted on a single hardware configuration. The precise speedup ratios and performance crossover points are dependent on the specific CPU and GPU models used. We expect that higher-end hardware would yield larger absolute performance numbers, but the overall qualitative trends and architectural insights are likely to remain consistent.

## 6. Conclusion and Future Work

**6.1. Summary of Findings.** This study conducted an empirical performance comparison of dense matrix multiplication on a consumer-grade laptop equipped with an 8-core AMD Ryzen

7 CPU and an NVIDIA GeForce GTX 1650 Mobile GPU. Our results reaffirm that for computationally intensive, data-parallel tasks, the choice of architecture has a profound impact on performance, even on widely accessible hardware. We demonstrated that while an optimized, multi-threaded CPU implementation provides a substantial and consistent speedup over sequential execution, the massively parallel architecture of the mobile GPU offers a performance advantage that scales dramatically with the problem size. For large-scale matrices, the GPU implementation was orders of magnitude faster than the sequential baseline and significantly outperformed the parallel CPU version by a factor of over 45x. These findings underscore that the immense throughput capability of many-core architectures is not confined to high-end data-center hardware, but is a critical and accessible tool for accelerating demanding computational kernels on modern heterogeneous systems.

**6.2. Future Work.** The results of this study suggest several promising directions for future research.

First, the GPU implementation could be enhanced by incorporating more advanced optimization techniques. While our current kernel effectively uses shared memory to reduce global memory traffic, further performance gains could be realized by implementing sophisticated memory tiling and register blocking strategies [5]. Such methods could further improve data locality and reduce the latency of memory operations, potentially pushing performance closer to the hardware’s theoretical peak.

Second, the comparative methodology employed in this paper could be extended to other important HPC kernels. Matrix multiplication is just one of the foundational “dwarfs” of scientific computing [4]. A similar analysis of other common kernels, such as Fast Fourier Transforms (FFTs), stencil computations, or sparse matrix operations, would provide a more comprehensive understanding of the performance trade-offs between CPU and GPU architectures across a wider range of computational patterns. This would contribute valuable data to the broader discussion on workload characterization for heterogeneous systems.

Finally, a crucial next step would be to apply the optimized GPU kernel to a real-world scientific application to measure its end-to-end impact. While micro-benchmarks are essential for architectural analysis, integrating the accelerated kernel into a complete application—such as a deep learning training framework or a computational fluid dynamics simulation—would quantify the practical, wall-clock performance improvement on a complete scientific workflow. This would validate the real-world utility of the architectural advantages demonstrated in this paper.

## References

- [1] V. Eijkhout, *Introduction to high performance scientific computing*. Lulu. com, 2010.
- [2] Z. Huang, N. Ma, S. Wang, and Y. Peng, “GPU computing performance analysis on matrix multiplication,” *The Journal of Engineering*, vol. 2019, no. 23, pp. 9043–9048, 2019.
- [3] K. S. Mohamed, “Neuromorphic computing and beyond,” (*No Title*), 2020.
- [4] K. Asanovic, R. Bodik, B. Catanzaro, J. Gebis, P. Husbands, K. Keutzer, D. Patterson, W. Plishker, J. Shalf, and S. W. Williams, “The landscape of parallel computing research: A view from berkeley,” 2006.
- [5] S. Ryoo, C. I. Rodrigues, S. S. Baghsorkhi, S. S. Stone, D. B. Kirk, and W.-m. W. Hwu, “Optimization principles and application performance evaluation of a multithreaded gpu using cuda,” in *Proceedings of the 13th ACM SIGPLAN Symposium on Principles and practice of parallel programming*, 2008, pp. 73–82.
- [6] S. Mittal and J. S. Vetter, “A survey of cpu-gpu heterogeneous computing techniques,” *ACM Computing Surveys (CSUR)*, vol. 47, no. 4, pp. 1–35, 2015.
- [7] L. Dagum and R. Menon, “Openmp: an industry standard api for shared-memory programming,” *IEEE computational science and engineering*, vol. 5, no. 1, pp. 46–55, 1998.

- [8] NVIDIA Corporation, “CUDA C Programming Guide,” 2024, version retrieved in July 2025. [Online]. Available: <https://docs.nvidia.com/cuda/cuda-c-programming-guide/index.html>
- [9] V. W. Lee, C. Kim, J. Chhugani, M. Deisher, D. Kim, A. D. Nguyen, N. Satish, M. Smelyanskiy, S. Chen-nupaty, P. Hammarlund *et al.*, “Debunking the 100x gpu vs. cpu myth: an evaluation of throughput computing on cpu and gpu,” in *Proceedings of the 37th annual international symposium on Computer architecture*, 2010, pp. 451–460.
- [10] G. Teodoro, T. Kurc, G. Andrade, J. Kong, R. Ferreira, and J. Saltz, “Performance analysis and efficient execution on systems with multi-core cpus, gpus and mics,” *arXiv preprint arXiv:1505.03819*, 2015.
- [11] K. Fatahalian, J. Sugerman, and P. Hanrahan, “Understanding the efficiency of gpu algorithms for matrix-matrix multiplication,” in *Proceedings of the ACM SIGGRAPH/EUROGRAPHICS conference on Graphics hardware*, 2004, pp. 133–137.
- [12] J. Chen, N. Xiong, X. Liang, D. Tao, S. Li, K. Ouyang, K. Zhao, N. DeBardeleben, Q. Guan, and Z. Chen, “Tsm2: optimizing tall-and-skinny matrix-matrix multiplication on gpus,” in *Proceedings of the ACM International Conference on Supercomputing*, 2019, pp. 106–116.
- [13] A. Monakov, A. Lokhmotov, and A. Avetisyan, “Automatically tuning sparse matrix-vector multiplication for gpu architectures,” in *International Conference on High-Performance Embedded Architectures and Compilers*. Springer, 2010, pp. 111–125.
- [14] K. Karimi, N. G. Dickson, and F. Hamze, “A performance comparison of cuda and opencl,” *arXiv preprint arXiv:1005.2581*, 2010.

## Appendix A. Benchmark Source Code

The complete C++/CUDA source code used to generate all performance data in this paper is provided below for reproducibility.

```

1 #include <iostream>
2 #include <vector>
3 #include <string>
4 #include <iomanip>
5 #include <cstdlib>
6 #include <ctime>
7 #include <chrono>
8 #include <omp.h>
9 #include <cuda_runtime.h>
10
11 // =====
12 // CUDA Error Checking Wrapper
13 // =====
14 #define CUDA_CHECK(err) { \
15     cudaError_t err_ = (err); \
16     if (err_ != cudaSuccess) { \
17         std::cerr << "CUDA Error in " << __FILE__ << " line " << __LINE__ \
18             << ": " << cudaGetErrorString(err_) << std::endl; \
19         exit(EXIT_FAILURE); \
20     } \
21 }
22
23 // =====
24 // Matrix Multiplication Implementations
25 // =====
26
27 void initializeMatrix(std::vector<float>& matrix, int size) {
28     for (int i = 0; i < size * size; ++i) {
29         matrix[i] = static_cast<float>(rand()) / static_cast<float>(RAND_MAX);
30     }
31 }
32

```

```

33 void sequentialMatrixMultiply(const std::vector<float>& A, const std::vector<
34   float>& B, std::vector<float>& C, int size) {
35   for (int row = 0; row < size; ++row) {
36     for (int col = 0; col < size; ++col) {
37       float sum = 0.0f;
38       for (int k = 0; k < size; ++k) {
39         sum += A[row * size + k] * B[k * size + col];
40       }
41       C[row * size + col] = sum;
42     }
43   }
44
45 void openmpMatrixMultiply(const std::vector<float>& A, const std::vector<float>&
46   B, std::vector<float>& C, int size) {
47 #pragma omp parallel for collapse(2)
48   for (int row = 0; row < size; ++row) {
49     for (int col = 0; col < size; ++col) {
50       float sum = 0.0f;
51       for (int k = 0; k < size; ++k) {
52         sum += A[row * size + k] * B[k * size + col];
53       }
54       C[row * size + col] = sum;
55     }
56   }
57
58 __global__ void matrixMulGpu(const float* A, const float* B, float* C, int size)
59 {
60   int row = blockIdx.y * blockDim.y + threadIdx.y;
61   int col = blockIdx.x * blockDim.x + threadIdx.x;
62
63   if (row < size && col < size) {
64     float sum = 0.0f;
65     for (int k = 0; k < size; ++k) {
66       sum += A[row * size + k] * B[k * size + col];
67     }
68     C[row * size + col] = sum;
69   }
70
71 // =====
72 // Main Benchmark Runner
73 // =====
74 int main() {
75   srand(static_cast<unsigned>(time(0)));
76
77   std::vector<int> matrix_sizes = {128, 256, 512, 1024, 2048, 3072, 4096};
78
79   // Print the header for our CSV data table
80   std::cout << "Matrix_Size,Sequential_CPU_ms,Parallel_CPU_ms,Parallel_GPU_ms,
81   "
82   << "Speedup_CPU_vs_Seq,Speedup_GPU_vs_CPU,Speedup_GPU_vs_Seq" <<
83   std::endl;
84
85   for (int N : matrix_sizes) {

```

```

84 // --- Host Memory Allocation & Initialization ---
85 std::vector<float> h_A(N * N);
86 std::vector<float> h_B(N * N);
87 std::vector<float> h_C(N * N, 0.0f); // Re-use one C matrix for all
88 results
89
90 initializeMatrix(h_A, N);
91 initializeMatrix(h_B, N);
92
93 // --- 1. Sequential CPU Benchmark ---
94 auto start_seq = std::chrono::high_resolution_clock::now();
95 sequentialMatrixMultiply(h_A, h_B, h_C, N);
96 auto end_seq = std::chrono::high_resolution_clock::now();
97 std::chrono::duration<double, std::milli> duration_seq = end_seq -
98 start_seq;
99
100 // --- 2. Parallel CPU (OpenMP) Benchmark ---
101 auto start_omp = std::chrono::high_resolution_clock::now();
102 openmpMatrixMultiply(h_A, h_B, h_C, N);
103 auto end_omp = std::chrono::high_resolution_clock::now();
104 std::chrono::duration<double, std::milli> duration_omp = end_omp -
105 start_omp;
106
107 // --- 3. Parallel GPU (CUDA) Benchmark ---
108 float *d_A, *d_B, *d_C;
109 size_t matrix_bytes = N * N * sizeof(float);
110 CUDA_CHECK(cudaMalloc(&d_A, matrix_bytes));
111 CUDA_CHECK(cudaMalloc(&d_B, matrix_bytes));
112 CUDA_CHECK(cudaMalloc(&d_C, matrix_bytes));
113
114 auto start_gpu = std::chrono::high_resolution_clock::now();
115 CUDA_CHECK(cudaMemcpy(d_A, h_A.data(), matrix_bytes,
116 cudaMemcpyHostToDevice));
117 CUDA_CHECK(cudaMemcpy(d_B, h_B.data(), matrix_bytes,
118 cudaMemcpyHostToDevice));
119 dim3 threadsPerBlock(16, 16);
120 dim3 numBlocks((N + threadsPerBlock.x - 1) / threadsPerBlock.x, (N +
121 threadsPerBlock.y - 1) / threadsPerBlock.y);
122 matrixMulGpu<<<numBlocks, threadsPerBlock>>>(d_A, d_B, d_C, N);
123 CUDA_CHECK(cudaMemcpy(h_C.data(), d_C, matrix_bytes,
124 cudaMemcpyDeviceToHost));
125 auto end_gpu = std::chrono::high_resolution_clock::now();
126 std::chrono::duration<double, std::milli> duration_gpu = end_gpu -
127 start_gpu;
128
129 CUDA_CHECK(cudaFree(d_A));
130 CUDA_CHECK(cudaFree(d_B));
131 CUDA_CHECK(cudaFree(d_C));
132
133 // --- 4. Calculate Speedups ---
134 double speedup_cpu_vs_seq = duration_seq.count() / duration_omp.count();
135 double speedup_gpu_vs_cpu = duration_omp.count() / duration_gpu.count();
136 double speedup_gpu_vs_seq = duration_seq.count() / duration_gpu.count();
137
138 // --- 5. Report Results for this Size ---
139 std::cout << N << "x" << N << ","

```

```

132     << std::fixed << std::setprecision(4) << duration_seq.count()
133     << " , "
134     << std::fixed << std::setprecision(4) << duration_omp.count()
135     << " , "
136     << std::fixed << std::setprecision(4) << duration_gpu.count()
137     << " , "
138     << std::fixed << std::setprecision(2) << speedup_cpu_vs_seq <<
139     "x , "
140     << std::fixed << std::setprecision(2) << speedup_gpu_vs_cpu <<
141     "x , "
142     << std::fixed << std::setprecision(2) << speedup_gpu_vs_seq <<
143     "x" << std::endl;
144 }
145 std::cout << "\nBenchmark complete." << std::endl;
146 return 0;
}

```

LISTING 1. Complete C++/CUDA source code for the benchmark.

DEPARTMENT OF ELECTRICAL ENGINEERING AND COMPUTER SCIENCE, THE UNIVERSITY OF TOLEDO, TOLEDO,  
OHIO 43606, USA

*Email address:* mufakir.ansari@utoledo.edu

DEPARTMENT OF SCHOOL OF ACCOUNTING AND INFORMATION SYSTEMS, LAMAR UNIVERSITY, BEAUMONT,  
TEXAS 77710, USA

*Email address:* mansari2@lamar.edu