72 lines
2.5 KiB
Python
72 lines
2.5 KiB
Python
# Copyright The PyTorch Lightning team.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
"""Utilities for automatic parameters tying.
|
|
|
|
Reference:
|
|
https://github.com/pytorch/fairseq/blob/1f7ef9ed1e1061f8c7f88f8b94c7186834398690/fairseq/trainer.py#L110-L118
|
|
"""
|
|
from typing import Dict, List, Optional
|
|
|
|
from torch import nn
|
|
from torch.nn import Parameter
|
|
|
|
|
|
def find_shared_parameters(module: nn.Module) -> List[str]:
|
|
"""Returns a list of names of shared parameters set in the module."""
|
|
return _find_shared_parameters(module)
|
|
|
|
|
|
def _find_shared_parameters(module: nn.Module, tied_parameters: Optional[Dict] = None, prefix: str = "") -> List[str]:
|
|
if tied_parameters is None:
|
|
first_call = True
|
|
tied_parameters = {}
|
|
else:
|
|
first_call = False
|
|
for name, param in module._parameters.items():
|
|
param_prefix = prefix + ("." if prefix else "") + name
|
|
if param is None:
|
|
continue
|
|
if param not in tied_parameters:
|
|
tied_parameters[param] = []
|
|
tied_parameters[param].append(param_prefix)
|
|
for name, m in module._modules.items():
|
|
if m is None:
|
|
continue
|
|
submodule_prefix = prefix + ("." if prefix else "") + name
|
|
_find_shared_parameters(m, tied_parameters, submodule_prefix)
|
|
if first_call:
|
|
return [x for x in tied_parameters.values() if len(x) > 1]
|
|
|
|
|
|
def set_shared_parameters(module: nn.Module, shared_params: list) -> nn.Module:
|
|
for shared_param in shared_params:
|
|
ref = _get_module_by_path(module, shared_param[0])
|
|
for path in shared_param[1:]:
|
|
_set_module_by_path(module, path, ref)
|
|
return module
|
|
|
|
|
|
def _get_module_by_path(module: nn.Module, path: str) -> nn.Module:
|
|
path = path.split(".")
|
|
for name in path:
|
|
module = getattr(module, name)
|
|
return module
|
|
|
|
|
|
def _set_module_by_path(module: nn.Module, path: str, value: Parameter) -> None:
|
|
path = path.split(".")
|
|
for name in path[:-1]:
|
|
module = getattr(module, name)
|
|
setattr(module, path[-1], value)
|