Distributed RPC Framework¶
The distributed RPC framework provides mechanisms for multi-machine model training through a set of primitives to allow for remote communication, and a higher-level API to automatically differentiate models split across several machines.
Warning
APIs in the RPC package are stable. There are multiple ongoing work items to improve performance and error handling, which will ship in future releases.
Basics¶
The distributed RPC framework makes it easy to run functions remotely, supports referencing remote objects without copying the real data around, and provides autograd and optimizer APIs to transparently run backward and update parameters across RPC boundaries. These features can be categorized into four sets of APIs.
Remote Procedure Call (RPC) supports running a function on the specified destination worker with the given arguments and getting the return value back or creating a reference to the return value. There are three main RPC APIs:
rpc_sync()
(synchronous),rpc_async()
(asynchronous), andremote()
(asynchronous and returns a reference to the remote return value). Use the synchronous API if the user code cannot proceed without the return value. Otherwise, use the asynchronous API to get a future, and wait on the future when the return value is needed on the caller. Theremote()
API is useful when the requirement is to create something remotely but never need to fetch it to the caller. Imagine the case that a driver process is setting up a parameter server and a trainer. The driver can create an embedding table on the parameter server and then share the reference to the embedding table with the trainer, but itself will never use the embedding table locally. In this case,rpc_sync()
andrpc_async()
are no longer appropriate, as they always imply that the return value will be returned to the caller immediately or in the future.Remote Reference (RRef) serves as a distributed shared pointer to a local or remote object. It can be shared with other workers and reference counting will be handled transparently. Each RRef only has one owner and the object only lives on that owner. Non-owner workers holding RRefs can get copies of the object from the owner by explicitly requesting it. This is useful when a worker needs to access some data object, but itself is neither the creator (the caller of
remote()
) or the owner of the object. The distributed optimizer, as we will discuss below, is one example of such use cases.Distributed Autograd stitches together local autograd engines on all the workers involved in the forward pass, and automatically reach out to them during the backward pass to compute gradients. This is especially helpful if the forward pass needs to span multiple machines when conducting, e.g., distributed model parallel training, parameter-server training, etc. With this feature, user code no longer needs to worry about how to send gradients across RPC boundaries and in which order should the local autograd engines be launched, which can become quite complicated where there are nested and inter-dependent RPC calls in the forward pass.
Distributed Optimizer’s constructor takes a
Optimizer()
(e.g.,SGD()
,Adagrad()
, etc.) and a list of parameter RRefs, creates anOptimizer()
instance on each distinct RRef owner, and updates parameters accordingly when runningstep()
. When you have distributed forward and backward passes, parameters and gradients will be scattered across multiple workers, and hence it requires an optimizer on each of the involved workers. Distributed Optimizer wraps all those local optimizers into one, and provides a concise constructor andstep()
API.
RPC¶
Before using RPC and distributed autograd primitives, initialization must take
place. To initialize the RPC framework we need to use
init_rpc()
which would initialize the RPC
framework, RRef framework and distributed autograd.
-
torch.distributed.rpc.
init_rpc
(name, backend=BackendType.PROCESS_GROUP, rank=-1, world_size=None, rpc_backend_options=None)[source]¶ Initializes RPC primitives such as the local RPC agent and distributed autograd, which immediately makes the current process ready to send and receive RPCs.
- Parameters
backend (BackendType, optional) – The type of RPC backend implementation. Supported values include
BackendType.PROCESS_GROUP
(the default) andBackendType.TENSORPIPE
. See Backends for more information.name (str) – a globally unique name of this node. (e.g.,
Trainer3
,ParameterServer2
,Master
,Worker1
) Name can only contain number, alphabet, underscore, colon, and/or dash, and must be shorter than 128 characters.rank (int) – a globally unique id/rank of this node.
world_size (int) – The number of workers in the group.
rpc_backend_options (RpcBackendOptions, optional) – The options passed to the RpcAgent constructor. It must be an agent-specific subclass of
RpcBackendOptions
and contains agent-specific initialization configurations. By default, for all agents, it sets the default timeout to 60 seconds and performs the rendezvous with an underlying process group initialized usinginit_method = "env://"
, meaning that environment variablesMASTER_ADDR
andMASTER_PORT
need to be set properly. See Backends for more information and find which options are available.
The following APIs allow users to remotely execute functions as well as create
references (RRefs) to remote data objects. In these APIs, when passing a
Tensor
as an argument or a return value, the destination worker will try to
create a Tensor
with the same meta (i.e., shape, stride, etc.). We
intentionally disallow transmitting CUDA tensors because it might crash if the
device lists on source and destination workers do not match. In such cases,
applications can always explicitly move the input tensors to CPU on the caller
and move it to the desired devices on the callee if necessary.
Warning
TorchScript support in RPC is a prototype feature and subject to change. Since
v1.5.0, torch.distributed.rpc
supports calling TorchScript functions as
RPC target functions, and this will help improve parallelism on the callee
side as executing TorchScript functions does not require GIL.
-
torch.distributed.rpc.
rpc_sync
(to, func, args=None, kwargs=None, timeout=-1.0)[source]¶ Make a blocking RPC call to run function
func
on workerto
. RPC messages are sent and received in parallel to execution of Python code. This method is thread-safe.- Parameters
to (str or WorkerInfo) – id or name of the destination worker.
func (callable) – a callable function, such as Python callables, builtin operators (e.g.
add()
) and annotated TorchScript functions.args (tuple) – the argument tuple for the
func
invocation.kwargs (dict) – is a dictionary of keyword arguments for the
func
invocation.timeout (float, optional) – timeout in seconds to use for this RPC. If the RPC does not complete in this amount of time, an exception indicating it has timed out will be raised. A value of 0 indicates an infinite timeout, i.e. a timeout error will never be raised. If not provided, the default value set during initialization or with
_set_rpc_timeout
is used.
- Returns
Returns the result of running
func
withargs
andkwargs
.
Warning
Using GPU tensors as arguments or return values of
func
is not supported since we don’t support sending GPU tensors over the wire. You need to explicitly copy GPU tensors to CPU before using them as arguments or return values offunc
.- Example::
Make sure that
MASTER_ADDR
andMASTER_PORT
are set properly on both workers. Refer toinit_process_group()
API for more details. For example,>>> export MASTER_ADDR=localhost >>> export MASTER_PORT=5678
Then run the following code in two different processes:
>>> # On worker 0: >>> import torch >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker0", rank=0, world_size=2) >>> ret = rpc.rpc_sync("worker1", torch.add, args=(torch.ones(2), 3)) >>> rpc.shutdown()
>>> # On worker 1: >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker1", rank=1, world_size=2) >>> rpc.shutdown()
Below is an example of running a TorchScript function using RPC.
>>> # On both workers: >>> @torch.jit.script >>> def my_script_add(t1, t2): >>> return torch.add(t1, t2)
>>> # On worker 0: >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker0", rank=0, world_size=2) >>> ret = rpc.rpc_sync("worker1", my_script_add, args=(torch.ones(2), 3)) >>> rpc.shutdown()
>>> # On worker 1: >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker1", rank=1, world_size=2) >>> rpc.shutdown()
-
torch.distributed.rpc.
rpc_async
(to, func, args=None, kwargs=None, timeout=-1.0)[source]¶ Make a non-blocking RPC call to run function
func
on workerto
. RPC messages are sent and received in parallel to execution of Python code. This method is thread-safe. This method will immediately return aFuture
that can be awaited on.- Parameters
to (str or WorkerInfo) – id or name of the destination worker.
func (callable) – a callable function, such as Python callables, builtin operators (e.g.
add()
) and annotated TorchScript functions.args (tuple) – the argument tuple for the
func
invocation.kwargs (dict) – is a dictionary of keyword arguments for the
func
invocation.timeout (float, optional) – timeout in seconds to use for this RPC. If the RPC does not complete in this amount of time, an exception indicating it has timed out will be raised. A value of 0 indicates an infinite timeout, i.e. a timeout error will never be raised. If not provided, the default value set during initialization or with
_set_rpc_timeout
is used.
- Returns
Returns a
Future
object that can be waited on. When completed, the return value offunc
onargs
andkwargs
can be retrieved from theFuture
object.
Warning
Using GPU tensors as arguments or return values of
func
is not supported since we don’t support sending GPU tensors over the wire. You need to explicitly copy GPU tensors to CPU before using them as arguments or return values offunc
.Warning
The
rpc_async
API does not copy storages of argument tensors until sending them over the wire, which could be done by a different thread depending on the RPC backend type. The caller should make sure that the contents of those tensors stay intact until the returnedFuture
completes.- Example::
Make sure that
MASTER_ADDR
andMASTER_PORT
are set properly on both workers. Refer toinit_process_group()
API for more details. For example,>>> export MASTER_ADDR=localhost >>> export MASTER_PORT=5678
Then run the following code in two different processes:
>>> # On worker 0: >>> import torch >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker0", rank=0, world_size=2) >>> fut1 = rpc.rpc_async("worker1", torch.add, args=(torch.ones(2), 3)) >>> fut2 = rpc.rpc_async("worker1", min, args=(1, 2)) >>> result = fut1.wait() + fut2.wait() >>> rpc.shutdown()
>>> # On worker 1: >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker1", rank=1, world_size=2) >>> rpc.shutdown()
Below is an example of running a TorchScript function using RPC.
>>> # On both workers: >>> @torch.jit.script >>> def my_script_add(t1, t2): >>> return torch.add(t1, t2)
>>> # On worker 0: >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker0", rank=0, world_size=2) >>> fut = rpc.rpc_async("worker1", my_script_add, args=(torch.ones(2), 3)) >>> ret = fut.wait() >>> rpc.shutdown()
>>> # On worker 1: >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker1", rank=1, world_size=2) >>> rpc.shutdown()
-
torch.distributed.rpc.
remote
(to, func, args=None, kwargs=None, timeout=-1.0)[source]¶ Make a remote call to run
func
on workerto
and return anRRef
to the result value immediately. Workerto
will be the owner of the returnedRRef
, and the worker callingremote
is a user. The owner manages the global reference count of itsRRef
, and the ownerRRef
is only destructed when globally there are no living references to it.- Parameters
to (str or WorkerInfo) – id or name of the destination worker.
func (callable) – a callable function, such as Python callables, builtin operators (e.g.
add()
) and annotated TorchScript functions.args (tuple) – the argument tuple for the
func
invocation.kwargs (dict) – is a dictionary of keyword arguments for the
func
invocation.timeout (float, optional) – timeout in seconds for this remote call. If the creation of this
RRef
on workerto
is not successfully processed on this worker within this timeout, then the next time there is an attempt to use the RRef (such asto_here()
), a timeout will be raised indicating this failure. A value of 0 indicates an infinite timeout, i.e. a timeout error will never be raised. If not provided, the default value set during initialization or with_set_rpc_timeout
is used.
- Returns
A user
RRef
instance to the result value. Use the blocking APItorch.distributed.rpc.RRef.to_here()
to retrieve the result value locally.
Warning
Using GPU tensors as arguments or return values of
func
is not supported since we don’t support sending GPU tensors over the wire. You need to explicitly copy GPU tensors to CPU before using them as arguments or return values offunc
.Warning
The
remote
API does not copy storages of argument tensors until sending them over the wire, which could be done by a different thread depending on the RPC backend type. The caller should make sure that the contents of those tensors stay intact until the returned RRef is confirmed by the owner, which can be checked using thetorch.distributed.rpc.RRef.confirmed_by_owner()
API.Warning
Errors such as timeouts for the
remote
API are handled on a best-effort basis. This means that when remote calls initiated byremote
fail, such as with a timeout error, we take a best-effort approach to error handling. This means that errors are handled and set on the resulting RRef on an asynchronous basis. If the RRef has not been used by the application before this handling (such asto_here
or fork call), then future uses of theRRef
will appropriately raise errors. However, it is possible that the user application will use theRRef
before the errors are handled. In this case, errors may not be raised as they have not yet been handled.- Example::
Make sure that
MASTER_ADDR
andMASTER_PORT
are set properly on both workers. Refer toinit_process_group()
API for more details. For example,>>> export MASTER_ADDR=localhost >>> export MASTER_PORT=5678
Then run the following code in two different processes:
>>> # On worker 0: >>> import torch >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker0", rank=0, world_size=2) >>> rref1 = rpc.remote("worker1", torch.add, args=(torch.ones(2), 3)) >>> rref2 = rpc.remote("worker1", torch.add, args=(torch.ones(2), 1)) >>> x = rref1.to_here() + rref2.to_here() >>> rpc.shutdown()
>>> # On worker 1: >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker1", rank=1, world_size=2) >>> rpc.shutdown()
Below is an example of running a TorchScript function using RPC.
>>> # On both workers: >>> @torch.jit.script >>> def my_script_add(t1, t2): >>> return torch.add(t1, t2)
>>> # On worker 0: >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker0", rank=0, world_size=2) >>> rref = rpc.remote("worker1", my_script_add, args=(torch.ones(2), 3)) >>> rref.to_here() >>> rpc.shutdown()
>>> # On worker 1: >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker1", rank=1, world_size=2) >>> rpc.shutdown()
-
torch.distributed.rpc.
get_worker_info
(worker_name=None)[source]¶ Get
WorkerInfo
of a given worker name. Use thisWorkerInfo
to avoid passing an expensive string on every invocation.- Parameters
worker_name (str) – the string name of a worker. If
None
, return the the id of the current worker. (defaultNone
)- Returns
WorkerInfo
instance for the givenworker_name
orWorkerInfo
of the current worker ifworker_name
isNone
.
-
torch.distributed.rpc.
shutdown
(graceful=True)[source]¶ Perform a shutdown of the RPC agent, and then destroy the RPC agent. This stops the local agent from accepting outstanding requests, and shuts down the RPC framework by terminating all RPC threads. If
graceful=True
, this will block until all local and remote RPC processes reach this method and wait for all outstanding work to complete. Otherwise, ifgraceful=False
, this is a local shutdown, and it does not wait for other RPC processes to reach this method.Warning
For
Future
objects returned byrpc_async()
,future.wait()
should not be called aftershutdown()
.- Parameters
graceful (bool) – Whether to do a graceful shutdown or not. If True, this will 1) wait until there is no pending system messages for
UserRRefs
and delete them; 2) block until all local and remote RPC processes have reached this method and wait for all outstanding work to complete.
- Example::
Make sure that
MASTER_ADDR
andMASTER_PORT
are set properly on both workers. Refer toinit_process_group()
API for more details. For example,>>> export MASTER_ADDR=localhost >>> export MASTER_PORT=5678
Then run the following code in two different processes:
>>> # On worker 0: >>> import torch >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker0", rank=0, world_size=2) >>> # do some work >>> result = rpc.rpc_sync("worker1", torch.add, args=(torch.ones(1), 1)) >>> # ready to shutdown >>> rpc.shutdown()
>>> # On worker 1: >>> import torch.distributed.rpc as rpc >>> rpc.init_rpc("worker1", rank=1, world_size=2) >>> # wait for worker 0 to finish work, and then shutdown. >>> rpc.shutdown()
-
class
torch.distributed.rpc.
WorkerInfo
¶ A structure that encapsulates information of a worker in the system. Contains the name and ID of the worker. This class is not meant to be constructed directly, rather, an instance can be retrieved through
get_worker_info()
and the result can be passed in to functions such asrpc_sync()
,rpc_async
,remote()
to avoid copying a string on every invocation.-
property
id
¶ Globally unique id to identify the worker.
-
property
name
¶ The name of the worker.
-
property
The RPC package also provides decorators which allow applications to specify how a given function should be treated on the callee side.
Warning
The rpc.functions
package is a prototype feature and subject to change.
-
torch.distributed.rpc.functions.
async_execution
(fn)[source]¶ A decorator for a function indicating that the return value of the function is guaranteed to be a
Future
object and this function can run asynchronously on the RPC callee. More specifically, the callee extracts theFuture
returned by the wrapped function and installs subsequent processing steps as a callback to thatFuture
. The installed callback will read the value from theFuture
when completed and send the value back as the RPC response. That also means the returnedFuture
only exists on the callee side and is never sent through RPC. This decorator is useful when the wrapped function’s (fn
) execution needs to pause and resume due to, e.g., containingrpc_async()
or waiting for other signals.Note
To enable asynchronous execution, applications must pass the function object returned by this decorator to RPC APIs. Otherwise, RPC will not be able to detect the attributes installed by this decorator. However, this does not mean this decorator has to be outmost one when defining a function. For example, when combined with
@staticmethod
or@classmethod
,@rpc.functions.async_execution
needs to be the inner decorator to allow the target function be recognized as a static or class function. This target function can still execute asynchronously because, when accessed, the static or class method preserves attributes installed by@rpc.functions.async_execution
.Warning
autograd profiler does not work with
async_execution
functions.- Example::
The returned
Future
object can come fromrpc.rpc_async
,Future.then(cb)
, orFuture
constructor. The example below shows directly using theFuture
returned byFuture.then(cb)
.>>> from torch.distributed import rpc >>> >>> # omitting setup and shutdown RPC >>> >>> # On all workers >>> @rpc.functions.async_execution >>> def async_add_chained(to, x, y, z): >>> # This function runs on "worker1" and returns immediately when >>> # the callback is installed through the `then(cb)` API. In the >>> # mean time, the `rpc_async` to "worker2" can run concurrently. >>> # When the return value of that `rpc_async` arrives at >>> # "worker1", "worker1" will run the lambda function accordinly >>> # and set the value for the previously returned `Future`, which >>> # will then trigger RPC to send the result back to "worker0". >>> return rpc.rpc_async(to, torch.add, args=(x, y)).then( >>> lambda fut: fut.wait() + z >>> ) >>> >>> # On worker0 >>> ret = rpc.rpc_sync( >>> "worker1", >>> async_add_chained, >>> args=("worker2", torch.ones(2), 1, 1) >>> ) >>> print(ret) # prints tensor([3., 3.])
When combined with TorchScript decorators, this decorator must be the outmost one.
>>> from torch import Tensor >>> from torch.futures import Future >>> from torch.distributed import rpc >>> >>> # omitting setup and shutdown RPC >>> >>> # On all workers >>> @torch.jit.script >>> def script_add(x: Tensor, y: Tensor) -> Tensor: >>> return x + y >>> >>> @rpc.functions.async_execution >>> @torch.jit.script >>> def async_add(to: str, x: Tensor, y: Tensor) -> Future[Tensor]: >>> return rpc.rpc_async(to, script_add, (x, y)) >>> >>> # On worker0 >>> ret = rpc.rpc_sync( >>> "worker1", >>> async_add, >>> args=("worker2", torch.ones(2), 1) >>> ) >>> print(ret) # prints tensor([2., 2.])
When combined with static or class method, this decorator must be the inner one.
>>> from torch.distributed import rpc >>> >>> # omitting setup and shutdown RPC >>> >>> # On all workers >>> class AsyncExecutionClass: >>> >>> @staticmethod >>> @rpc.functions.async_execution >>> def static_async_add(to, x, y, z): >>> return rpc.rpc_async(to, torch.add, args=(x, y)).then( >>> lambda fut: fut.wait() + z >>> ) >>> >>> @classmethod >>> @rpc.functions.async_execution >>> def class_async_add(cls, to, x, y, z): >>> ret_fut = torch.futures.Future() >>> rpc.rpc_async(to, torch.add, args=(x, y)).then( >>> lambda fut: ret_fut.set_result(fut.wait() + z) >>> ) >>> return ret_fut >>> >>> # On worker0 >>> ret = rpc.rpc_sync( >>> "worker1", >>> AsyncExecutionClass.static_async_add, >>> args=("worker2", torch.ones(2), 1, 2) >>> ) >>> print(ret) # prints tensor([4., 4.]) >>> >>> ret = rpc.rpc_sync( >>> "worker1", >>> AsyncExecutionClass.class_async_add, >>> args=("worker2", torch.ones(2), 1, 2) >>> ) >>> print(ret) # prints tensor([4., 4.])
Backends¶
The RPC module can leverage different backends to perform the communication
between the nodes. The backend to be used can be specified in the
init_rpc()
function, by passing a certain value of
the BackendType
enum. Regardless of what backend
is used, the rest of the RPC API won’t change. Each backend also defines its own
subclass of the RpcBackendOptions
class, an
instance of which can also be passed to init_rpc()
to configure the backend’s behavior.
-
class
torch.distributed.rpc.
BackendType
¶ An enum class of available backends.
PyTorch ships with two builtin backends:
BackendType.PROCESS_GROUP
andBackendType.TENSORPIPE
. Additional ones can be registered using theregister_backend()
function.
-
class
torch.distributed.rpc.
RpcBackendOptions
¶ An abstract structure encapsulating the options passed into the RPC backend. An instance of this class can be passed in to
init_rpc()
in order to initialize RPC with specific configurations, such as the RPC timeout andinit_method
to be used.-
property
init_method
¶ URL specifying how to initialize the process group. Default is
env://
-
property
rpc_timeout
¶ A float indicating the timeout to use for all RPCs. If an RPC does not complete in this timeframe, it will complete with an exception indicating that it has timed out.
-
property
Process Group Backend¶
The Process Group agent, which is the default, instantiates a process group from
the distributed
module and utilizes its point-to-point
communication capabilities to send RPC messages across. Internally, the process
group uses the Gloo library.
Gloo has been hardened by years of extensive use in PyTorch and is thus very reliable. However, as it was designed to perform collective communication, it may not always be the best fit for RPC. For example, each networking operation is synchronous and blocking, which means that it cannot be run in parallel with others. Moreover, it opens a connection between all pairs of nodes, and brings down all of them when one fails, thus reducing the resiliency and the elasticity of the system.
Example:
>>> import os
>>> from torch.distributed import rpc
>>> os.environ['MASTER_ADDR'] = 'localhost'
>>> os.environ['MASTER_PORT'] = '29500'
>>>
>>> rpc.init_rpc(
>>> "worker1",
>>> rank=0,
>>> world_size=2,
>>> rpc_backend_options=rpc.ProcessGroupRpcBackendOptions(
>>> num_send_recv_threads=16,
>>> rpc_timeout=20 # 20 second timeout
>>> )
>>> )
>>>
>>> # omitting init_rpc invocation on worker2
-
class
torch.distributed.rpc.
ProcessGroupRpcBackendOptions
¶ The backend options class for
ProcessGroupAgent
, which is derived fromRpcBackendOptions
.- Parameters
num_send_recv_threads (int, optional) – The number of threads in the thread-pool used by
ProcessGroupAgent
(default: 4).rpc_timeout (float, optional) – The default timeout, in seconds, for RPC requests (default: 60 seconds). If the RPC has not completed in this timeframe, an exception indicating so will be raised. Callers can override this timeout for individual RPCs in
rpc_sync()
andrpc_async()
if necessary.init_method (str, optional) – The URL to initialize
ProcessGroupGloo
(default:env://
).
-
property
init_method
¶ URL specifying how to initialize the process group. Default is
env://
-
property
num_send_recv_threads
¶ The number of threads in the thread-pool used by ProcessGroupAgent.
-
property
rpc_timeout
¶ A float indicating the timeout to use for all RPCs. If an RPC does not complete in this timeframe, it will complete with an exception indicating that it has timed out.
TensorPipe Backend¶
Warning
The TensorPipe backend is a beta feature.
The TensorPipe agent leverages the TensorPipe library, which provides a natively point-to-point communication primitive specifically suited for machine learning that fundamentally addresses some of the limitations of Gloo. Compared to Gloo, it has the advantage of being asynchronous, which allows a large number of transfers to occur simultaneously, each at their own speed, without blocking each other. It will only open pipes between pairs of nodes when needed, on demand, and when one node fails only its incident pipes will be closed, while all other ones will keep working as normal. In addition, it is able to support multiple different transports (TCP, of course, but also shared memory, NVLink, InfiniBand, …) and can automatically detect their availability and negotiate the best transport to use for each pipe.
The TensorPipe backend has been introduced in PyTorch v1.6 and is being actively developed. At the moment, it only supports CPU tensors, with GPU support coming soon. It comes with a TCP-based transport, just like Gloo. It is also able to automatically chunk and multiplex large tensors over multiple sockets and threads in order to achieve very high bandwidths. In addition to that, it packs two Linux-specific transports for communication between processes on a same machine (one based on ringbuffers stored in shared memory, the other on the cross-memory attach syscalls) which can achieve lower latencies than TCP. The agent will be able to pick the best transport on its own, with no intervention required.
Example:
>>> import os
>>> from torch.distributed import rpc
>>> os.environ['MASTER_ADDR'] = 'localhost'
>>> os.environ['MASTER_PORT'] = '29500'
>>>
>>> rpc.init_rpc(
>>> "worker1",
>>> rank=0,
>>> world_size=2,
>>> backend=rpc.BackendType.TENSORPIPE,
>>> rpc_backend_options=rpc.TensorPipeRpcBackendOptions(
>>> num_worker_threads=8,
>>> rpc_timeout=20 # 20 second timeout
>>> )
>>> )
>>>
>>> # omitting init_rpc invocation on worker2
-
class
torch.distributed.rpc.
TensorPipeRpcBackendOptions
¶ The backend options for
TensorPipeAgent
, derived fromRpcBackendOptions
.- Parameters
num_worker_threads (int, optional) – The number of threads in the thread-pool used by
TensorPipeAgent
to execute requests (default: 16).rpc_timeout (float, optional) – The default timeout, in seconds, for RPC requests (default: 60 seconds). If the RPC has not completed in this timeframe, an exception indicating so will be raised. Callers can override this timeout for individual RPCs in
rpc_sync()
andrpc_async()
if necessary.init_method (str, optional) – The URL to initialize the distributed store used for rendezvous. It takes any value accepted for the same argument of
init_process_group()
(default:env://
).
-
property
init_method
¶ URL specifying how to initialize the process group. Default is
env://
-
property
num_worker_threads
¶ The number of threads in the thread-pool used by
TensorPipeAgent
to execute requests.
-
property
rpc_timeout
¶ A float indicating the timeout to use for all RPCs. If an RPC does not complete in this timeframe, it will complete with an exception indicating that it has timed out.
RRef¶
An RRef
(Remote REFerence) is a reference to a value of some type T
(e.g. Tensor
) on a remote worker. This handle keeps the referenced remote
value alive on the owner, but there is no implication that the value will be
transferred to the local worker in the future. RRefs can be used in
multi-machine training by holding references to nn.Modules that exist on
other workers, and calling the appropriate functions to retrieve or modify their
parameters during training. See Remote Reference Protocol for more
details.
-
class
torch.distributed.rpc.
RRef
[source]¶ -
confirmed_by_owner
(self: torch.distributed.rpc.RRef) → bool¶ Returns whether this
RRef
has been confirmed by the owner.OwnerRRef
always returns true, whileUserRRef
only returns true when the owner knowns about thisUserRRef
.
-
is_owner
(self: torch.distributed.rpc.RRef) → bool¶ Returns whether or not the current node is the owner of this
RRef
.
-
local_value
(self: torch.distributed.rpc.RRef) → object¶ If the current node is the owner, returns a reference to the local value. Otherwise, throws an exception.
-
owner
(self: torch.distributed.rpc.RRef) → torch.distributed.rpc.WorkerInfo¶ Returns worker information of the node that owns this
RRef
.
-
owner_name
(self: torch.distributed.rpc.RRef) → str¶ Returns worker name of the node that owns this
RRef
.
-
remote
(self: torch.distributed.rpc.RRef) → object¶ Create a helper proxy to easily launch a
remote
using the owner of the RRef as the destination to run functions on the object referenced by this RRef. More specifically,rref.remote().func_name(*args, **kwargs)
is the same as the following:>>> def run(rref, func_name, args, kwargs): >>> return getattr(rref.local_value(), func_name)(*args, **kwargs) >>> >>> rpc.remote(rref.owner(), run, args=(rref, func_name, args, kwargs))
- Example::
>>> from torch.distributed import rpc >>> rref = rpc.remote("worker1", torch.add, args=(torch.zeros(2, 2), 1)) >>> rref.remote().size().to_here() # returns torch.Size([2, 2]) >>> rref.remote().view(1, 4).to_here() # returns tensor([[1., 1., 1., 1.]])
-
rpc_async
(self: torch.distributed.rpc.RRef) → object¶ Create a helper proxy to easily launch an
rpc_async
using the owner of the RRef as the destination to run functions on the object referenced by this RRef. More specifically,rref.rpc_async().func_name(*args, **kwargs)
is the same as the following:>>> def run(rref, func_name, args, kwargs): >>> return getattr(rref.local_value(), func_name)(*args, **kwargs) >>> >>> rpc.rpc_async(rref.owner(), run, args=(rref, func_name, args, kwargs))
- Example::
>>> from torch.distributed import rpc >>> rref = rpc.remote("worker1", torch.add, args=(torch.zeros(2, 2), 1)) >>> rref.rpc_async().size().wait() # returns torch.Size([2, 2]) >>> rref.rpc_async().view(1, 4).wait() # returns tensor([[1., 1., 1., 1.]])
-
rpc_sync
(self: torch.distributed.rpc.RRef) → object¶ Create a helper proxy to easily launch an
rpc_sync
using the owner of the RRef as the destination to run functions on the object referenced by this RRef. More specifically,rref.rpc_sync().func_name(*args, **kwargs)
is the same as the following:>>> def run(rref, func_name, args, kwargs): >>> return getattr(rref.local_value(), func_name)(*args, **kwargs) >>> >>> rpc.rpc_sync(rref.owner(), run, args=(rref, func_name, args, kwargs))
- Example::
>>> from torch.distributed import rpc >>> rref = rpc.remote("worker1", torch.add, args=(torch.zeros(2, 2), 1)) >>> rref.rpc_sync().size() # returns torch.Size([2, 2]) >>> rref.rpc_sync().view(1, 4) # returns tensor([[1., 1., 1., 1.]])
-
to_here
(self: torch.distributed.rpc.RRef, timeout: float = -1.0) → object¶ Blocking call that copies the value of the RRef from the owner to the local node and returns it. If the current node is the owner, returns a reference to the local value.
- Parameters
timeout (float, optional) – Timeout for
to_here
. If the call does not complete within this timeframe, an exception indicating so will be raised. If this argument is not provided, the default RPC timeout (60s) will be used.
-
Distributed Autograd Framework¶
This module provides an RPC-based distributed autograd framework that can be used for applications such as model parallel training. In short, applications may send and receive gradient recording tensors over RPC. In the forward pass, we record when gradient recording tensors are sent over RPC and during the backward pass we use this information to perform a distributed backward pass using RPC. For more details see Distributed Autograd Design.
-
class
torch.distributed.autograd.
context
[source]¶ Context object to wrap forward and backward passes when using distributed autograd. The
context_id
generated in thewith
statement is required to uniquely identify a distributed backward pass on all workers. Each worker stores metadata associated with thiscontext_id
, which is required to correctly execute a distributed autograd pass.- Example::
>>> import torch.distributed.autograd as dist_autograd >>> with dist_autograd.context() as context_id: >>> t1 = torch.rand((3, 3), requires_grad=True) >>> t2 = torch.rand((3, 3), requires_grad=True) >>> loss = rpc.rpc_sync("worker1", torch.add, args=(t1, t2)).sum() >>> dist_autograd.backward(context_id, [loss])
-
torch.distributed.autograd.
backward
(context_id: int, roots: List[Tensor], retain_graph = False) → None¶ Kicks off the distributed backward pass using the provided roots. This currently implements the FAST mode algorithm which assumes all RPC messages sent in the same distributed autograd context across workers would be part of the autograd graph during the backward pass.
We use the provided roots to discover the autograd graph and compute appropriate dependencies. This method blocks until the entire autograd computation is done.
We accumulate the gradients in the appropriate
torch.distributed.autograd.context
on each of the nodes. The autograd context to be used is looked up given thecontext_id
that is passed in whentorch.distributed.autograd.backward()
is called. If there is no valid autograd context corresponding to the given ID, we throw an error. You can retrieve the accumulated gradients using theget_gradients()
API.- Parameters
context_id (int) – The autograd context id for which we should retrieve the gradients.
roots (list) – Tensors which represent the roots of the autograd computation. All the tensors should be scalars.
retain_graph (bool, optional) – If False, the graph used to compute the grad will be freed. Note that in nearly all cases setting this option to True is not needed and often can be worked around in a much more efficient way. Usually, you need to set this to True to run backward multiple times.
- Example::
>>> import torch.distributed.autograd as dist_autograd >>> with dist_autograd.context() as context_id: >>> pred = model.forward() >>> loss = loss_func(pred, loss) >>> dist_autograd.backward(context_id, loss)
-
torch.distributed.autograd.
get_gradients
(context_id: int) → Dict[Tensor, Tensor]¶ Retrieves a map from Tensor to the appropriate gradient for that Tensor accumulated in the provided context corresponding to the given
context_id
as part of the distributed autograd backward pass.- Parameters
context_id (int) – The autograd context id for which we should retrieve the gradients.
- Returns
A map where the key is the Tensor and the value is the associated gradient for that Tensor.
- Example::
>>> import torch.distributed.autograd as dist_autograd >>> with dist_autograd.context() as context_id: >>> t1 = torch.rand((3, 3), requires_grad=True) >>> t2 = torch.rand((3, 3), requires_grad=True) >>> loss = t1 + t2 >>> dist_autograd.backward(context_id, [loss.sum()]) >>> grads = dist_autograd.get_gradients(context_id) >>> print(grads[t1]) >>> print(grads[t2])
Distributed Optimizer¶
torch.distributed.optim
exposes DistributedOptimizer, which takes a list
of remote parameters (RRef
) and runs the
optimizer locally on the workers where the parameters live. The distributed
optimizer can use any of the local optimizer Algorithms to
apply the gradients on each worker.
-
class
torch.distributed.optim.
DistributedOptimizer
(optimizer_class, params_rref, *args, **kwargs)[source]¶ DistributedOptimizer takes remote references to parameters scattered across workers and applies the given optimizer locally for each parameter.
This class uses
get_gradients()
in order to retrieve the gradients for specific parameters.Concurrent calls to
step()
, either from the same or different clients, will be serialized on each worker – as each worker’s optimizer can only work on one set of gradients at a time. However, there is no guarantee that the full forward-backward-optimizer sequence will execute for one client at a time. This means that the gradients being applied may not correspond to the latest forward pass executed on a given worker. Also, there is no guaranteed ordering across workers.- Parameters
optimizer_class (optim.Optimizer) – the class of optimizer to instantiate on each worker.
params_rref (list[RRef]) – list of RRefs to local or remote parameters to optimize.
args – arguments to pass to the optimizer constructor on each worker.
kwargs – arguments to pass to the optimizer constructor on each worker.
- Example::
>>> import torch.distributed.autograd as dist_autograd >>> import torch.distributed.rpc as rpc >>> from torch import optim >>> from torch.distributed.optim import DistributedOptimizer >>> >>> with dist_autograd.context() as context_id: >>> # Forward pass. >>> rref1 = rpc.remote("worker1", torch.add, args=(torch.ones(2), 3)) >>> rref2 = rpc.remote("worker1", torch.add, args=(torch.ones(2), 1)) >>> loss = rref1.to_here() + rref2.to_here() >>> >>> # Backward pass. >>> dist_autograd.backward(context_id, [loss.sum()]) >>> >>> # Optimizer. >>> dist_optim = DistributedOptimizer( >>> optim.SGD, >>> [rref1, rref2], >>> lr=0.05, >>> ) >>> dist_optim.step(context_id)
-
step
(context_id)[source]¶ Performs a single optimization step.
This will call
torch.optim.Optimizer.step()
on each worker containing parameters to be optimized, and will block until all workers return. The providedcontext_id
will be used to retrieve the correspondingcontext
that contains the gradients that should be applied to the parameters.- Parameters
context_id – the autograd context id for which we should run the optimizer step.
Design Notes¶
The distributed autograd design note covers the design of the RPC-based distributed autograd framework that is useful for applications such as model parallel training.
The RRef design note covers the design of the RRef (Remote REFerence) protocol used to refer to values on remote workers by the framework.
Tutorials¶
The RPC tutorial introduces users to the RPC framework and provides two example applications using torch.distributed.rpc APIs.