-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathCalibNet.py
84 lines (79 loc) · 3.73 KB
/
CalibNet.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
import torch
from torch import nn
import torch.nn.functional as F
from Modules import resnet18
class Aggregation(nn.Module):
def __init__(self,inplanes=768,planes=96,final_feat=(5,2)):
super(Aggregation,self).__init__()
self.conv1 = nn.Conv2d(in_channels=inplanes,out_channels=planes*4,kernel_size=3,stride=2,padding=1)
self.bn1 = nn.BatchNorm2d(planes*4)
self.conv2 = nn.Conv2d(in_channels=planes*4,out_channels=planes*4,kernel_size=3,stride=2,padding=1)
self.bn2 = nn.BatchNorm2d(planes*4)
self.conv3 = nn.Conv2d(in_channels=planes*4,out_channels=planes*2,kernel_size=(2,1),stride=2)
self.bn3 = nn.BatchNorm2d(planes*2)
self.tr_conv = nn.Conv2d(in_channels=planes*2,out_channels=planes,kernel_size=1,stride=1)
self.tr_bn = nn.BatchNorm2d(planes)
self.rot_conv = nn.Conv2d(in_channels=planes*2,out_channels=planes,kernel_size=1,stride=1)
self.rot_bn = nn.BatchNorm2d(planes)
self.tr_drop = nn.Dropout2d(p=0.2)
self.rot_drop = nn.Dropout2d(p=0.2)
self.tr_pool = nn.AdaptiveAvgPool2d(output_size=final_feat)
self.rot_pool = nn.AdaptiveAvgPool2d(output_size=final_feat)
self.fc1 = nn.Linear(planes*final_feat[0]*final_feat[1],3) # 96*10
self.fc2 = nn.Linear(planes*final_feat[0]*final_feat[1],3) # 96*10
for m in self.modules():
if isinstance(m, nn.Conv2d) or isinstance(m, nn.ConvTranspose2d):
nn.init.kaiming_normal_(m.weight.data, mode='fan_in')
if m.bias is not None:
m.bias.data.zero_()
nn.init.xavier_normal_(self.fc1.weight,0.1)
nn.init.xavier_normal_(self.fc2.weight,0.1)
def forward(self,x:torch.Tensor):
x = self.conv1(x)
x = self.bn1(x)
x = self.conv2(x)
x = self.bn2(x)
x = self.conv3(x)
x = self.bn3(x)
x_tr = self.tr_conv(x)
x_tr = self.tr_bn(x_tr)
x_tr = self.tr_drop(x_tr)
x_tr = self.tr_pool(x_tr) # (19,6)
x_tr = self.fc1(x_tr.view(x_tr.shape[0],-1))
x_rot = self.rot_conv(x)
x_rot = self.rot_bn(x_rot)
x_rot = self.rot_drop(x_rot)
x_rot = self.rot_pool(x_rot) # (19.6)
x_rot = self.fc2(x_rot.view(x_rot.shape[0],-1))
return x_rot, x_tr
class CalibNet(nn.Module):
def __init__(self,backbone_pretrained=False,depth_scale=100.0):
super(CalibNet,self).__init__()
self.scale = depth_scale
self.rgb_resnet = resnet18(inplanes=3,planes=64) # outplanes = 512
self.depth_resnet = nn.Sequential(
nn.MaxPool2d(kernel_size=5,stride=1,padding=2), # outplanes = 256
resnet18(inplanes=1,planes=32),
)
self.aggregation = Aggregation(inplanes=512+256,planes=96)
self.device = torch.device("cuda:0" if torch.cuda.is_available() else "cpu")
if backbone_pretrained:
self.rgb_resnet.load_state_dict(torch.load("resnetV1C.pth")['state_dict'],strict=False)
self.to(self.device)
def forward(self,rgb:torch.Tensor,depth:torch.Tensor):
# rgb: [B,3,H,W]
# depth: [B,1,H,W]
x1,x2 = rgb,depth.clone() # clone dpeth, or it will change depth in '/' operation
x2 /= self.scale
x1 = self.rgb_resnet(x1)[-1]
x2 = self.depth_resnet(x2)[-1]
feat = torch.cat((x1,x2),dim=1) # [B,C1+C2,H,W]
x_rot, x_tr = self.aggregation(feat)
return x_rot, x_tr
if __name__=="__main__":
x = (torch.rand(2,3,1242,375).cuda(),torch.rand(2,1,1242,375).cuda())
model = CalibNet(backbone_pretrained=False).cuda()
model.eval()
rotation,translation = model(*x)
print("translation size:{}".format(translation.size()))
print("rotation size:{}".format(rotation.size()))