464 строки
18 KiB
C++
464 строки
18 KiB
C++
/*************************************************************************
|
|
* Copyright (c) 2022 Advanced Micro Devices, Inc. All rights reserved.
|
|
*
|
|
* See LICENSE.txt for license information
|
|
************************************************************************/
|
|
|
|
#include "CollectiveArgs.hpp"
|
|
#include "PrepDataFuncs.hpp"
|
|
#include <cstdio>
|
|
#include <hip/hip_runtime.h>
|
|
|
|
namespace RcclUnitTesting
|
|
{
|
|
ErrCode DefaultPrepareDataFunc(CollectiveArgs &collArgs)
|
|
{
|
|
switch (collArgs.funcType)
|
|
{
|
|
case ncclCollBroadcast: return DefaultPrepData_Broadcast(collArgs);
|
|
case ncclCollReduce: return DefaultPrepData_Reduce(collArgs, false);
|
|
case ncclCollAllGather: return DefaultPrepData_Gather(collArgs, true);
|
|
case ncclCollReduceScatter: return DefaultPrepData_ReduceScatter(collArgs);
|
|
case ncclCollAllReduce: return DefaultPrepData_Reduce(collArgs, true);
|
|
case ncclCollGather: return DefaultPrepData_Gather(collArgs, false);
|
|
case ncclCollScatter: return DefaultPrepData_Scatter(collArgs);
|
|
case ncclCollAlltoAll: return DefaultPrepData_AllToAll(collArgs);
|
|
case ncclCollAlltoAllv: return DefaultPrepData_AllToAllv(collArgs);
|
|
case ncclCollSend: return DefaultPrepData_Send(collArgs);
|
|
case ncclCollRecv: return DefaultPrepData_Recv(collArgs);
|
|
default:
|
|
ERROR("Unknown func type %d\n", collArgs.funcType);
|
|
return TEST_FAIL;
|
|
}
|
|
}
|
|
|
|
ErrCode CheckAllocation(CollectiveArgs const& collArgs)
|
|
{
|
|
if (collArgs.numInputElements > collArgs.numInputElementsAllocated)
|
|
{
|
|
ERROR("Number of input elements (%lu) exceeds the number of allocated input elements (%lu)\n",
|
|
collArgs.numInputElements, collArgs.numInputElementsAllocated);
|
|
return TEST_FAIL;
|
|
}
|
|
|
|
if (collArgs.numOutputElements > collArgs.numOutputElementsAllocated)
|
|
{
|
|
ERROR("Number of output elements (%lu) exceeds the number of allocated output elements (%lu)\n",
|
|
collArgs.numOutputElements, collArgs.numOutputElementsAllocated);
|
|
return TEST_FAIL;
|
|
}
|
|
|
|
// Check bias allocation if bias is enabled
|
|
if (collArgs.options.useBias)
|
|
{
|
|
if (collArgs.numBiasElements == 0 || collArgs.numBiasBytesAllocated == 0)
|
|
{
|
|
ERROR("Bias is enabled but bias buffers are not allocated\n");
|
|
return TEST_FAIL;
|
|
}
|
|
if (collArgs.numBiasElements != collArgs.numOutputElements)
|
|
{
|
|
ERROR("Number of bias elements (%lu) must match number of output elements (%lu)\n",
|
|
collArgs.numBiasElements, collArgs.numOutputElements);
|
|
return TEST_FAIL;
|
|
}
|
|
}
|
|
return TEST_SUCCESS;
|
|
}
|
|
|
|
ErrCode DefaultPrepData_Broadcast(CollectiveArgs &collArgs)
|
|
{
|
|
CHECK_CALL(CheckAllocation(collArgs));
|
|
if (collArgs.numInputElements != collArgs.numOutputElements)
|
|
{
|
|
ERROR("Number of input elements must match number of output elements for Broadcast\n");
|
|
return TEST_FAIL;
|
|
}
|
|
|
|
size_t const numBytes = collArgs.numInputElements * DataTypeToBytes(collArgs.dataType);
|
|
|
|
// Clear output for all ranks (done before filling input in case of in-place)
|
|
CHECK_CALL(collArgs.outputGpu.ClearGpuMem(numBytes));
|
|
|
|
// Only root needs input pattern
|
|
if (collArgs.globalRank == collArgs.options.root)
|
|
CHECK_CALL(collArgs.inputGpu.FillPattern(collArgs.dataType,
|
|
collArgs.numInputElements,
|
|
collArgs.options.root, true));
|
|
|
|
// Otherwise all other ranks expected output is the same as input of root
|
|
return collArgs.expected.FillPattern(collArgs.dataType,
|
|
collArgs.numInputElements,
|
|
collArgs.options.root,
|
|
false);
|
|
}
|
|
|
|
ErrCode DefaultPrepData_Reduce(CollectiveArgs &collArgs, bool const isAllReduce)
|
|
{
|
|
CHECK_CALL(CheckAllocation(collArgs));
|
|
if (collArgs.numInputElements != collArgs.numOutputElements)
|
|
{
|
|
ERROR("Number of input elements must match number of output elements for Reduce\n");
|
|
return TEST_FAIL;
|
|
}
|
|
|
|
size_t const numBytes = collArgs.numInputElements * DataTypeToBytes(collArgs.dataType);
|
|
|
|
// Clear output for all ranks (done before filling input in case of in-place)
|
|
CHECK_CALL(collArgs.outputGpu.ClearGpuMem(numBytes));
|
|
|
|
// Clear expected buffer for holding reduction
|
|
PtrUnion result;
|
|
CHECK_CALL(result.Attach(collArgs.expected));
|
|
CHECK_CALL(result.ClearCpuMem(numBytes));
|
|
|
|
// If average or custom reduction operator is used, perform a summation instead
|
|
ncclRedOp_t const tempOp = (collArgs.options.redOp >= ncclAvg ? ncclSum : collArgs.options.redOp);
|
|
|
|
// Loop over each rank and generate their input into a temp buffer, then reduce
|
|
PtrUnion scalarsPerRank;
|
|
scalarsPerRank.Attach(collArgs.options.scalarTransport.ptr);
|
|
|
|
PtrUnion tempInputCpu;
|
|
CHECK_CALL(tempInputCpu.Attach(collArgs.outputCpu));
|
|
for (int rank = 0; rank < collArgs.totalRanks; ++rank)
|
|
{
|
|
// Generate temporary input for this rank
|
|
if (collArgs.options.inputConstantValue >= 0)
|
|
{
|
|
// Use constant value for all input elements across all ranks
|
|
// This is useful for ncclProd at high rank counts to avoid factorial overflow
|
|
for (size_t i = 0; i < collArgs.numInputElements; i++)
|
|
{
|
|
CHECK_CALL(tempInputCpu.Set(collArgs.dataType, i,
|
|
collArgs.options.inputConstantValue,
|
|
(double)collArgs.options.inputConstantValue));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Use rank-based pattern: value[rank][i] = (rank + i) % 256 (default behavior)
|
|
CHECK_CALL(tempInputCpu.FillPattern(collArgs.dataType, collArgs.numInputElements, rank, false));
|
|
}
|
|
|
|
// Copy the pre-scaled input into GPU memory for the correct rank
|
|
if (rank == collArgs.globalRank)
|
|
{
|
|
CHECK_HIP(hipMemcpy(collArgs.inputGpu.ptr, tempInputCpu.ptr, numBytes, hipMemcpyHostToDevice));
|
|
}
|
|
|
|
// Scale the temporary input by local scalar for this rank
|
|
// (Used by custom reduction ops)
|
|
if (collArgs.options.scalarMode >= 0)
|
|
{
|
|
CHECK_CALL(tempInputCpu.Scale(collArgs.dataType, collArgs.numInputElements,
|
|
scalarsPerRank, rank));
|
|
}
|
|
|
|
// Any rank that requires output reduces the scaled-inputs
|
|
if (isAllReduce || collArgs.options.root == collArgs.globalRank)
|
|
{
|
|
if (rank == 0)
|
|
{
|
|
memcpy(result.ptr, tempInputCpu.ptr, numBytes);
|
|
}
|
|
else
|
|
{
|
|
CHECK_CALL(result.Reduce(collArgs.dataType, collArgs.numInputElements,
|
|
tempInputCpu, tempOp));
|
|
}
|
|
}
|
|
}
|
|
|
|
// Perform averaging if necessary
|
|
if (collArgs.options.redOp == ncclAvg && (isAllReduce || collArgs.options.root == collArgs.globalRank))
|
|
{
|
|
CHECK_CALL(result.DivideByInt(collArgs.dataType, collArgs.numInputElements, collArgs.totalRanks));
|
|
}
|
|
|
|
// Add bias to expected output if bias is enabled
|
|
if (collArgs.options.useBias && (isAllReduce || collArgs.options.root == collArgs.globalRank))
|
|
{
|
|
// Initialize bias data on CPU
|
|
if (collArgs.options.biasConstantValue >= 0)
|
|
{
|
|
// Use constant value for all bias elements (useful for ncclProd to avoid overflow)
|
|
for (size_t i = 0; i < collArgs.numBiasElements; i++)
|
|
{
|
|
CHECK_CALL(collArgs.biasCpu.Set(collArgs.dataType, i,
|
|
collArgs.options.biasConstantValue,
|
|
(double)collArgs.options.biasConstantValue));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Use incremental pattern: bias[i] = i (default behavior)
|
|
CHECK_CALL(collArgs.biasCpu.FillPattern(collArgs.dataType, collArgs.numBiasElements, 0, false));
|
|
}
|
|
|
|
// Copy bias data to GPU
|
|
size_t const biasBytes = collArgs.numBiasBytesAllocated;
|
|
CHECK_HIP(hipMemcpy(collArgs.biasGpu.ptr, collArgs.biasCpu.ptr, biasBytes, hipMemcpyHostToDevice));
|
|
|
|
// Apply bias to expected output using the SAME reduction operation as AllReduce
|
|
CHECK_CALL(result.Reduce(collArgs.dataType, collArgs.numInputElements, collArgs.biasCpu, tempOp));
|
|
|
|
// Update the biasPtr in options to point to the GPU buffer
|
|
collArgs.options.biasPtr = collArgs.biasGpu.ptr;
|
|
}
|
|
|
|
return TEST_SUCCESS;
|
|
}
|
|
|
|
ErrCode DefaultPrepData_Gather(CollectiveArgs &collArgs, bool const isAllGather)
|
|
{
|
|
CHECK_CALL(CheckAllocation(collArgs));
|
|
if (collArgs.totalRanks * collArgs.numInputElements != collArgs.numOutputElements)
|
|
{
|
|
ERROR("# of output elements must be total ranks * # input elements for AllGather\n");
|
|
return TEST_FAIL;
|
|
}
|
|
|
|
// Clear output for all ranks (done before filling input in case of in-place)
|
|
size_t const numInputBytes = collArgs.numInputElements * DataTypeToBytes(collArgs.dataType);
|
|
size_t const numOutputBytes = collArgs.numOutputElements * DataTypeToBytes(collArgs.dataType);
|
|
CHECK_CALL(collArgs.inputGpu.ClearGpuMem(numInputBytes));
|
|
CHECK_CALL(collArgs.outputGpu.ClearGpuMem(numOutputBytes));
|
|
|
|
PtrUnion result;
|
|
CHECK_CALL(result.Attach(collArgs.expected.ptr));
|
|
CHECK_CALL(result.ClearCpuMem(numOutputBytes));
|
|
|
|
// Use outputCpu buffer to store temporary input
|
|
PtrUnion tempInputCpu;
|
|
CHECK_CALL(tempInputCpu.Attach(collArgs.outputCpu.ptr));
|
|
|
|
for (int rank = 0; rank < collArgs.totalRanks; ++rank)
|
|
{
|
|
CHECK_CALL(tempInputCpu.FillPattern(collArgs.dataType, collArgs.numInputElements, rank, false));
|
|
if (rank == collArgs.globalRank)
|
|
{
|
|
CHECK_HIP(hipMemcpy(collArgs.inputGpu.ptr, tempInputCpu.ptr, numInputBytes, hipMemcpyHostToDevice));
|
|
}
|
|
if (isAllGather || collArgs.options.root == collArgs.globalRank)
|
|
{
|
|
memcpy(result.I1 + (rank * numInputBytes), tempInputCpu.ptr, numInputBytes);
|
|
}
|
|
}
|
|
return TEST_SUCCESS;
|
|
}
|
|
|
|
ErrCode DefaultPrepData_ReduceScatter(CollectiveArgs &collArgs)
|
|
{
|
|
CHECK_CALL(CheckAllocation(collArgs));
|
|
if (collArgs.numInputElements != collArgs.numOutputElements * collArgs.totalRanks)
|
|
{
|
|
ERROR("# of input elements must be total ranks * # output elements for ReduceScatter\n");
|
|
return TEST_FAIL;
|
|
}
|
|
|
|
size_t const numInputBytes = collArgs.numInputElements * DataTypeToBytes(collArgs.dataType);
|
|
size_t const numOutputBytes = collArgs.numOutputElements * DataTypeToBytes(collArgs.dataType);
|
|
|
|
// Clear output for all ranks (done before filling input in case of in-place)
|
|
CHECK_CALL(collArgs.outputGpu.ClearGpuMem(numOutputBytes));
|
|
|
|
PtrUnion tempInputCpu;
|
|
PtrUnion tempResultCpu;
|
|
|
|
CHECK_CALL(tempInputCpu.AllocateCpuMem(numInputBytes));
|
|
CHECK_CALL(tempResultCpu.AllocateCpuMem(numInputBytes));
|
|
CHECK_CALL(tempResultCpu.ClearCpuMem(numInputBytes));
|
|
|
|
// If average or custom reduction operator is used, perform a summation instead
|
|
ncclRedOp_t const tempOp = (collArgs.options.redOp >= ncclAvg ? ncclSum : collArgs.options.redOp);
|
|
|
|
// Loop over each rank and generate the input / scale / reduce
|
|
PtrUnion scalarsPerRank;
|
|
scalarsPerRank.Attach(collArgs.options.scalarTransport.ptr);
|
|
for (int rank = 0; rank < collArgs.totalRanks; ++rank)
|
|
{
|
|
CHECK_CALL(tempInputCpu.FillPattern(collArgs.dataType, collArgs.numInputElements, rank, false));
|
|
|
|
if (rank == collArgs.globalRank)
|
|
{
|
|
if (hipMemcpy(collArgs.inputGpu.ptr, tempInputCpu.ptr, numInputBytes, hipMemcpyHostToDevice) != hipSuccess)
|
|
{
|
|
ERROR("hipMemcpy to input failed\n");
|
|
CHECK_CALL(tempInputCpu.FreeCpuMem());
|
|
CHECK_CALL(tempResultCpu.FreeCpuMem());
|
|
return TEST_FAIL;
|
|
}
|
|
}
|
|
|
|
// Scale the temporary input by local scalar for this rank
|
|
// (Used by custom reduction ops)
|
|
if (collArgs.options.scalarMode >= 0)
|
|
{
|
|
CHECK_CALL(tempInputCpu.Scale(collArgs.dataType, collArgs.numInputElements,
|
|
scalarsPerRank, rank));
|
|
}
|
|
|
|
if (rank == 0)
|
|
{
|
|
memcpy(tempResultCpu.ptr, tempInputCpu.ptr, numInputBytes);
|
|
}
|
|
else
|
|
{
|
|
CHECK_CALL(tempResultCpu.Reduce(collArgs.dataType, collArgs.numInputElements,
|
|
tempInputCpu, tempOp));
|
|
}
|
|
}
|
|
|
|
// Perform averaging if necessary
|
|
if (collArgs.options.redOp == ncclAvg)
|
|
{
|
|
CHECK_CALL(tempResultCpu.DivideByInt(collArgs.dataType, collArgs.numInputElements, collArgs.totalRanks));
|
|
}
|
|
|
|
// Copy over portion of result
|
|
memcpy(collArgs.expected.I1,
|
|
tempResultCpu.I1 + collArgs.globalRank * numOutputBytes,
|
|
numOutputBytes);
|
|
CHECK_CALL(tempInputCpu.FreeCpuMem());
|
|
CHECK_CALL(tempResultCpu.FreeCpuMem());
|
|
return TEST_SUCCESS;
|
|
}
|
|
|
|
ErrCode DefaultPrepData_Scatter(CollectiveArgs &collArgs)
|
|
{
|
|
CHECK_CALL(CheckAllocation(collArgs));
|
|
if (collArgs.numInputElements != collArgs.numOutputElements * collArgs.totalRanks)
|
|
{
|
|
ERROR("# of input elements must be total ranks * # output elements for Scatter\n");
|
|
return TEST_FAIL;
|
|
}
|
|
|
|
size_t const numInputBytes = collArgs.numInputElements * DataTypeToBytes(collArgs.dataType);
|
|
size_t const numOutputBytes = collArgs.numOutputElements * DataTypeToBytes(collArgs.dataType);
|
|
|
|
// Clear outputs on all ranks (prior to input in case of in-place)
|
|
collArgs.outputGpu.ClearGpuMem(numOutputBytes);
|
|
|
|
// Generate input as if on root rank - each rank will receive a portion
|
|
PtrUnion tempInput;
|
|
tempInput.AllocateCpuMem(numInputBytes);
|
|
tempInput.FillPattern(collArgs.dataType, collArgs.numInputElements, collArgs.options.root, false);
|
|
|
|
// Copy input to root rank
|
|
if (collArgs.globalRank == collArgs.options.root)
|
|
{
|
|
if (hipMemcpy(collArgs.inputGpu.ptr, tempInput.ptr, numInputBytes, hipMemcpyHostToDevice) != hipSuccess)
|
|
{
|
|
ERROR("hipMemcpy to input failed\n");
|
|
tempInput.FreeCpuMem();
|
|
return TEST_FAIL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
collArgs.inputGpu.ClearGpuMem(numInputBytes);
|
|
}
|
|
|
|
// Each rank receive a portion of the input
|
|
memcpy(collArgs.expected.U1, tempInput.U1 + (collArgs.globalRank * numOutputBytes), numOutputBytes);
|
|
|
|
tempInput.FreeCpuMem();
|
|
return TEST_SUCCESS;
|
|
}
|
|
|
|
ErrCode DefaultPrepData_AllToAll(CollectiveArgs &collArgs)
|
|
{
|
|
CHECK_CALL(CheckAllocation(collArgs));
|
|
if (collArgs.numInputElements != collArgs.numOutputElements)
|
|
{
|
|
ERROR("Number of input elements must match number of output elements for AllToAll\n");
|
|
return TEST_FAIL;
|
|
}
|
|
if (collArgs.numInputElements % collArgs.totalRanks)
|
|
{
|
|
ERROR("Input / Output size for AllToAll must be a multiple of %d\n", collArgs.totalRanks);
|
|
return TEST_FAIL;
|
|
}
|
|
size_t const numInputBytes = collArgs.numInputElements * DataTypeToBytes(collArgs.dataType);
|
|
size_t const numOutputBytes = collArgs.numOutputElements * DataTypeToBytes(collArgs.dataType);
|
|
size_t const numBytes = numInputBytes / collArgs.totalRanks;
|
|
|
|
// Clear outputs on all ranks (prior to input in case of in-place)
|
|
collArgs.outputGpu.ClearGpuMem(numOutputBytes);
|
|
|
|
// Generate input on root rank - each rank will receive a portion
|
|
PtrUnion tempInput;
|
|
tempInput.Attach(collArgs.outputCpu);
|
|
|
|
for (int rank = 0; rank < collArgs.totalRanks; ++rank)
|
|
{
|
|
tempInput.FillPattern(collArgs.dataType, collArgs.numInputElements, rank, false);
|
|
|
|
// Copy input
|
|
if (rank == collArgs.globalRank)
|
|
{
|
|
CHECK_HIP(hipMemcpy(collArgs.inputGpu.ptr, tempInput.ptr, numInputBytes, hipMemcpyHostToDevice));
|
|
}
|
|
memcpy(collArgs.expected.U1 + (numBytes * rank), tempInput.U1 + (numBytes * collArgs.globalRank), numBytes);
|
|
}
|
|
return TEST_SUCCESS;
|
|
}
|
|
|
|
ErrCode DefaultPrepData_AllToAllv(CollectiveArgs &collArgs)
|
|
{
|
|
|
|
CHECK_CALL(CheckAllocation(collArgs));
|
|
size_t const numInputBytes = collArgs.numInputElements * DataTypeToBytes(collArgs.dataType);
|
|
size_t const numOutputBytes = collArgs.numOutputElements * DataTypeToBytes(collArgs.dataType);
|
|
|
|
// calculating maxNumElements as the maximum number of input bytes out of all the ranks
|
|
size_t maxNumElements = 0;
|
|
for (int sendRank = 0; sendRank < collArgs.totalRanks; ++sendRank)
|
|
for (int recvRank = 0; recvRank < collArgs.totalRanks; ++recvRank)
|
|
{
|
|
size_t rankSendCount = collArgs.options.sdispls[(sendRank)*collArgs.totalRanks+recvRank] + collArgs.options.sendcounts[(sendRank)*collArgs.totalRanks+recvRank];
|
|
maxNumElements = std::max(maxNumElements, rankSendCount);
|
|
}
|
|
|
|
// Clear outputs on all ranks (prior to input in case of in-place)
|
|
collArgs.outputGpu.ClearGpuMem(numOutputBytes);
|
|
|
|
// Generate input on root rank - each rank will receive a portion
|
|
PtrUnion tempInput;
|
|
tempInput.AllocateCpuMem(maxNumElements*DataTypeToBytes(collArgs.dataType));
|
|
|
|
for (int sendRank = 0; sendRank < collArgs.totalRanks; ++sendRank)
|
|
{
|
|
tempInput.FillPattern(collArgs.dataType, maxNumElements, sendRank, false);
|
|
size_t recvDspls = collArgs.options.rdispls[collArgs.globalRank*collArgs.totalRanks + sendRank] * DataTypeToBytes(collArgs.dataType);
|
|
size_t sendDspls = collArgs.options.sdispls[sendRank*collArgs.totalRanks + collArgs.globalRank] * DataTypeToBytes(collArgs.dataType);
|
|
size_t numBytes = collArgs.options.recvcounts[collArgs.globalRank*collArgs.totalRanks + sendRank] * DataTypeToBytes(collArgs.dataType);
|
|
memcpy(collArgs.expected.U1 + recvDspls, tempInput.U1 + sendDspls, numBytes);
|
|
}
|
|
tempInput.FillPattern(collArgs.dataType, collArgs.numInputElements, collArgs.globalRank, false);
|
|
|
|
CHECK_HIP(hipMemcpy(collArgs.inputGpu.ptr, tempInput.ptr, numInputBytes, hipMemcpyHostToDevice));
|
|
|
|
tempInput.FreeCpuMem();
|
|
return TEST_SUCCESS;
|
|
}
|
|
|
|
ErrCode DefaultPrepData_Send(CollectiveArgs &collArgs)
|
|
{
|
|
CHECK_CALL(CheckAllocation(collArgs));
|
|
return collArgs.inputGpu.FillPattern(collArgs.dataType,
|
|
collArgs.numInputElements,
|
|
collArgs.globalRank, true);
|
|
}
|
|
|
|
ErrCode DefaultPrepData_Recv(CollectiveArgs &collArgs)
|
|
{
|
|
CHECK_CALL(CheckAllocation(collArgs));
|
|
return collArgs.expected.FillPattern(collArgs.dataType,
|
|
collArgs.numOutputElements,
|
|
collArgs.options.root,
|
|
false);
|
|
}
|
|
}
|