-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.m
157 lines (130 loc) · 5.18 KB
/
main.m
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
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
%% Control variables
vo_freq = 16;
lidar_freq = 12.5;
ins_freq = 50;
gps_freq = 5;
milli = 1000000;
merge_step_size = 0.2; %0.5; %m
alpha = 0.984; % 1 == vo, 0 == lidar
%alpha = 1
close all
%% Grab Data
if ~exist('DATA_IS_LOADED','var') || ~DATA_IS_LOADED
fprintf("Loading Data... ")
% date = "2014-06-25-16-22-15";
date = "2015-11-13-10-28-08";
% Load data
[vo,vo_time,scale] = get_vo(date);
[scans,lidar_time] = get_lidar(date,1);
[ins,ins_time] = get_ins(date);
[gps, gps_time] = get_gps(date);
% Find the earliest time stamp
[first_time,first_ind] = ...
min([vo_time(1),lidar_time(1),vo_time(1)]);
[last_time,last_ind] = ...
max([vo_time(end),lidar_time(end),vo_time(end)]);
% Adjust time to be in seconds
vo_time_s = (vo_time-first_time)/milli;
lidar_time_s = (lidar_time-first_time)/milli;
ins_time_s = (ins_time-first_time)/milli;
gps_time_s = (gps_time-first_time)/milli;
DATA_IS_LOADED = 1;
fprintf("Done!\n")
vo(:,4:6) = vo(:,4:6)/scale; % Remove scaling from rotation values
%vo_state = odometryToState([0,0,0,0,0,-pi/2]',vo);
vo_state = odometryToState(zeros(6,1),vo');
vo_state(1:3,:) = vo_state(1:3,:)/scale; % Remove scaling from translation values
vo(:,1:3) = vo(:,1:3)/scale;
end
% How close lidar and vo need to be to count as same time
vo_lidar_time_epsilon = 2*abs((1/vo_freq) - (1/lidar_freq));
gps_lidar_time_epsilon = 2*abs((1/gps_freq) - (1/lidar_freq));
ins_lidar_time_epsilon = 2*abs((1/ins_freq) - (1/lidar_freq));
% Initalization
next_lidar_scan_index = 1;
global_pointcloud = [];
% Keep track of state since vo and lidar last aligned
state_at_last_sync = [];
vo_index_at_last_sync = [];
state_at_each_timestep = [];
error = [];
last_gps_idx = 1;
nan_flag = 0;
% Lidar starts before vo, bypass early measuremnts
while lidar_time_s(next_lidar_scan_index) < vo_time_s(1)
next_lidar_scan_index = next_lidar_scan_index + 1;
end
initial_lidar_scan_index = next_lidar_scan_index;
% Loop through each vo state
for i = 1:size(vo_state,2)-1
% Display current vo iteration
if (mod(i,100) == 0)
fprintf("VO Iteration %d/%d\n",i,size(vo_state,2)-1);
end
% If there are no more lidar scans
if next_lidar_scan_index > size(scans,1)
new_state_estimate = state_at_last_sync + vo_state(:,i) - vo_state(:,i-1);
state_at_each_timestep = [state_at_each_timestep,new_state_estimate];
state_at_last_sync = new_state_estimate;
continue;
end
% If vo_time in s is close to next lidar scan
if abs(vo_time_s(i) - lidar_time_s(next_lidar_scan_index)) < vo_lidar_time_epsilon
vo_index_at_last_sync = i;
% IF first time set global point cloud to new scan
if next_lidar_scan_index == initial_lidar_scan_index
% Calculate transform to global frame for lidar scan from vo
[vo_aff3d] = stateToAffine3d(vo_state(:,i));
state_at_last_sync = vo_state(:,i);% + [0,0,0,0,0,-pi/2]'
% Transform lidar scan to global frame
new_points_global = ...
pctransform(scans{next_lidar_scan_index},vo_aff3d);
global_pointcloud = new_points_global;
next_lidar_scan_index = next_lidar_scan_index + 1;
%vo_index_at_last_sync = i;
continue
end
% Else merge into global pointcloud
% COMPLEMENTARY FILTER
vo_state_diff = vo_state(:,i) - vo_state(:,vo_index_at_last_sync);
% Get diff in state from last sync via lidar
rig3d = pcregistericp(global_pointcloud, scans{next_lidar_scan_index});
lidar_state_diff = [rig3d.Translation rotm2eul(rig3d.Rotation)]';
% Apply filter
comp = alpha*vo_state_diff + (1-alpha)*lidar_state_diff;
% Predict our new state
new_state_estimate = ...
prediction_step(state_at_last_sync,[],comp);
% Create affine3d transformation
[comp_aff3d] = stateToAffine3d(new_state_estimate(:,1));
% Trasnform scan to global coords
new_points_global = ...
pctransform(scans{next_lidar_scan_index},vo_aff3d);
% Update global pointcloud
global_pointcloud = ...
pcmerge(global_pointcloud,new_points_global,merge_step_size);
% Update last
state_at_last_sync = new_state_estimate;
%vo_index_at_last_sync = i;
state_at_each_timestep = [state_at_each_timestep new_state_estimate];
% Compare with GPS data
for j = last_gps_idx:size(gps_time,1)
if (abs(gps_time_s(j) - vo_time_s(i)) < gps_lidar_time_epsilon)
error = [error ; norm((ins(1:3,j)-ins(1:3,1))-new_state_estimate(1:3,1))];
last_gps_idx = j+1;
break;
end
end
% Update next scan
next_lidar_scan_index = next_lidar_scan_index + 1;
if (mod(next_lidar_scan_index,20) == 0)
fprintf("Lidar Scan %d/%d\n",next_lidar_scan_index,numel(scans))
end
end
end
% Display Results
pcshow(global_pointcloud)
xlabel("X")
ylabel("Y tho")
zlabel("Z")
visualize_two_state(ins, state_at_each_timestep,"error");