174 lines
5.2 KiB
C++
174 lines
5.2 KiB
C++
/*
|
|
* Copyright (c) 2018-2020, NVIDIA CORPORATION. All rights reserved.
|
|
*
|
|
* NVIDIA CORPORATION and its licensors retain all intellectual property
|
|
* and proprietary rights in and to this software, related documentation
|
|
* and any modifications thereto. Any use, reproduction, disclosure or
|
|
* distribution of this software and related documentation without an express
|
|
* license agreement from NVIDIA CORPORATION is strictly prohibited.
|
|
*
|
|
* See COPYRIGHT.txt for license information
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <assert.h>
|
|
#include <string.h>
|
|
#include <getopt.h>
|
|
#include <sys/time.h>
|
|
#include <cuda.h>
|
|
#include <cuda_runtime.h>
|
|
#include "utils.h"
|
|
|
|
int lat(void *data_d, void *data_d_local, int sizeBytes, int pe, int iter, int skip,
|
|
putget_issue_t iss, dir_t dir, cudaStream_t strm, cudaEvent_t sev, cudaEvent_t eev,
|
|
float *ms, float *us) {
|
|
int status = 0;
|
|
int peer = !pe;
|
|
struct timeval start, stop;
|
|
|
|
if (iss.type == ON_STREAM) {
|
|
if (dir.type == WRITE) {
|
|
for (int i = 0; i < (iter + skip); i++) {
|
|
if (i == skip) CUDA_CHECK(cudaEventRecord(sev, strm));
|
|
nvshmemx_putmem_on_stream((void *)data_d, (void *)data_d_local, sizeBytes, peer,
|
|
strm);
|
|
}
|
|
} else {
|
|
for (int i = 0; i < (iter + skip); i++) {
|
|
if (i == skip) CUDA_CHECK(cudaEventRecord(sev, strm));
|
|
nvshmemx_getmem_on_stream((void *)data_d_local, (void *)data_d, sizeBytes, peer,
|
|
strm);
|
|
}
|
|
}
|
|
CUDA_CHECK(cudaEventRecord(eev, strm));
|
|
CUDA_CHECK(cudaEventSynchronize(eev));
|
|
CUDA_CHECK(cudaEventElapsedTime(ms, sev, eev));
|
|
} else {
|
|
if (dir.type == WRITE) {
|
|
for (int i = 0; i < (iter + skip); i++) {
|
|
if (i == skip) gettimeofday(&start, NULL);
|
|
nvshmem_putmem((void *)data_d, (void *)data_d_local, sizeBytes, peer);
|
|
}
|
|
} else {
|
|
for (int i = 0; i < (iter + skip); i++) {
|
|
if (i == skip) gettimeofday(&start, NULL);
|
|
nvshmem_getmem((void *)data_d_local, (void *)data_d, sizeBytes, peer);
|
|
}
|
|
}
|
|
nvshmem_quiet();
|
|
gettimeofday(&stop, NULL);
|
|
*us = (stop.tv_usec - start.tv_usec) + (stop.tv_sec - start.tv_sec) * 1000000;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
int main(int argc, char *argv[]) {
|
|
int status = 0;
|
|
int mype, npes;
|
|
char *data_d = NULL, *data_d_local = NULL;
|
|
void *data_h_local = NULL;
|
|
uint64_t *size_array = NULL;
|
|
double *latency_array = NULL;
|
|
int num_entries;
|
|
int i;
|
|
read_args(argc, argv);
|
|
|
|
int iter = iters;
|
|
int skip = warmup_iters;
|
|
|
|
init_wrapper(&argc, &argv);
|
|
|
|
mype = nvshmem_my_pe();
|
|
npes = nvshmem_n_pes();
|
|
|
|
if (npes != 2) {
|
|
fprintf(stderr, "This test requires exactly two processes \n");
|
|
status = -1;
|
|
goto finalize;
|
|
}
|
|
|
|
num_entries = floor(log2((float)max_size)) - floor(log2((float)min_size)) + 1;
|
|
size_array = (uint64_t *)calloc(sizeof(uint64_t), num_entries);
|
|
if (!size_array) {
|
|
status = -1;
|
|
goto finalize;
|
|
}
|
|
|
|
latency_array = (double *)calloc(sizeof(double), num_entries);
|
|
if (!latency_array) {
|
|
status = -1;
|
|
goto finalize;
|
|
}
|
|
|
|
data_d = (char *)nvshmem_malloc(max_size);
|
|
CUDA_CHECK(cudaMemset(data_d, 0, max_size));
|
|
|
|
data_h_local = (double *)malloc(sizeof(double));
|
|
if (!data_h_local) {
|
|
fprintf(stderr, "malloc failed \n");
|
|
status = -1;
|
|
goto finalize;
|
|
}
|
|
|
|
memset(data_h_local, 0, sizeof(double));
|
|
|
|
#ifdef _NVSHMEM_REGISTRATION_CACHE_ENABLED
|
|
CUDA_CHECK(cudaMalloc((void **)&data_d_local, max_size));
|
|
#else
|
|
data_d_local = (char *)nvshmem_malloc(max_size);
|
|
#endif
|
|
CUDA_CHECK(cudaMemset(data_d_local, 0, max_size));
|
|
|
|
cudaStream_t strm;
|
|
CUDA_CHECK(cudaStreamCreateWithFlags(&strm, cudaStreamNonBlocking));
|
|
|
|
CUDA_CHECK(cudaDeviceSynchronize());
|
|
|
|
if (mype == 0) {
|
|
float ms, us;
|
|
cudaEvent_t sev, eev;
|
|
CUDA_CHECK(cudaEventCreate(&sev));
|
|
CUDA_CHECK(cudaEventCreate(&eev));
|
|
i = 0;
|
|
for (int size = min_size; size <= max_size; size *= step_factor) {
|
|
lat(data_d, data_d_local, size, mype, iter, skip, putget_issue, dir, strm, sev, eev,
|
|
&ms, &us);
|
|
size_array[i] = size;
|
|
if (putget_issue.type == ON_STREAM) {
|
|
latency_array[i] = ms * 1000 / iter;
|
|
} else {
|
|
latency_array[i] = us / iter;
|
|
}
|
|
i++;
|
|
}
|
|
|
|
print_table_basic("Latency", "None", "size (Bytes)", "latency", "us", '-', size_array,
|
|
latency_array, i);
|
|
CUDA_CHECK(cudaEventDestroy(sev));
|
|
CUDA_CHECK(cudaEventDestroy(eev));
|
|
|
|
nvshmem_barrier_all();
|
|
|
|
} else {
|
|
nvshmem_barrier_all();
|
|
}
|
|
|
|
finalize:
|
|
CUDA_CHECK(cudaStreamDestroy(strm));
|
|
|
|
if (data_d) nvshmem_free(data_d);
|
|
if (size_array) free(size_array);
|
|
if (latency_array) free(latency_array);
|
|
|
|
#ifdef _NVSHMEM_REGISTRATION_CACHE_ENABLED
|
|
if (data_d_local) cudaFree(data_d_local);
|
|
#else
|
|
if (data_d_local) nvshmem_free(data_d_local);
|
|
#endif
|
|
|
|
finalize_wrapper();
|
|
|
|
return status;
|
|
}
|