Source code for opacus.utils.module_utils

#!/usr/bin/env python3
# Copyright (c) Meta Platforms, Inc. and affiliates.
#
# 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.

import io
import logging
import sys
from typing import Dict, Iterable, List, Tuple

import torch
import torch.nn as nn


logging.basicConfig(
    format="%(asctime)s:%(levelname)s:%(message)s",
    datefmt="%m/%d/%Y %H:%M:%S",
    stream=sys.stderr,
)
logger = logging.getLogger(__name__)
logger.setLevel(level=logging.INFO)


def has_trainable_params(module: nn.Module) -> bool:
    return any(p.requires_grad for p in module.parameters(recurse=False))


[docs] def parametrized_modules(module: nn.Module) -> Iterable[Tuple[str, nn.Module]]: """ Recursively iterates over all submodules, returning those that have parameters (as opposed to "wrapper modules" that just organize modules). """ yield from ( (m_name, m) for (m_name, m) in module.named_modules() if any(p is not None for p in m.parameters(recurse=False)) )
[docs] def trainable_modules(module: nn.Module) -> Iterable[Tuple[str, nn.Module]]: """ Recursively iterates over all submodules, returning those that have parameters and are trainable (ie they want a grad). """ yield from ( (m_name, m) for (m_name, m) in parametrized_modules(module) if any(p.requires_grad for p in m.parameters(recurse=False)) )
[docs] def trainable_parameters(module: nn.Module) -> Iterable[Tuple[str, nn.Parameter]]: """ Recursively iterates over all parameters, returning those that are trainable (ie they want a grad). """ yield from ( (p_name, p) for (p_name, p) in module.named_parameters() if p.requires_grad )
[docs] def requires_grad(module: nn.Module, *, recurse: bool = False) -> bool: """ Checks if any parameters in a specified module require gradients. Args: module: PyTorch module whose parameters are to be examined. recurse: Flag specifying if the gradient requirement check should be applied recursively to submodules of the specified module Returns: Flag indicate if any parameters require gradients """ requires_grad = any(p.requires_grad for p in module.parameters(recurse)) return requires_grad
[docs] def clone_module(module: nn.Module) -> nn.Module: """ Handy utility to clone an nn.Module. PyTorch doesn't always support copy.deepcopy(), so it is just easier to serialize the model to a BytesIO and read it from there. When ``weights_only=False``, ``torch.load()`` uses "pickle" module implicity, which is known to be insecure. Only load the model you trust. Args: module: The module to clone Returns: The clone of ``module`` """ with io.BytesIO() as bytesio: torch.save(module, bytesio) bytesio.seek(0) module_copy = torch.load(bytesio, weights_only=False) next_param = next( module.parameters(), None ) # Eg, InstanceNorm with affine=False has no params return module_copy.to(next_param.device) if next_param is not None else module_copy
[docs] def get_submodule(module: nn.Module, target: str) -> nn.Module: """ Returns the submodule given by target if it exists, otherwise throws an error. This is copy-pasta of Pytorch 1.9's ``get_submodule()`` implementation; and is included here to also support Pytorch 1.8. This function can be removed in favour of ``module.get_submodule()`` once Opacus abandons support for torch 1.8. See more details at https://pytorch.org/docs/stable/generated/torch.nn.Module.html?highlight=get_submodule#torch.nn.Module.get_submodule Args: module: module target: submodule string Returns: The submodule given by target if it exists Raises: AttributeError If submodule doesn't exist """ if target == "": return module atoms: List[str] = target.split(".") mod: nn.Module = module for item in atoms: if not hasattr(mod, item): raise AttributeError( mod._get_name() + " has no " "attribute `" + item + "`" ) mod = getattr(mod, item) if not isinstance(mod, torch.nn.Module): raise AttributeError("`" + item + "` is not " "an nn.Module") return mod
[docs] def are_state_dict_equal(sd1: Dict, sd2: Dict): """ Compares two state dicts, while logging discrepancies """ if len(sd1) != len(sd2): return False for k1, v1 in sd1.items(): # check that all keys are accounted for. if k1 not in sd2: return False # check that value tensors are equal. v2 = sd2[k1] if not torch.allclose(v1, v2): return False return True