first commit
This commit is contained in:
19
rtdetr_paddle/ppdet/modeling/losses/__init__.py
Normal file
19
rtdetr_paddle/ppdet/modeling/losses/__init__.py
Normal file
@@ -0,0 +1,19 @@
|
||||
# Copyright (c) 2020 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.
|
||||
|
||||
from .iou_loss import *
|
||||
from .gfocal_loss import *
|
||||
from .detr_loss import *
|
||||
from .focal_loss import *
|
||||
from .smooth_l1_loss import *
|
||||
578
rtdetr_paddle/ppdet/modeling/losses/detr_loss.py
Normal file
578
rtdetr_paddle/ppdet/modeling/losses/detr_loss.py
Normal file
@@ -0,0 +1,578 @@
|
||||
# Copyright (c) 2021 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.
|
||||
|
||||
from __future__ import absolute_import
|
||||
from __future__ import division
|
||||
from __future__ import print_function
|
||||
|
||||
import paddle
|
||||
import paddle.nn as nn
|
||||
import paddle.nn.functional as F
|
||||
from ppdet.core.workspace import register
|
||||
from .iou_loss import GIoULoss
|
||||
from ..transformers import bbox_cxcywh_to_xyxy, sigmoid_focal_loss, varifocal_loss_with_logits
|
||||
from ..bbox_utils import bbox_iou
|
||||
|
||||
__all__ = ['DETRLoss', 'DINOLoss']
|
||||
|
||||
|
||||
@register
|
||||
class DETRLoss(nn.Layer):
|
||||
__shared__ = ['num_classes', 'use_focal_loss']
|
||||
__inject__ = ['matcher']
|
||||
|
||||
def __init__(self,
|
||||
num_classes=80,
|
||||
matcher='HungarianMatcher',
|
||||
loss_coeff={
|
||||
'class': 1,
|
||||
'bbox': 5,
|
||||
'giou': 2,
|
||||
'no_object': 0.1,
|
||||
'mask': 1,
|
||||
'dice': 1
|
||||
},
|
||||
aux_loss=True,
|
||||
use_focal_loss=False,
|
||||
use_vfl=False,
|
||||
use_uni_match=False,
|
||||
uni_match_ind=0):
|
||||
r"""
|
||||
Args:
|
||||
num_classes (int): The number of classes.
|
||||
matcher (HungarianMatcher): It computes an assignment between the targets
|
||||
and the predictions of the network.
|
||||
loss_coeff (dict): The coefficient of loss.
|
||||
aux_loss (bool): If 'aux_loss = True', loss at each decoder layer are to be used.
|
||||
use_focal_loss (bool): Use focal loss or not.
|
||||
"""
|
||||
super(DETRLoss, self).__init__()
|
||||
|
||||
self.num_classes = num_classes
|
||||
self.matcher = matcher
|
||||
self.loss_coeff = loss_coeff
|
||||
self.aux_loss = aux_loss
|
||||
self.use_focal_loss = use_focal_loss
|
||||
self.use_vfl = use_vfl
|
||||
self.use_uni_match = use_uni_match
|
||||
self.uni_match_ind = uni_match_ind
|
||||
|
||||
if not self.use_focal_loss:
|
||||
self.loss_coeff['class'] = paddle.full([num_classes + 1],
|
||||
loss_coeff['class'])
|
||||
self.loss_coeff['class'][-1] = loss_coeff['no_object']
|
||||
self.giou_loss = GIoULoss()
|
||||
|
||||
def _get_loss_class(self,
|
||||
logits,
|
||||
gt_class,
|
||||
match_indices,
|
||||
bg_index,
|
||||
num_gts,
|
||||
postfix="",
|
||||
iou_score=None):
|
||||
# logits: [b, query, num_classes], gt_class: list[[n, 1]]
|
||||
name_class = "loss_class" + postfix
|
||||
|
||||
target_label = paddle.full(logits.shape[:2], bg_index, dtype='int64')
|
||||
bs, num_query_objects = target_label.shape
|
||||
num_gt = sum(len(a) for a in gt_class)
|
||||
if num_gt > 0:
|
||||
index, updates = self._get_index_updates(num_query_objects,
|
||||
gt_class, match_indices)
|
||||
target_label = paddle.scatter(
|
||||
target_label.reshape([-1, 1]), index, updates.astype('int64'))
|
||||
target_label = target_label.reshape([bs, num_query_objects])
|
||||
if self.use_focal_loss:
|
||||
target_label = F.one_hot(target_label,
|
||||
self.num_classes + 1)[..., :-1]
|
||||
if iou_score is not None and self.use_vfl:
|
||||
target_score = paddle.zeros([bs, num_query_objects])
|
||||
if num_gt > 0:
|
||||
target_score = paddle.scatter(
|
||||
target_score.reshape([-1, 1]), index, iou_score)
|
||||
target_score = target_score.reshape(
|
||||
[bs, num_query_objects, 1]) * target_label
|
||||
loss_ = self.loss_coeff['class'] * varifocal_loss_with_logits(
|
||||
logits, target_score, target_label,
|
||||
num_gts / num_query_objects)
|
||||
else:
|
||||
loss_ = self.loss_coeff['class'] * sigmoid_focal_loss(
|
||||
logits, target_label, num_gts / num_query_objects)
|
||||
else:
|
||||
loss_ = F.cross_entropy(
|
||||
logits, target_label, weight=self.loss_coeff['class'])
|
||||
return {name_class: loss_}
|
||||
|
||||
def _get_loss_bbox(self, boxes, gt_bbox, match_indices, num_gts,
|
||||
postfix=""):
|
||||
# boxes: [b, query, 4], gt_bbox: list[[n, 4]]
|
||||
name_bbox = "loss_bbox" + postfix
|
||||
name_giou = "loss_giou" + postfix
|
||||
|
||||
loss = dict()
|
||||
if sum(len(a) for a in gt_bbox) == 0:
|
||||
loss[name_bbox] = paddle.to_tensor([0.])
|
||||
loss[name_giou] = paddle.to_tensor([0.])
|
||||
return loss
|
||||
|
||||
src_bbox, target_bbox = self._get_src_target_assign(boxes, gt_bbox,
|
||||
match_indices)
|
||||
loss[name_bbox] = self.loss_coeff['bbox'] * F.l1_loss(
|
||||
src_bbox, target_bbox, reduction='sum') / num_gts
|
||||
loss[name_giou] = self.giou_loss(
|
||||
bbox_cxcywh_to_xyxy(src_bbox), bbox_cxcywh_to_xyxy(target_bbox))
|
||||
loss[name_giou] = loss[name_giou].sum() / num_gts
|
||||
loss[name_giou] = self.loss_coeff['giou'] * loss[name_giou]
|
||||
return loss
|
||||
|
||||
def _get_loss_mask(self, masks, gt_mask, match_indices, num_gts,
|
||||
postfix=""):
|
||||
# masks: [b, query, h, w], gt_mask: list[[n, H, W]]
|
||||
name_mask = "loss_mask" + postfix
|
||||
name_dice = "loss_dice" + postfix
|
||||
|
||||
loss = dict()
|
||||
if sum(len(a) for a in gt_mask) == 0:
|
||||
loss[name_mask] = paddle.to_tensor([0.])
|
||||
loss[name_dice] = paddle.to_tensor([0.])
|
||||
return loss
|
||||
|
||||
src_masks, target_masks = self._get_src_target_assign(masks, gt_mask,
|
||||
match_indices)
|
||||
src_masks = F.interpolate(
|
||||
src_masks.unsqueeze(0),
|
||||
size=target_masks.shape[-2:],
|
||||
mode="bilinear")[0]
|
||||
loss[name_mask] = self.loss_coeff['mask'] * F.sigmoid_focal_loss(
|
||||
src_masks,
|
||||
target_masks,
|
||||
paddle.to_tensor(
|
||||
[num_gts], dtype='float32'))
|
||||
loss[name_dice] = self.loss_coeff['dice'] * self._dice_loss(
|
||||
src_masks, target_masks, num_gts)
|
||||
return loss
|
||||
|
||||
def _dice_loss(self, inputs, targets, num_gts):
|
||||
inputs = F.sigmoid(inputs)
|
||||
inputs = inputs.flatten(1)
|
||||
targets = targets.flatten(1)
|
||||
numerator = 2 * (inputs * targets).sum(1)
|
||||
denominator = inputs.sum(-1) + targets.sum(-1)
|
||||
loss = 1 - (numerator + 1) / (denominator + 1)
|
||||
return loss.sum() / num_gts
|
||||
|
||||
def _get_loss_aux(self,
|
||||
boxes,
|
||||
logits,
|
||||
gt_bbox,
|
||||
gt_class,
|
||||
bg_index,
|
||||
num_gts,
|
||||
dn_match_indices=None,
|
||||
postfix="",
|
||||
masks=None,
|
||||
gt_mask=None):
|
||||
loss_class = []
|
||||
loss_bbox, loss_giou = [], []
|
||||
loss_mask, loss_dice = [], []
|
||||
if dn_match_indices is not None:
|
||||
match_indices = dn_match_indices
|
||||
elif self.use_uni_match:
|
||||
match_indices = self.matcher(
|
||||
boxes[self.uni_match_ind],
|
||||
logits[self.uni_match_ind],
|
||||
gt_bbox,
|
||||
gt_class,
|
||||
masks=masks[self.uni_match_ind] if masks is not None else None,
|
||||
gt_mask=gt_mask)
|
||||
for i, (aux_boxes, aux_logits) in enumerate(zip(boxes, logits)):
|
||||
aux_masks = masks[i] if masks is not None else None
|
||||
if not self.use_uni_match and dn_match_indices is None:
|
||||
match_indices = self.matcher(
|
||||
aux_boxes,
|
||||
aux_logits,
|
||||
gt_bbox,
|
||||
gt_class,
|
||||
masks=aux_masks,
|
||||
gt_mask=gt_mask)
|
||||
if self.use_vfl:
|
||||
if sum(len(a) for a in gt_bbox) > 0:
|
||||
src_bbox, target_bbox = self._get_src_target_assign(
|
||||
aux_boxes.detach(), gt_bbox, match_indices)
|
||||
iou_score = bbox_iou(
|
||||
bbox_cxcywh_to_xyxy(src_bbox).split(4, -1),
|
||||
bbox_cxcywh_to_xyxy(target_bbox).split(4, -1))
|
||||
else:
|
||||
iou_score = None
|
||||
else:
|
||||
iou_score = None
|
||||
loss_class.append(
|
||||
self._get_loss_class(aux_logits, gt_class, match_indices,
|
||||
bg_index, num_gts, postfix, iou_score)[
|
||||
'loss_class' + postfix])
|
||||
loss_ = self._get_loss_bbox(aux_boxes, gt_bbox, match_indices,
|
||||
num_gts, postfix)
|
||||
loss_bbox.append(loss_['loss_bbox' + postfix])
|
||||
loss_giou.append(loss_['loss_giou' + postfix])
|
||||
if masks is not None and gt_mask is not None:
|
||||
loss_ = self._get_loss_mask(aux_masks, gt_mask, match_indices,
|
||||
num_gts, postfix)
|
||||
loss_mask.append(loss_['loss_mask' + postfix])
|
||||
loss_dice.append(loss_['loss_dice' + postfix])
|
||||
loss = {
|
||||
"loss_class_aux" + postfix: paddle.add_n(loss_class),
|
||||
"loss_bbox_aux" + postfix: paddle.add_n(loss_bbox),
|
||||
"loss_giou_aux" + postfix: paddle.add_n(loss_giou)
|
||||
}
|
||||
if masks is not None and gt_mask is not None:
|
||||
loss["loss_mask_aux" + postfix] = paddle.add_n(loss_mask)
|
||||
loss["loss_dice_aux" + postfix] = paddle.add_n(loss_dice)
|
||||
return loss
|
||||
|
||||
def _get_index_updates(self, num_query_objects, target, match_indices):
|
||||
batch_idx = paddle.concat([
|
||||
paddle.full_like(src, i) for i, (src, _) in enumerate(match_indices)
|
||||
])
|
||||
src_idx = paddle.concat([src for (src, _) in match_indices])
|
||||
src_idx += (batch_idx * num_query_objects)
|
||||
target_assign = paddle.concat([
|
||||
paddle.gather(
|
||||
t, dst, axis=0) for t, (_, dst) in zip(target, match_indices)
|
||||
])
|
||||
return src_idx, target_assign
|
||||
|
||||
def _get_src_target_assign(self, src, target, match_indices):
|
||||
src_assign = paddle.concat([
|
||||
paddle.gather(
|
||||
t, I, axis=0) if len(I) > 0 else paddle.zeros([0, t.shape[-1]])
|
||||
for t, (I, _) in zip(src, match_indices)
|
||||
])
|
||||
target_assign = paddle.concat([
|
||||
paddle.gather(
|
||||
t, J, axis=0) if len(J) > 0 else paddle.zeros([0, t.shape[-1]])
|
||||
for t, (_, J) in zip(target, match_indices)
|
||||
])
|
||||
return src_assign, target_assign
|
||||
|
||||
def _get_num_gts(self, targets, dtype="float32"):
|
||||
num_gts = sum(len(a) for a in targets)
|
||||
num_gts = paddle.to_tensor([num_gts], dtype=dtype)
|
||||
if paddle.distributed.get_world_size() > 1:
|
||||
paddle.distributed.all_reduce(num_gts)
|
||||
num_gts /= paddle.distributed.get_world_size()
|
||||
num_gts = paddle.clip(num_gts, min=1.)
|
||||
return num_gts
|
||||
|
||||
def _get_prediction_loss(self,
|
||||
boxes,
|
||||
logits,
|
||||
gt_bbox,
|
||||
gt_class,
|
||||
masks=None,
|
||||
gt_mask=None,
|
||||
postfix="",
|
||||
dn_match_indices=None,
|
||||
num_gts=1):
|
||||
if dn_match_indices is None:
|
||||
match_indices = self.matcher(
|
||||
boxes, logits, gt_bbox, gt_class, masks=masks, gt_mask=gt_mask)
|
||||
else:
|
||||
match_indices = dn_match_indices
|
||||
|
||||
if self.use_vfl:
|
||||
if sum(len(a) for a in gt_bbox) > 0:
|
||||
src_bbox, target_bbox = self._get_src_target_assign(
|
||||
boxes.detach(), gt_bbox, match_indices)
|
||||
iou_score = bbox_iou(
|
||||
bbox_cxcywh_to_xyxy(src_bbox).split(4, -1),
|
||||
bbox_cxcywh_to_xyxy(target_bbox).split(4, -1))
|
||||
else:
|
||||
iou_score = None
|
||||
else:
|
||||
iou_score = None
|
||||
|
||||
loss = dict()
|
||||
loss.update(
|
||||
self._get_loss_class(logits, gt_class, match_indices,
|
||||
self.num_classes, num_gts, postfix, iou_score))
|
||||
loss.update(
|
||||
self._get_loss_bbox(boxes, gt_bbox, match_indices, num_gts,
|
||||
postfix))
|
||||
if masks is not None and gt_mask is not None:
|
||||
loss.update(
|
||||
self._get_loss_mask(masks, gt_mask, match_indices, num_gts,
|
||||
postfix))
|
||||
return loss
|
||||
|
||||
def forward(self,
|
||||
boxes,
|
||||
logits,
|
||||
gt_bbox,
|
||||
gt_class,
|
||||
masks=None,
|
||||
gt_mask=None,
|
||||
postfix="",
|
||||
**kwargs):
|
||||
r"""
|
||||
Args:
|
||||
boxes (Tensor): [l, b, query, 4]
|
||||
logits (Tensor): [l, b, query, num_classes]
|
||||
gt_bbox (List(Tensor)): list[[n, 4]]
|
||||
gt_class (List(Tensor)): list[[n, 1]]
|
||||
masks (Tensor, optional): [l, b, query, h, w]
|
||||
gt_mask (List(Tensor), optional): list[[n, H, W]]
|
||||
postfix (str): postfix of loss name
|
||||
"""
|
||||
|
||||
dn_match_indices = kwargs.get("dn_match_indices", None)
|
||||
num_gts = kwargs.get("num_gts", None)
|
||||
if num_gts is None:
|
||||
num_gts = self._get_num_gts(gt_class)
|
||||
|
||||
total_loss = self._get_prediction_loss(
|
||||
boxes[-1],
|
||||
logits[-1],
|
||||
gt_bbox,
|
||||
gt_class,
|
||||
masks=masks[-1] if masks is not None else None,
|
||||
gt_mask=gt_mask,
|
||||
postfix=postfix,
|
||||
dn_match_indices=dn_match_indices,
|
||||
num_gts=num_gts)
|
||||
|
||||
if self.aux_loss:
|
||||
total_loss.update(
|
||||
self._get_loss_aux(
|
||||
boxes[:-1],
|
||||
logits[:-1],
|
||||
gt_bbox,
|
||||
gt_class,
|
||||
self.num_classes,
|
||||
num_gts,
|
||||
dn_match_indices,
|
||||
postfix,
|
||||
masks=masks[:-1] if masks is not None else None,
|
||||
gt_mask=gt_mask))
|
||||
|
||||
return total_loss
|
||||
|
||||
|
||||
@register
|
||||
class DINOLoss(DETRLoss):
|
||||
def forward(self,
|
||||
boxes,
|
||||
logits,
|
||||
gt_bbox,
|
||||
gt_class,
|
||||
masks=None,
|
||||
gt_mask=None,
|
||||
postfix="",
|
||||
dn_out_bboxes=None,
|
||||
dn_out_logits=None,
|
||||
dn_meta=None,
|
||||
**kwargs):
|
||||
num_gts = self._get_num_gts(gt_class)
|
||||
total_loss = super(DINOLoss, self).forward(
|
||||
boxes, logits, gt_bbox, gt_class, num_gts=num_gts)
|
||||
|
||||
if dn_meta is not None:
|
||||
dn_positive_idx, dn_num_group = \
|
||||
dn_meta["dn_positive_idx"], dn_meta["dn_num_group"]
|
||||
assert len(gt_class) == len(dn_positive_idx)
|
||||
|
||||
# denoising match indices
|
||||
dn_match_indices = self.get_dn_match_indices(
|
||||
gt_class, dn_positive_idx, dn_num_group)
|
||||
|
||||
# compute denoising training loss
|
||||
num_gts *= dn_num_group
|
||||
dn_loss = super(DINOLoss, self).forward(
|
||||
dn_out_bboxes,
|
||||
dn_out_logits,
|
||||
gt_bbox,
|
||||
gt_class,
|
||||
postfix="_dn",
|
||||
dn_match_indices=dn_match_indices,
|
||||
num_gts=num_gts)
|
||||
total_loss.update(dn_loss)
|
||||
else:
|
||||
total_loss.update(
|
||||
{k + '_dn': paddle.to_tensor([0.])
|
||||
for k in total_loss.keys()})
|
||||
|
||||
return total_loss
|
||||
|
||||
@staticmethod
|
||||
def get_dn_match_indices(labels, dn_positive_idx, dn_num_group):
|
||||
dn_match_indices = []
|
||||
for i in range(len(labels)):
|
||||
num_gt = len(labels[i])
|
||||
if num_gt > 0:
|
||||
gt_idx = paddle.arange(end=num_gt, dtype="int64")
|
||||
gt_idx = gt_idx.tile([dn_num_group])
|
||||
assert len(dn_positive_idx[i]) == len(gt_idx)
|
||||
dn_match_indices.append((dn_positive_idx[i], gt_idx))
|
||||
else:
|
||||
dn_match_indices.append((paddle.zeros(
|
||||
[0], dtype="int64"), paddle.zeros(
|
||||
[0], dtype="int64")))
|
||||
return dn_match_indices
|
||||
|
||||
|
||||
@register
|
||||
class MaskDINOLoss(DETRLoss):
|
||||
__shared__ = ['num_classes', 'use_focal_loss', 'num_sample_points']
|
||||
__inject__ = ['matcher']
|
||||
|
||||
def __init__(self,
|
||||
num_classes=80,
|
||||
matcher='HungarianMatcher',
|
||||
loss_coeff={
|
||||
'class': 4,
|
||||
'bbox': 5,
|
||||
'giou': 2,
|
||||
'mask': 5,
|
||||
'dice': 5
|
||||
},
|
||||
aux_loss=True,
|
||||
use_focal_loss=False,
|
||||
num_sample_points=12544,
|
||||
oversample_ratio=3.0,
|
||||
important_sample_ratio=0.75):
|
||||
super(MaskDINOLoss, self).__init__(num_classes, matcher, loss_coeff,
|
||||
aux_loss, use_focal_loss)
|
||||
assert oversample_ratio >= 1
|
||||
assert important_sample_ratio <= 1 and important_sample_ratio >= 0
|
||||
|
||||
self.num_sample_points = num_sample_points
|
||||
self.oversample_ratio = oversample_ratio
|
||||
self.important_sample_ratio = important_sample_ratio
|
||||
self.num_oversample_points = int(num_sample_points * oversample_ratio)
|
||||
self.num_important_points = int(num_sample_points *
|
||||
important_sample_ratio)
|
||||
self.num_random_points = num_sample_points - self.num_important_points
|
||||
|
||||
def forward(self,
|
||||
boxes,
|
||||
logits,
|
||||
gt_bbox,
|
||||
gt_class,
|
||||
masks=None,
|
||||
gt_mask=None,
|
||||
postfix="",
|
||||
dn_out_bboxes=None,
|
||||
dn_out_logits=None,
|
||||
dn_out_masks=None,
|
||||
dn_meta=None,
|
||||
**kwargs):
|
||||
num_gts = self._get_num_gts(gt_class)
|
||||
total_loss = super(MaskDINOLoss, self).forward(
|
||||
boxes,
|
||||
logits,
|
||||
gt_bbox,
|
||||
gt_class,
|
||||
masks=masks,
|
||||
gt_mask=gt_mask,
|
||||
num_gts=num_gts)
|
||||
|
||||
if dn_meta is not None:
|
||||
dn_positive_idx, dn_num_group = \
|
||||
dn_meta["dn_positive_idx"], dn_meta["dn_num_group"]
|
||||
assert len(gt_class) == len(dn_positive_idx)
|
||||
|
||||
# denoising match indices
|
||||
dn_match_indices = DINOLoss.get_dn_match_indices(
|
||||
gt_class, dn_positive_idx, dn_num_group)
|
||||
|
||||
# compute denoising training loss
|
||||
num_gts *= dn_num_group
|
||||
dn_loss = super(MaskDINOLoss, self).forward(
|
||||
dn_out_bboxes,
|
||||
dn_out_logits,
|
||||
gt_bbox,
|
||||
gt_class,
|
||||
masks=dn_out_masks,
|
||||
gt_mask=gt_mask,
|
||||
postfix="_dn",
|
||||
dn_match_indices=dn_match_indices,
|
||||
num_gts=num_gts)
|
||||
total_loss.update(dn_loss)
|
||||
else:
|
||||
total_loss.update(
|
||||
{k + '_dn': paddle.to_tensor([0.])
|
||||
for k in total_loss.keys()})
|
||||
|
||||
return total_loss
|
||||
|
||||
def _get_loss_mask(self, masks, gt_mask, match_indices, num_gts,
|
||||
postfix=""):
|
||||
# masks: [b, query, h, w], gt_mask: list[[n, H, W]]
|
||||
name_mask = "loss_mask" + postfix
|
||||
name_dice = "loss_dice" + postfix
|
||||
|
||||
loss = dict()
|
||||
if sum(len(a) for a in gt_mask) == 0:
|
||||
loss[name_mask] = paddle.to_tensor([0.])
|
||||
loss[name_dice] = paddle.to_tensor([0.])
|
||||
return loss
|
||||
|
||||
src_masks, target_masks = self._get_src_target_assign(masks, gt_mask,
|
||||
match_indices)
|
||||
# sample points
|
||||
sample_points = self._get_point_coords_by_uncertainty(src_masks)
|
||||
sample_points = 2.0 * sample_points.unsqueeze(1) - 1.0
|
||||
|
||||
src_masks = F.grid_sample(
|
||||
src_masks.unsqueeze(1), sample_points,
|
||||
align_corners=False).squeeze([1, 2])
|
||||
|
||||
target_masks = F.grid_sample(
|
||||
target_masks.unsqueeze(1), sample_points,
|
||||
align_corners=False).squeeze([1, 2]).detach()
|
||||
|
||||
loss[name_mask] = self.loss_coeff[
|
||||
'mask'] * F.binary_cross_entropy_with_logits(
|
||||
src_masks, target_masks,
|
||||
reduction='none').mean(1).sum() / num_gts
|
||||
loss[name_dice] = self.loss_coeff['dice'] * self._dice_loss(
|
||||
src_masks, target_masks, num_gts)
|
||||
return loss
|
||||
|
||||
def _get_point_coords_by_uncertainty(self, masks):
|
||||
# Sample points based on their uncertainty.
|
||||
masks = masks.detach()
|
||||
num_masks = masks.shape[0]
|
||||
sample_points = paddle.rand(
|
||||
[num_masks, 1, self.num_oversample_points, 2])
|
||||
|
||||
out_mask = F.grid_sample(
|
||||
masks.unsqueeze(1), 2.0 * sample_points - 1.0,
|
||||
align_corners=False).squeeze([1, 2])
|
||||
out_mask = -paddle.abs(out_mask)
|
||||
|
||||
_, topk_ind = paddle.topk(out_mask, self.num_important_points, axis=1)
|
||||
batch_ind = paddle.arange(end=num_masks, dtype=topk_ind.dtype)
|
||||
batch_ind = batch_ind.unsqueeze(-1).tile([1, self.num_important_points])
|
||||
topk_ind = paddle.stack([batch_ind, topk_ind], axis=-1)
|
||||
|
||||
sample_points = paddle.gather_nd(sample_points.squeeze(1), topk_ind)
|
||||
if self.num_random_points > 0:
|
||||
sample_points = paddle.concat(
|
||||
[
|
||||
sample_points,
|
||||
paddle.rand([num_masks, self.num_random_points, 2])
|
||||
],
|
||||
axis=1)
|
||||
return sample_points
|
||||
138
rtdetr_paddle/ppdet/modeling/losses/focal_loss.py
Normal file
138
rtdetr_paddle/ppdet/modeling/losses/focal_loss.py
Normal file
@@ -0,0 +1,138 @@
|
||||
# Copyright (c) 2020 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.
|
||||
|
||||
from __future__ import absolute_import
|
||||
from __future__ import division
|
||||
from __future__ import print_function
|
||||
|
||||
import paddle
|
||||
import paddle.nn.functional as F
|
||||
import paddle.nn as nn
|
||||
from ppdet.core.workspace import register
|
||||
|
||||
__all__ = ['FocalLoss', 'Weighted_FocalLoss']
|
||||
|
||||
@register
|
||||
class FocalLoss(nn.Layer):
|
||||
"""A wrapper around paddle.nn.functional.sigmoid_focal_loss.
|
||||
Args:
|
||||
use_sigmoid (bool): currently only support use_sigmoid=True
|
||||
alpha (float): parameter alpha in Focal Loss
|
||||
gamma (float): parameter gamma in Focal Loss
|
||||
loss_weight (float): final loss will be multiplied by this
|
||||
"""
|
||||
def __init__(self,
|
||||
use_sigmoid=True,
|
||||
alpha=0.25,
|
||||
gamma=2.0,
|
||||
loss_weight=1.0):
|
||||
super(FocalLoss, self).__init__()
|
||||
assert use_sigmoid == True, \
|
||||
'Focal Loss only supports sigmoid at the moment'
|
||||
self.use_sigmoid = use_sigmoid
|
||||
self.alpha = alpha
|
||||
self.gamma = gamma
|
||||
self.loss_weight = loss_weight
|
||||
|
||||
def forward(self, pred, target, reduction='none'):
|
||||
"""forward function.
|
||||
Args:
|
||||
pred (Tensor): logits of class prediction, of shape (N, num_classes)
|
||||
target (Tensor): target class label, of shape (N, )
|
||||
reduction (str): the way to reduce loss, one of (none, sum, mean)
|
||||
"""
|
||||
num_classes = pred.shape[1]
|
||||
target = F.one_hot(target, num_classes+1).cast(pred.dtype)
|
||||
target = target[:, :-1].detach()
|
||||
loss = F.sigmoid_focal_loss(
|
||||
pred, target, alpha=self.alpha, gamma=self.gamma,
|
||||
reduction=reduction)
|
||||
return loss * self.loss_weight
|
||||
|
||||
|
||||
@register
|
||||
class Weighted_FocalLoss(FocalLoss):
|
||||
"""A wrapper around paddle.nn.functional.sigmoid_focal_loss.
|
||||
Args:
|
||||
use_sigmoid (bool): currently only support use_sigmoid=True
|
||||
alpha (float): parameter alpha in Focal Loss
|
||||
gamma (float): parameter gamma in Focal Loss
|
||||
loss_weight (float): final loss will be multiplied by this
|
||||
"""
|
||||
def __init__(self,
|
||||
use_sigmoid=True,
|
||||
alpha=0.25,
|
||||
gamma=2.0,
|
||||
loss_weight=1.0,
|
||||
reduction="mean"):
|
||||
super(FocalLoss, self).__init__()
|
||||
assert use_sigmoid == True, \
|
||||
'Focal Loss only supports sigmoid at the moment'
|
||||
self.use_sigmoid = use_sigmoid
|
||||
self.alpha = alpha
|
||||
self.gamma = gamma
|
||||
self.loss_weight = loss_weight
|
||||
self.reduction = reduction
|
||||
|
||||
def forward(self, pred, target, weight=None, avg_factor=None, reduction_override=None):
|
||||
"""forward function.
|
||||
Args:
|
||||
pred (Tensor): logits of class prediction, of shape (N, num_classes)
|
||||
target (Tensor): target class label, of shape (N, )
|
||||
reduction (str): the way to reduce loss, one of (none, sum, mean)
|
||||
"""
|
||||
assert reduction_override in (None, 'none', 'mean', 'sum')
|
||||
reduction = (
|
||||
reduction_override if reduction_override else self.reduction)
|
||||
num_classes = pred.shape[1]
|
||||
target = F.one_hot(target, num_classes + 1).astype(pred.dtype)
|
||||
target = target[:, :-1].detach()
|
||||
loss = F.sigmoid_focal_loss(
|
||||
pred, target, alpha=self.alpha, gamma=self.gamma,
|
||||
reduction='none')
|
||||
|
||||
if weight is not None:
|
||||
if weight.shape != loss.shape:
|
||||
if weight.shape[0] == loss.shape[0]:
|
||||
# For most cases, weight is of shape (num_priors, ),
|
||||
# which means it does not have the second axis num_class
|
||||
weight = weight.reshape((-1, 1))
|
||||
else:
|
||||
# Sometimes, weight per anchor per class is also needed. e.g.
|
||||
# in FSAF. But it may be flattened of shape
|
||||
# (num_priors x num_class, ), while loss is still of shape
|
||||
# (num_priors, num_class).
|
||||
assert weight.numel() == loss.numel()
|
||||
weight = weight.reshape((loss.shape[0], -1))
|
||||
assert weight.ndim == loss.ndim
|
||||
loss = loss * weight
|
||||
|
||||
# if avg_factor is not specified, just reduce the loss
|
||||
if avg_factor is None:
|
||||
if reduction == 'mean':
|
||||
loss = loss.mean()
|
||||
elif reduction == 'sum':
|
||||
loss = loss.sum()
|
||||
else:
|
||||
# if reduction is mean, then average the loss by avg_factor
|
||||
if reduction == 'mean':
|
||||
# Avoid causing ZeroDivisionError when avg_factor is 0.0,
|
||||
# i.e., all labels of an image belong to ignore index.
|
||||
eps = 1e-10
|
||||
loss = loss.sum() / (avg_factor + eps)
|
||||
# if reduction is 'none', then do nothing, otherwise raise an error
|
||||
elif reduction != 'none':
|
||||
raise ValueError('avg_factor can not be used with reduction="sum"')
|
||||
|
||||
return loss * self.loss_weight
|
||||
217
rtdetr_paddle/ppdet/modeling/losses/gfocal_loss.py
Normal file
217
rtdetr_paddle/ppdet/modeling/losses/gfocal_loss.py
Normal file
@@ -0,0 +1,217 @@
|
||||
# Copyright (c) 2021 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.
|
||||
|
||||
# The code is based on:
|
||||
# https://github.com/open-mmlab/mmdetection/blob/master/mmdet/models/losses/gfocal_loss.py
|
||||
|
||||
from __future__ import absolute_import
|
||||
from __future__ import division
|
||||
from __future__ import print_function
|
||||
import numpy as np
|
||||
import paddle
|
||||
import paddle.nn as nn
|
||||
import paddle.nn.functional as F
|
||||
from ppdet.core.workspace import register, serializable
|
||||
from ppdet.modeling import ops
|
||||
|
||||
__all__ = ['QualityFocalLoss', 'DistributionFocalLoss']
|
||||
|
||||
|
||||
def quality_focal_loss(pred, target, beta=2.0, use_sigmoid=True):
|
||||
"""
|
||||
Quality Focal Loss (QFL) is from `Generalized Focal Loss: Learning
|
||||
Qualified and Distributed Bounding Boxes for Dense Object Detection
|
||||
<https://arxiv.org/abs/2006.04388>`_.
|
||||
Args:
|
||||
pred (Tensor): Predicted joint representation of classification
|
||||
and quality (IoU) estimation with shape (N, C), C is the number of
|
||||
classes.
|
||||
target (tuple([Tensor])): Target category label with shape (N,)
|
||||
and target quality label with shape (N,).
|
||||
beta (float): The beta parameter for calculating the modulating factor.
|
||||
Defaults to 2.0.
|
||||
Returns:
|
||||
Tensor: Loss tensor with shape (N,).
|
||||
"""
|
||||
assert len(target) == 2, """target for QFL must be a tuple of two elements,
|
||||
including category label and quality label, respectively"""
|
||||
# label denotes the category id, score denotes the quality score
|
||||
label, score = target
|
||||
if use_sigmoid:
|
||||
func = F.binary_cross_entropy_with_logits
|
||||
else:
|
||||
func = F.binary_cross_entropy
|
||||
|
||||
# negatives are supervised by 0 quality score
|
||||
pred_sigmoid = F.sigmoid(pred) if use_sigmoid else pred
|
||||
scale_factor = pred_sigmoid
|
||||
zerolabel = paddle.zeros(pred.shape, dtype='float32')
|
||||
loss = func(pred, zerolabel, reduction='none') * scale_factor.pow(beta)
|
||||
|
||||
# FG cat_id: [0, num_classes -1], BG cat_id: num_classes
|
||||
bg_class_ind = pred.shape[1]
|
||||
pos = paddle.logical_and((label >= 0),
|
||||
(label < bg_class_ind)).nonzero().squeeze(1)
|
||||
if pos.shape[0] == 0:
|
||||
return loss.sum(axis=1)
|
||||
pos_label = paddle.gather(label, pos, axis=0)
|
||||
pos_mask = np.zeros(pred.shape, dtype=np.int32)
|
||||
pos_mask[pos.numpy(), pos_label.numpy()] = 1
|
||||
pos_mask = paddle.to_tensor(pos_mask, dtype='bool')
|
||||
score = score.unsqueeze(-1).expand([-1, pred.shape[1]]).cast('float32')
|
||||
# positives are supervised by bbox quality (IoU) score
|
||||
scale_factor_new = score - pred_sigmoid
|
||||
|
||||
loss_pos = func(
|
||||
pred, score, reduction='none') * scale_factor_new.abs().pow(beta)
|
||||
loss = loss * paddle.logical_not(pos_mask) + loss_pos * pos_mask
|
||||
loss = loss.sum(axis=1)
|
||||
return loss
|
||||
|
||||
|
||||
def distribution_focal_loss(pred, label):
|
||||
"""Distribution Focal Loss (DFL) is from `Generalized Focal Loss: Learning
|
||||
Qualified and Distributed Bounding Boxes for Dense Object Detection
|
||||
<https://arxiv.org/abs/2006.04388>`_.
|
||||
Args:
|
||||
pred (Tensor): Predicted general distribution of bounding boxes
|
||||
(before softmax) with shape (N, n+1), n is the max value of the
|
||||
integral set `{0, ..., n}` in paper.
|
||||
label (Tensor): Target distance label for bounding boxes with
|
||||
shape (N,).
|
||||
Returns:
|
||||
Tensor: Loss tensor with shape (N,).
|
||||
"""
|
||||
dis_left = label.cast('int64')
|
||||
dis_right = dis_left + 1
|
||||
weight_left = dis_right.cast('float32') - label
|
||||
weight_right = label - dis_left.cast('float32')
|
||||
loss = F.cross_entropy(pred, dis_left, reduction='none') * weight_left \
|
||||
+ F.cross_entropy(pred, dis_right, reduction='none') * weight_right
|
||||
return loss
|
||||
|
||||
|
||||
@register
|
||||
@serializable
|
||||
class QualityFocalLoss(nn.Layer):
|
||||
r"""Quality Focal Loss (QFL) is a variant of `Generalized Focal Loss:
|
||||
Learning Qualified and Distributed Bounding Boxes for Dense Object
|
||||
Detection <https://arxiv.org/abs/2006.04388>`_.
|
||||
Args:
|
||||
use_sigmoid (bool): Whether sigmoid operation is conducted in QFL.
|
||||
Defaults to True.
|
||||
beta (float): The beta parameter for calculating the modulating factor.
|
||||
Defaults to 2.0.
|
||||
reduction (str): Options are "none", "mean" and "sum".
|
||||
loss_weight (float): Loss weight of current loss.
|
||||
"""
|
||||
|
||||
def __init__(self,
|
||||
use_sigmoid=True,
|
||||
beta=2.0,
|
||||
reduction='mean',
|
||||
loss_weight=1.0):
|
||||
super(QualityFocalLoss, self).__init__()
|
||||
self.use_sigmoid = use_sigmoid
|
||||
self.beta = beta
|
||||
assert reduction in ('none', 'mean', 'sum')
|
||||
self.reduction = reduction
|
||||
self.loss_weight = loss_weight
|
||||
|
||||
def forward(self, pred, target, weight=None, avg_factor=None):
|
||||
"""Forward function.
|
||||
Args:
|
||||
pred (Tensor): Predicted joint representation of
|
||||
classification and quality (IoU) estimation with shape (N, C),
|
||||
C is the number of classes.
|
||||
target (tuple([Tensor])): Target category label with shape
|
||||
(N,) and target quality label with shape (N,).
|
||||
weight (Tensor, optional): The weight of loss for each
|
||||
prediction. Defaults to None.
|
||||
avg_factor (int, optional): Average factor that is used to average
|
||||
the loss. Defaults to None.
|
||||
"""
|
||||
|
||||
loss = self.loss_weight * quality_focal_loss(
|
||||
pred, target, beta=self.beta, use_sigmoid=self.use_sigmoid)
|
||||
|
||||
if weight is not None:
|
||||
loss = loss * weight
|
||||
if avg_factor is None:
|
||||
if self.reduction == 'none':
|
||||
return loss
|
||||
elif self.reduction == 'mean':
|
||||
return loss.mean()
|
||||
elif self.reduction == 'sum':
|
||||
return loss.sum()
|
||||
else:
|
||||
# if reduction is mean, then average the loss by avg_factor
|
||||
if self.reduction == 'mean':
|
||||
loss = loss.sum() / avg_factor
|
||||
# if reduction is 'none', then do nothing, otherwise raise an error
|
||||
elif self.reduction != 'none':
|
||||
raise ValueError(
|
||||
'avg_factor can not be used with reduction="sum"')
|
||||
return loss
|
||||
|
||||
|
||||
@register
|
||||
@serializable
|
||||
class DistributionFocalLoss(nn.Layer):
|
||||
"""Distribution Focal Loss (DFL) is a variant of `Generalized Focal Loss:
|
||||
Learning Qualified and Distributed Bounding Boxes for Dense Object
|
||||
Detection <https://arxiv.org/abs/2006.04388>`_.
|
||||
Args:
|
||||
reduction (str): Options are `'none'`, `'mean'` and `'sum'`.
|
||||
loss_weight (float): Loss weight of current loss.
|
||||
"""
|
||||
|
||||
def __init__(self, reduction='mean', loss_weight=1.0):
|
||||
super(DistributionFocalLoss, self).__init__()
|
||||
assert reduction in ('none', 'mean', 'sum')
|
||||
self.reduction = reduction
|
||||
self.loss_weight = loss_weight
|
||||
|
||||
def forward(self, pred, target, weight=None, avg_factor=None):
|
||||
"""Forward function.
|
||||
Args:
|
||||
pred (Tensor): Predicted general distribution of bounding
|
||||
boxes (before softmax) with shape (N, n+1), n is the max value
|
||||
of the integral set `{0, ..., n}` in paper.
|
||||
target (Tensor): Target distance label for bounding boxes
|
||||
with shape (N,).
|
||||
weight (Tensor, optional): The weight of loss for each
|
||||
prediction. Defaults to None.
|
||||
avg_factor (int, optional): Average factor that is used to average
|
||||
the loss. Defaults to None.
|
||||
"""
|
||||
loss = self.loss_weight * distribution_focal_loss(pred, target)
|
||||
if weight is not None:
|
||||
loss = loss * weight
|
||||
if avg_factor is None:
|
||||
if self.reduction == 'none':
|
||||
return loss
|
||||
elif self.reduction == 'mean':
|
||||
return loss.mean()
|
||||
elif self.reduction == 'sum':
|
||||
return loss.sum()
|
||||
else:
|
||||
# if reduction is mean, then average the loss by avg_factor
|
||||
if self.reduction == 'mean':
|
||||
loss = loss.sum() / avg_factor
|
||||
# if reduction is 'none', then do nothing, otherwise raise an error
|
||||
elif self.reduction != 'none':
|
||||
raise ValueError(
|
||||
'avg_factor can not be used with reduction="sum"')
|
||||
return loss
|
||||
295
rtdetr_paddle/ppdet/modeling/losses/iou_loss.py
Normal file
295
rtdetr_paddle/ppdet/modeling/losses/iou_loss.py
Normal file
@@ -0,0 +1,295 @@
|
||||
# Copyright (c) 2020 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.
|
||||
|
||||
from __future__ import absolute_import
|
||||
from __future__ import division
|
||||
from __future__ import print_function
|
||||
|
||||
import numpy as np
|
||||
import math
|
||||
import paddle
|
||||
|
||||
from ppdet.core.workspace import register, serializable
|
||||
from ..bbox_utils import bbox_iou
|
||||
|
||||
__all__ = ['IouLoss', 'GIoULoss', 'DIouLoss', 'SIoULoss']
|
||||
|
||||
|
||||
@register
|
||||
@serializable
|
||||
class IouLoss(object):
|
||||
"""
|
||||
iou loss, see https://arxiv.org/abs/1908.03851
|
||||
loss = 1.0 - iou * iou
|
||||
Args:
|
||||
loss_weight (float): iou loss weight, default is 2.5
|
||||
max_height (int): max height of input to support random shape input
|
||||
max_width (int): max width of input to support random shape input
|
||||
ciou_term (bool): whether to add ciou_term
|
||||
loss_square (bool): whether to square the iou term
|
||||
"""
|
||||
|
||||
def __init__(self,
|
||||
loss_weight=2.5,
|
||||
giou=False,
|
||||
diou=False,
|
||||
ciou=False,
|
||||
loss_square=True):
|
||||
self.loss_weight = loss_weight
|
||||
self.giou = giou
|
||||
self.diou = diou
|
||||
self.ciou = ciou
|
||||
self.loss_square = loss_square
|
||||
|
||||
def __call__(self, pbox, gbox):
|
||||
iou = bbox_iou(
|
||||
pbox, gbox, giou=self.giou, diou=self.diou, ciou=self.ciou)
|
||||
if self.loss_square:
|
||||
loss_iou = 1 - iou * iou
|
||||
else:
|
||||
loss_iou = 1 - iou
|
||||
|
||||
loss_iou = loss_iou * self.loss_weight
|
||||
return loss_iou
|
||||
|
||||
|
||||
@register
|
||||
@serializable
|
||||
class GIoULoss(object):
|
||||
"""
|
||||
Generalized Intersection over Union, see https://arxiv.org/abs/1902.09630
|
||||
Args:
|
||||
loss_weight (float): giou loss weight, default as 1
|
||||
eps (float): epsilon to avoid divide by zero, default as 1e-10
|
||||
reduction (string): Options are "none", "mean" and "sum". default as none
|
||||
"""
|
||||
|
||||
def __init__(self, loss_weight=1., eps=1e-10, reduction='none'):
|
||||
self.loss_weight = loss_weight
|
||||
self.eps = eps
|
||||
assert reduction in ('none', 'mean', 'sum')
|
||||
self.reduction = reduction
|
||||
|
||||
def bbox_overlap(self, box1, box2, eps=1e-10):
|
||||
"""calculate the iou of box1 and box2
|
||||
Args:
|
||||
box1 (Tensor): box1 with the shape (..., 4)
|
||||
box2 (Tensor): box1 with the shape (..., 4)
|
||||
eps (float): epsilon to avoid divide by zero
|
||||
Return:
|
||||
iou (Tensor): iou of box1 and box2
|
||||
overlap (Tensor): overlap of box1 and box2
|
||||
union (Tensor): union of box1 and box2
|
||||
"""
|
||||
x1, y1, x2, y2 = box1
|
||||
x1g, y1g, x2g, y2g = box2
|
||||
|
||||
xkis1 = paddle.maximum(x1, x1g)
|
||||
ykis1 = paddle.maximum(y1, y1g)
|
||||
xkis2 = paddle.minimum(x2, x2g)
|
||||
ykis2 = paddle.minimum(y2, y2g)
|
||||
w_inter = (xkis2 - xkis1).clip(0)
|
||||
h_inter = (ykis2 - ykis1).clip(0)
|
||||
overlap = w_inter * h_inter
|
||||
|
||||
area1 = (x2 - x1) * (y2 - y1)
|
||||
area2 = (x2g - x1g) * (y2g - y1g)
|
||||
union = area1 + area2 - overlap + eps
|
||||
iou = overlap / union
|
||||
|
||||
return iou, overlap, union
|
||||
|
||||
def __call__(self, pbox, gbox, iou_weight=1., loc_reweight=None):
|
||||
x1, y1, x2, y2 = paddle.split(pbox, num_or_sections=4, axis=-1)
|
||||
x1g, y1g, x2g, y2g = paddle.split(gbox, num_or_sections=4, axis=-1)
|
||||
box1 = [x1, y1, x2, y2]
|
||||
box2 = [x1g, y1g, x2g, y2g]
|
||||
iou, overlap, union = self.bbox_overlap(box1, box2, self.eps)
|
||||
xc1 = paddle.minimum(x1, x1g)
|
||||
yc1 = paddle.minimum(y1, y1g)
|
||||
xc2 = paddle.maximum(x2, x2g)
|
||||
yc2 = paddle.maximum(y2, y2g)
|
||||
|
||||
area_c = (xc2 - xc1) * (yc2 - yc1) + self.eps
|
||||
miou = iou - ((area_c - union) / area_c)
|
||||
if loc_reweight is not None:
|
||||
loc_reweight = paddle.reshape(loc_reweight, shape=(-1, 1))
|
||||
loc_thresh = 0.9
|
||||
giou = 1 - (1 - loc_thresh
|
||||
) * miou - loc_thresh * miou * loc_reweight
|
||||
else:
|
||||
giou = 1 - miou
|
||||
if self.reduction == 'none':
|
||||
loss = giou
|
||||
elif self.reduction == 'sum':
|
||||
loss = paddle.sum(giou * iou_weight)
|
||||
else:
|
||||
loss = paddle.mean(giou * iou_weight)
|
||||
return loss * self.loss_weight
|
||||
|
||||
|
||||
@register
|
||||
@serializable
|
||||
class DIouLoss(GIoULoss):
|
||||
"""
|
||||
Distance-IoU Loss, see https://arxiv.org/abs/1911.08287
|
||||
Args:
|
||||
loss_weight (float): giou loss weight, default as 1
|
||||
eps (float): epsilon to avoid divide by zero, default as 1e-10
|
||||
use_complete_iou_loss (bool): whether to use complete iou loss
|
||||
"""
|
||||
|
||||
def __init__(self, loss_weight=1., eps=1e-10, use_complete_iou_loss=True):
|
||||
super(DIouLoss, self).__init__(loss_weight=loss_weight, eps=eps)
|
||||
self.use_complete_iou_loss = use_complete_iou_loss
|
||||
|
||||
def __call__(self, pbox, gbox, iou_weight=1.):
|
||||
x1, y1, x2, y2 = paddle.split(pbox, num_or_sections=4, axis=-1)
|
||||
x1g, y1g, x2g, y2g = paddle.split(gbox, num_or_sections=4, axis=-1)
|
||||
cx = (x1 + x2) / 2
|
||||
cy = (y1 + y2) / 2
|
||||
w = x2 - x1
|
||||
h = y2 - y1
|
||||
|
||||
cxg = (x1g + x2g) / 2
|
||||
cyg = (y1g + y2g) / 2
|
||||
wg = x2g - x1g
|
||||
hg = y2g - y1g
|
||||
|
||||
x2 = paddle.maximum(x1, x2)
|
||||
y2 = paddle.maximum(y1, y2)
|
||||
|
||||
# A and B
|
||||
xkis1 = paddle.maximum(x1, x1g)
|
||||
ykis1 = paddle.maximum(y1, y1g)
|
||||
xkis2 = paddle.minimum(x2, x2g)
|
||||
ykis2 = paddle.minimum(y2, y2g)
|
||||
|
||||
# A or B
|
||||
xc1 = paddle.minimum(x1, x1g)
|
||||
yc1 = paddle.minimum(y1, y1g)
|
||||
xc2 = paddle.maximum(x2, x2g)
|
||||
yc2 = paddle.maximum(y2, y2g)
|
||||
|
||||
intsctk = (xkis2 - xkis1) * (ykis2 - ykis1)
|
||||
intsctk = intsctk * paddle.greater_than(
|
||||
xkis2, xkis1) * paddle.greater_than(ykis2, ykis1)
|
||||
unionk = (x2 - x1) * (y2 - y1) + (x2g - x1g) * (y2g - y1g
|
||||
) - intsctk + self.eps
|
||||
iouk = intsctk / unionk
|
||||
|
||||
# DIOU term
|
||||
dist_intersection = (cx - cxg) * (cx - cxg) + (cy - cyg) * (cy - cyg)
|
||||
dist_union = (xc2 - xc1) * (xc2 - xc1) + (yc2 - yc1) * (yc2 - yc1)
|
||||
diou_term = (dist_intersection + self.eps) / (dist_union + self.eps)
|
||||
|
||||
# CIOU term
|
||||
ciou_term = 0
|
||||
if self.use_complete_iou_loss:
|
||||
ar_gt = wg / hg
|
||||
ar_pred = w / h
|
||||
arctan = paddle.atan(ar_gt) - paddle.atan(ar_pred)
|
||||
ar_loss = 4. / np.pi / np.pi * arctan * arctan
|
||||
alpha = ar_loss / (1 - iouk + ar_loss + self.eps)
|
||||
alpha.stop_gradient = True
|
||||
ciou_term = alpha * ar_loss
|
||||
|
||||
diou = paddle.mean((1 - iouk + ciou_term + diou_term) * iou_weight)
|
||||
|
||||
return diou * self.loss_weight
|
||||
|
||||
|
||||
@register
|
||||
@serializable
|
||||
class SIoULoss(GIoULoss):
|
||||
"""
|
||||
see https://arxiv.org/pdf/2205.12740.pdf
|
||||
Args:
|
||||
loss_weight (float): siou loss weight, default as 1
|
||||
eps (float): epsilon to avoid divide by zero, default as 1e-10
|
||||
theta (float): default as 4
|
||||
reduction (str): Options are "none", "mean" and "sum". default as none
|
||||
"""
|
||||
|
||||
def __init__(self, loss_weight=1., eps=1e-10, theta=4., reduction='none'):
|
||||
super(SIoULoss, self).__init__(loss_weight=loss_weight, eps=eps)
|
||||
self.loss_weight = loss_weight
|
||||
self.eps = eps
|
||||
self.theta = theta
|
||||
self.reduction = reduction
|
||||
|
||||
def __call__(self, pbox, gbox):
|
||||
x1, y1, x2, y2 = paddle.split(pbox, num_or_sections=4, axis=-1)
|
||||
x1g, y1g, x2g, y2g = paddle.split(gbox, num_or_sections=4, axis=-1)
|
||||
|
||||
box1 = [x1, y1, x2, y2]
|
||||
box2 = [x1g, y1g, x2g, y2g]
|
||||
iou = bbox_iou(box1, box2)
|
||||
|
||||
cx = (x1 + x2) / 2
|
||||
cy = (y1 + y2) / 2
|
||||
w = x2 - x1 + self.eps
|
||||
h = y2 - y1 + self.eps
|
||||
|
||||
cxg = (x1g + x2g) / 2
|
||||
cyg = (y1g + y2g) / 2
|
||||
wg = x2g - x1g + self.eps
|
||||
hg = y2g - y1g + self.eps
|
||||
|
||||
x2 = paddle.maximum(x1, x2)
|
||||
y2 = paddle.maximum(y1, y2)
|
||||
|
||||
# A or B
|
||||
xc1 = paddle.minimum(x1, x1g)
|
||||
yc1 = paddle.minimum(y1, y1g)
|
||||
xc2 = paddle.maximum(x2, x2g)
|
||||
yc2 = paddle.maximum(y2, y2g)
|
||||
|
||||
cw_out = xc2 - xc1
|
||||
ch_out = yc2 - yc1
|
||||
|
||||
ch = paddle.maximum(cy, cyg) - paddle.minimum(cy, cyg)
|
||||
cw = paddle.maximum(cx, cxg) - paddle.minimum(cx, cxg)
|
||||
|
||||
# angle cost
|
||||
dist_intersection = paddle.sqrt((cx - cxg)**2 + (cy - cyg)**2)
|
||||
sin_angle_alpha = ch / dist_intersection
|
||||
sin_angle_beta = cw / dist_intersection
|
||||
thred = paddle.pow(paddle.to_tensor(2), 0.5) / 2
|
||||
thred.stop_gradient = True
|
||||
sin_alpha = paddle.where(sin_angle_alpha > thred, sin_angle_beta,
|
||||
sin_angle_alpha)
|
||||
angle_cost = paddle.cos(paddle.asin(sin_alpha) * 2 - math.pi / 2)
|
||||
|
||||
# distance cost
|
||||
gamma = 2 - angle_cost
|
||||
# gamma.stop_gradient = True
|
||||
beta_x = ((cxg - cx) / cw_out)**2
|
||||
beta_y = ((cyg - cy) / ch_out)**2
|
||||
dist_cost = 1 - paddle.exp(-gamma * beta_x) + 1 - paddle.exp(-gamma *
|
||||
beta_y)
|
||||
|
||||
# shape cost
|
||||
omega_w = paddle.abs(w - wg) / paddle.maximum(w, wg)
|
||||
omega_h = paddle.abs(hg - h) / paddle.maximum(h, hg)
|
||||
omega = (1 - paddle.exp(-omega_w))**self.theta + (
|
||||
1 - paddle.exp(-omega_h))**self.theta
|
||||
siou_loss = 1 - iou + (omega + dist_cost) / 2
|
||||
|
||||
if self.reduction == 'mean':
|
||||
siou_loss = paddle.mean(siou_loss)
|
||||
elif self.reduction == 'sum':
|
||||
siou_loss = paddle.sum(siou_loss)
|
||||
|
||||
return siou_loss * self.loss_weight
|
||||
60
rtdetr_paddle/ppdet/modeling/losses/smooth_l1_loss.py
Normal file
60
rtdetr_paddle/ppdet/modeling/losses/smooth_l1_loss.py
Normal file
@@ -0,0 +1,60 @@
|
||||
# Copyright (c) 2020 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.
|
||||
|
||||
from __future__ import absolute_import
|
||||
from __future__ import division
|
||||
from __future__ import print_function
|
||||
|
||||
import paddle
|
||||
import paddle.nn as nn
|
||||
import paddle.nn.functional as F
|
||||
from ppdet.core.workspace import register
|
||||
|
||||
__all__ = ['SmoothL1Loss']
|
||||
|
||||
@register
|
||||
class SmoothL1Loss(nn.Layer):
|
||||
"""Smooth L1 Loss.
|
||||
Args:
|
||||
beta (float): controls smooth region, it becomes L1 Loss when beta=0.0
|
||||
loss_weight (float): the final loss will be multiplied by this
|
||||
"""
|
||||
def __init__(self,
|
||||
beta=1.0,
|
||||
loss_weight=1.0):
|
||||
super(SmoothL1Loss, self).__init__()
|
||||
assert beta >= 0
|
||||
self.beta = beta
|
||||
self.loss_weight = loss_weight
|
||||
|
||||
def forward(self, pred, target, reduction='none'):
|
||||
"""forward function, based on fvcore.
|
||||
Args:
|
||||
pred (Tensor): prediction tensor
|
||||
target (Tensor): target tensor, pred.shape must be the same as target.shape
|
||||
reduction (str): the way to reduce loss, one of (none, sum, mean)
|
||||
"""
|
||||
assert reduction in ('none', 'sum', 'mean')
|
||||
target = target.detach()
|
||||
if self.beta < 1e-5:
|
||||
loss = paddle.abs(pred - target)
|
||||
else:
|
||||
n = paddle.abs(pred - target)
|
||||
cond = n < self.beta
|
||||
loss = paddle.where(cond, 0.5 * n ** 2 / self.beta, n - 0.5 * self.beta)
|
||||
if reduction == 'mean':
|
||||
loss = loss.mean() if loss.size > 0 else 0.0 * loss.sum()
|
||||
elif reduction == 'sum':
|
||||
loss = loss.sum()
|
||||
return loss * self.loss_weight
|
||||
152
rtdetr_paddle/ppdet/modeling/losses/varifocal_loss.py
Normal file
152
rtdetr_paddle/ppdet/modeling/losses/varifocal_loss.py
Normal file
@@ -0,0 +1,152 @@
|
||||
# Copyright (c) 2021 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.
|
||||
|
||||
# The code is based on:
|
||||
# https://github.com/open-mmlab/mmdetection/blob/master/mmdet/models/losses/varifocal_loss.py
|
||||
|
||||
from __future__ import absolute_import
|
||||
from __future__ import division
|
||||
from __future__ import print_function
|
||||
import numpy as np
|
||||
import paddle
|
||||
import paddle.nn as nn
|
||||
import paddle.nn.functional as F
|
||||
from ppdet.core.workspace import register, serializable
|
||||
from ppdet.modeling import ops
|
||||
|
||||
__all__ = ['VarifocalLoss']
|
||||
|
||||
|
||||
def varifocal_loss(pred,
|
||||
target,
|
||||
alpha=0.75,
|
||||
gamma=2.0,
|
||||
iou_weighted=True,
|
||||
use_sigmoid=True):
|
||||
"""`Varifocal Loss <https://arxiv.org/abs/2008.13367>`_
|
||||
|
||||
Args:
|
||||
pred (Tensor): The prediction with shape (N, C), C is the
|
||||
number of classes
|
||||
target (Tensor): The learning target of the iou-aware
|
||||
classification score with shape (N, C), C is the number of classes.
|
||||
alpha (float, optional): A balance factor for the negative part of
|
||||
Varifocal Loss, which is different from the alpha of Focal Loss.
|
||||
Defaults to 0.75.
|
||||
gamma (float, optional): The gamma for calculating the modulating
|
||||
factor. Defaults to 2.0.
|
||||
iou_weighted (bool, optional): Whether to weight the loss of the
|
||||
positive example with the iou target. Defaults to True.
|
||||
"""
|
||||
# pred and target should be of the same size
|
||||
assert pred.shape == target.shape
|
||||
if use_sigmoid:
|
||||
pred_new = F.sigmoid(pred)
|
||||
else:
|
||||
pred_new = pred
|
||||
target = target.cast(pred.dtype)
|
||||
if iou_weighted:
|
||||
focal_weight = target * (target > 0.0).cast('float32') + \
|
||||
alpha * (pred_new - target).abs().pow(gamma) * \
|
||||
(target <= 0.0).cast('float32')
|
||||
else:
|
||||
focal_weight = (target > 0.0).cast('float32') + \
|
||||
alpha * (pred_new - target).abs().pow(gamma) * \
|
||||
(target <= 0.0).cast('float32')
|
||||
|
||||
if use_sigmoid:
|
||||
loss = F.binary_cross_entropy_with_logits(
|
||||
pred, target, reduction='none') * focal_weight
|
||||
else:
|
||||
loss = F.binary_cross_entropy(
|
||||
pred, target, reduction='none') * focal_weight
|
||||
loss = loss.sum(axis=1)
|
||||
return loss
|
||||
|
||||
|
||||
@register
|
||||
@serializable
|
||||
class VarifocalLoss(nn.Layer):
|
||||
def __init__(self,
|
||||
use_sigmoid=True,
|
||||
alpha=0.75,
|
||||
gamma=2.0,
|
||||
iou_weighted=True,
|
||||
reduction='mean',
|
||||
loss_weight=1.0):
|
||||
"""`Varifocal Loss <https://arxiv.org/abs/2008.13367>`_
|
||||
|
||||
Args:
|
||||
use_sigmoid (bool, optional): Whether the prediction is
|
||||
used for sigmoid or softmax. Defaults to True.
|
||||
alpha (float, optional): A balance factor for the negative part of
|
||||
Varifocal Loss, which is different from the alpha of Focal
|
||||
Loss. Defaults to 0.75.
|
||||
gamma (float, optional): The gamma for calculating the modulating
|
||||
factor. Defaults to 2.0.
|
||||
iou_weighted (bool, optional): Whether to weight the loss of the
|
||||
positive examples with the iou target. Defaults to True.
|
||||
reduction (str, optional): The method used to reduce the loss into
|
||||
a scalar. Defaults to 'mean'. Options are "none", "mean" and
|
||||
"sum".
|
||||
loss_weight (float, optional): Weight of loss. Defaults to 1.0.
|
||||
"""
|
||||
super(VarifocalLoss, self).__init__()
|
||||
assert alpha >= 0.0
|
||||
self.use_sigmoid = use_sigmoid
|
||||
self.alpha = alpha
|
||||
self.gamma = gamma
|
||||
self.iou_weighted = iou_weighted
|
||||
self.reduction = reduction
|
||||
self.loss_weight = loss_weight
|
||||
|
||||
def forward(self, pred, target, weight=None, avg_factor=None):
|
||||
"""Forward function.
|
||||
|
||||
Args:
|
||||
pred (Tensor): The prediction.
|
||||
target (Tensor): The learning target of the prediction.
|
||||
weight (Tensor, optional): The weight of loss for each
|
||||
prediction. Defaults to None.
|
||||
avg_factor (int, optional): Average factor that is used to average
|
||||
the loss. Defaults to None.
|
||||
Returns:
|
||||
Tensor: The calculated loss
|
||||
"""
|
||||
loss = self.loss_weight * varifocal_loss(
|
||||
pred,
|
||||
target,
|
||||
alpha=self.alpha,
|
||||
gamma=self.gamma,
|
||||
iou_weighted=self.iou_weighted,
|
||||
use_sigmoid=self.use_sigmoid)
|
||||
|
||||
if weight is not None:
|
||||
loss = loss * weight
|
||||
if avg_factor is None:
|
||||
if self.reduction == 'none':
|
||||
return loss
|
||||
elif self.reduction == 'mean':
|
||||
return loss.mean()
|
||||
elif self.reduction == 'sum':
|
||||
return loss.sum()
|
||||
else:
|
||||
# if reduction is mean, then average the loss by avg_factor
|
||||
if self.reduction == 'mean':
|
||||
loss = loss.sum() / avg_factor
|
||||
# if reduction is 'none', then do nothing, otherwise raise an error
|
||||
elif self.reduction != 'none':
|
||||
raise ValueError(
|
||||
'avg_factor can not be used with reduction="sum"')
|
||||
return loss
|
||||
Reference in New Issue
Block a user