From 700398242ef113e7288ccad999926debd03c5495 Mon Sep 17 00:00:00 2001 From: Dmytro Panchenko Date: Tue, 25 May 2021 17:17:52 +0300 Subject: [PATCH 1/6] Make YOLOv3 neck more flexible --- mmdet/models/necks/yolo_neck.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/mmdet/models/necks/yolo_neck.py b/mmdet/models/necks/yolo_neck.py index 8f6aac7d45b..8b228944e07 100644 --- a/mmdet/models/necks/yolo_neck.py +++ b/mmdet/models/necks/yolo_neck.py @@ -109,7 +109,7 @@ def __init__(self, self.detect1 = DetectionBlock(in_channels[0], out_channels[0], **cfg) for i in range(1, self.num_scales): in_c, out_c = self.in_channels[i], self.out_channels[i] - self.add_module(f'conv{i}', ConvModule(in_c, out_c, 1, **cfg)) + self.add_module(f'conv{i}', ConvModule(out_channels[i - 1], out_c, 1, **cfg)) # in_c + out_c : High-lvl feats will be cat with low-lvl feats self.add_module(f'detect{i+1}', DetectionBlock(in_c + out_c, out_c, **cfg)) From 717723fbab469d86dd03c1f47a8abf2be8456104 Mon Sep 17 00:00:00 2001 From: Dmytro Panchenko Date: Wed, 26 May 2021 08:45:03 +0300 Subject: [PATCH 2/6] Fix linter warning --- mmdet/models/necks/yolo_neck.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/mmdet/models/necks/yolo_neck.py b/mmdet/models/necks/yolo_neck.py index 8b228944e07..165d2f27637 100644 --- a/mmdet/models/necks/yolo_neck.py +++ b/mmdet/models/necks/yolo_neck.py @@ -109,7 +109,8 @@ def __init__(self, self.detect1 = DetectionBlock(in_channels[0], out_channels[0], **cfg) for i in range(1, self.num_scales): in_c, out_c = self.in_channels[i], self.out_channels[i] - self.add_module(f'conv{i}', ConvModule(out_channels[i - 1], out_c, 1, **cfg)) + inter_c = out_channels[i - 1] + self.add_module(f'conv{i}', ConvModule(inter_c, out_c, 1, **cfg)) # in_c + out_c : High-lvl feats will be cat with low-lvl feats self.add_module(f'detect{i+1}', DetectionBlock(in_c + out_c, out_c, **cfg)) From 2fd9922eb70c841dbc410d3a6d1211eafb96e232 Mon Sep 17 00:00:00 2001 From: RangiLyu Date: Wed, 9 Jun 2021 16:57:04 +0800 Subject: [PATCH 3/6] update docstring --- mmdet/models/necks/yolo_neck.py | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/mmdet/models/necks/yolo_neck.py b/mmdet/models/necks/yolo_neck.py index 165d2f27637..eff7897f254 100644 --- a/mmdet/models/necks/yolo_neck.py +++ b/mmdet/models/necks/yolo_neck.py @@ -76,12 +76,13 @@ class YOLOV3Neck(BaseModule): Args: num_scales (int): The number of scales / stages. - in_channels (int): The number of input channels. - out_channels (int): The number of output channels. - conv_cfg (dict): Config dict for convolution layer. Default: None. - norm_cfg (dict): Dictionary to construct and config norm layer. - Default: dict(type='BN', requires_grad=True) - act_cfg (dict): Config dict for activation layer. + in_channels (List[int]): The number of input channels per scale. + out_channels (List[int]): The number of output channels per scale. + conv_cfg (dict, optional): Config dict for convolution layer. + Default: None. + norm_cfg (dict, optional): Dictionary to construct and config norm + layer. Default: dict(type='BN', requires_grad=True) + act_cfg (dict, optional): Config dict for activation layer. Default: dict(type='LeakyReLU', negative_slope=0.1). init_cfg (dict or list[dict], optional): Initialization config dict. Default: None From 29eb02183124d52dc93b3c8250a1e59eea20da5a Mon Sep 17 00:00:00 2001 From: RangiLyu Date: Wed, 9 Jun 2021 16:59:32 +0800 Subject: [PATCH 4/6] add unit test of yolov3 neck --- tests/test_models/test_necks.py | 52 ++++++++++++++++++++++++++++++++- 1 file changed, 51 insertions(+), 1 deletion(-) diff --git a/tests/test_models/test_necks.py b/tests/test_models/test_necks.py index 8305770e38c..21e7278ac9a 100644 --- a/tests/test_models/test_necks.py +++ b/tests/test_models/test_necks.py @@ -2,7 +2,8 @@ import torch from torch.nn.modules.batchnorm import _BatchNorm -from mmdet.models.necks import FPN, ChannelMapper, CTResNetNeck, DilatedEncoder +from mmdet.models.necks import (FPN, ChannelMapper, CTResNetNeck, + DilatedEncoder, YOLOV3Neck) def test_fpn(): @@ -288,3 +289,52 @@ def test_ct_resnet_neck(): feat = feat.cuda() out_feat = ct_resnet_neck([feat])[0] assert out_feat.shape == (1, num_filters[-1], 16, 16) + + +def test_yolov3_neck(): + # num_scales, in_channels, out_channels must be same length + with pytest.raises(AssertionError): + YOLOV3Neck(num_scales=3, in_channels=[16, 8, 4], out_channels=[8, 4]) + + # len(feats) must equal to num_scales + with pytest.raises(AssertionError): + neck = YOLOV3Neck( + num_scales=3, in_channels=[16, 8, 4], out_channels=[8, 4, 2]) + feats = (torch.rand(1, 4, 16, 16), torch.rand(1, 8, 16, 16)) + neck(feats) + + # test normal channels + s = 32 + in_channels = [16, 8, 4] + out_channels = [8, 4, 2] + feat_sizes = [s // 2**i for i in range(len(in_channels) - 1, -1, -1)] + feats = [ + torch.rand(1, in_channels[i], feat_sizes[i], feat_sizes[i]) + for i in range(len(in_channels) - 1, -1, -1) + ] + neck = YOLOV3Neck( + num_scales=3, in_channels=in_channels, out_channels=out_channels) + outs = neck(feats) + + assert len(outs) == len(feats) + for i in range(len(outs)): + assert outs[i].shape == \ + (1, out_channels[i], feat_sizes[i], feat_sizes[i]) + + # test more flexible setting + s = 32 + in_channels = [32, 8, 16] + out_channels = [8, 16, 4] + feat_sizes = [s // 2**i for i in range(len(in_channels) - 1, -1, -1)] + feats = [ + torch.rand(1, in_channels[i], feat_sizes[i], feat_sizes[i]) + for i in range(len(in_channels) - 1, -1, -1) + ] + neck = YOLOV3Neck( + num_scales=3, in_channels=in_channels, out_channels=out_channels) + outs = neck(feats) + + assert len(outs) == len(feats) + for i in range(len(outs)): + assert outs[i].shape == \ + (1, out_channels[i], feat_sizes[i], feat_sizes[i]) From 79996c8e9d914290cb0cb2838f9f830ddab6c037 Mon Sep 17 00:00:00 2001 From: RangiLyu Date: Fri, 11 Jun 2021 14:42:39 +0800 Subject: [PATCH 5/6] update --- tests/test_models/test_necks.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/tests/test_models/test_necks.py b/tests/test_models/test_necks.py index 21e7278ac9a..0e2ba378ae7 100644 --- a/tests/test_models/test_necks.py +++ b/tests/test_models/test_necks.py @@ -324,7 +324,7 @@ def test_yolov3_neck(): # test more flexible setting s = 32 in_channels = [32, 8, 16] - out_channels = [8, 16, 4] + out_channels = [19, 21, 5] feat_sizes = [s // 2**i for i in range(len(in_channels) - 1, -1, -1)] feats = [ torch.rand(1, in_channels[i], feat_sizes[i], feat_sizes[i]) From b962b6eac91fd517bd82d79b383d945c60d4b406 Mon Sep 17 00:00:00 2001 From: RangiLyu Date: Tue, 15 Jun 2021 13:15:57 +0800 Subject: [PATCH 6/6] fix docstring --- mmdet/models/necks/yolo_neck.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/mmdet/models/necks/yolo_neck.py b/mmdet/models/necks/yolo_neck.py index eff7897f254..999fb483779 100644 --- a/mmdet/models/necks/yolo_neck.py +++ b/mmdet/models/necks/yolo_neck.py @@ -76,7 +76,7 @@ class YOLOV3Neck(BaseModule): Args: num_scales (int): The number of scales / stages. - in_channels (List[int]): The number of input channels per scale. + in_channels (List[int]): The number of input channels per scale. out_channels (List[int]): The number of output channels per scale. conv_cfg (dict, optional): Config dict for convolution layer. Default: None.