345 líneas
12 KiB
C++
345 líneas
12 KiB
C++
/*************************************************************************
|
|
* Copyright (c) 2015-2020, NVIDIA CORPORATION. All rights reserved.
|
|
* Modifications Copyright (c) 2019-2020 Advanced Micro Devices, Inc. All rights reserved.
|
|
*
|
|
* See LICENSE.txt for license information
|
|
************************************************************************/
|
|
|
|
#include "group.h"
|
|
#include "debug.h"
|
|
#include "enqueue.h"
|
|
#include "transport.h"
|
|
|
|
#define MAX_ASYNC_OPS 128
|
|
thread_local pthread_t ncclGroupThreads[MAX_ASYNC_OPS];
|
|
thread_local int ncclGroupIndex = 0;
|
|
thread_local int ncclGroupMode = 0;
|
|
thread_local ncclResult_t ncclGroupError = ncclSuccess;
|
|
|
|
bool ncclAsyncMode() {
|
|
return ncclGroupMode > 0;
|
|
}
|
|
|
|
ncclResult_t ncclAsyncErrCheck(ncclResult_t ret) {
|
|
if (ncclGroupError == ncclSuccess || ret != ncclSuccess) ncclGroupError = ret;
|
|
return ret;
|
|
}
|
|
|
|
struct ncclInitArgs {
|
|
ncclInitFunc_t func;
|
|
int cudaDev;
|
|
ncclComm_t* newcomm;
|
|
int ndev;
|
|
ncclUniqueId commId;
|
|
int myrank;
|
|
};
|
|
struct ncclCollArgs {
|
|
ncclComm_t comm;
|
|
int connect;
|
|
};
|
|
|
|
enum ncclAsyncFuncType {
|
|
ASYNC_FUNC_INVALID = 0,
|
|
ASYNC_FUNC_INIT = 1,
|
|
ASYNC_FUNC_COLL = 2,
|
|
};
|
|
struct ncclAsyncArgs {
|
|
ncclResult_t ret;
|
|
enum ncclAsyncFuncType funcType;
|
|
union {
|
|
ncclCollArgs coll;
|
|
ncclInitArgs init;
|
|
};
|
|
};
|
|
|
|
thread_local struct ncclAsyncArgs ncclGroupArgs[MAX_ASYNC_OPS];
|
|
|
|
#define NCCLCHECKTHREAD(a) do { \
|
|
if ((args->ret = (a)) != ncclSuccess) { \
|
|
INFO(NCCL_INIT,"%s:%d -> %d [Async thread]", __FILE__, __LINE__, args->ret); \
|
|
return args; \
|
|
} \
|
|
} while(0)
|
|
|
|
#define CUDACHECKTHREAD(a) do { \
|
|
if ((a) != hipSuccess) { \
|
|
INFO(NCCL_INIT,"%s:%d -> %d [Async thread]", __FILE__, __LINE__, args->ret); \
|
|
args->ret = ncclUnhandledCudaError; \
|
|
return args; \
|
|
} \
|
|
} while(0)
|
|
|
|
void* ncclAsyncThreadMain(void* args_) {
|
|
struct ncclAsyncArgs* args = (struct ncclAsyncArgs*)args_;
|
|
NCCLCHECKTHREAD(args->init.func(args->init.newcomm, args->init.ndev, args->init.commId, args->init.myrank, args->init.cudaDev));
|
|
return args;
|
|
}
|
|
|
|
ncclResult_t ncclAsyncInit(ncclInitFunc_t func, ncclComm_t* newcomm, int ndev, ncclUniqueId commId, int myrank, int cudaDev) {
|
|
if (ncclGroupIndex >= MAX_ASYNC_OPS) {
|
|
WARN("Too many async operations in progress, max is %d", MAX_ASYNC_OPS);
|
|
return ncclAsyncErrCheck(ncclInvalidUsage);
|
|
}
|
|
int index = ncclGroupIndex++;
|
|
struct ncclAsyncArgs* args = ncclGroupArgs+index;
|
|
args->funcType = ASYNC_FUNC_INIT;
|
|
args->init.func = func;
|
|
args->init.cudaDev = cudaDev;
|
|
args->init.newcomm = newcomm;
|
|
args->init.ndev = ndev;
|
|
memcpy(&args->init.commId, &commId, sizeof(commId));
|
|
args->init.myrank = myrank;
|
|
return ncclSuccess;
|
|
}
|
|
|
|
ncclResult_t ncclAsyncColl(ncclComm_t comm) {
|
|
struct ncclAsyncArgs* args = ncclGroupArgs;
|
|
for (int i=0; i<ncclGroupIndex; i++) {
|
|
if (args->coll.comm == comm) return ncclSuccess;
|
|
args++;
|
|
}
|
|
if (ncclGroupIndex >= MAX_ASYNC_OPS) {
|
|
WARN("Too many async operations in progress, max is %d", MAX_ASYNC_OPS);
|
|
return ncclAsyncErrCheck(ncclInvalidUsage);
|
|
}
|
|
ncclGroupIndex++;
|
|
args->funcType = ASYNC_FUNC_COLL;
|
|
args->coll.comm = comm;
|
|
return ncclSuccess;
|
|
}
|
|
|
|
NCCL_API(ncclResult_t, ncclGroupStart);
|
|
ncclResult_t ncclGroupStart() {
|
|
if (ncclGroupMode == 0) {
|
|
memset(ncclGroupArgs, 0, sizeof(struct ncclAsyncArgs)*MAX_ASYNC_OPS);
|
|
}
|
|
ncclGroupMode++;
|
|
return ncclSuccess;
|
|
}
|
|
|
|
static ncclResult_t scheduleSendRecv(struct ncclComm* comm, int delta, int channelId, ssize_t recvbytes, void* recvbuff, ssize_t sendbytes, const void* sendbuff) {
|
|
struct ncclInfo info = { ncclCollSendRecv, "SendRecv",
|
|
sendbuff, recvbuff, (size_t)std::max<ssize_t>(sendbytes,recvbytes), ncclInt8, ncclSum, -1, comm, comm->userStream, /* Args */
|
|
1, 1 };
|
|
info.delta = delta;
|
|
info.channelId = channelId;
|
|
info.sendbytes = sendbytes;
|
|
info.recvbytes = recvbytes;
|
|
if (delta == 0 && sendbytes != recvbytes) return ncclInvalidUsage;
|
|
NCCLCHECK(ncclSaveKernel(&info));
|
|
return ncclSuccess;
|
|
}
|
|
|
|
void* ncclAsyncThreadPreconnect(void* args_) {
|
|
struct ncclAsyncArgs* args = (struct ncclAsyncArgs*)args_;
|
|
CUDACHECKTHREAD(hipSetDevice(args->coll.comm->cudaDev));
|
|
for (int c=0; c<args->coll.comm->p2pnChannels; c++) {
|
|
struct ncclComm* comm = args->coll.comm;
|
|
struct ncclChannel* channel = comm->channels+c;
|
|
struct ncclP2PConnect* connect = &comm->p2plist.connect;
|
|
NCCLCHECKTHREAD(ncclTransportP2pSetup(comm, NULL, channel, connect->nrecv[c], connect->recv+c*comm->nRanks, connect->nsend[c], connect->send+c*comm->nRanks));
|
|
connect->nrecv[c] = 0;
|
|
connect->nsend[c] = 0;
|
|
}
|
|
return args;
|
|
}
|
|
|
|
NCCL_API(ncclResult_t, ncclGroupEnd);
|
|
ncclResult_t ncclGroupEnd() {
|
|
if (ncclGroupMode == 0) {
|
|
WARN("ncclGroupEnd: not in a group call.");
|
|
return ncclInvalidUsage;
|
|
}
|
|
ncclGroupMode--;
|
|
if (ncclGroupMode > 0) return ncclSuccess;
|
|
int savedDev;
|
|
CUDACHECK(hipGetDevice(&savedDev));
|
|
int activeThreads = 0;
|
|
int doneArray[MAX_ASYNC_OPS];
|
|
for (int i=0; i<ncclGroupIndex; i++) doneArray[i] = 1;
|
|
ncclResult_t ret = ncclGroupError;
|
|
if (ret != ncclSuccess) goto group_cleanup;
|
|
|
|
/* Launch async ncclCommInitRank */
|
|
for (int i=0; i<ncclGroupIndex; i++) {
|
|
struct ncclAsyncArgs* args = ncclGroupArgs+i;
|
|
if (args->funcType == ASYNC_FUNC_INIT) {
|
|
pthread_create(ncclGroupThreads+i, NULL, ncclAsyncThreadMain, args);
|
|
activeThreads++;
|
|
doneArray[i] = 0;
|
|
}
|
|
}
|
|
/* For init, since we use threads, we just wait for threads to complete */
|
|
while (activeThreads) {
|
|
for (int i=0; i<ncclGroupIndex; i++) {
|
|
struct ncclAsyncArgs* args = ncclGroupArgs+i;
|
|
if (args->funcType == ASYNC_FUNC_INIT && doneArray[i] == 0) {
|
|
int err = pthread_tryjoin_np(ncclGroupThreads[i], NULL);
|
|
if (err == EBUSY) continue;
|
|
if (err != 0) ret = ncclSystemError;
|
|
if (args->ret != ncclSuccess) ret = args->ret;
|
|
doneArray[i] = 1;
|
|
activeThreads--;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (int i=0; i<ncclGroupIndex; i++) {
|
|
struct ncclAsyncArgs* args = ncclGroupArgs+i;
|
|
if (args->funcType == ASYNC_FUNC_COLL) {
|
|
struct ncclP2Plist* p2plist = &args->coll.comm->p2plist;
|
|
if (p2plist->count != 0) {
|
|
struct ncclComm* comm = args->coll.comm;
|
|
args->coll.connect = 0;
|
|
for (int c=0; c<comm->p2pnChannels; c++)
|
|
args->coll.connect += comm->p2plist.connect.nsend[c] + comm->p2plist.connect.nrecv[c];
|
|
if (args->coll.connect) {
|
|
pthread_create(ncclGroupThreads+i, NULL, ncclAsyncThreadPreconnect, args);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (int i=0; i<ncclGroupIndex; i++) {
|
|
struct ncclAsyncArgs* args = ncclGroupArgs+i;
|
|
if (args->funcType == ASYNC_FUNC_COLL && (args->coll.connect)) {
|
|
int err = pthread_join(ncclGroupThreads[i], NULL);
|
|
if (err != 0) {
|
|
WARN("Error waiting for pthread_join : %s\n", strerror(errno));
|
|
return ncclSystemError;
|
|
}
|
|
NCCLCHECKGOTO(args->ret, ret, end);
|
|
}
|
|
}
|
|
|
|
for (int i=0; i<ncclGroupIndex; i++) {
|
|
struct ncclAsyncArgs* args = ncclGroupArgs+i;
|
|
if (args->funcType == ASYNC_FUNC_COLL) {
|
|
struct ncclComm* comm = args->coll.comm;
|
|
int rank = comm->rank;
|
|
int nRanks = comm->nRanks;
|
|
struct ncclP2Plist* p2plist = &args->coll.comm->p2plist;
|
|
if (p2plist->count) {
|
|
for (int delta=0; delta<nRanks; delta++) {
|
|
uint32_t from = (rank+nRanks-delta)%nRanks;
|
|
uint32_t to = (rank+delta)%nRanks;
|
|
|
|
// Compute how much to split operations
|
|
// Natural step size matching buffer steps.
|
|
ssize_t stepSize = 4*comm->buffSizes[NCCL_PROTO_SIMPLE] / NCCL_STEPS;
|
|
// Split each operation on p2pnChannelsPerPeer max.
|
|
ssize_t recvChunkSize = DIVUP(p2plist->peerlist[from].recvbytes, comm->p2pnChannelsPerPeer);
|
|
ssize_t sendChunkSize = DIVUP(p2plist->peerlist[to].sendbytes, comm->p2pnChannelsPerPeer);
|
|
recvChunkSize = std::max((ssize_t)1, DIVUP(recvChunkSize, stepSize)) * stepSize;
|
|
sendChunkSize = std::max((ssize_t)1, DIVUP(sendChunkSize, stepSize)) * stepSize;
|
|
|
|
ssize_t sendOffset = 0;
|
|
ssize_t recvOffset = 0;
|
|
int remaining = 1;
|
|
int chunk = 0;
|
|
while (remaining) {
|
|
int channelId = (delta+comm->p2pChannels[chunk%comm->p2pnChannelsPerPeer]) % comm->p2pnChannels;
|
|
remaining = 0;
|
|
ssize_t recvbytes = p2plist->peerlist[from].recvbytes-recvOffset;
|
|
ssize_t sendbytes = p2plist->peerlist[to].sendbytes-sendOffset;
|
|
if (recvbytes > recvChunkSize) { remaining = 1; recvbytes = recvChunkSize; } else p2plist->peerlist[from].recvbytes = -1;
|
|
if (sendbytes > sendChunkSize) { remaining = 1; sendbytes = sendChunkSize; } else p2plist->peerlist[to].sendbytes = -1;
|
|
if (sendbytes >= 0 || recvbytes >= 0) {
|
|
NCCLCHECKGOTO(scheduleSendRecv(comm, delta, channelId,
|
|
recvbytes, ((char*)(p2plist->peerlist[from].recvbuff)) + recvOffset,
|
|
sendbytes, ((const char*)(p2plist->peerlist[to].sendbuff)) + sendOffset), ret, end);
|
|
}
|
|
recvOffset += recvChunkSize;
|
|
sendOffset += sendChunkSize;
|
|
chunk++;
|
|
}
|
|
}
|
|
p2plist->count = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Collectives are done in three steps :
|
|
* 1. Barrier Check In. Only the last call may call cudaLaunchKernel[cooperative]
|
|
* 2. Barrier Wait. No CUDA call is permitted
|
|
* 3. Enqueue Events. CUDA event wait/enqueue.
|
|
* This is needed because step 2 cannot call any CUDA primitive, otherwise if
|
|
* hipFree happens between 1 and 3, it could block that CUDA call and
|
|
* prevent some ranks from launching their network threads, which would
|
|
* prevent the NCCL call from completing, blocking the hipFree call.
|
|
*/
|
|
for (int i=0; i<ncclGroupIndex; i++) {
|
|
struct ncclAsyncArgs* args = ncclGroupArgs+i;
|
|
if (args->funcType == ASYNC_FUNC_COLL) {
|
|
if (args->coll.comm->userStream == NULL)
|
|
CUDACHECKGOTO(hipSetDevice(args->coll.comm->cudaDev), ret, end);
|
|
NCCLCHECKGOTO(ncclBarrierEnqueue(args->coll.comm), ret, end);
|
|
}
|
|
}
|
|
for (int i=0; i<ncclGroupIndex; i++) {
|
|
struct ncclAsyncArgs* args = ncclGroupArgs+i;
|
|
if (args->funcType == ASYNC_FUNC_COLL) {
|
|
CUDACHECKGOTO(hipSetDevice(args->coll.comm->cudaDev), ret, end);
|
|
NCCLCHECKGOTO(ncclBarrierEnqueueWait(args->coll.comm), ret, end);
|
|
}
|
|
}
|
|
for (int i=0; i<ncclGroupIndex; i++) {
|
|
struct ncclAsyncArgs* args = ncclGroupArgs+i;
|
|
if (args->funcType == ASYNC_FUNC_COLL) {
|
|
if (args->coll.comm->userStream == NULL)
|
|
CUDACHECKGOTO(hipSetDevice(args->coll.comm->cudaDev), ret, end);
|
|
NCCLCHECKGOTO(ncclEnqueueEvents(args->coll.comm), ret, end);
|
|
}
|
|
}
|
|
|
|
goto end;
|
|
group_cleanup:
|
|
if (ret != ncclSuccess) {
|
|
// At least one call in the group failed. Since we want to make that group
|
|
// an atomic operation, we need to cancel all operations.
|
|
for (int i=0; i<ncclGroupIndex; i++) {
|
|
struct ncclAsyncArgs* args = ncclGroupArgs+i;
|
|
if (args->funcType == ASYNC_FUNC_INIT) {
|
|
if (args->init.newcomm) ncclCommDestroy(*args->init.newcomm);
|
|
*args->init.newcomm = NULL;
|
|
} else {
|
|
struct ncclComm* comm = args->coll.comm;
|
|
for (int c=0; c<comm->p2pnChannels; c++) {
|
|
struct ncclChannel* channel = comm->channels+c;
|
|
for (int i=0; i<channel->collCount; i++) {
|
|
channel->collectives[(channel->collStart + i)%NCCL_MAX_OPS].active = 0;
|
|
}
|
|
channel->collFifoTail = channel->collStart;
|
|
channel->collCount = 0;
|
|
}
|
|
/* Cancel all proxy ops : mark them as ncclProxyOpNone and they should be freed later on */
|
|
struct ncclProxyState* state = &comm->proxyState;
|
|
struct ncclProxyArgs *op, *start;
|
|
pthread_mutex_lock(&state->mutex);
|
|
op = start = state->ops;
|
|
while (op) {
|
|
if (op->opCount >= comm->lastOpCount) op->state = ncclProxyOpNone;
|
|
struct ncclProxyArgs* peerOp = op->nextPeer;
|
|
while (peerOp) {
|
|
if (peerOp->opCount >= comm->lastOpCount) peerOp->state = ncclProxyOpNone;
|
|
peerOp = peerOp->nextPeer;
|
|
}
|
|
op = op->next;
|
|
if (op == start) break;
|
|
}
|
|
comm->opCount = comm->lastOpCount;
|
|
pthread_cond_signal(&state->cond);
|
|
pthread_mutex_unlock(&state->mutex);
|
|
|
|
comm->myParams->gridDim.x = comm->myParams->blockDim.x = 0;
|
|
comm->userStreamSet = false;
|
|
}
|
|
}
|
|
}
|
|
end:
|
|
ncclGroupError = ncclSuccess;
|
|
ncclGroupIndex = 0;
|
|
CUDACHECK(hipSetDevice(savedDev)); // do other clean-ups first before calling hipSetDevice, because this call can fail too
|
|
return ret;
|
|
}
|