Ardupilot2/libraries/AP_Scripting/examples/copter_pingpong.md
2023-09-27 08:14:15 +10:00

189 lines
8.1 KiB
Markdown

# Drone Ping-Pong Code
This script makes the drone go forward and backward a defined distance and number of times.
## Stages
- 0: Change to Guided mode
- 1: Takeoff to the height defined by `takeoff_alt`
- 2: Wait until reaching the takeoff altitude
- 3: Go forward to the defined distance
- 4: Go back to the initial position
- 5: Change to Land mode
## Variables:
- `takeoff_alt`: Takeoff height (m)
- `copter_guided_mode_num`: Guided mode number
- `copter_land_mode_num`: Land mode number
- `stage`: current stage
- `count`: Number of times the drone has gone forward
- `max_count`: Maximum number of times the drone should go forward
- `ping_pong_distance`: Distance up to which the drone should go forward (m)
- `vel`: Drone velocity (m/s)
## Understand the code:
First, there is a comment explaining what the code is about and how it works. Then, the local variables that will be used are declared.
The `copter_guided_mode_num` and `copter_land_mode_num `are standard numbers, defined by ArduPilot, for the Guided and Land flight modes, respectively. The other variables are user-defined settings according to the desired behavior, as indicated in the comments in front of each one.
```lua
-- This script makes the drone go forward and backward at a defined distance and number of times.
-- The stages are:
-- 0) Change to Guided mode
-- 1) Takeoff to the height defined by takeoff_alt
-- 2) Wait until reaching the takeoff altitude
-- 3) Go forward to the defined distance
-- 4) Go back to the initial position
-- 5) Change to Land mode
local takeoff_alt = 3 -- Takeoff height
local copter_guided_mode_num = 4
local copter_land_mode_num = 9
local stage = 0
local count = 0 -- Number of times the drone has gone forward
local max_count = 2 -- Maximum number of times the drone should go forward
local ping_pong_distance = 10 -- Distance up to which the drone should go forward (m)
local vel = 1
```
Next, there's the main function of the code, the `update()` function, which will be called once the code is started (since it wasn't indicated a waiting time at the last line of the code) and its `return` indicates that this same fucntion will be called again 100ms after finishing its executuion.
```lua
function update()
-- [Code]
return update(), 100
end
return update()
```
In the presented if, the code is waiting for the drone to be armed to change to the next stage. The is_armed() function from the arming library is used to check if the drone is armed or not. The send_text() function from the gcs library is used to send a message ("Arming") with severity 6 (information) to the GCS (Ground Control Station). Thus, while the drone is not armed, it remains in stage 0 of the code.
``` lua
if not arming:is_armed() then
stage = 0
gcs:send_text(6, "Arming")
```
If the drone is armed, it moves to the else section, which presents a behavior for each stage.
In `stage 0`, the drone flight mode is changed to GUIDED MODE. To do this, it's used the function `set_mode()` of `vehicle` library, which receives the desired flight mode number(copter_guided_mode_num, defined in the local variables). When it verifies the drone has switched to the desired flight mode, it moves to the next stage.
``` lua
else
if stage == 0 then
if vehicle:set_mode(copter_guided_mode_num) then
stage = stage + 1
end
```
in stage 1, it's used the `start_takeoff()` function from the `vehicle` library for the drone to take off to a height defined by the variable `takeoff_alt`.
``` lua
elseif stage == 1 then
gcs:send_text(6, "Taking off")
if vehicle:start_takeoff(takeoff_alt) then
stage = stage + 1
end
```
In stage 2, it's used the `ahrs:get_home()` function to get the drone's takeoff location, and the `ahrs:get_position()` function to get the drone's current position. In line 4, it checks that the obtained values are not null. Then, the `home:get_distance_NED()` function stores in `vec_from_home` a 3D vector, starting from `curr_loc` and ending at `home`. In line 7, the code sends to the GCS the value contained in the z-coordinate of `vec_from_home`, i.e., the current altitude of the drone (multiplied by -1 since the vector points towards home, which is at a lower altitude than the current position).
When the difference between `takeoff_alt` and `vec_from_home:z()` is less than 1, indicating that the drone has reached the takeoff altitude, it moves to the next stage (`math.abs()` is used to get the absolute value, and we performed an addition instead of subtraction because the z component is negative).
``` lua
elseif stage == 2 then
local home = ahrs:get_home()
local curr_loc = ahrs:get_position()
if home and curr_loc then
local vec_from_home = home:get_distance_NED(curr_loc)
gcs:send_text(6, "Altitude above home: " .. tostring(math.floor(-vec_from_home:z())))
if math.abs(takeoff_alt + vec_from_home:z()) < 1 then
stage = stage + 1
end
end
```
In `stage3`, first, the drone checks if it has already performed all the requested loops specified in `max_count`. If so, it switches to `stage5`. If not, it executes the commands of `stage3`.
First, it creates a 3D vector `target_vel` to store the desired velocity. Then, it sets the value of each of the components of this created vector (0 for y and z, and `vel` for x).
Then, it uses the `vehicle:set_target_velocity_NED(target_vel)` function to set the drone's velocity as that of `target_vel`. If it encounters any issues, it sends a warning message.
``` lua
elseif (stage == 3) then -- Stage 3: Moving Forward
-- If the maximum number of times is exceeded, move to stage 5
if (count >= max_count) then
stage = stage + 2
end
-- Calculate velocity vector
local target_vel = Vector3f()
target_vel:x(vel)
target_vel:y(0)
target_vel:z(0)
-- Send velocity request
if not (vehicle:set_target_velocity_NED(target_vel)) then
gcs:send_text(6, "Failed to execute velocity command")
end
```
After that, it uses the same strategy as shown before to calculate the distance vector from the takeoff location to verify the distance `x` traveled. When the difference between the distance `x` that the drone traveled and the `ping_pong_distance` is less than 1, it increments the `count` and moves to the next stage.
``` lua
-- checking if reached stop point
local home = ahrs:get_home()
local curr_loc = ahrs:get_position()
if home and curr_loc then
local vec_from_home = home:get_distance_NED(curr_loc)
gcs:send_text(6, "Distance from home: " .. tostring(math.floor(vec_from_home:x())))
if(math.abs(ping_pong_distance - vec_from_home:x()) < 1) then
count = count + 1
stage = stage + 1
end
end
end
```
In `Stage4`, the velocity in `x` is multiplied by -1 for the drone to fly in the opposite direction, returning to the takeoff position. When the difference between the drone's position `x` and the takeoff position's `x` is less than 1, it goes back to `Stage3` to start the forward movement again (or not, if the `count` has reached `max_count`).
``` lua
elseif (stage == 4) then -- Stage 4: Moving Back
-- calculate velocity vector
local target_vel = Vector3f()
target_vel:x(-vel)
target_vel:y(0)
target_vel:z(0)
-- send velocity request
if not (vehicle:set_target_velocity_NED(target_vel)) then
gcs:send_text(6, "Failed to execute velocity command")
end
-- checking if reached stop point
local home = ahrs:get_home()
local curr_loc = ahrs:get_position()
if home and curr_loc then
local vec_from_home = home:get_distance_NED(curr_loc)
gcs:send_text(6, "Distance from home: " .. tostring(math.floor(vec_from_home:x())))
if(math.abs(vec_from_home:x()) < 1) then
stage = stage - 1
end
end
end
```
In `stage5`, the drone simply changes to the Land flight mode and lands, indicating the completion of the code through a message.
``` lua
elseif (stage == 5) then -- Stage 5: Change to LAND mode
vehicle:set_mode(copter_rtl_mode_num)
stage = stage + 1
gcs:send_text(6, "Finished pingpong, switching to LAND")
end
```