import argparse
import os
import os.path as osp
import warnings
import mmcv
import torch
import numpy as np
from apex import amp
from mmcv import Config, DictAction
from mmcv.cnn import fuse_conv_bn
from mmcv.fileio.io import file_handlers
from mmcv.parallel import MMDataParallel, MMDistributedDataParallel
from mmcv.runner import get_dist_info, init_dist, load_checkpoint
from mmcv.runner.fp16_utils import wrap_fp16_model
from mmaction.datasets import build_dataloader, build_dataset
from mmaction.models import build_model
from mmaction.utils import register_module_hooks
try:
from mmcv.engine import multi_gpu_test, single_gpu_test
except (ImportError, ModuleNotFoundError):
warnings.warn(
'DeprecationWarning: single_gpu_test, multi_gpu_test, '
'collect_results_cpu, collect_results_gpu from mmaction2 will be '
'deprecated. Please install mmcv through master branch.')
from mmaction.apis import multi_gpu_test, single_gpu_test
def parse_args():
parser = argparse.ArgumentParser(
description='MMAction2 test (and eval) a model')
parser.add_argument('--data_root', type=str, default='.')
parser.add_argument('--config', default='config/tsm_k400_pretrained_r50_1x1x8_25e_ucf101_rgb.py',
help='test config file path')
parser.add_argument('--checkpoint', default='./result/epoch_24.pth',
help='checkpoint file')
parser.add_argument('--out', default='result.json', help='output result file in pkl/yaml/json format')
parser.add_argument('--fuse-conv-bn', action='store_true',
help='Whether to fuse conv and bn, this will slightly increase the inference speed')
parser.add_argument(
'--eval',
type=str,
nargs='+',
default='top_k_accuracy',
help='evaluation metrics, which depends on the dataset, e.g.,'
' "top_k_accuracy", "mean_class_accuracy" for video dataset')
parser.add_argument(
'--gpu-collect',
action='store_true',
help='whether to use gpu to collect results')
parser.add_argument(
'--tmpdir',
help='tmp directory used for collecting results from multiple '
'workers, available when gpu-collect is not specified')
parser.add_argument(
'--options',
nargs='+',
action=DictAction,
default={},
help='custom options for evaluation, the key-value pair in xxx=yyy '
'format will be kwargs for dataset.evaluate() function (deprecate), '
'change to --eval-options instead.')
parser.add_argument(
'--eval-options',
nargs='+',
action=DictAction,
default={},
help='custom options for evaluation, the key-value pair in xxx=yyy '
'format will be kwargs for dataset.evaluate() function')
parser.add_argument(
'--cfg-options',
nargs='+',
action=DictAction,
default={},
help='override some settings in the used config, the key-value pair '
'in xxx=yyy format will be merged into config file. For example, '
"'--cfg-options model.backbone.depth=18 model.backbone.with_cp=True'")
parser.add_argument(
'--average-clips',
choices=['score', 'prob', None],
default=None,
help='average type when averaging test clips')
parser.add_argument(
'--launcher',
choices=['none', 'pytorch', 'slurm', 'mpi'],
default='none',
help='job launcher')
parser.add_argument('--local_rank', type=int, default=0)
parser.add_argument(
'--onnx',
action='store_true',
help='Whether to test with onnx model or not')
parser.add_argument(
'--tensorrt',
action='store_true',
help='Whether to test with TensorRT engine or not')
args = parser.parse_args()
if 'LOCAL_RANK' not in os.environ:
os.environ['LOCAL_RANK'] = str(args.local_rank)
if args.options and args.eval_options:
raise ValueError(
'--options and --eval-options cannot be both '
'specified, --options is deprecated in favor of --eval-options')
if args.options:
warnings.warn('--options is deprecated in favor of --eval-options')
args.eval_options = args.options
return args
def turn_off_pretrained(cfg):
if 'pretrained' in cfg:
cfg.pretrained = None
for sub_cfg in cfg.values():
if isinstance(sub_cfg, dict):
turn_off_pretrained(sub_cfg)
def inference_pytorch(args, cfg, distributed, data_loader):
"""Get predictions by pytorch models."""
if args.average_clips is not None:
if cfg.model.get('test_cfg') is None and cfg.get('test_cfg') is None:
cfg.model.setdefault('test_cfg',
dict(average_clips=args.average_clips))
else:
if cfg.model.get('test_cfg') is not None:
cfg.model.test_cfg.average_clips = args.average_clips
else:
cfg.test_cfg.average_clips = args.average_clips
turn_off_pretrained(cfg.model)
model = build_model(
cfg.model, train_cfg=None, test_cfg=cfg.get('test_cfg'))
if cfg.AMP:
model = amp.initialize(model.npu(), opt_level=cfg.OPT_LEVEL, loss_scale=cfg.LOSS_SCALE)
if len(cfg.module_hooks) > 0:
register_module_hooks(model, cfg.module_hooks)
fp16_cfg = cfg.get('fp16', None)
if fp16_cfg is not None:
wrap_fp16_model(model)
load_checkpoint(model, args.checkpoint, map_location='cpu')
if args.fuse_conv_bn:
model = fuse_conv_bn(model)
if not distributed:
model = model.npu()
outputs = single_gpu_test(model, data_loader)
else:
model = MMDistributedDataParallel(
model.npu(),
device_ids=[torch.npu.current_device()],
broadcast_buffers=False)
outputs, times= multi_gpu_test(model, data_loader, args.tmpdir,
args.gpu_collect, True)
return outputs, times
def main():
args = parse_args()
if args.tensorrt and args.onnx:
raise ValueError(
'Cannot set onnx mode and tensorrt mode at the same time.')
cfg = Config.fromfile(args.config)
cfg.merge_from_dict(args.cfg_options)
if args.data_root != '.':
cfg.data.train.ann_file = os.path.join(args.data_root, cfg.data.train.ann_file[9:])
cfg.data.train.data_prefix = os.path.join(args.data_root, cfg.data.train.data_prefix[9:])
cfg.data.val.ann_file = os.path.join(args.data_root, cfg.data.val.ann_file[9:])
cfg.data.val.data_prefix = os.path.join(args.data_root, cfg.data.val.data_prefix[9:])
cfg.data.test.ann_file = os.path.join(args.data_root, cfg.data.test.ann_file[9:])
cfg.data.test.data_prefix = os.path.join(args.data_root, cfg.data.test.data_prefix[9:])
output_config = cfg.get('output_config', {})
if args.out:
output_config = Config._merge_a_into_b(
dict(out=args.out), output_config)
eval_config = cfg.get('eval_config', {})
if args.eval:
eval_config = Config._merge_a_into_b(
dict(metrics=args.eval), eval_config)
if args.eval_options:
eval_config = Config._merge_a_into_b(args.eval_options, eval_config)
assert output_config or eval_config, \
('Please specify at least one operation (save or eval the '
'results) with the argument "--out" or "--eval"')
dataset_type = cfg.data.test.type
if output_config.get('out', None):
if 'output_format' in output_config:
warnings.warn(
'Skip checking `output_format` in localization task.')
else:
out = output_config['out']
mmcv.mkdir_or_exist(osp.dirname(out))
_, suffix = osp.splitext(out)
if dataset_type == 'AVADataset':
assert suffix[1:] == 'csv', ('For AVADataset, the format of '
'the output file should be csv')
else:
assert suffix[1:] in file_handlers, (
'The format of the output '
'file should be json, pickle or yaml')
if cfg.get('cudnn_benchmark', False):
torch.backends.cudnn.benchmark = True
cfg.data.test.test_mode = True
if args.launcher == 'none':
distributed = False
device = torch.device('npu:{}'.format(cfg.DEVICE_ID))
torch.npu.set_device(device)
else:
distributed = True
init_dist(args.launcher, **cfg.dist_params)
cfg.setdefault('module_hooks', [])
dataset = build_dataset(cfg.data.test, dict(test_mode=True))
temp = dataset[0]
dataloader_setting = dict(
videos_per_gpu=cfg.data.get('videos_per_gpu', 1),
workers_per_gpu=cfg.data.get('workers_per_gpu', 1),
dist=distributed,
shuffle=False)
dataloader_setting = dict(dataloader_setting,
**cfg.data.get('test_dataloader', {}))
data_loader = build_dataloader(dataset, **dataloader_setting)
outputs, times = inference_pytorch(args, cfg, distributed, data_loader)
rank, _ = get_dist_info()
if rank == 0:
if output_config.get('out', None):
out = output_config['out']
print(f'\nwriting results to {out}')
dataset.dump_results(outputs, **output_config)
if eval_config:
eval_res = dataset.evaluate(outputs, **eval_config)
for name, val in eval_res.items():
print(f'{name}: {val:.04f}')
times = times[5:-1]
avg_fps = np.average(times) * 8
print('Average Evaluation FPS: {}'.format(avg_fps))
if __name__ == '__main__':
main()