| |
| import os.path as osp |
| import platform |
| import shutil |
| import time |
| import warnings |
|
|
| import torch |
| from torch.optim import Optimizer |
|
|
| import annotator.mmpkg.mmcv as mmcv |
| from .base_runner import BaseRunner |
| from .builder import RUNNERS |
| from .checkpoint import save_checkpoint |
| from .hooks import IterTimerHook |
| from .utils import get_host_info |
|
|
|
|
| class IterLoader: |
|
|
| def __init__(self, dataloader): |
| self._dataloader = dataloader |
| self.iter_loader = iter(self._dataloader) |
| self._epoch = 0 |
|
|
| @property |
| def epoch(self): |
| return self._epoch |
|
|
| def __next__(self): |
| try: |
| data = next(self.iter_loader) |
| except StopIteration: |
| self._epoch += 1 |
| if hasattr(self._dataloader.sampler, 'set_epoch'): |
| self._dataloader.sampler.set_epoch(self._epoch) |
| time.sleep(2) |
| self.iter_loader = iter(self._dataloader) |
| data = next(self.iter_loader) |
|
|
| return data |
|
|
| def __len__(self): |
| return len(self._dataloader) |
|
|
|
|
| @RUNNERS.register_module() |
| class IterBasedRunner(BaseRunner): |
| """Iteration-based Runner. |
| |
| This runner train models iteration by iteration. |
| """ |
|
|
| def train(self, data_loader, **kwargs): |
| self.model.train() |
| self.mode = 'train' |
| self.data_loader = data_loader |
| self._epoch = data_loader.epoch |
| data_batch = next(data_loader) |
| self.call_hook('before_train_iter') |
| outputs = self.model.train_step(data_batch, self.optimizer, **kwargs) |
| if not isinstance(outputs, dict): |
| raise TypeError('model.train_step() must return a dict') |
| if 'log_vars' in outputs: |
| self.log_buffer.update(outputs['log_vars'], outputs['num_samples']) |
| self.outputs = outputs |
| self.call_hook('after_train_iter') |
| self._inner_iter += 1 |
| self._iter += 1 |
|
|
| @torch.no_grad() |
| def val(self, data_loader, **kwargs): |
| self.model.eval() |
| self.mode = 'val' |
| self.data_loader = data_loader |
| data_batch = next(data_loader) |
| self.call_hook('before_val_iter') |
| outputs = self.model.val_step(data_batch, **kwargs) |
| if not isinstance(outputs, dict): |
| raise TypeError('model.val_step() must return a dict') |
| if 'log_vars' in outputs: |
| self.log_buffer.update(outputs['log_vars'], outputs['num_samples']) |
| self.outputs = outputs |
| self.call_hook('after_val_iter') |
| self._inner_iter += 1 |
|
|
| def run(self, data_loaders, workflow, max_iters=None, **kwargs): |
| """Start running. |
| |
| Args: |
| data_loaders (list[:obj:`DataLoader`]): Dataloaders for training |
| and validation. |
| workflow (list[tuple]): A list of (phase, iters) to specify the |
| running order and iterations. E.g, [('train', 10000), |
| ('val', 1000)] means running 10000 iterations for training and |
| 1000 iterations for validation, iteratively. |
| """ |
| assert isinstance(data_loaders, list) |
| assert mmcv.is_list_of(workflow, tuple) |
| assert len(data_loaders) == len(workflow) |
| if max_iters is not None: |
| warnings.warn( |
| 'setting max_iters in run is deprecated, ' |
| 'please set max_iters in runner_config', DeprecationWarning) |
| self._max_iters = max_iters |
| assert self._max_iters is not None, ( |
| 'max_iters must be specified during instantiation') |
|
|
| work_dir = self.work_dir if self.work_dir is not None else 'NONE' |
| self.logger.info('Start running, host: %s, work_dir: %s', |
| get_host_info(), work_dir) |
| self.logger.info('Hooks will be executed in the following order:\n%s', |
| self.get_hook_info()) |
| self.logger.info('workflow: %s, max: %d iters', workflow, |
| self._max_iters) |
| self.call_hook('before_run') |
|
|
| iter_loaders = [IterLoader(x) for x in data_loaders] |
|
|
| self.call_hook('before_epoch') |
|
|
| while self.iter < self._max_iters: |
| for i, flow in enumerate(workflow): |
| self._inner_iter = 0 |
| mode, iters = flow |
| if not isinstance(mode, str) or not hasattr(self, mode): |
| raise ValueError( |
| 'runner has no method named "{}" to run a workflow'. |
| format(mode)) |
| iter_runner = getattr(self, mode) |
| for _ in range(iters): |
| if mode == 'train' and self.iter >= self._max_iters: |
| break |
| iter_runner(iter_loaders[i], **kwargs) |
|
|
| time.sleep(1) |
| self.call_hook('after_epoch') |
| self.call_hook('after_run') |
|
|
| def resume(self, |
| checkpoint, |
| resume_optimizer=True, |
| map_location='default'): |
| """Resume model from checkpoint. |
| |
| Args: |
| checkpoint (str): Checkpoint to resume from. |
| resume_optimizer (bool, optional): Whether resume the optimizer(s) |
| if the checkpoint file includes optimizer(s). Default to True. |
| map_location (str, optional): Same as :func:`torch.load`. |
| Default to 'default'. |
| """ |
| if map_location == 'default': |
| device_id = torch.cuda.current_device() |
| checkpoint = self.load_checkpoint( |
| checkpoint, |
| map_location=lambda storage, loc: storage.cuda(device_id)) |
| else: |
| checkpoint = self.load_checkpoint( |
| checkpoint, map_location=map_location) |
|
|
| self._epoch = checkpoint['meta']['epoch'] |
| self._iter = checkpoint['meta']['iter'] |
| self._inner_iter = checkpoint['meta']['iter'] |
| if 'optimizer' in checkpoint and resume_optimizer: |
| if isinstance(self.optimizer, Optimizer): |
| self.optimizer.load_state_dict(checkpoint['optimizer']) |
| elif isinstance(self.optimizer, dict): |
| for k in self.optimizer.keys(): |
| self.optimizer[k].load_state_dict( |
| checkpoint['optimizer'][k]) |
| else: |
| raise TypeError( |
| 'Optimizer should be dict or torch.optim.Optimizer ' |
| f'but got {type(self.optimizer)}') |
|
|
| self.logger.info(f'resumed from epoch: {self.epoch}, iter {self.iter}') |
|
|
| def save_checkpoint(self, |
| out_dir, |
| filename_tmpl='iter_{}.pth', |
| meta=None, |
| save_optimizer=True, |
| create_symlink=True): |
| """Save checkpoint to file. |
| |
| Args: |
| out_dir (str): Directory to save checkpoint files. |
| filename_tmpl (str, optional): Checkpoint file template. |
| Defaults to 'iter_{}.pth'. |
| meta (dict, optional): Metadata to be saved in checkpoint. |
| Defaults to None. |
| save_optimizer (bool, optional): Whether save optimizer. |
| Defaults to True. |
| create_symlink (bool, optional): Whether create symlink to the |
| latest checkpoint file. Defaults to True. |
| """ |
| if meta is None: |
| meta = {} |
| elif not isinstance(meta, dict): |
| raise TypeError( |
| f'meta should be a dict or None, but got {type(meta)}') |
| if self.meta is not None: |
| meta.update(self.meta) |
| |
| |
| |
| |
| meta.update(epoch=self.epoch + 1, iter=self.iter) |
|
|
| filename = filename_tmpl.format(self.iter + 1) |
| filepath = osp.join(out_dir, filename) |
| optimizer = self.optimizer if save_optimizer else None |
| save_checkpoint(self.model, filepath, optimizer=optimizer, meta=meta) |
| |
| |
| if create_symlink: |
| dst_file = osp.join(out_dir, 'latest.pth') |
| if platform.system() != 'Windows': |
| mmcv.symlink(filename, dst_file) |
| else: |
| shutil.copy(filepath, dst_file) |
|
|
| def register_training_hooks(self, |
| lr_config, |
| optimizer_config=None, |
| checkpoint_config=None, |
| log_config=None, |
| momentum_config=None, |
| custom_hooks_config=None): |
| """Register default hooks for iter-based training. |
| |
| Checkpoint hook, optimizer stepper hook and logger hooks will be set to |
| `by_epoch=False` by default. |
| |
| Default hooks include: |
| |
| +----------------------+-------------------------+ |
| | Hooks | Priority | |
| +======================+=========================+ |
| | LrUpdaterHook | VERY_HIGH (10) | |
| +----------------------+-------------------------+ |
| | MomentumUpdaterHook | HIGH (30) | |
| +----------------------+-------------------------+ |
| | OptimizerStepperHook | ABOVE_NORMAL (40) | |
| +----------------------+-------------------------+ |
| | CheckpointSaverHook | NORMAL (50) | |
| +----------------------+-------------------------+ |
| | IterTimerHook | LOW (70) | |
| +----------------------+-------------------------+ |
| | LoggerHook(s) | VERY_LOW (90) | |
| +----------------------+-------------------------+ |
| | CustomHook(s) | defaults to NORMAL (50) | |
| +----------------------+-------------------------+ |
| |
| If custom hooks have same priority with default hooks, custom hooks |
| will be triggered after default hooks. |
| """ |
| if checkpoint_config is not None: |
| checkpoint_config.setdefault('by_epoch', False) |
| if lr_config is not None: |
| lr_config.setdefault('by_epoch', False) |
| if log_config is not None: |
| for info in log_config['hooks']: |
| info.setdefault('by_epoch', False) |
| super(IterBasedRunner, self).register_training_hooks( |
| lr_config=lr_config, |
| momentum_config=momentum_config, |
| optimizer_config=optimizer_config, |
| checkpoint_config=checkpoint_config, |
| log_config=log_config, |
| timer_config=IterTimerHook(), |
| custom_hooks_config=custom_hooks_config) |
|
|