-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathauv_control.orogen
456 lines (355 loc) · 17.3 KB
/
auv_control.orogen
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
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
name "auv_control"
version "0.2"
import_types_from "base"
import_types_from "6dControl.hpp"
using_library "motor_controller"
using_library "base-logging", typekit: false
typekit do
export_types "/motor_controller/PIDSettings", "/motor_controller/ParallelPIDSettings"
export_types "/base/LinearAngular6DWaypoint"
end
# Definition of the base interface for all AUV control components
#
# It also implements the base functionality of merging and validating the
# declared inputs, as well as managing the command input ports.
task_context "Base" do
needs_configuration
abstract
# This property defines which parts of the command input is expected to be
# set once we merged all the declared input ports.
property("expected_inputs", "/auv_control/ExpectedInputs")
.dynamic
# If true, the component will send a zero command before getting into an
# exception state. Otherwise, it will not do anything
property "keep_position_on_exception", "bool", true
# If true write NaN on all axis, in keep position case
property "nan_on_keep_position", "bool", false
# The target state for this controller
#
# This port is statically defined for simplicity reasons, additional ports
# can be created using the addCommandInput operation
input_port("cmd_in", "/base/LinearAngular6DCommand")
# This property defines the timeout for the cmd_in input port. 0
# means that the timeout would be ignored. Default is 1 second.
property "timeout_in", "/base/Time"
# When used in a cascade, this input port can be used to feed the output of
# the controllers before.
#
# Leave unconnected if you are not cascading controllers
#
# This port is statically defined for simplicity reasons, additional ports
# can be created using the addCommandInput operation
input_port("cmd_cascade", "/base/LinearAngular6DCommand")
# This property defines the timeout for the cascade input port. 0
# means that the timeout would be ignored. Default is 1 second.
property "timeout_cascade", "/base/Time"
# This property defines the safty behavior ath the merging of the input-ports.
# If the property is on true (default) the merged command need to be like in
# the expected_inputs property defined. Else the expected_inputs are ignored
# while the merged comand are unic.
property "safe_mode", "bool", true
# Create a new input to merge
#
# Returns true if the port got added, and false if a port with the same name
# already exists
operation("addCommandInput")
.argument("name", "string")
.argument("timeout", "/base/Time")
.returns("bool")
# This property allows additional command input ports to be added on configuration.
property "additional_command_input", "/std/vector<auv_control/InputPortConfig>"
dynamic_input_port(/cmd_\w+/, "/base/LinearAngular6DCommand")
runtime_states :CONTROLLING, :CONTROLLING_UNSAFE
error_states :INPUT_MISSING, :INPUT_COLLIDING, :INPUT_UNEXPECTED, :TIMEOUT, :WAIT_FOR_CONNECTED_INPUT_PORT
# Deployed with a 10ms period by default
periodic 0.01
end
# Base implementation of all the tasks that use one PID controller per axis to
# generate commands
task_context "BasePIDController", subclasses: "Base" do
abstract
# Ideal Settings for the PID controllers
property("pid_settings", "/base/LinearAngular6DPIDSettings")
.dynamic
# Parallel Settings for the PID controllers
property("parallel_pid_settings", "/base/LinearAngular6DParallelPIDSettings")
.dynamic
# Use the Ideal (false) or Parallel PID-Settings
property "use_parallel_pid_settings", "bool", false
# Defines if the derivative will be applied to the error or to the output in the PID
# controller
property "apply_derivative_to_error", "bool", false
# Under this Value the axis are not controled. Output is 0.
property "variance_threshold", "double"
# This property defines the timeout for the pose_samples input port.
# 0 means that the timeout would be ignored. Default is 1 second
property "timeout_pose", "/base/Time"
# The system state. Only the parts of the state that are controlled needs to
# be available (i.e. if the command involves only orientation, only the
# orientation part is really needed)
input_port "pose_samples", "/base/samples/RigidBodyState"
# The output command. It is a velocity command expressed in the aligned
# frame.
output_port "cmd_out", "/base/LinearAngular6DCommand"
# The states of the pid-controllers
output_port "pid_state", "/auv_control/LinearAngular6DPIDState"
error_states :UNSURE_POSE_SAMPLE
exception_states :POSE_SAMPLE_INVALID, :POSE_TIMEOUT
end
# Controller that takes either positions or velocities, expressed in either the
# world or aligned frames, and outputs "whatever" in the same frame
task_context "PIDController", subclasses: "BasePIDController" do
# The command domain (true:position or false:velocity)
property "position_control", "bool"
# The command frame (true:world or false:aligned)
property "world_frame", "bool"
end
# Task that get a target position in the world frame and outputs a position
# command in the aligned frame.
#
# It will work in any of the three control domains (position, velocity and
# effor), you just have to set position_control to true for the position domain.
#
# Note that this task expects to either have both x,y inputs or none. Giving it
# only X or only Y will result in a failure to start.
task_context "WorldToAligned", subclasses: "Base" do
# The domain of what we are converting (true:position or false:velocity or
# efforts)
property "position_control", "bool"
# Indicates how the angular velocity is being represented (true: euler-rate or
# false: axis-angle). This is NOT used for position control domain.
property "ang_vel_euler_rate", "bool", false
# The system state. What is required depends on which parts of the state are
# given as input
input_port "pose_samples", "/base/samples/RigidBodyState"
# This property defines the timeout for the pose_samples input port.
# 0 means that the timeout would be ignored. Default is 1 second
property "timeout_pose", "/base/Time"
# The output command.
output_port "cmd_out", "/base/LinearAngular6DCommand"
runtime_states :WAIT_FOR_POSE_SAMPLE
exception_states :POSE_SAMPLE_INVALID, :POSE_TIMEOUT
end
# Controller that takes either velocities or efforts expressed in the aligned
# frame as input and outputs the same commands, but expressed in the body frame.
task_context "AlignedToBody", subclasses: "Base" do
# The system pose in the world frame
#
# Only the orientation is being used (to be more precise, only the pitch and
# roll angles)
input_port "orientation_samples", "/base/samples/RigidBodyState"
# This property defines the timeout for the orientation_samples input port.
# 0 means that the timeout would be ignored. Default is 1 second
property "timeout_orientation", "/base/Time"
# The output command. It is expressed in the body frame, and is of the same
# nature than the input (efforts if the inputs are efforts, velocities if
# the inputs are velocities)
output_port "cmd_out", "/base/LinearAngular6DCommand"
runtime_states :WAIT_FOR_ORIENTATION_SAMPLE
exception_states :ORIENTATION_SAMPLE_INVALID, :ORIENTATION_TIMEOUT
end
# Generates thruster commands based on a thruster matrix and a force-torque
# input expressed in the system's body frame.
#
# It assumes zero values on the axis that are not set
#
# It basically projects the (merged) input vector onto the thrusters using the
# matrix
task_context "AccelerationController", subclasses: "Base" do
# Matrix with size of n * 6. n means the count of thrusters that are used.
# The columns 0 to 2 of the matrix are the linear axis. The columns 3 to 5 of the
# matrix are the angular axis.
#
# Warning: if one writes this property programmatically (as opposed to loading it from
# a YAML file), you must transpose it before writing it. See the comments in configure
# Hook for more details.
property "matrix", "/base/MatrixXd"
# Weights that indicate which thrusters should be prioritized in
# cases where multiple solutions are possible. The thrusters with lower
# weights will be prioritized. The property size should be equal to the number of
# thrusters and it must have only positive numbers. If there's no preference
# between the thrusters, just assign the same weight to all of them.
property "thrusters_weights", "/base/VectorXd"
# Names of the thrusters
#
# Leave empty to use no names
property "names", "/std/vector</std/string>"
# Limits of the thrusters
#
# Leave empty if you don't want to limit anything (is that really a good
# idea ?)
property "limits", "/base/JointLimits"
# Lists which command parameter are being controlled on a per-joint basis.
#
# If left empty, uses RAW by default
property "control_modes", "/std/vector</base/JointState/MODE>"
# TRUE: allows the SVD solution for calculating the thrusters commands
# similarly to pseudo-inverse solution.
# FALSE: the thruster commands will be calcultated by transposing the
# thruster matrix and multiplying it by the input.
property "svd_calculation", "bool", true
# Generated motor commands
output_port "cmd_out", "/base/commands/Joints"
# The expected generated effort (as opposed to the input effort)
output_port "expected_effort", "/base/LinearAngular6DCommand"
exception_states :WRONG_SIZE_OF_CONTROLMODES, :WRONG_SIZE_OF_LIMITS, :WRONG_SIZE_OF_NAMES, :INVALID_NAME_IN_LIMITS
end
# Generates a single constant command in the configured domain
task_context "ConstantCommand" do
# The desired command
property "cmd", "/base/LinearAngular6DCommand"
# The output command.
output_port "cmd_out", "/base/LinearAngular6DCommand"
periodic(0.01)
end
# The GroundFollower task provides a depth command to the control chain that keeps a fixed
# distance from the ground provided depth and altitude readings. In the case of altitude
# dropout, the task will still provide a command for a fixed period of time (given as a
# property). If this time is exceeded, keeping a fixed depth is considered an unsafe
# behaviour and therefore an expection state is emitted by the task as a call for help.
task_context "GroundFollower" do
# The desired distance to ground, should be a positive number.
property "distance_to_ground", "double", 2.0
# The timeout the task should wait to get an altimeter reading
property "altimeter_timeout", "/base/Time"
# The timeout the task should wait to get a depth reading
property "depth_timeout", "/base/Time"
# This timeout is when there is a reading on the altimeter but it is a nan which means
# that the altimeter is running but cannot see the ground, in this case the task will
# run for the specified time keeping the last valid ground distance before going into
# an exception state
property "altimeter_dropout_timeout", "/base/Time"
# Minimum saftey altitude, if altitude goes below this value a warning state is
# emitted, this indicates that the control chain is not keeping the desired depth
# correctly
property "safety_distance", "double", 1.0
input_port "altimeter", "/base/samples/RigidBodyState"
input_port "depth", "/base/samples/RigidBodyState"
output_port "floor_position", "double"
# The output command.
output_port "cmd_out", "/base/LinearAngular6DCommand"
runtime_states(:NO_DEPTH_READING, :NO_ALTIMETER_READING, :ALTIMETER_DROPOUT, :WARNING_LOW_ALTITUDE, :NO_VALID_GROUND_DISTANCE)
exception_states(:DEPTH_TIMEOUT, :ALTIMETER_TIMEOUT, :INVALID_DEPTH_READING, :INVALID_NEGATIVE_ALTIMETER_READING, :ALTIMETER_DROPOUT_TIMEOUT)
periodic(0.01)
end
# Follows a set of waypoints, where each waypoint is defined
task_context "WaypointNavigator" do
needs_configuration
# The trajectory to follow, expressed as a set of waypoints in the world
# frame
input_port "trajectory", "/std/vector</base/LinearAngular6DWaypoint>"
# The current system pose
input_port "pose_sample", "/base/samples/RigidBodyState"
# The output command, as a pose in world frame that can be used by the
# auv_control controllers
output_port "cmd_out", "/base/LinearAngular6DCommand"
# Shows error between current and desired waypoint for debuging,
# current waypoint is navigated by this controller and remaining waypoints to follow
output_port("waypoint_info", "/base/LinearAngular6DWaypointInfo")
runtime_states :WAIT_FOR_WAYPOINTS, :KEEP_WAYPOINT, :FINISHED,
:FOLLOWING_WAYPOINTS, :POSE_SAMPLE_MISSING
periodic(0.01)
end
# Task to convert base/commands/Motion2D into a velocity
#
# This allows to reuse control algorithms that deal with 2D motions (such as the
# trajectory_follower)
task_context "MotionCommand2DConverter" do
needs_configuration
# Command input as a velocity motion in the plan
input_port "cmd_in", "/base/commands/Motion2D"
# Command output as an aligned velocity. Only the heading and X velocities
# are set
output_port "cmd_out", "/base/LinearAngular6DCommand"
port_driven "cmd_in"
end
# Task that allows a vehicle to drive long distances in an optimal
# orientation. So the vehicle can drive faster to his target
# position. This controler works with Commands in AlignedPosition
# Frame.
task_context "OptimalHeadingController", subclasses: "Base" do
# The optimal orientation of the vehicle, if the target position
# is on an orientation of 0.0
property "optimal_heading", "double", 0.0
# The distance to target position, over that the vehicle used
# the optimal heading
property "optimal_heading_distance", "double"
# The system state. What is required depends on which parts of the state are
# given as input
input_port "orientation_samples", "/base/samples/RigidBodyState"
# The output command.
output_port "cmd_out", "/base/LinearAngular6DCommand"
error_states :WAIT_FOR_ORIENTATION_SAMPLE
end
# Base task for ThrustersInput and ThrustersFeedback
task_context "ThrustersBase" do
needs_configuration
abstract
# Convert thruster signal into forces, in positive direction or CW.
# Should have a size equal to the number of thrusters
# Thruster[N] = Coeff * rotation * |rotation|
property "thruster_coeff_pos", "base::VectorXd"
# Convert thruster signal into forces, in negative direction or CCW.
# Should have a size equal to the number of thrusters
# Thruster[N] = Coeff * rotation * |rotation|
property "thruster_coeff_neg", "/base/VectorXd"
# If left empty, uses RAW by default
property "control_modes", "/std/vector</base/JointState/MODE>"
# In case the control_modes is RAW (pwm), used to convert the signal into DC Voltage
# Thruster[N] = Coeff * voltage * |voltage|
property "thruster_voltage", "double", 0
# The description about the ports are specified in ThrustersInput and
# ThrustersFeedback tasks
input_port "cmd_in", "/base/commands/Joints"
output_port "cmd_out", "/base/commands/Joints"
exception_states :UNSET_THRUSTER_INPUT, :UNEXPECTED_THRUSTER_INPUT
port_driven "cmd_in"
end
# Task that convert thruster forces to thruster's rotation.
#
# AccelerationController.cmd_out --> ThrustersInput --> Demultiplexer.cmd_in
#
task_context "ThrustersInput", subclasses: "ThrustersBase" do
needs_configuration
# cmd_in: forces that each thruster should apply. Should contain data in cmd_in.effort
# cmd_out: Generated motor commands
end
# Task that converts thruster's feedback from RPM or PWM to thruster forces.
#
# Multiplexer.joint_samples --> ThrustersFeedback --> ThrusterForce2BodyEffort.thruster_forces
#
task_context "ThrustersFeedback", subclasses: "ThrustersBase" do
needs_configuration
# cmd_in: thrusters rotation feedback
# cmd_out: thrusters forces
end
# Task that converts thruster's forces to body efforts.
#
# ThrustersFeedback.cmd_out --> ThrusterForce2BodyEffort --> Wherever body efforts are required
#
task_context "ThrusterForce2BodyEffort" do
needs_configuration
# Matrix with size of 6 * n. n means the count of thrusters that are used.
# The rows 0 to 2 of the matrix are the linear axis. The lines 3 to 5 of the
# matrix are the angular axis.
property "thruster_configuration_matrix", "/base/MatrixXd"
# Thruster individual forces
input_port "thruster_forces", "/base/commands/Joints"
# Body efforts once the thruster configuration matrix is applied to the
# thruster forces
output_port "body_efforts", "/base/LinearAngular6DCommand"
exception_states :UNSET_THRUSTER_INPUT, :UNEXPECTED_THRUSTER_INPUT
port_driven "thruster_forces"
end
# Task that works as switch betwen two commands
task_context "CommandInjection", subclasses: "Base" do
# Timeout for the injection command
property "cmd_injection_timeout", "double", 0.05
# Needs a keep alive command (NaN's in cmd_injection) in order to forward the cascade command
property "keep_alive", "bool", false
# Non NaN values will be injected in the cascade input
input_port "cmd_injection", "/base/LinearAngular6DCommand"
# The output command
output_port "cmd_out", "/base/LinearAngular6DCommand"
port_driven
end