You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
277 lines
11 KiB
277 lines
11 KiB
# Copyright (c) 2022 PaddlePaddle Authors. All Rights Reserved. |
|
# |
|
# Licensed under the Apache License, Version 2.0 (the "License"); |
|
# you may not use this file except in compliance with the License. |
|
# You may obtain a copy of the License at |
|
# |
|
# http://www.apache.org/licenses/LICENSE-2.0 |
|
# |
|
# Unless required by applicable law or agreed to in writing, software |
|
# distributed under the License is distributed on an "AS IS" BASIS, |
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
|
# See the License for the specific language governing permissions and |
|
# limitations under the License. |
|
""" |
|
This code is based on https://github.com/Z-Zheng/FarSeg |
|
Ths copyright of Z-Zheng/FarSeg is as follows: |
|
Apache License [see LICENSE for details] |
|
""" |
|
|
|
import math |
|
import paddle.nn as nn |
|
import paddle.nn.functional as F |
|
from paddle.vision.models import resnet50 |
|
from paddle import nn |
|
import paddle.nn.functional as F |
|
from .layers import (Identity, ConvReLU, kaiming_normal_init, constant_init) |
|
|
|
|
|
class FPN(nn.Layer): |
|
""" |
|
Module that adds FPN on top of a list of feature maps. |
|
The feature maps are currently supposed to be in increasing depth |
|
order, and must be consecutive |
|
""" |
|
|
|
def __init__(self, |
|
in_channels_list, |
|
out_channels, |
|
conv_block=ConvReLU, |
|
top_blocks=None): |
|
super(FPN, self).__init__() |
|
self.inner_blocks = [] |
|
self.layer_blocks = [] |
|
for idx, in_channels in enumerate(in_channels_list, 1): |
|
inner_block = "fpn_inner{}".format(idx) |
|
layer_block = "fpn_layer{}".format(idx) |
|
if in_channels == 0: |
|
continue |
|
inner_block_module = conv_block(in_channels, out_channels, 1) |
|
layer_block_module = conv_block(out_channels, out_channels, 3, 1) |
|
self.add_sublayer(inner_block, inner_block_module) |
|
self.add_sublayer(layer_block, layer_block_module) |
|
for module in [inner_block_module, layer_block_module]: |
|
for m in module.sublayers(): |
|
if isinstance(m, nn.Conv2D): |
|
kaiming_normal_init(m.weight) |
|
self.inner_blocks.append(inner_block) |
|
self.layer_blocks.append(layer_block) |
|
self.top_blocks = top_blocks |
|
|
|
def forward(self, x): |
|
last_inner = getattr(self, self.inner_blocks[-1])(x[-1]) |
|
results = [getattr(self, self.layer_blocks[-1])(last_inner)] |
|
for feature, inner_block, layer_block in zip( |
|
x[:-1][::-1], self.inner_blocks[:-1][::-1], |
|
self.layer_blocks[:-1][::-1]): |
|
if not inner_block: |
|
continue |
|
inner_top_down = F.interpolate( |
|
last_inner, scale_factor=2, mode="nearest") |
|
inner_lateral = getattr(self, inner_block)(feature) |
|
last_inner = inner_lateral + inner_top_down |
|
results.insert(0, getattr(self, layer_block)(last_inner)) |
|
if isinstance(self.top_blocks, LastLevelP6P7): |
|
last_results = self.top_blocks(x[-1], results[-1]) |
|
results.extend(last_results) |
|
elif isinstance(self.top_blocks, LastLevelMaxPool): |
|
last_results = self.top_blocks(results[-1]) |
|
results.extend(last_results) |
|
return tuple(results) |
|
|
|
|
|
class LastLevelMaxPool(nn.Layer): |
|
def forward(self, x): |
|
return [F.max_pool2d(x, 1, 2, 0)] |
|
|
|
|
|
class LastLevelP6P7(nn.Layer): |
|
""" |
|
This module is used in RetinaNet to generate extra layers, P6 and P7. |
|
""" |
|
|
|
def __init__(self, in_channels, out_channels): |
|
super(LastLevelP6P7, self).__init__() |
|
self.p6 = nn.Conv2D(in_channels, out_channels, 3, 2, 1) |
|
self.p7 = nn.Conv2D(out_channels, out_channels, 3, 2, 1) |
|
for module in [self.p6, self.p7]: |
|
for m in module.sublayers(): |
|
kaiming_normal_init(m.weight) |
|
constant_init(m.bias, value=0) |
|
self.use_P5 = in_channels == out_channels |
|
|
|
def forward(self, c5, p5): |
|
x = p5 if self.use_P5 else c5 |
|
p6 = self.p6(x) |
|
p7 = self.p7(F.relu(p6)) |
|
return [p6, p7] |
|
|
|
|
|
class SceneRelation(nn.Layer): |
|
def __init__(self, |
|
in_channels, |
|
channel_list, |
|
out_channels, |
|
scale_aware_proj=True): |
|
super(SceneRelation, self).__init__() |
|
self.scale_aware_proj = scale_aware_proj |
|
if scale_aware_proj: |
|
self.scene_encoder = nn.LayerList([ |
|
nn.Sequential( |
|
nn.Conv2D(in_channels, out_channels, 1), |
|
nn.ReLU(), nn.Conv2D(out_channels, out_channels, 1)) |
|
for _ in range(len(channel_list)) |
|
]) |
|
else: |
|
# 2mlp |
|
self.scene_encoder = nn.Sequential( |
|
nn.Conv2D(in_channels, out_channels, 1), |
|
nn.ReLU(), |
|
nn.Conv2D(out_channels, out_channels, 1), ) |
|
self.content_encoders = nn.LayerList() |
|
self.feature_reencoders = nn.LayerList() |
|
for c in channel_list: |
|
self.content_encoders.append( |
|
nn.Sequential( |
|
nn.Conv2D(c, out_channels, 1), |
|
nn.BatchNorm2D(out_channels), nn.ReLU())) |
|
self.feature_reencoders.append( |
|
nn.Sequential( |
|
nn.Conv2D(c, out_channels, 1), |
|
nn.BatchNorm2D(out_channels), nn.ReLU())) |
|
self.normalizer = nn.Sigmoid() |
|
|
|
def forward(self, scene_feature, features: list): |
|
content_feats = [ |
|
c_en(p_feat) |
|
for c_en, p_feat in zip(self.content_encoders, features) |
|
] |
|
if self.scale_aware_proj: |
|
scene_feats = [op(scene_feature) for op in self.scene_encoder] |
|
relations = [ |
|
self.normalizer((sf * cf).sum(axis=1, keepdim=True)) |
|
for sf, cf in zip(scene_feats, content_feats) |
|
] |
|
else: |
|
scene_feat = self.scene_encoder(scene_feature) |
|
relations = [ |
|
self.normalizer((scene_feat * cf).sum(axis=1, keepdim=True)) |
|
for cf in content_feats |
|
] |
|
p_feats = [ |
|
op(p_feat) for op, p_feat in zip(self.feature_reencoders, features) |
|
] |
|
refined_feats = [r * p for r, p in zip(relations, p_feats)] |
|
return refined_feats |
|
|
|
|
|
class AssymetricDecoder(nn.Layer): |
|
def __init__(self, |
|
in_channels, |
|
out_channels, |
|
in_feat_output_strides=(4, 8, 16, 32), |
|
out_feat_output_stride=4, |
|
norm_fn=nn.BatchNorm2D, |
|
num_groups_gn=None): |
|
super(AssymetricDecoder, self).__init__() |
|
if norm_fn == nn.BatchNorm2D: |
|
norm_fn_args = dict(num_features=out_channels) |
|
elif norm_fn == nn.GroupNorm: |
|
if num_groups_gn is None: |
|
raise ValueError( |
|
'When norm_fn is nn.GroupNorm, num_groups_gn is needed.') |
|
norm_fn_args = dict( |
|
num_groups=num_groups_gn, num_channels=out_channels) |
|
else: |
|
raise ValueError('Type of {} is not support.'.format(type(norm_fn))) |
|
self.blocks = nn.LayerList() |
|
for in_feat_os in in_feat_output_strides: |
|
num_upsample = int(math.log2(int(in_feat_os))) - int( |
|
math.log2(int(out_feat_output_stride))) |
|
num_layers = num_upsample if num_upsample != 0 else 1 |
|
self.blocks.append( |
|
nn.Sequential(*[ |
|
nn.Sequential( |
|
nn.Conv2D( |
|
in_channels if idx == 0 else out_channels, |
|
out_channels, |
|
3, |
|
1, |
|
1, |
|
bias_attr=False), |
|
norm_fn(**norm_fn_args) |
|
if norm_fn is not None else Identity(), |
|
nn.ReLU(), |
|
nn.UpsamplingBilinear2D(scale_factor=2) if num_upsample |
|
!= 0 else Identity(), ) for idx in range(num_layers) |
|
])) |
|
|
|
def forward(self, feat_list: list): |
|
inner_feat_list = [] |
|
for idx, block in enumerate(self.blocks): |
|
decoder_feat = block(feat_list[idx]) |
|
inner_feat_list.append(decoder_feat) |
|
out_feat = sum(inner_feat_list) / 4. |
|
return out_feat |
|
|
|
|
|
class ResNet50Encoder(nn.Layer): |
|
def __init__(self, pretrained=True): |
|
super(ResNet50Encoder, self).__init__() |
|
self.resnet = resnet50(pretrained=pretrained) |
|
|
|
def forward(self, inputs): |
|
x = inputs |
|
x = self.resnet.conv1(x) |
|
x = self.resnet.bn1(x) |
|
x = self.resnet.relu(x) |
|
x = self.resnet.maxpool(x) |
|
c2 = self.resnet.layer1(x) |
|
c3 = self.resnet.layer2(c2) |
|
c4 = self.resnet.layer3(c3) |
|
c5 = self.resnet.layer4(c4) |
|
return [c2, c3, c4, c5] |
|
|
|
|
|
class FarSeg(nn.Layer): |
|
''' |
|
The FarSeg implementation based on PaddlePaddle. |
|
|
|
The original article refers to |
|
Zheng, Zhuo, et al. "Foreground-Aware Relation Network for Geospatial Object Segmentation in High Spatial Resolution Remote Sensing Imagery" |
|
(https://openaccess.thecvf.com/content_CVPR_2020/papers/Zheng_Foreground-Aware_Relation_Network_for_Geospatial_Object_Segmentation_in_High_Spatial_CVPR_2020_paper.pdf) |
|
''' |
|
|
|
def __init__(self, |
|
num_classes=16, |
|
fpn_ch_list=(256, 512, 1024, 2048), |
|
mid_ch=256, |
|
out_ch=128, |
|
sr_ch_list=(256, 256, 256, 256), |
|
encoder_pretrained=True): |
|
super(FarSeg, self).__init__() |
|
self.en = ResNet50Encoder(encoder_pretrained) |
|
self.fpn = FPN(in_channels_list=fpn_ch_list, out_channels=mid_ch) |
|
self.decoder = AssymetricDecoder( |
|
in_channels=mid_ch, out_channels=out_ch) |
|
self.cls_pred_conv = nn.Conv2D(out_ch, num_classes, 1) |
|
self.upsample4x_op = nn.UpsamplingBilinear2D(scale_factor=4) |
|
self.scene_relation = True if sr_ch_list is not None else False |
|
if self.scene_relation: |
|
self.gap = nn.AdaptiveAvgPool2D(1) |
|
self.sr = SceneRelation(fpn_ch_list[-1], sr_ch_list, mid_ch) |
|
|
|
def forward(self, x): |
|
feat_list = self.en(x) |
|
fpn_feat_list = self.fpn(feat_list) |
|
if self.scene_relation: |
|
c5 = feat_list[-1] |
|
c6 = self.gap(c5) |
|
refined_fpn_feat_list = self.sr(c6, fpn_feat_list) |
|
else: |
|
refined_fpn_feat_list = fpn_feat_list |
|
final_feat = self.decoder(refined_fpn_feat_list) |
|
cls_pred = self.cls_pred_conv(final_feat) |
|
cls_pred = self.upsample4x_op(cls_pred) |
|
cls_pred = F.softmax(cls_pred, axis=1) |
|
return [cls_pred]
|
|
|