This is a very basic "controller" written in Python, using ROS2 and MAVROS to control a PX4-based vehicle. It commands all current vehicles to fly in a 1m radius circle at a height of 2.5m.
It begins sending offboard setpoints, sets the mode to OFFBOARD
for px4 or GUIDED
for ardupilot, arms the vehicle, takes off, and finally begins moving the setpoint around the sky.
-
In the initialisation phase, the controller sends a stream of setpoints to the vehicle while it waits for a valid position. This is needed in PX4, as it will not switch into
OFFBOARD
mode without a valid setpoint stream. Ardupilot also needs this stream to set its home position before flight. In this phase, the controller also initialises the initial position of the vehicle. -
PX4 follows various types of setpoints when in
OFFBOARD
mode. Similarly, Ardupilot follows setpoints whilst inGUIDED
mode. The controller uses a ROS service to command the vehicle to switch into this mode. It will retry the mode switch once per second until it succeeds. Once this is complete it will wait for a ros message from/mission_start
before continuing on to ARM. -
With the vehicle in the correct mode, the controller uses another ROS service to command the vehicle to arm. Once the vehicle is armed, the rotors begin to spin and the vehicle will begin trying to reach the setpoints it is provided with. Again the controller attempts this once per second until success.
-
Once the vehicle is armed and in
OFFBOARD
/GUIDED
mode, the flight can begin. For PX4, the controller uses the initial position it recorded earlier and begins to generate a setpoint that rises above this. For Ardupilot, a specific takeoff command is given. Once the setpoint is at the desired takeoff altitude, the controller waits for the actual vehicle position to reach some predetermined value before it considers the takeoff to be complete. -
With takeoff complete, the controller begins sending setpoints for the vehicle. In this case, these follow a circle around the local coordinate system origin.
In the 'fenswood' scenario runner, during the flight, the controller attempts to follow a hard coded trajectory. It performs linear interpolation between the trajectory points based on the times at which the drone should arrive at each trajectory point. Therefore at each controller cycle, it will send a setpoint for the next interpolated location. Once the arrival time for the current target point is reached, the controller interpolates to the next point and so on.
Note: If a message is sent on the
/emergency_stop
topic this controller will send the PX4 e-STOP command. This stops the motors of the drones.
If this controller is run with multiple SITL or real drone instances, each broadcasting their topics in the form <drone_name>/mavros/...
, then one controller is deployed for each drone instance.
By default, launch/example_fly_all.launch.py
is the launch file which produces this behaviour. It provides an example for how to write an offboard multi-drone controller.
ROS and MAVROS use Front-Left-Up (FLU) coordinate systems while PX4 uses a Front-Right-Down (FRD) coordinate system. Coordinates and vectors transferred between the two systems are automatically transformed to the appropriate convention by MAVROS. You should provide MAVROS with setpoints in FLU and expect position data from it in FLU.
Another aspect to be aware of is PX4's treatment of the local coordinate system. Generally, this coordinate system will be centred where the vehicle was powered up. However, internal estimates are often poor during ground handling, so there may be a signficant offset from your expectation. Coordinating the local coordinate systems of multiple vehicles can become complex.
If external local positioning information is provided, for example from Vicon, PX4's coordinate system will share the same origin. This makes dealing with multiple vehicles muc easier.
PX4's global coordinate system as exposed by MAVROS comes in two forms. The first is a PoseStamped message, which is relative to either a provided map origin, the home position, or the first GNSS fix received. The other is a full geographic coordinate formed of latitude, longitude and altitude as part of a NavSatFix message.
The controller should be mostly compatible with ArduPilot. The main difference is the name that ArduPilot gives to the mode needed to obey setpoints. The closest equivalent is ArduPilot's "GUIDED" mode. The controller has not been tested with ArduPilot yet.
There are a few ROS specifics that may look a little strange to the
uninitiated. The first is the blank file under resource/
. This file is in
fact quite critical. It is used by the ROS index to let it know that this
package has been installed. The empty __init__.py
file is there for similar
mysterious reasons. The second oddity is the setup.cfg
file. This is again
a ROS specific configuration ensuring that things get installed into the
correct places.
package.xml
provides ROS with information about the package. You should list
any dependencies your controller has here. For almost all of the controllers,
you will want to list mavros_msgs
as a dependency. This is already installed from the package manager into
the base controller image. A lot of
information from here then has to be duplicated into the setup.py
script. If
only there was a way to automate the building of build scripts...