-
Notifications
You must be signed in to change notification settings - Fork 357
/
Copy pathbenchmark_hlif.hpp
205 lines (168 loc) · 6.36 KB
/
benchmark_hlif.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
/*
* SPDX-FileCopyrightText: Copyright (c) 2022-2024 NVIDIA CORPORATION & AFFILIATES.
* All rights reserved. SPDX-License-Identifier: LicenseRef-NvidiaProprietary
*
* NVIDIA CORPORATION, its affiliates and licensors retain all intellectual
* property and proprietary rights in and to this material, related
* documentation and any modifications thereto. Any use, reproduction,
* disclosure or distribution of this material and related documentation
* without an express license agreement from NVIDIA CORPORATION or
* its affiliates is strictly prohibited.
*/
#pragma once
// Benchmark performance from the binary data file fname
#include <vector>
#include <numeric>
// TODO: Our Windows x86 CI images for some reason do not contain the NVTX headers,
// whereas the regular Windows CTK installations do contain them. Check why.
#if defined(__x86_64) && !defined(_MSC_VER)
#define NVTX_ENABLED
#endif // defined(__x86_64) && !defined(_MSC_VER)
#ifdef NVTX_ENABLED
#include <nvtx3/nvToolsExt.h>
#endif // NVTX_ENABLED
#include "benchmark_common.h"
#include "nvcomp.hpp"
#include "nvcomp/nvcompManagerFactory.hpp"
using namespace nvcomp;
const int chunk_size = 1 << 16;
template<typename T = uint8_t>
void run_benchmark(
const std::vector<T>& data, nvcompManagerBase& batch_manager, int verbose_memory,
cudaStream_t stream, const int benchmark_exec_count = 1, const bool warmup = true)
{
size_t input_element_count = data.size();
// Make sure dataset fits on GPU to benchmark total compression
size_t freeMem;
size_t totalMem;
CUDA_CHECK(cudaMemGetInfo(&freeMem, &totalMem));
if (freeMem < input_element_count * sizeof(T)) {
std::cout << "Insufficient GPU memory to perform compression." << std::endl;
exit(1);
}
std::cout << "----------" << std::endl;
std::cout << "uncompressed (B): " << data.size() * sizeof(T) << std::endl;
T* d_in_data;
const size_t in_bytes = sizeof(T) * input_element_count;
CUDA_CHECK(cudaMalloc(&d_in_data, in_bytes));
CUDA_CHECK(
cudaMemcpy(d_in_data, data.data(), in_bytes, cudaMemcpyHostToDevice));
auto compress_config = batch_manager.configure_compression(in_bytes);
size_t comp_out_bytes = compress_config.max_compressed_buffer_size;
benchmark_assert(
comp_out_bytes > 0, "Output size must be greater than zero.");
// Allocate temp workspace
uint8_t* d_comp_out;
CUDA_CHECK(cudaMalloc(&d_comp_out, comp_out_bytes));
// Launch compression
cudaEvent_t start, end;
CUDA_CHECK(cudaEventCreate(&start));
CUDA_CHECK(cudaEventCreate(&end));
if (warmup) {
#ifdef NVTX_ENABLED
nvtxRangePush("compress_warmup");
#endif // NVTX_ENABLED
batch_manager.compress(
d_in_data,
d_comp_out,
compress_config);
cudaStreamSynchronize(stream);
#ifdef NVTX_ENABLED
nvtxRangePop();
#endif // NVTX_ENABLED
}
std::vector<float> compress_run_times(benchmark_exec_count);
for (int ix_run = 0; ix_run < benchmark_exec_count; ++ix_run) {
#ifdef NVTX_ENABLED
nvtxRangePush("compress");
#endif // NVTX_ENABLED
CUDA_CHECK(cudaEventRecord(start, stream));
batch_manager.compress(
d_in_data,
d_comp_out,
compress_config);
CUDA_CHECK(cudaEventRecord(end, stream));
CUDA_CHECK(cudaStreamSynchronize(stream));
#ifdef NVTX_ENABLED
nvtxRangePop();
#endif // NVTX_ENABLED
comp_out_bytes = batch_manager.get_compressed_output_size(d_comp_out);
float compress_ms;
CUDA_CHECK(cudaEventElapsedTime(&compress_ms, start, end));
compress_run_times[ix_run] = compress_ms;
}
// compute average run time.
std::cout << "comp_size: " << comp_out_bytes
<< ", compressed ratio: " << std::fixed << std::setprecision(2)
<< (double)data.size() * sizeof(T) / comp_out_bytes << std::endl;
std::cout << "compression throughput (GB/s): "
<< average_gbs(compress_run_times, data.size() * sizeof(T)) << std::endl;
CUDA_CHECK(cudaFree(d_in_data));
std::vector<float> decompress_run_times(benchmark_exec_count);
auto decomp_config = batch_manager.configure_decompression(d_comp_out);
// allocate output buffer
const size_t decomp_bytes = decomp_config.decomp_data_size;
uint8_t* decomp_out_ptr;
CUDA_CHECK(cudaMalloc(&decomp_out_ptr, decomp_bytes));
if (warmup) {
#ifdef NVTX_ENABLED
nvtxRangePush("decomp warmup");
#endif // NVTX_ENABLED
batch_manager.decompress(decomp_out_ptr, d_comp_out, decomp_config);
CUDA_CHECK(cudaStreamSynchronize(stream));
#ifdef NVTX_ENABLED
nvtxRangePop();
#endif // NVTX_ENABLED
}
for (int ix_run = 0; ix_run < benchmark_exec_count; ++ix_run) {
// get output size
if (verbose_memory) {
std::cout << "decompression memory (input+output+temp) (B): "
<< (decomp_bytes + comp_out_bytes)
<< std::endl;
}
#ifdef NVTX_ENABLED
nvtxRangePush("decomp");
#endif // NVTX_ENABLED
CUDA_CHECK(cudaEventRecord(start, stream));
// execute decompression (asynchronous)
batch_manager.decompress(decomp_out_ptr, d_comp_out, decomp_config);
CUDA_CHECK(cudaEventRecord(end, stream));
CUDA_CHECK(cudaStreamSynchronize(stream));
#ifdef NVTX_ENABLED
nvtxRangePop();
#endif // NVTX_ENABLED
float decompress_ms;
CUDA_CHECK(cudaEventElapsedTime(&decompress_ms, start, end));
decompress_run_times[ix_run] = decompress_ms;
}
CUDA_CHECK(cudaEventDestroy(start));
CUDA_CHECK(cudaEventDestroy(end));
std::cout << "decompression throughput (GB/s): "
<< average_gbs(decompress_run_times, decomp_bytes) << std::endl
<< "decompression time: "
<< std::accumulate(decompress_run_times.begin(), decompress_run_times.end(), 0.0) / benchmark_exec_count
<< " ms."
<< std::endl;
CUDA_CHECK(cudaFree(d_comp_out));
benchmark_assert(
decomp_bytes == input_element_count * sizeof(T),
"Decompressed result incorrect size.");
std::vector<T> res(input_element_count);
cudaMemcpy(
res.data(),
decomp_out_ptr,
input_element_count * sizeof(T),
cudaMemcpyDeviceToHost);
CUDA_CHECK(cudaFree(decomp_out_ptr));
// check the size
#if VERBOSE > 1
// dump output data
std::cout << "Output" << std::endl;
for (size_t i = 0; i < data.size(); i++)
std::cout << reinterpret_cast<T*>(decomp_out_ptr)[i] << " ";
std::cout << std::endl;
#endif
benchmark_assert(res == data, "Decompressed data does not match input.");
}
#undef NVTX_ENABLED