-
Notifications
You must be signed in to change notification settings - Fork 13
/
planCoverRRTDubins.m
197 lines (166 loc) · 5.84 KB
/
planCoverRRTDubins.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
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
%% planRRT
%% - basic coverage algorithm
%%
%% Last Modified - 11/15/2010 - R. Beard
function path=planCoverRRTDubins(wpp_start, R, map)
% desired down position is down position of start node
pd = wpp_start(3);
% specify start node from wpp_start
start_node = [wpp_start(1), wpp_start(2), pd, 0, 0, 0];
% format is [N, E, D, chi, cost, parent]
% return map
returnMapSize = 30; % this is a critical parameter!
return_map = 50*ones(returnMapSize,returnMapSize)+ rand(returnMapSize,returnMapSize);
plotReturnMap(return_map), %pause
% construct search path by doing N search cycles
SEARCH_CYCLES = 100; % number of search cycles
% look ahead tree parameters
L = 2*R+1; % segment Length
vartheta = pi/4; % search angle
depth = 5; % depth of look ahead tree
% initialize path and tree
path = start_node;
for i=1:SEARCH_CYCLES,
tree = extendTree(path(end,:),L,vartheta,depth,map,return_map,pd);
next_path = findMaxReturnPath(tree);
path = [path; next_path(1,:)];
% update the return map
return_map = updateReturnMap(next_path(1,:),return_map,map);
plotReturnMap(return_map), %pause
% set the end of the path as the root of the tree
end
% remove path segments where there is no turn
path_=path;
path = path(1,:);
for i=2:size(path_,1),
if path_(i,4)~=path_(i-1,4);
path = [path; path_(i,:)];
end
end
% % specify that these are straight-line paths.
% for i=1:size(path,1),
% path(i,4)=-9999;
% end
end
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% extendTree
%% extend tree by randomly selecting point and growing tree toward that
%% point
function tree = extendTree(start_node,L,vartheta,depth,map,return_map,pd)
tree_ = [start_node, 0]; % the last variable marks node as expanded
% extend tree initially
for d = 1:depth,
newnodes = [];
for j=1:size(tree_,1),
if tree_(j,7)~=1, % process unexpanded nodes
for i=1:3,
if i==1, theta = tree_(j,4)-vartheta;
elseif i==2, theta = tree_(j,4);
elseif i==3, theta = tree_(j,4)+vartheta;
end
newnode_ = [...
tree_(j,1)+L*cos(theta),...
tree_(j,2)+L*sin(theta),...
tree_(j,3),...
theta,...
0,...
j,...
0,...
];
if collision(tree_(j,:), newnode_, map)==0,
newnode_(5) = tree_(j,5)...
+findReturn(newnode_(1),newnode_(2),return_map,map);
newnodes = [newnodes; newnode_];
end
end
tree_(j,7)=1; % mark as expanded
end
end
tree_ = [tree_; newnodes];
end
tree = tree_(:,1:6);
end
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% collision
%% check to see if a node is in collsion with obstacles
function collision_flag = collision(start_node, end_node, map)
collision_flag = 0;
sigma = 0:.1:1; % define where collisions will be checked
for i=1:length(sigma),
X = (1-sigma(i))*start_node(1) + sigma(i)*end_node(1);
Y = (1-sigma(i))*start_node(2) + sigma(i)*end_node(2);
Z = (1-sigma(i))*start_node(3) + sigma(i)*end_node(3);
if Z >= downAtNE(map, X, Y),
collision_flag = 1;
end
% check to see if outside of world
if (X>map.width) | (X<0) | (Y>map.width) | (Y<0),
collision_flag = 1;
end
end
end
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% downAtNE
%% find the map down coordinate at a specified (n,e) location
function down = downAtNE(map, n, e)
[d_n,idx_n] = min(abs(n - map.buildings_n));
[d_e,idx_e] = min(abs(e - map.buildings_e));
if (d_n<=map.BuildingWidth) & (d_e<=map.BuildingWidth),
down = -map.heights(idx_e,idx_n);
else
down = 0;
end
end
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% findReturn
%% compute the return value at a particular location
function return_value = findReturn(pn,pe,return_map,map);
[pn_max,pe_max] = size(return_map);
fn = pn_max*pn/map.width;
fn = min(pn_max,round(fn));
fn = max(1,fn);
fe = pe_max*pe/map.width;
fe = min(pe_max,round(fe));
fe = max(1,fe);
return_value = return_map(fn,fe);
end
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% findMaxReturnPath
%% find the maximum return path in the tree
function path = findMaxReturnPath(tree)
% find node with max return
[tmp,idx] = max(tree(:,5));
% construct path with maximum return
path = tree(idx,:);
parent_node = tree(idx,6);
%while parent_node>1,
while parent_node>1,
path = [tree(parent_node,:); path];
parent_node = tree(parent_node,6);
end
end
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% updateReturnMap
%% update the return map to indicate where MAV has been
function new_return_map = updateReturnMap(path,return_map,map)
new_return_map = return_map;
for i=1:size(path,1),
pn = path(i,1);
pe = path(i,2);
[pn_max,pe_max] = size(return_map);
fn = pn_max*pn/map.width;
fn = min(pn_max,round(fn));
fn = max(1,fn);
fe = pe_max*pe/map.width;
fe = min(pe_max,round(fe));
fe = max(1,fe);
new_return_map(fn,fe) = return_map(fn,fe) - 50;
end
end
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
%% plotReturnMap
%% plot the return map
function plotReturnMap(return_map)
figure(2), clf
mesh(return_map)
end