| | |
| | import torch |
| | import torch.nn as nn |
| | from mmengine.model import BaseModule |
| |
|
| | from mmdet.registry import MODELS |
| |
|
| |
|
| | @MODELS.register_module() |
| | class GlobalAveragePooling(BaseModule): |
| | """Global Average Pooling neck. |
| | |
| | Note that we use `view` to remove extra channel after pooling. We do not |
| | use `squeeze` as it will also remove the batch dimension when the tensor |
| | has a batch dimension of size 1, which can lead to unexpected errors. |
| | """ |
| |
|
| | def __init__(self, kernel_size=None, stride=None): |
| | super(GlobalAveragePooling, self).__init__() |
| | if kernel_size is None and stride is None: |
| | self.gap = nn.AdaptiveAvgPool2d((1, 1)) |
| | else: |
| | self.gap = nn.AvgPool2d(kernel_size, stride) |
| |
|
| | def forward(self, inputs): |
| | if isinstance(inputs, tuple): |
| | outs = tuple([self.gap(x) for x in inputs]) |
| | outs = tuple([ |
| | out.view(x.size(0), |
| | torch.tensor(out.size()[1:]).prod()) |
| | for out, x in zip(outs, inputs) |
| | ]) |
| | elif isinstance(inputs, torch.Tensor): |
| | outs = self.gap(inputs) |
| | outs = outs.view( |
| | inputs.size(0), |
| | torch.tensor(outs.size()[1:]).prod()) |
| | else: |
| | raise TypeError('neck inputs should be tuple or torch.tensor') |
| | return outs |
| |
|