diff --git a/Tools/autotest/arducopter.py b/Tools/autotest/arducopter.py index 8b46a62473ec3..2d968b900d9fa 100644 --- a/Tools/autotest/arducopter.py +++ b/Tools/autotest/arducopter.py @@ -8304,6 +8304,57 @@ def WatchAlts(self): self.do_RTL() + def TestTetherStuck(self): + """Test tethered vehicle stuck because of tether""" + # Enable tether simulation + self.set_parameters({ + "SIM_TETH_ENABLE": 1, + }) + self.delay_sim_time(2) + self.reboot_sitl() + + # Set tether line length + self.set_parameters({ + "SIM_TETH_LINELEN": 10, + }) + self.delay_sim_time(2) + + # Prepare and take off + self.wait_ready_to_arm() + self.arm_vehicle() + self.takeoff(10, mode='LOITER') + + # Simulate vehicle getting stuck by increasing RC throttle + self.set_rc(3, 1900) + self.delay_sim_time(5, reason='let tether get stuck') + + # Monitor behavior for 10 seconds + tstart = self.get_sim_time() + initial_alt = self.get_altitude() + stuck = True # Assume it's stuck unless proven otherwise + + while self.get_sim_time() - tstart < 10: + # Fetch current altitude + current_alt = self.get_altitude() + self.progress(f"current_alt={current_alt}") + + # Fetch and log battery status + battery_status = self.mav.recv_match(type='BATTERY_STATUS', blocking=True, timeout=1) + if battery_status: + self.progress(f"Battery: {battery_status}") + + # Check if the vehicle is stuck. + # We assume the vehicle is stuck if the current is high and the altitude is not changing + if battery_status and (battery_status.current_battery < 6500 or abs(current_alt - initial_alt) > 2): + stuck = False # Vehicle moved or current is abnormal + break + + if not stuck: + raise NotAchievedException("Vehicle did not get stuck as expected") + + # Land and disarm the vehicle to ensure we can go down + self.land_and_disarm() + def fly_rangefinder_drivers_fly(self, rangefinders): '''ensure rangefinder gives height-above-ground''' self.change_mode('GUIDED') @@ -12330,6 +12381,7 @@ def tests2b(self): # this block currently around 9.5mins here self.MAV_CMD_MISSION_START_p1_p2, self.ScriptingAHRSSource, self.CommonOrigin, + self.TestTetherStuck, ]) return ret diff --git a/libraries/SITL/SIM_Aircraft.cpp b/libraries/SITL/SIM_Aircraft.cpp index c0469abc7cf20..138e5ae875534 100644 --- a/libraries/SITL/SIM_Aircraft.cpp +++ b/libraries/SITL/SIM_Aircraft.cpp @@ -839,6 +839,11 @@ void Aircraft::update_dynamics(const Vector3f &rot_accel) sitl->models.slung_payload_sim.update(get_position_relhome(), velocity_ef, accel_earth, wind_ef); #endif + // update tether +#if AP_SIM_TETHER_ENABLED + sitl->models.tether_sim.update(location); +#endif + // allow for changes in physics step adjust_frame_time(constrain_float(sitl->loop_rate_hz, rate_hz-1, rate_hz+1)); } @@ -1273,18 +1278,26 @@ void Aircraft::add_twist_forces(Vector3f &rot_accel) } } -#if AP_SIM_SLUNGPAYLOAD_ENABLED -// add body-frame force due to slung payload -void Aircraft::add_slungpayload_forces(Vector3f &body_accel) +// add body-frame force due to slung payload and tether +void Aircraft::add_external_forces(Vector3f &body_accel) { - Vector3f forces_ef; - sitl->models.slung_payload_sim.get_forces_on_vehicle(forces_ef); + Vector3f total_force; +#if AP_SIM_SLUNGPAYLOAD_ENABLED + Vector3f forces_ef_slung; + sitl->models.slung_payload_sim.get_forces_on_vehicle(forces_ef_slung); + total_force += forces_ef_slung; +#endif + +#if AP_SIM_TETHER_ENABLED + Vector3f forces_ef_tether; + sitl->models.tether_sim.get_forces_on_vehicle(forces_ef_tether); + total_force += forces_ef_tether; +#endif // convert ef forces to body-frame accelerations (acceleration = force / mass) - const Vector3f accel_bf = dcm.transposed() * forces_ef / mass; - body_accel += accel_bf; + const Vector3f accel_bf_tether = dcm.transposed() * total_force / mass; + body_accel += accel_bf_tether; } -#endif /* get position relative to home diff --git a/libraries/SITL/SIM_Aircraft.h b/libraries/SITL/SIM_Aircraft.h index e84a078ac8faf..a6d7b273c09b6 100644 --- a/libraries/SITL/SIM_Aircraft.h +++ b/libraries/SITL/SIM_Aircraft.h @@ -322,10 +322,8 @@ class Aircraft { void add_shove_forces(Vector3f &rot_accel, Vector3f &body_accel); void add_twist_forces(Vector3f &rot_accel); -#if AP_SIM_SLUNGPAYLOAD_ENABLED - // add body-frame force due to slung payload - void add_slungpayload_forces(Vector3f &body_accel); -#endif + // add body-frame force due to payload + void add_external_forces(Vector3f &body_accel); // get local thermal updraft float get_local_updraft(const Vector3d ¤tPos); diff --git a/libraries/SITL/SIM_Multicopter.cpp b/libraries/SITL/SIM_Multicopter.cpp index 4e019dde9b4de..273c2fc06dad7 100644 --- a/libraries/SITL/SIM_Multicopter.cpp +++ b/libraries/SITL/SIM_Multicopter.cpp @@ -49,12 +49,10 @@ void MultiCopter::calculate_forces(const struct sitl_input &input, Vector3f &rot add_shove_forces(rot_accel, body_accel); add_twist_forces(rot_accel); -#if AP_SIM_SLUNGPAYLOAD_ENABLED - // add forces from slung payload - add_slungpayload_forces(body_accel); -#endif + // add forces from slung payload or tether payload + add_external_forces(body_accel); } - + /* update the multicopter simulation by one time step */ diff --git a/libraries/SITL/SIM_Tether.cpp b/libraries/SITL/SIM_Tether.cpp new file mode 100644 index 0000000000000..005347b4c25ce --- /dev/null +++ b/libraries/SITL/SIM_Tether.cpp @@ -0,0 +1,309 @@ +/* + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program 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 General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + */ +/* + simulate a static tether attached to the vehicle and ground +*/ + +#include "SIM_config.h" + +#if AP_SIM_TETHER_ENABLED + +#include "SIM_Tether.h" +#include "SITL.h" +#include +#include "SIM_Aircraft.h" +#include +#include +#include + +using namespace SITL; + +// TetherSim parameters +const AP_Param::GroupInfo TetherSim::var_info[] = { + // @Param: ENABLE + // @DisplayName: Tether Simulation Enable/Disable + // @Description: Enable or disable the tether simulation + // @Values: 0:Disabled,1:Enabled + // @User: Advanced + AP_GROUPINFO_FLAGS("ENABLE", 1, TetherSim, enable, 0, AP_PARAM_FLAG_ENABLE), + + // @Param: DENSITY + // @DisplayName: Tether Wire Density + // @Description: Linear mass density of the tether wire + // @Range: 0 1 + // @User: Advanced + AP_GROUPINFO("DENSITY", 2, TetherSim, tether_linear_density, 0.0167), + + // @Param: LINELEN + // @DisplayName: Tether Maximum Line Length + // @Description: Maximum length of the tether line in meters + // @Units: m + // @Range: 0 100 + // @User: Advanced + AP_GROUPINFO("LINELEN", 3, TetherSim, max_line_length, 100.0), + + // @Param: SYSID + // @DisplayName: Tether Simulation MAVLink System ID + // @Description: MAVLink system ID for the tether simulation, used to distinguish it from other systems on the network + // @Range: 0 255 + // @User: Advanced + AP_GROUPINFO("SYSID", 4, TetherSim, sys_id, 2), + + // @Param: STUCK + // @DisplayName: Tether Stuck Enable/Disable + // @Description: Enable or disable a stuck tether simulation + // @Values: 0:Disabled,1:Enabled + // @User: Advanced + AP_GROUPINFO("STUCK", 5, TetherSim, tether_stuck, 0), + + // @Param: SPGCNST + // @DisplayName: Tether Spring Constant + // @Description: Spring constant for the tether to simulate elastic forces when stretched beyond its maximum length + // @Range: 0 255 + // @User: Advanced + AP_GROUPINFO("SPGCNST", 6, TetherSim, tether_spring_constant, 100), + + AP_GROUPEND +}; + +// TetherSim handles interaction with main vehicle +TetherSim::TetherSim() +{ + AP_Param::setup_object_defaults(this, var_info); +} + + +void TetherSim::update(const Location& veh_pos) +{ + if (!enable) { + return; + } + + if (veh_pos.is_zero()) { + return; + } + + // initialise fixed tether ground location + const uint32_t now_us = AP_HAL::micros(); + if (!initialised) { + // capture EKF origin + auto *sitl = AP::sitl(); + const Location ekf_origin = sitl->state.home; + if (ekf_origin.lat == 0 && ekf_origin.lng == 0) { + return; + } + + // more initialisation + last_update_us = now_us; + initialised = true; + } + + // calculate dt and update tether forces + const float dt = (now_us - last_update_us)*1.0e-6; + last_update_us = now_us; + + update_tether_force(veh_pos, dt); + + // send tether location to GCS at 5hz + // TO-Do: add provision to make the tether movable + const uint32_t now_ms = AP_HAL::millis(); + if (now_ms - last_report_ms >= reporting_period_ms) { + last_report_ms = now_ms; + send_report(); + write_log(); + } +} + +// get earth-frame forces on the vehicle from the tether +// returns true on success and fills in forces_ef argument, false on failure +bool TetherSim::get_forces_on_vehicle(Vector3f& forces_ef) const +{ + if (!enable) { + return false; + } + + forces_ef = veh_forces_ef; + return true; +} + +// send a report to the vehicle control code over MAVLink +void TetherSim::send_report(void) +{ + if (!mavlink_connected && mav_socket.connect(target_address, target_port)) { + ::printf("Tether System connected to %s:%u\n", target_address, (unsigned)target_port); + mavlink_connected = true; + } + if (!mavlink_connected) { + return; + } + + // get current time + uint32_t now_ms = AP_HAL::millis(); + + // send heartbeat at 1hz + const uint8_t component_id = MAV_COMP_ID_USER11; + if (now_ms - last_heartbeat_ms >= 1000) { + last_heartbeat_ms = now_ms; + + const mavlink_heartbeat_t heartbeat{ + custom_mode: 0, + type : MAV_TYPE_GROUND_ROVER, + autopilot : MAV_AUTOPILOT_INVALID, + base_mode: 0, + system_status: 0, + mavlink_version: 0, + }; + + mavlink_message_t msg; + mavlink_msg_heartbeat_encode_status( + sys_id.get(), + component_id, + &mav_status, + &msg, + &heartbeat); + uint8_t buf[300]; + const uint16_t len = mavlink_msg_to_send_buffer(buf, &msg); + mav_socket.send(buf, len); + } + + // send a GLOBAL_POSITION_INT messages + { + Location tether_anchor_loc; + int32_t alt_amsl_cm, alt_rel_cm; + if (!get_tether_ground_location(tether_anchor_loc) || + !tether_anchor_loc.get_alt_cm(Location::AltFrame::ABSOLUTE, alt_amsl_cm) || + !tether_anchor_loc.get_alt_cm(Location::AltFrame::ABOVE_HOME, alt_rel_cm)) { + return; + } + const mavlink_global_position_int_t global_position_int{ + time_boot_ms: now_ms, + lat: tether_anchor_loc.lat, + lon: tether_anchor_loc.lng, + alt: alt_amsl_cm * 10, // amsl alt in mm + relative_alt: alt_rel_cm * 10, // relative alt in mm + vx: 0, // velocity in cm/s + vy: 0, // velocity in cm/s + vz: 0, // velocity in cm/s + hdg: 0 // heading in centi-degrees + }; + mavlink_message_t msg; + mavlink_msg_global_position_int_encode_status(sys_id, component_id, &mav_status, &msg, &global_position_int); + uint8_t buf[300]; + const uint16_t len = mavlink_msg_to_send_buffer(buf, &msg); + if (len > 0) { + mav_socket.send(buf, len); + } + } +} + +void TetherSim::write_log() +{ +#if HAL_LOGGING_ENABLED + // write log of tether state + // @LoggerMessage: TETH + // @Description: Tether state + // @Field: TimeUS: Time since system startup + // @Field: Len: Tether length + // @Field: VFN: Force on vehicle in North direction + // @Field: VFE: Force on vehicle in East direction + // @Field: VFD: Force on vehicle in Down direction + AP::logger().WriteStreaming("TETH", + "TimeUS,Len,VFN,VFE,VFD", // labels + "s----", // units + "F----", // multipliers + "Qffff", // format + AP_HAL::micros64(), + (float)tether_length, + (double)veh_forces_ef.x, + (double)veh_forces_ef.y, + (double)veh_forces_ef.z); +#endif +} +// returns true on success and fills in tether_ground_loc argument, false on failure +bool TetherSim::get_tether_ground_location(Location& tether_ground_loc) const +{ + // get EKF origin + auto *sitl = AP::sitl(); + if (sitl == nullptr) { + return false; + } + const Location ekf_origin = sitl->state.home; + if (ekf_origin.lat == 0 && ekf_origin.lng == 0) { + return false; + } + + tether_ground_loc = ekf_origin; + return true; +} + +void TetherSim::update_tether_force(const Location& veh_pos, float dt) +{ + + Location tether_anchor_loc; + if (!get_tether_ground_location(tether_anchor_loc)) { + return; + } + + // Step 1: Calculate the vector from the tether anchor to the vehicle + Vector3f tether_vector = veh_pos.get_distance_NED(tether_anchor_loc); + tether_length = tether_vector.length(); + + // Step 2: Check if tether is taut (length exceeds maximum allowed length) or stuck + if (tether_length > max_line_length) { + + // Calculate the stretch beyond the maximum length + float stretch = MAX(tether_length - max_line_length, 0.0f); + + // Apply a spring-like penalty force proportional to the stretch + float penalty_force_magnitude = tether_spring_constant * stretch; + + // Direction of force is along the tether, pulling toward the anchor + veh_forces_ef = tether_vector.normalized() * penalty_force_magnitude; + + GCS_SEND_TEXT(MAV_SEVERITY_WARNING, "Tether: Exceeded maximum length."); + + return; + } + + if (tether_stuck) { + + // Calculate the stretch beyond the maximum length + float stretch = MAX(tether_length - tether_not_stuck_length, 0.0f); + + // Apply a spring-like penalty force proportional to the stretch + float penalty_force_magnitude = tether_spring_constant * stretch; + + // Direction of force is directly along the tether, towards the tether anchor point + veh_forces_ef = tether_vector.normalized() * penalty_force_magnitude; + + GCS_SEND_TEXT(MAV_SEVERITY_WARNING, "Tether: Stuck."); + + return; + } else { + tether_not_stuck_length = tether_length; + } + + // Step 3: Calculate the weight of the tether being lifted + // The weight is proportional to the current tether length + const float tether_weight_force = tether_linear_density * tether_length * GRAVITY_MSS; + + // Step 4: Calculate the tension force + Vector3f tension_force_NED = tether_vector.normalized() * tether_weight_force; + + // Step 5: Apply the force to the vehicle + veh_forces_ef = tension_force_NED; +} + +#endif diff --git a/libraries/SITL/SIM_Tether.h b/libraries/SITL/SIM_Tether.h new file mode 100644 index 0000000000000..4a7366e1089cd --- /dev/null +++ b/libraries/SITL/SIM_Tether.h @@ -0,0 +1,95 @@ +/* + This program is free software: you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation, either version 3 of the License, or + (at your option) any later version. + + This program 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 General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program. If not, see . + */ + +/* + Simulate a tethered vehicle. + Models the forces exerted by the tether and reports them to the vehicle simulation. The dynamics are not accurate but provide a very rough approximation intended to test a "stuck tether". + */ + +#pragma once + +#include "SIM_config.h" + +#if AP_SIM_TETHER_ENABLED + +#include +#include +#include +#include + +namespace SITL { + +// TetherSim simulates a tethered to a vehicle +class TetherSim { +public: + // constructor + TetherSim(); + + // Update the tether simulation state using the vehicle's earth-frame position + void update(const Location& veh_pos); + + // Retrieve earth-frame forces on the vehicle due to the tether + // Returns true on success and fills in the forces_ef argument; false on failure + bool get_forces_on_vehicle(Vector3f& forces_ef) const; + + // Parameter table for configuration + static const struct AP_Param::GroupInfo var_info[]; + +private: + // Parameters + AP_Int8 enable; // Enable or disable the tether simulation + AP_Float tether_linear_density; // Linear mass density of the tether in kg/m + AP_Float max_line_length; // Maximum allowed tether length in meters + AP_Int8 sys_id; // MAVLink system ID for GCS reporting + AP_Int8 tether_stuck; // Set to 1 to simulate a stuck tether + AP_Float tether_spring_constant; // Spring constant for modeling tether stretch + + // Send MAVLink messages to the GCS + void send_report(); + + // Write tether simulation state to onboard log + void write_log(); + + // Retrieve the location of the tether anchor point on the ground + // Returns true on success and fills in the tether_anchor_loc argument; false on failure + bool get_tether_ground_location(Location& tether_anchor_loc) const; + + // Update forces exerted by the tether based on the vehicle's position + // Takes the vehicle's position and the simulation time step (dt) as inputs + void update_tether_force(const Location& veh_pos, float dt); + + // Socket connection variables for MAVLink communication + const char *target_address = "127.0.0.1"; // Address for MAVLink socket communication + const uint16_t target_port = 5763; // Port for MAVLink socket communication + SocketAPM_native mav_socket { false }; // Socket for MAVLink communication + bool initialised; // True if the simulation class is initialized + uint32_t last_update_us; // Timestamp of the last update in microseconds + + // MAVLink reporting variables + const float reporting_period_ms = 100; // Reporting period in milliseconds + uint32_t last_report_ms; // Timestamp of the last MAVLink report sent to GCS + uint32_t last_heartbeat_ms; // Timestamp of the last MAVLink heartbeat sent to GCS + bool mavlink_connected; // True if MAVLink connection is established + mavlink_status_t mav_status; // Status of MAVLink communication + + // Tether simulation variables + Vector3f veh_forces_ef; // Earth-frame forces on the vehicle due to the tether + float tether_length = 0.0f; // Current tether length in meters + float tether_not_stuck_length = 0.0f; // Tether length when the tether is not stuck +}; + +} // namespace SITL + +#endif // AP_SIM_TETHER_ENABLED diff --git a/libraries/SITL/SIM_config.h b/libraries/SITL/SIM_config.h index d417577249278..12fca3b30789c 100644 --- a/libraries/SITL/SIM_config.h +++ b/libraries/SITL/SIM_config.h @@ -57,6 +57,10 @@ #define AP_SIM_SLUNGPAYLOAD_ENABLED (CONFIG_HAL_BOARD == HAL_BOARD_SITL) #endif +#ifndef AP_SIM_TETHER_ENABLED +#define AP_SIM_TETHER_ENABLED (CONFIG_HAL_BOARD == HAL_BOARD_SITL) +#endif + #ifndef AP_SIM_FLIGHTAXIS_ENABLED #define AP_SIM_FLIGHTAXIS_ENABLED (CONFIG_HAL_BOARD == HAL_BOARD_SITL) #endif diff --git a/libraries/SITL/SITL.cpp b/libraries/SITL/SITL.cpp index e329ed232a960..03ba5f1771718 100644 --- a/libraries/SITL/SITL.cpp +++ b/libraries/SITL/SITL.cpp @@ -1367,6 +1367,12 @@ const AP_Param::GroupInfo SIM::ModelParm::var_info[] = { AP_SUBGROUPPTR(flightaxis_ptr, "RFL_", 5, SIM::ModelParm, FlightAxis), #endif +#if AP_SIM_TETHER_ENABLED + // @Group: TETH_ + // @Path: ./SIM_Tether.cpp + AP_SUBGROUPINFO(tether_sim, "TETH_", 6, SIM::ModelParm, TetherSim), +#endif + AP_GROUPEND }; diff --git a/libraries/SITL/SITL.h b/libraries/SITL/SITL.h index 4edda5efd21e4..54d935043718a 100644 --- a/libraries/SITL/SITL.h +++ b/libraries/SITL/SITL.h @@ -28,6 +28,7 @@ #include "SIM_IntelligentEnergy24.h" #include "SIM_Ship.h" #include "SIM_SlungPayload.h" +#include "SIM_Tether.h" #include "SIM_GPS.h" #include "SIM_DroneCANDevice.h" #include "SIM_ADSB_Sagetech_MXS.h" @@ -338,6 +339,9 @@ class SIM { #if AP_SIM_SLUNGPAYLOAD_ENABLED SlungPayloadSim slung_payload_sim; #endif +#if AP_SIM_TETHER_ENABLED + TetherSim tether_sim; +#endif #if AP_SIM_FLIGHTAXIS_ENABLED FlightAxis *flightaxis_ptr; #endif