|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
from torch import nn
|
|
|
|
|
|
|
|
|
class ModuleProxyWrapper(nn.Module):
|
|
|
"""
|
|
|
Wrap a DistributedDataParallel module and forward requests for missing
|
|
|
attributes to the module wrapped by DDP (the twice-wrapped module).
|
|
|
Also forward calls to :func:`state_dict` and :func:`load_state_dict`.
|
|
|
Usage::
|
|
|
module.xyz = "hello world"
|
|
|
wrapped_module = DistributedDataParallel(module, **ddp_args)
|
|
|
wrapped_module = ModuleProxyWrapper(wrapped_module)
|
|
|
assert wrapped_module.xyz == "hello world"
|
|
|
assert wrapped_module.state_dict().keys() == module.state_dict().keys()
|
|
|
Args:
|
|
|
module (nn.Module): module to wrap
|
|
|
"""
|
|
|
|
|
|
def __init__(self, module: nn.Module):
|
|
|
super().__init__()
|
|
|
assert hasattr(module, "module"), \
|
|
|
"ModuleProxyWrapper expects input to wrap another module"
|
|
|
self.module = module
|
|
|
|
|
|
def __getattr__(self, name):
|
|
|
"""Forward missing attributes to twice-wrapped module."""
|
|
|
try:
|
|
|
|
|
|
return super().__getattr__(name)
|
|
|
except AttributeError:
|
|
|
try:
|
|
|
|
|
|
return getattr(self.module, name)
|
|
|
except AttributeError:
|
|
|
|
|
|
return getattr(self.module.module, name)
|
|
|
|
|
|
def state_dict(self, *args, **kwargs):
|
|
|
"""Forward to the twice-wrapped module."""
|
|
|
return self.module.module.state_dict(*args, **kwargs)
|
|
|
|
|
|
def load_state_dict(self, *args, **kwargs):
|
|
|
"""Forward to the twice-wrapped module."""
|
|
|
return self.module.module.load_state_dict(*args, **kwargs)
|
|
|
|
|
|
def forward(self, *args, **kwargs):
|
|
|
return self.module(*args, **kwargs) |