-
Notifications
You must be signed in to change notification settings - Fork 449
/
RandomPath.m
243 lines (218 loc) · 8.47 KB
/
RandomPath.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
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
%RandomPath Vehicle driver class
%
% Create a "driver" object capable of steering a Vehicle subclass object through random
% waypoints within a rectangular region and at constant speed.
%
% The driver object is connected to a Vehicle object by the latter's
% add_driver() method. The driver's demand() method is invoked on every
% call to the Vehicle's step() method.
%
% Methods::
% init reset the random number generator
% demand speed and steer angle to next waypoint
% display display the state and parameters in human readable form
% char convert to string
%plot
% Properties::
% goal current goal/waypoint coordinate
% veh the Vehicle object being controlled
% dim dimensions of the work space (2x1) [m]
% speed speed of travel [m/s]
% dthresh proximity to waypoint at which next is chosen [m]
%
% Example::
%
% veh = Bicycle(V);
% veh.add_driver( RandomPath(20, 2) );
%
% Notes::
% - It is possible in some cases for the vehicle to move outside the desired
% region, for instance if moving to a waypoint near the edge, the limited
% turning circle may cause the vehicle to temporarily move outside.
% - The vehicle chooses a new waypoint when it is closer than property
% closeenough to the current waypoint.
% - Uses its own random number stream so as to not influence the performance
% of other randomized algorithms such as path planning.
%
% Reference::
%
% Robotics, Vision & Control, Chap 6,
% Peter Corke,
% Springer 2011
%
% See also Vehicle, Bicycle, Unicycle.
% Copyright (C) 1993-2017, by Peter I. Corke
%
% This file is part of The Robotics Toolbox for MATLAB (RTB).
%
% RTB is free software: you can redistribute it and/or modify
% it under the terms of the GNU Lesser General Public License as published by
% the Free Software Foundation, either version 3 of the License, or
% (at your option) any later version.
%
% RTB is distributed in the hope that it will be useful,
% but WITHOUT ANY WARRANTY; without even the implied warranty of
% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
% GNU Lesser General Public License for more details.
%
% You should have received a copy of the GNU Leser General Public License
% along with RTB. If not, see <http://www.gnu.org/licenses/>.
%
% http://www.petercorke.com
% TODO
% should be a subclass of VehicleDriver
% Vehicle should be an abstract superclass
% dim should be checked, can be a 4-vector like axis()
classdef RandomPath < handle
properties
goal % current goal
h_goal % graphics handle for goal
veh % the vehicle we are driving
xrange
yrange
speed % speed of travel
dthresh % proximity to goal before choosing new one
d_prev
randstream % random stream just for Sensors
show
verbose
end
methods
function driver = RandomPath(dim, varargin)
%RandomPath.RandomPath Create a driver object
%
% D = RandomPath(D, OPTIONS) returns a "driver" object capable of driving
% a Vehicle subclass object through random waypoints. The waypoints are positioned
% inside a rectangular region of dimension D interpreted as:
% - D scalar; X: -D to +D, Y: -D to +D
% - D (1x2); X: -D(1) to +D(1), Y: -D(2) to +D(2)
% - D (1x4); X: D(1) to D(2), Y: D(3) to D(4)
%
% Options::
% 'speed',S Speed along path (default 1m/s).
% 'dthresh',D Distance from goal at which next goal is chosen.
%
% See also Vehicle.
% TODO options to specify region, maybe accept a Map object?
% dim can be a 4-vector
switch length(dim)
case 1
driver.xrange = [-dim dim];
driver.yrange = [-dim dim];
case 2
driver.xrange = [-dim(1) dim(1)];
driver.yrange = [-dim(2) dim(2)];
case 4
driver.xrange = [dim(1) dim(2)];
driver.yrange = [dim(3) dim(4)];
otherwise
error('bad dimension specified');
end
opt.speed = 1;
opt.dthresh = 0.05 * diff(driver.xrange) / 2;
opt.show = true;
driver = tb_optparse(opt, varargin, driver);
drive.d_prev = Inf;
driver.randstream = RandStream.create('mt19937ar');
end
function init(driver)
%RandomPath.init Reset random number generator
%
% R.init() resets the random number generator used to create the waypoints.
% This enables the sequence of random waypoints to be repeated.
%
% Notes::
% - Called by Vehicle.run.
%
% See also RANDSTREAM.
driver.goal = [];
driver.randstream.reset();
delete(driver.h_goal); % delete the goal
driver.h_goal = [];
end
% called by Vehicle superclass
function plot(driver)
clf
axis([driver.xrange driver.yrange]);
hold on
xlabel('x');
ylabel('y');
end
% private method, invoked from demand() to compute a new waypoint
function setgoal(driver)
% choose a uniform random goal within inner 80% of driving area
while true
r = driver.randstream.rand()*0.8+0.1;
gx = driver.xrange * [r; 1-r];
r = driver.randstream.rand()*0.8+0.1;
gy = driver.yrange * [r; 1-r];
driver.goal = [gx; gy];
%driver.goal = 0.8 * driver.dim * (r - 0.5)*2;
if norm(driver.goal - driver.veh.x(1:2)) > 2*driver.dthresh
break;
end
end
if driver.verbose
fprintf('set goal: (%.1f %.1f)\n', driver.goal);
end
if driver.show && isempty(driver.h_goal)
driver.h_goal = plot(driver.goal(1), driver.goal(2), 'rd', 'MarkerSize', 12, 'MarkerFaceColor', 'r');
else
set(driver.h_goal, 'Xdata', driver.goal(1), 'Ydata', driver.goal(2));
end
end
function [speed, steer] = demand(driver)
%RandomPath.demand Compute speed and heading to waypoint
%
% [SPEED,STEER] = R.demand() is the speed and steer angle to
% drive the vehicle toward the next waypoint. When the vehicle is
% within R.dtresh a new waypoint is chosen.
%
% See also Vehicle.
if isempty(driver.goal)
driver.setgoal();
end
speed = driver.speed;
goal_heading = atan2(driver.goal(2)-driver.veh.x(2), ...
driver.goal(1)-driver.veh.x(1));
d_heading = angdiff(goal_heading, driver.veh.x(3));
steer = d_heading;
% if nearly at goal point, choose the next one
d = norm(driver.veh.x(1:2) - driver.goal);
if d < driver.dthresh
driver.setgoal();
elseif d > 2*driver.d_prev
driver.setgoal();
end
driver.d_prev = d;
end
function display(driver)
%RandomPath.display Display driver parameters and state
%
% R.display() displays driver parameters and state in compact
% human readable form.
%
% Notes::
% - This method is invoked implicitly at the command line when the result
% of an expression is a RandomPath object and the command has no trailing
% semicolon.
%
% See also RandomPath.char.
loose = strcmp( get(0, 'FormatSpacing'), 'loose');
if loose
disp(' ');
end
disp([inputname(1), ' = '])
disp( char(driver) );
end % display()
function s = char(driver)
%RandomPath.char Convert to string
%
% s = R.char() is a string showing driver parameters and state in in
% a compact human readable format.
s = 'RandomPath driver object';
s = char(s, sprintf(' current goal=(%g,%g), X %f : %f; Y %f : %f, dthresh %g', ...
driver.goal, driver.xrange, driver.yrange, driver.dthresh));
end
end % methods
end % classdef