forked from tyagi-iiitv/PointPillars
-
Notifications
You must be signed in to change notification settings - Fork 1
/
loss_v2.py
87 lines (66 loc) · 3.59 KB
/
loss_v2.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
import tensorflow as tf
import tensorflow_probability as tfp
from tensorflow.python.keras import backend as K
from config_v2 import Parameters
class PointPillarNetworkLoss:
def __init__(self, params: Parameters):
self.alpha = float(params.alpha)
self.gamma = float(params.gamma)
self.focal_weight = float(params.focal_weight)
self.loc_weight = float(params.loc_weight)
self.size_weight = float(params.size_weight)
self.angle_weight = float(params.angle_weight)
self.heading_weight = float(params.heading_weight)
# self.class_weight = float(params.class_weight)
def losses(self):
return [self.focal_loss, self.loc_loss, self.size_loss, self.angle_loss, self.heading_loss]
def focal_loss(self, y_true: tf.Tensor, y_pred: tf.Tensor):
""" y_true value from occ in {-1, 0, 1}, i.e. {bad match, neg box, pos box} """
self.mask = tf.equal(y_true, 1)
cross_entropy = K.binary_crossentropy(y_true, y_pred)
p_t = y_true * y_pred + (tf.subtract(1.0, y_true) * tf.subtract(1.0, y_pred))
gamma_factor = tf.pow(1.0 - p_t, self.gamma)
alpha_factor = y_true * self.alpha + (1.0 - y_true) * (1.0 - self.alpha)
focal_loss = gamma_factor * alpha_factor * cross_entropy
neg_mask = tf.equal(y_true, 0)
thr = tfp.stats.percentile(tf.boolean_mask(focal_loss, neg_mask), 90.)
hard_neg_mask = tf.greater(focal_loss, thr)
# mask = tf.logical_or(tf.equal(y_true, 0), tf.equal(y_true, 1))
mask = tf.logical_or(self.mask, tf.logical_and(neg_mask, hard_neg_mask))
masked_loss = tf.boolean_mask(focal_loss, mask)
return self.focal_weight * tf.reduce_mean(masked_loss)
def loc_loss(self, y_true: tf.Tensor, y_pred: tf.Tensor):
mask = tf.tile(tf.expand_dims(self.mask, -1), [1, 1, 1, 1, 3])
loss = tf.compat.v1.losses.huber_loss(y_true,
y_pred,
delta=3.0,
reduction="none")
masked_loss = tf.boolean_mask(loss, mask)
return self.loc_weight * tf.reduce_mean(masked_loss)
def size_loss(self, y_true: tf.Tensor, y_pred: tf.Tensor):
mask = tf.tile(tf.expand_dims(self.mask, -1), [1, 1, 1, 1, 3])
loss = tf.compat.v1.losses.huber_loss(y_true,
y_pred,
delta=3.0,
reduction="none")
masked_loss = tf.boolean_mask(loss, mask)
return self.size_weight * tf.reduce_mean(masked_loss)
def add_sin_difference(self, y_true, y_pred, factor=1.0):
if factor != 1.0:
y_true = factor * y_true
y_pred = factor * y_pred
rad_pred_encoding = tf.math.sin(y_pred) * tf.math.cos(y_true)
rad_tg_encoding = tf.math.cos(y_pred) * tf.math.sin(y_true)
return rad_tg_encoding, rad_pred_encoding
def angle_loss(self, y_true: tf.Tensor, y_pred: tf.Tensor):
y_true, y_pred = self.add_sin_difference(y_true, y_pred)
loss = tf.compat.v1.losses.huber_loss(y_true,
y_pred,
delta=3.0,
reduction="none")
masked_loss = tf.boolean_mask(loss, self.mask)
return self.angle_weight * tf.reduce_mean(masked_loss)
def heading_loss(self, y_true: tf.Tensor, y_pred: tf.Tensor):
loss = K.binary_crossentropy(y_true, y_pred)
masked_loss = tf.boolean_mask(loss, self.mask)
return self.heading_weight * tf.reduce_mean(masked_loss)