| | |
| | import importlib |
| | import importlib.util |
| | import logging |
| | import numpy as np |
| | import os |
| | import random |
| | import sys |
| | from datetime import datetime |
| | import torch |
| |
|
| | __all__ = ["seed_all_rng"] |
| |
|
| |
|
| | TORCH_VERSION = tuple(int(x) for x in torch.__version__.split(".")[:2]) |
| | """ |
| | PyTorch version as a tuple of 2 ints. Useful for comparison. |
| | """ |
| |
|
| |
|
| | DOC_BUILDING = os.getenv("_DOC_BUILDING", False) |
| | """ |
| | Whether we're building documentation. |
| | """ |
| |
|
| |
|
| | def seed_all_rng(seed=None): |
| | """ |
| | Set the random seed for the RNG in torch, numpy and python. |
| | |
| | Args: |
| | seed (int): if None, will use a strong random seed. |
| | """ |
| | if seed is None: |
| | seed = ( |
| | os.getpid() |
| | + int(datetime.now().strftime("%S%f")) |
| | + int.from_bytes(os.urandom(2), "big") |
| | ) |
| | logger = logging.getLogger(__name__) |
| | logger.info("Using a generated random seed {}".format(seed)) |
| | np.random.seed(seed) |
| | torch.manual_seed(seed) |
| | random.seed(seed) |
| | torch.cuda.manual_seed_all(str(seed)) |
| | os.environ["PYTHONHASHSEED"] = str(seed) |
| |
|
| |
|
| | |
| | def _import_file(module_name, file_path, make_importable=False): |
| | spec = importlib.util.spec_from_file_location(module_name, file_path) |
| | module = importlib.util.module_from_spec(spec) |
| | spec.loader.exec_module(module) |
| | if make_importable: |
| | sys.modules[module_name] = module |
| | return module |
| |
|
| |
|
| | def _configure_libraries(): |
| | """ |
| | Configurations for some libraries. |
| | """ |
| | |
| | |
| | disable_cv2 = int(os.environ.get("DETECTRON2_DISABLE_CV2", False)) |
| | if disable_cv2: |
| | sys.modules["cv2"] = None |
| | else: |
| | |
| | |
| | os.environ["OPENCV_OPENCL_RUNTIME"] = "disabled" |
| | try: |
| | import cv2 |
| |
|
| | if int(cv2.__version__.split(".")[0]) >= 3: |
| | cv2.ocl.setUseOpenCL(False) |
| | except ModuleNotFoundError: |
| | |
| | |
| | |
| | pass |
| |
|
| | def get_version(module, digit=2): |
| | return tuple(map(int, module.__version__.split(".")[:digit])) |
| |
|
| | |
| | assert get_version(torch) >= (1, 4), "Requires torch>=1.4" |
| | import fvcore |
| | assert get_version(fvcore, 3) >= (0, 1, 2), "Requires fvcore>=0.1.2" |
| | import yaml |
| | assert get_version(yaml) >= (5, 1), "Requires pyyaml>=5.1" |
| | |
| |
|
| |
|
| | _ENV_SETUP_DONE = False |
| |
|
| |
|
| | def setup_environment(): |
| | """Perform environment setup work. The default setup is a no-op, but this |
| | function allows the user to specify a Python source file or a module in |
| | the $DETECTRON2_ENV_MODULE environment variable, that performs |
| | custom setup work that may be necessary to their computing environment. |
| | """ |
| | global _ENV_SETUP_DONE |
| | if _ENV_SETUP_DONE: |
| | return |
| | _ENV_SETUP_DONE = True |
| |
|
| | _configure_libraries() |
| |
|
| | custom_module_path = os.environ.get("DETECTRON2_ENV_MODULE") |
| |
|
| | if custom_module_path: |
| | setup_custom_environment(custom_module_path) |
| | else: |
| | |
| | pass |
| |
|
| |
|
| | def setup_custom_environment(custom_module): |
| | """ |
| | Load custom environment setup by importing a Python source file or a |
| | module, and run the setup function. |
| | """ |
| | if custom_module.endswith(".py"): |
| | module = _import_file("detectron2.utils.env.custom_module", custom_module) |
| | else: |
| | module = importlib.import_module(custom_module) |
| | assert hasattr(module, "setup_environment") and callable(module.setup_environment), ( |
| | "Custom environment module defined in {} does not have the " |
| | "required callable attribute 'setup_environment'." |
| | ).format(custom_module) |
| | module.setup_environment() |
| |
|
| |
|
| | def fixup_module_metadata(module_name, namespace, keys=None): |
| | """ |
| | Fix the __qualname__ of module members to be their exported api name, so |
| | when they are referenced in docs, sphinx can find them. Reference: |
| | https://github.com/python-trio/trio/blob/6754c74eacfad9cc5c92d5c24727a2f3b620624e/trio/_util.py#L216-L241 |
| | """ |
| | if not DOC_BUILDING: |
| | return |
| | seen_ids = set() |
| |
|
| | def fix_one(qualname, name, obj): |
| | |
| | |
| | if id(obj) in seen_ids: |
| | return |
| | seen_ids.add(id(obj)) |
| |
|
| | mod = getattr(obj, "__module__", None) |
| | if mod is not None and (mod.startswith(module_name) or mod.startswith("fvcore.")): |
| | obj.__module__ = module_name |
| | |
| | |
| | |
| | if hasattr(obj, "__name__") and "." not in obj.__name__: |
| | obj.__name__ = name |
| | obj.__qualname__ = qualname |
| | if isinstance(obj, type): |
| | for attr_name, attr_value in obj.__dict__.items(): |
| | fix_one(objname + "." + attr_name, attr_name, attr_value) |
| |
|
| | if keys is None: |
| | keys = namespace.keys() |
| | for objname in keys: |
| | if not objname.startswith("_"): |
| | obj = namespace[objname] |
| | fix_one(objname, objname, obj) |
| |
|