diff --git a/libraries/AP_Scripting/applets/Bendy_ruler.lua b/libraries/AP_Scripting/applets/Bendy_ruler.lua new file mode 100644 index 0000000000000..1265c59d3bf0c --- /dev/null +++ b/libraries/AP_Scripting/applets/Bendy_ruler.lua @@ -0,0 +1,896 @@ +-- MAV_COLLISION_THREAT_LEVEL +MAV_COLLISION_THREAT_LEVEL = { + NONE = 0, -- Not a threat + LOW = 1, -- Mild concern about this threat + HIGH = 2, -- Craft is panicking and may take action to avoid + ENUM_END = 3 -- End of enum +} +-- MAV_COLLISION_SRC +MAV_COLLISION_SRC = { + ADSB = 0, -- Source is ADSB_VEHICLE packets + MAVLINK_GPS_GLOBAL_INT = 1, -- Source is MAVLink GPS_GLOBAL_INT + ENUM_END = 2 -- End of enum +} +MAV_COLLISION_ACTION={ + NONE=0, -- Ignore any potential collisions + REPORT=1, -- Report potential collision + ASCEND_OR_DESCEND=2, -- Ascend or Descend to avoid threat + MOVE_HORIZONTALLY=3, -- Move horizontally to avoid threat + MOVE_PERPENDICULAR=4, -- Aircraft to move perpendicular to the collision's velocity vector + RTL=5, -- Aircraft to fly directly back to its launch point + HOVER=6, -- Aircraft to stop in place + } ; + + +local gcs_threat = { + src = nil, + src_id = 0, + timestamp_ms = 0, + _location = nil, + _velocity = nil, + threat_level = nil, + closest_approach_xy = 0.0, + closest_approach_z = 0.0, + time_to_closest_approach = 0.0, + distance_to_closest_approach = 0.0 +} + + +local avoidance_request = { + current_loc = nil, + target_loc = nil, + groundspeed = 0.0, + airspeed = 0.0, + wind_dir_rad = 0.0, + wind_speed = 0.0, + request_time_ms = 0 +} +local avoid_req2 = { + current_loc = nil, + target_loc = nil, + groundspeed = 0.0, + airspeed = 0.0, + wind_dir_rad = 0.0, + wind_speed = 0.0, + request_time_ms = 0 +} +local avoidance_result = { + target_loc = nil, + new_target_loc = nil, + result_time_ms = 0, + avoidance_needed = nil +} + +local PARAM_TABLE_KEY = 7 +local PARAM_TABLE_PREFIX = "BENDY_" +local OPTION_IGNORE_HEIGHT=1 +local ALT_FRAME_ABSOLUTE = 0 + +-- bind a parameter to a variable +function bind_param(name) + local p = Parameter() + assert(p:init(name), string.format('could not find %s parameter', name)) + return p +end + +-- add a parameter and bind it to a variable +function bind_add_param(name, idx, default_value) + assert(param:add_param(PARAM_TABLE_KEY, idx, name, default_value), string.format('could not add param %s', name)) + return bind_param(PARAM_TABLE_PREFIX .. name) +end + +-- setup SHIP specific parameters +assert(param:add_table(PARAM_TABLE_KEY, PARAM_TABLE_PREFIX, 7), 'could not add param table') +--[[ + // @Param: BENDY_OPTIONS + // @DisplayName: ADS-B avoidance options + // @Description: Bitmask of behaviour options + // @Values: 0:Disabled,1:Enabled + // @User: Advanced +--]] +BENDY_OPTIONS = bind_add_param('OPTIONS', 1, 1) + +--[[ + // @Param: BENDY_MARGIN_FENCE + // @DisplayName: fence margin + // @Description: Avoidance margin for fence + // @User: Advanced +--]] +BENDY_MARGIN_FENCE = bind_add_param('MARGIN_FENCE', 2, 50) + +--[[ + // @Param: BENDY_MARGIN_DYN + // @DisplayName: dynamic margin + // @Description: Avoidance margin for dynamic objects + // @User: Advanced +--]] +BENDY_MARGIN_DYN = bind_add_param('MARGIN_DYN', 3, 20) + +--[[ + // @Param: BENDY_MARGIN_EXCL + // @DisplayName: exclusion zone margin + // @Description: Avoidance margin for exclusion zones + // @User: Advanced +--]] +BENDY_MARGIN_EXCL = bind_add_param('MARGIN_EXCL', 4, 20) + +--[[ + // @Param: BENDY_MARGIN_WIDE + // @DisplayName: wide avoidance margin + // @Description: Avoidance margin for wide avoidance + // @User: Advanced +--]] +BENDY_MARGIN_WIDE = bind_add_param('MARGIN_WIDE', 5, 30) + +--[[ + // @Param: BENDY_MARGIN_HGT + // @DisplayName: height avoidance margin + // @Description: Avoidance margin for height avoidance + // @User: Advanced +--]] +BENDY_MARGIN_HGT = bind_add_param('MARGIN_HGT', 6, 60) + +--[[ + // @Param: BENDY_LOOKAHEAD + // @DisplayName: avoidance lookahead distance + // @Description: Avoidance lookahead distance + // @Units: metres + // @User: Advanced +--]] +BENDY_LOOKAHEAD = bind_add_param('LOOKAHEAD', 7, 500) + +WARN_DIST_XY = bind_param("AVD_W_DIST_XY") +WARN_ACTION = bind_param("AVD_W_ACTION") +local warn_act= WARN_ACTION:get() +AVD_ENABLE = bind_param("AVD_ENABLE") +local enabled=AVD_ENABLE:get() +ROLL_LIMIT_DEG = bind_param("ROLL_LIMIT_DEG") +local roll_limit_cd=ROLL_LIMIT_DEG:get() *100 +local lookahead_param=BENDY_LOOKAHEAD:get() +local margin_fence=BENDY_MARGIN_FENCE:get() +local enabled=AVD_ENABLE:get() +local current_loc=Location() +local target_loc=Location() +local thread_created = false +local current_lookahead =0 +local obstacle_count +local collision_detected +local GRAVITY_MSS=9.80665 +local LOCATION_SCALING_FACTOR_INV=89.83204953368922 +local new_target_loc = Location() +local current_target_s +local flag=0 + +--Auxiliary functions +function wrap_180(angle) + local res = wrap_360(angle) + if res > 180 then + res = res - 360 + end + return res +end + +function wrap_360(angle) + local res = math.fmod(angle, 360.0) + if res < 0 then + res = res + 360.0 + end + return res + end + +function length_squared(w) + return (w:x() * w:x()) + (w:y() * w:y()) +end + +function dot_product_2vector(p,w) + return (w:x() * p:x()) + (w:y() * p:y()) +end + +function closest_point(p, w) + local l2 = length_squared(w) + if l2 < 1e-6 then + return w -- case v == w + end + local t = dot_product_2vector(p,w) / l2 + if t <= 0 then + local Vector2=Vector2f() + Vector2:x(0) + Vector2:y(0) + return Vector2 + elseif t >= 1 then + return w + else + w:x(w:x()*t) + w:y(w:y()*t) + return w + end +end + +function closest_distance_between_radial_and_point(w, p) + local closest = closest_point(p, w) + return math.sqrt(length_squared(closest - p)) +end + +function closest_approach_xy(my_loc, my_vel, obstacle_loc, obstacle_vel, time_horizon) + local delta_vel_ne = Vector2f() + delta_vel_ne:x(obstacle_vel:x() - my_vel:x()) + delta_vel_ne:y(obstacle_vel:y() - my_vel:y()) + local delta_pos_ne = obstacle_loc:get_distance_NE(my_loc) + local line_segment_ne =Vector2f() + line_segment_ne:x(delta_vel_ne:x() * time_horizon) + line_segment_ne:y(delta_vel_ne:y() * time_horizon) + local ret = closest_distance_between_radial_and_point(line_segment_ne, delta_pos_ne) + --print(string.format(" time_horizon: (%d)", time_horizon)) + --print(string.format(" delta pos: (y=%f, x=%f)", delta_pos_ne.x, delta_pos_ne.y)) + --print(string.format(" delta vel: (y=%f, x=%f)", delta_vel_ne.x, delta_vel_ne.y)) + --print(string.format(" line segment: (y=%f, x=%f)", line_segment_ne.x, line_segment_ne.y)) + --print(string.format(" closest: (%f)", ret)) + return ret +end + +function longitude_scale(lat) + local DEG_TO_RAD = math.pi / 180 + local scale = math.cos(lat * (1.0e-7 * DEG_TO_RAD)) + return math.max(scale, 0.01) +end + +function limit_latitude(lat) + if lat > 900000000 then + lat = 1800000000 - lat + elseif lat < -900000000 then + lat = -(1800000000 + lat) + end + return lat +end + +function wrap_longitude(lon) + if lon > 1800000000 then + lon = lon - 3600000000 + elseif lon < -1800000000 then + lon = lon + 3600000000 + end + return lon +end + + -- Extrapolate latitude/longitude given bearing and distance + function offset_bearing(lat, lng, bearing_deg, distance) + local radians = math.rad + local ofs_north = math.cos(math.rad(bearing_deg)) * distance + local ofs_east = math.sin(math.rad(bearing_deg)) * distance + local dlat = ofs_north * LOCATION_SCALING_FACTOR_INV + local dlng = (ofs_east * LOCATION_SCALING_FACTOR_INV) / longitude_scale(lat + dlat / 2) + lat = lat + dlat + lat = limit_latitude(lat) + lng = wrap_longitude(dlng + lng) + return lat, lng +end + +function location_project(loc1, bearing_deg, distance) + -- Create a copy of the location + local loc2 = Location() + --loc2=loc1 + --loc2:offset_bearing(bearing_deg, distance) + local lat, lon=offset_bearing(loc1:lat(), loc1:lng(), bearing_deg, distance) + loc2:lat(lat) + loc2:lng(lon) + loc2:alt(loc1:alt()) + --gcs:send_text(0, string.format("IN: Lat: %.2f, Lon: %.2f", loc1:lat(),loc1:lng())) + --gcs:send_text(0, string.format("Dist: %.2f, Bear: %.2f", distance,bearing_deg)) + --gcs:send_text(0, string.format("Out: Lat: %.2f, Lon: %.2f", loc2:lat(),loc2:lng())) + return loc2 +end + +--get the avoidance radius in meters of a given obstacle type +function get_avoidance_radius(obstacle) + src_id=obstacle.src_id + if src_id < 20000 then + -- fixed wing, 300m radius + return 300 + elseif src_id < 30000 then + -- weather, radius 150 at ground, 300m at 3000m, 173m at 1500ft + return 173 + elseif src_id < 40000 then + -- migratory bird, 100m + return 100 + elseif src_id < 50000 then + -- bird of prey, 200m + return 200 + else + -- default to 300, which is worst case + return 300 + end +end + +--check if we are within the height range to need to avoid an obstacle +function within_avoidance_height(obstacle,margin, deltat) + local src_id=obstacle.src_id + local loc=obstacle.location + local velocity=obstacle.velocity + + + if BENDY_OPTIONS:get()==1 and OPTION_IGNORE_HEIGHT==1 then + return true + end + + if src_id >= 20000 and src_id < 30000 then + --weather, always avoid + gcs:send_text(0, "Weather Alt") + return true + end + local alt_cm + + if not loc:change_alt_frame(ALT_FRAME_ABSOLUTE) then + return true + else + alt_cm=loc:alt() + end + local obstacle_alt = alt_cm * 0.01 + local alt_min, alt_max + + if src_id < 20000 or (src_id >= 30000 and src_id < 40000) then + -- fixed wing or migrating bird, height range 150m, deltat seconds of height change + alt_min = obstacle_alt - (75 + margin) + alt_max = obstacle_alt + (75 + margin) + else + -- bird of prey, from location to ground + alt_max = obstacle_alt + margin + alt_min = -10000 + end + -- note that velocity is NED + if velocity:z() < 0 then + alt_max = alt_max - (deltat * velocity:z()) + else + alt_min = alt_min - (deltat * velocity:z()) + end + local myloc = Location() + local myloc = ahrs:get_position() + if not myloc:change_alt_frame(ALT_FRAME_ABSOLUTE) then + return true + else + alt_cm=myloc:alt() + end + local alt = alt_cm * 0.01 + -- are we in the range of avoidance heights? + return alt > alt_min and alt < alt_max +end + +--get height difference between an obstacle and our location + function obstacle_height_difference(obstacle) + location=obstacle.location + local alt1_cm=0; + location:change_alt_frame(ALT_FRAME_ABSOLUTE) + alt1_cm=location:alt() + + local alt2_cm=0; + local myloc = ahrs:get_position() + myloc:change_alt_frame(ALT_FRAME_ABSOLUTE) + alt2_cm=myloc:alt() + return (alt1_cm - alt2_cm) * 0.01; + end + + function mission_clear(current_loc, xy_clearance, z_clearance, time_s) + if enabled==0 or warn_act ~= 2 then + gcs:send_text(0, string.format("AVD_ENABLE %d: ,WARN_ACTION: %d", enabled, warn_act)) + gcs:send_text(0, "Params not enabled for clear checking") + return true + end + local timeout_ms = 5000 + local now = millis() + --assume we are not moving + + local my_vel = Vector3f() + my_vel:x(0) + my_vel:y(0) + my_vel:z(0) + local obs_count = 0 + local closest_idx = -1 + local closest_dist = 5000 + local closest_radius = 0 + obstacle_count=avoid:num_obstacles() + if obstacle_count == 0 then + --gcs:send_text(0, "No obstacles detected") + else + --gcs:send_text(0, string.format("Num of Obstacle: %d", obstacle_count)) + end + for i = 0, obstacle_count-1 do + local obstacle = {} + obstacle.timestamp_ms=avoid:get_obstacle_time(i) + obstacle.velocity=avoid:get_obstacle_vel(i) + obstacle.location=avoid:get_obstacle_loc(i) + obstacle.src_id=avoid:get_obstacle_id(i) + if (now - obstacle.timestamp_ms) > timeout_ms then + goto continue + end + obs_count = obs_count + 1 + + if not within_avoidance_height(obstacle, z_clearance, time_s) then + goto continue + end + -- get updated obstacle position + local obstacle_loc =Location() + obstacle_loc = obstacle.location + local obstacle_velocity = Vector2f() + obstacle_velocity:x(obstacle.velocity:x()) + obstacle_velocity:y(obstacle.velocity:y()) + local dt = (now - obstacle.timestamp_ms):tofloat() * 0.001 + obstacle_loc:offset(obstacle_velocity:x() * dt, obstacle_velocity:y() * dt) + + local closest_xy = closest_approach_xy(current_loc, my_vel, obstacle_loc, obstacle.velocity, time_s) + local radius = get_avoidance_radius(obstacle) + + if closest_xy < closest_dist then + closest_dist = closest_xy + closest_idx = i + closest_radius = radius + end + ::continue:: + end + + gcs_threat.src = obs_count + gcs_threat.closest_approach_xy = closest_dist - closest_radius + gcs_threat.closest_approach_z = 0 + gcs_threat.threat_level = MAV_COLLISION_THREAT_LEVEL.NONE + gcs_action = MAV_COLLISION_ACTION.NONE + if closest_dist < (xy_clearance + closest_radius) then + -- it could come within the radius in the given time + local obstacle = {} + obstacle.timestamp_ms=avoid:get_obstacle_time(closest_idx) + obstacle.velocity=avoid:get_obstacle_vel(closest_idx) + obstacle.location=avoid:get_obstacle_loc(closest_idx) + obstacle.src_id=avoid:get_obstacle_id(closest_idx) + gcs_threat.src_id = obstacle.src_id + gcs_threat.threat_level = 2 -- MAV_COLLISION_THREAT_LEVEL_HIGH + gcs_threat.time_to_closest_approach = 0 + gcs_threat.closest_approach_z = obstacle_height_difference(obstacle) + gcs_action = MAV_COLLISION_ACTION.REPORT + return false + end + -- All clear + return true + +end + +function mission_avoidance_margin(our_loc, our_velocity, avoid_sec) + local timeout_ms = 5000 + local margin = math.max(BENDY_MARGIN_WIDE:get(), WARN_DIST_XY:get()) + local num_outside_height_range = 0 + local num_timed_out = 0 + local closest_dist = 0 + local closest_id = 0 + gcs_threat.closest_approach_z = 1000 + local obs_count = 0 + for i = 0, obstacle_count-1 do + -- obstacles can update via MAVLink while we are calculating + -- avoidance margins + local obstacle = {} + obstacle.timestamp_ms=avoid:get_obstacle_time(i) + obstacle.velocity=avoid:get_obstacle_vel(i) + obstacle.location=avoid:get_obstacle_loc(i) + obstacle.src_id=avoid:get_obstacle_id(i) + + local now =millis() + if now - obstacle.timestamp_ms > timeout_ms then + num_timed_out = num_timed_out + 1 + goto continue + end + obs_count = obs_count + 1 + if not within_avoidance_height(obstacle, BENDY_MARGIN_HGT :get(), avoid_sec) then + num_outside_height_range = num_outside_height_range + 1 + goto continue + end + -- use our starting point as origin + local obstacle_position = our_loc:get_distance_NE(obstacle.location) + -- update obstacle position by delta time since we logged its position + local obstacle_velocity = Vector2f() + obstacle_velocity:x(obstacle.velocity:x()) + obstacle_velocity:y(obstacle.velocity:y()) + local dt = (now - obstacle.timestamp_ms):tofloat() * 0.001 + obstacle_position:x(obstacle_position:x() + (obstacle_velocity:x() * dt)) + obstacle_position:y(obstacle_position:y() + (obstacle_velocity:y() * dt)) + -- get our velocity relative to obstacle + local relative_velocity = our_velocity - obstacle_velocity + local final_pos = Vector2f() + final_pos:x(relative_velocity:x() * avoid_sec) + final_pos:y(relative_velocity:y() * avoid_sec) + -- lookup the min distance to keep from the object + local radius = get_avoidance_radius(obstacle) + -- assume that messages about aircraft position could be up to 2s old when we get them + local position_lag = 2.0 + local position_error = position_lag * obstacle_velocity:length() + local dist = closest_distance_between_radial_and_point(final_pos, obstacle_position) - (radius + position_error) + dist = dist - BENDY_MARGIN_DYN:get() + --gcs:send_text(0, string.format("distance:%f, margin: %d",dist, margin)) + if dist < margin then + margin = dist + closest_dist = dist + closest_id = i + end + ::continue:: + end + if closest_dist > 0 then + -- update threat report for GCS + local obstacle = {} + obstacle.timestamp_ms=avoid:get_obstacle_time(closest_id) + obstacle.velocity=avoid:get_obstacle_vel(closest_id) + obstacle.location=avoid:get_obstacle_loc(closest_id) + obstacle.src_id=avoid:get_obstacle_id(closest_id) + + gcs_threat.src = obs_count + gcs_threat.src_id = obstacle.src_id + gcs_threat.threat_level = MAV_COLLISION_THREAT_LEVEL_LOW + gcs_threat.time_to_closest_approach = 0 + gcs_threat.closest_approach_xy = closest_dist + BENDY_MARGIN_DYN:get() + gcs_threat.closest_approach_z = obstacle_height_difference(obstacle) + end + --gcs:send_text(0, string.format("to:%d oh:%d t:%d", num_timed_out, num_outside_height_range, obstacle_count)) + --gcs:send_text(0, string.format("margin: %f",margin)) + return margin +end + +function calc_avoidance_margin(loc1, loc2, our_velocity, avoid_sec) + -- By projecting 1m along the line we avoid a problem with the + -- exclusion avoidance being happy to skirt along a line parallel + -- to an exclusion zone + local bearing_deg = math.deg(loc1:get_bearing(loc2)) + local loc1_shifted = location_project(loc1, bearing_deg, 1) + local obs_margin = mission_avoidance_margin(loc1_shifted, our_velocity, avoid_sec) + return obs_margin +end + +function effective_groundspeed(airspeed, bearing_deg, wind_dir_rad, wind_speed) + -- Ensure airspeed is at least 1.0 + airspeed = math.max(airspeed, 1.0) + -- Convert bearing to radians + local bearing_rad = math.rad(bearing_deg) + -- Calculate the angle between wind direction and bearing + local temp = math.pi - (wind_dir_rad - bearing_rad) + local dangle = wind_speed * math.sin(temp) / airspeed + -- If dangle is out of valid range, return 0 + if dangle > 1.0 or dangle < -1.0 then + return 0 + end + -- Calculate the angle alpha using arcsine + local alpha = math.asin(dangle) + -- Calculate yaw + local yaw = bearing_rad - alpha + -- Calculate beta, the angle between wind direction and yaw + local beta = math.pi - (wind_dir_rad - yaw) + -- Calculate ground speed squared (gs2) + local gs2 = airspeed^2 + wind_speed^2 - 2 * airspeed * wind_speed * math.cos(beta) + -- If gs2 is negative or zero, return 0 + if gs2 <= 0 then + return 0 + end + -- Calculate the final ground speed + local gs = math.sqrt(gs2) + --gcs:send_text(0, string.format("as:%.1f bear:%.1f wind_dir:%.1f ws:%.1f -> gs:%.1f", airspeed, bearing_deg, math.deg(wind_dir_rad), wind_speed, gs)) + return gs +end + +function have_collided(current_loc) + local timeout_ms = 5000 + local ret = false + local now = millis() + + for i = 0, obstacle_count-1 do + local obstacle = {} + obstacle.timestamp_ms=avoid:get_obstacle_time(i) + obstacle.velocity=avoid:get_obstacle_vel(i) + obstacle.location=avoid:get_obstacle_loc(i) + obstacle.src_id=avoid:get_obstacle_id(i) + -- If the obstacle data is too old, skip it + if now - obstacle.timestamp_ms > timeout_ms then + goto continue + end + -- Check if the obstacle is within the avoidance height + if not within_avoidance_height(obstacle, 0, 0) then + goto continue + end + -- Get updated obstacle position + local obstacle_loc = avoid:get_obstacle_loc(i) + local obstacle_velocity = Vector2f() + obstacle_velocity:x(obstacle.velocity:x()) + obstacle_velocity:y(obstacle.velocity:y()) + local dt = (now - obstacle.timestamp_ms):tofloat() * 0.001 + -- Update the obstacle's position using its velocity + obstacle_loc:offset(obstacle_velocity:x() * dt, obstacle_velocity:y() * dt) + local radius = get_avoidance_radius(obstacle) + local distance = current_loc:get_distance(obstacle_loc) + -- If the distance to the obstacle is less than its radius, a collision is detected + if distance < radius then + --DEBUG(1, string.format("Collided with %u %.0fm", obstacle.src_id, distance)) + gcs:send_text(0, string.format("Collision Alert")) + ret = true + end + ::continue:: + end + + collision_detected = ret + return ret +end + +function update_mission_avoidance(avd, new_target_loc) + --gcs:send_text(0, "Started mission avoidance update") + local airspeed = math.max(avd.airspeed, 1.0) + local current_loc = avd.current_loc + current_lookahead = math.min(math.max(current_lookahead, lookahead_param * 0.5), lookahead_param) + --gcs:send_text(0, "got current lookahead") + local full_distance = current_loc:get_distance(new_target_loc) + -- the distance we look ahead is adjusted dynamically based on avoidance results + local avoid_step1_m = current_lookahead + local avoid_step2_m = current_lookahead * 2 + -- test for flying past the waypoint, so if we are close, we have room to dodge after the waypoint + local avoid_max = math.min(avoid_step1_m, full_distance + math.min(margin_fence / 2, 100)) + local avoid_sec1 = avoid_max / airspeed + local bearing_inc_cd = 1500 + local distance = airspeed * avoid_sec1 + local bearing_cd =math.deg(current_loc:get_bearing(new_target_loc))*100 + -- report collisions + have_collided(current_loc) + -- get the current ground course + local ground_course_deg + ground_course_deg = wrap_180(math.deg(ahrs:groundspeed_vector():angle())) + --gcs:send_text(0, string.format("ground_course_deg : %.2f",ground_course_deg )) + + -- If the full distance is less than 20m, no avoidance is needed + if full_distance < 20 then + return false + end + -- Try 5 degree increments around a circle, alternating left and right. Check each one to see if flying in that direction would avoid all obstacles. + local best_bearing = bearing_cd * 0.01 + local have_best_bearing = false + local best_margin = -10000 + local best_margin_bearing = best_bearing + local rate_of_turn_dps = math.deg(GRAVITY_MSS * math.tan(math.rad(roll_limit_cd * 0.01 * 0.6)) / (airspeed + 0.1)) + for i = 0, 360 / (bearing_inc_cd / 100) do + local bearing_delta_cd = i * bearing_inc_cd / 2 + --gcs:send_text(0, string.format("i: %d", i)) + if i % 2 == 1 then + -- Alternate between left and right of the target + bearing_delta_cd = -bearing_delta_cd + end + --gcs:send_text(0, string.format("bearing_delta_cd: %d",bearing_delta_cd)) + -- Test bearing + local bearing_test = wrap_180((bearing_cd*0.01) + (bearing_delta_cd*0.01)) + --gcs:send_text(0, string.format("bearing_cd: %.2f, bearing_delta_cd: %.2f, bearing_test: %.2f",bearing_cd, bearing_delta_cd, bearing_test)) + --gcs:send_text(0, string.format("bearing_test: %.2f",bearing_test)) + local course_change_deg = wrap_180(bearing_test - ground_course_deg) + --gcs:send_text(0, string.format("course change: %.2f",course_change_deg)) + local ground_speed = effective_groundspeed(airspeed, bearing_test, avd.wind_dir_rad, avd.wind_speed) + + if math.abs(course_change_deg) > 170 then + -- Skip 180-degree turns as we can't predict the turn direction + goto continue + end + -- Calculate how long it will take to change course + local turn_time_s = math.abs(course_change_deg / rate_of_turn_dps) + -- Approximate turn by flying forward for half of the turn time + local projected_loc = location_project(current_loc, ground_course_deg, ground_speed * turn_time_s * 0.5) + -- If turning more than 90 degrees, add sideways movement + if math.abs(course_change_deg) > 90 then + local direction = course_change_deg > 0 and (ground_course_deg + 90) or (ground_course_deg - 90) + local proportion = math.sin(math.rad(math.abs(course_change_deg) - 90)) + projected_loc = location_project(projected_loc, direction, ground_speed * proportion * turn_time_s * 0.5) + end + -- Position after one step + local loc_test = location_project(projected_loc, bearing_test, distance) + -- Calculate velocity and margin + local loc_diff = projected_loc:get_distance_NE(loc_test) + local our_velocity = Vector2f() + our_velocity:x(loc_diff:x() / avoid_sec1) + our_velocity:y(loc_diff:y() / avoid_sec1) + local margin = calc_avoidance_margin(projected_loc, loc_test, our_velocity, avoid_sec1) + --gcs:send_text(0, string.format("Mission avoidance margin: %f", margin)) + if margin > best_margin then + best_margin_bearing = bearing_test + best_margin = margin + end + --gcs:send_text(0, string.format("Mission avoidance margin: %d, Mission Wide: %d",margin,BENDY_MARGIN_WIDE:get())) + if margin > BENDY_MARGIN_WIDE:get() then + --gcs:send_text(0, string.format("Inside first if")) + -- This direction avoids all obstacles for one step. Check if it leads to a clear path for a longer distance. + if not have_best_bearing then + best_bearing = bearing_test + have_best_bearing = true + elseif math.abs(wrap_180(ground_course_deg - bearing_test)) < math.abs(wrap_180(ground_course_deg - best_bearing)) then + -- Replace with a closer direction + best_bearing = bearing_test + end + local test_bearings = { 0, 45, -45 } + local target_bearing = math.deg(loc_test:get_bearing(new_target_loc)) + for _, delta in ipairs(test_bearings) do + local new_bearing = target_bearing + delta + local target_distance2 = loc_test:get_distance(new_target_loc) + local distance2 = math.min(math.max(avoid_step2_m, 10), target_distance2) + local avoid_sec2 = distance2 / airspeed + local loc_test2 = location_project(loc_test, new_bearing, distance2) + local loc_diff2 = loc_test:get_distance_NE(loc_test2) + local our_velocity2 = Vector2f() + our_velocity2:x(loc_diff2:x() / avoid_sec2) + our_velocity2:y(loc_diff2:y() / avoid_sec2) + local margin2 = calc_avoidance_margin(loc_test, loc_test2, our_velocity2, avoid_sec2) + + if margin2 > BENDY_MARGIN_WIDE:get() then + --gcs:send_text(0, string.format("Inside second if")) + -- Project the new target in the chosen direction by the full distance + local new_loc = location_project(projected_loc, bearing_test, full_distance) + new_target_loc:lat(new_loc:lat()) + new_target_loc:lng(new_loc:lng()) + new_target_loc:alt(new_loc:alt()) + --gcs:send_text(0, string.format("M2 Lat: %.2f, M2 lon: %.2f", new_target_loc:lat(),new_target_loc:lng())) + current_lookahead = math.min(lookahead_param, current_lookahead * 1.1) + gcs_action = (i ~= 0 or delta ~= 0) and 1 or 0 + --gcs:send_text(0, string.format("delta: %d",delta)) + --gcs:send_text(0, string.format("Margin: %f, Margin2: %f", margin, margin2)) + return i ~= 0 or delta ~= 0 + end + end + --else + -- gcs:send_text(0, string.format("No first if")) + end + ::continue:: + end + -- If no good direction was found, choose the best based on margin + local chosen_bearing + if have_best_bearing then + chosen_bearing = best_bearing + current_lookahead = math.min(lookahead_param, current_lookahead * 1.05) + gcs_action = 2 + else + chosen_bearing = best_margin_bearing + current_lookahead = math.max(lookahead_param * 0.5, current_lookahead * 0.9) + gcs_action = 3 + end + -- Calculate new target location based on the best effort + local new_loc = location_project(current_loc, chosen_bearing, full_distance) + new_target_loc:lat(new_loc:lat()) + new_target_loc:lng(new_loc:lng()) + new_target_loc:alt(new_loc:alt()) + --gcs:send_text(0, string.format("Mission avoidance updated?: True")) + return true +end + + +function avoidance_thread() + -- Local variables for target location and new target location + local local_target_loc = Location() + -- Get the current time + local now = millis() + -- If the request is too old (more than 2000 ms), skip processing + if now - avoidance_request.request_time_ms > 2000 then + --gcs:send_text(0, string.format("Time passed: %.2f", (now - avoidance_request.request_time_ms):tofloat())) + return -- Skip the rest of this loop iteration + end + --gcs:send_text(0, string.format("Time passed: %.2f", (now - avoidance_request.request_time_ms):tofloat())) + -- Copy the avoidance request into a local variable + avoid_req2 = avoidance_request + local_target_loc = avoid_req2.target_loc:copy() + new_target_loc = local_target_loc:copy() + --gcs:send_text(0, string.format("Local1 Lat: %.2f, Local1 lon: %.2f", local_target_loc:lat(), local_target_loc:lng())) + -- Call the function to update the mission avoidance with the request and new target location + local res = update_mission_avoidance(avoid_req2, new_target_loc) + --[[if res then + gcs:send_text(0, string.format("res: True")) + gcs:send_text(0, string.format("NT Lat: %.2f, NT lon: %.2f", new_target_loc:lat(),new_target_loc:lng())) + else + gcs:send_text(0, string.format("res: False")) + end ]] + + -- Update the avoidance result directly without semaphore + --gcs:send_text(0, string.format("Local2 Lat: %.2f, Local2 lon: %.2f", local_target_loc:lat(), local_target_loc:lng())) + avoidance_result.target_loc = local_target_loc + avoidance_result.new_target_loc = res and new_target_loc or local_target_loc + avoidance_result.result_time_ms = millis() + avoidance_result.avoidance_needed = res +end + +function mission_avoidance(current_loc, target_loc, groundspeed) + -- Check if avoidance is disabled or warning action is not set to 2 + if enabled==0 or warn_act ~= 2 then + -- Reset threat and action if conditions aren't met + gcs_threat = {} -- Clear GCS threat data + gcs_action = 0 -- Reset GCS action + return false + end + -- Simulate thread creation for avoidance logic + avoidance_thread() + -- Get current time (milliseconds simulation) + local now = millis() -- Populate avoidance request with current data + avoidance_request.current_loc = current_loc + avoidance_request.target_loc = target_loc + avoidance_request.groundspeed = groundspeed + -- Estimate airspeed (use groundspeed if no airspeed estimate available) + if not ahrs:airspeed_estimate() then + avoidance_request.airspeed = groundspeed + else + avoidance_request.airspeed = ahrs:airspeed_estimate() + end -- Get wind estimate and convert to 2D + local wind_3d = ahrs:wind_estimate() + local wind_2d = Vector2f() + wind_2d:x(wind_3d:x()) + wind_2d:y(wind_3d:y()) + -- Calculate wind direction and speed + avoidance_request.wind_dir_rad = wind_2d:angle() + avoidance_request.wind_speed = wind_2d:length() + -- Record request time + avoidance_request.request_time_ms = now + -- Check if previous result is still valid and matches the current target location + if avoidance_result.target_loc then + --gcs:send_text(0, string.format("lat_tar: .%.2f, res_lat: %.2f", target_loc:lat(), avoidance_result.target_loc:lat())) + if target_loc:lat() == avoidance_result.target_loc:lat() and target_loc:lng() == avoidance_result.target_loc:lng() and (now - avoidance_result.result_time_ms) < 2000 then + -- Update target location with the new avoidance result location + target_loc:lat(avoidance_result.new_target_loc:lat()) + target_loc:lng(avoidance_result.new_target_loc:lng()) + -- Return whether avoidance is needed based on the result + --gcs:send_text(0, string.format("Up Lat: %.2f, Up lon: %.2f", avoidance_result.new_target_loc:lat(),avoidance_result.new_target_loc:lng())) + return avoidance_result.avoidance_needed + end + else + --gcs:send_text(0, "No calculation performed") + end + + -- Return false if no valid avoidance result is available + return false +end + +function update() + --gcs:send_text(0, "Lua running") + -- Check if the 'avoid' function is available + if avoid == nil then + gcs:send_text(0, "AP_Avoidance singleton not available") + return + end + + + current_loc = ahrs:get_position() + target_loc= vehicle:get_target_location() + local current_target + if target_loc ~=nil then + current_target=target_loc:copy() + --gcs:send_text(0, "Update_Current_target") + end + --gcs:send_text(0, string.format("Original pos: Lat: %.2f, Lon: %.2f", target_loc:lat(),target_loc:lng())) + + + if flag==0 and target_loc ~=nil then + current_target_s=target_loc:copy() + end + --gcs:send_text(0, string.format("Current_targ Latitude: %.2f, Current_targ longitude: %.2f", current_target:lat(),current_target:lng())) + local groundspeed=ahrs:groundspeed_vector():length() + if not current_loc then + gcs:send_text(0, "Not current position") + return update, 100 + else + if mission_clear(current_loc, 400, 100, 25)== true then + gcs:send_text(6, "Mission clear") + if target_loc ~=nil then + if target_loc:lng()~=current_target_s:lng() and flag==1 then + vehicle:update_target_location(current_target, current_target_s) + flag=0 + gcs:send_text(4, "Returning to path") + end + end + else + if target_loc then + --gcs:send_text(0, "Calculating mission avoidance") + if mission_avoidance(current_loc, target_loc, groundspeed)==true then + --gcs:send_text(0, "Updating target") + --vehicle:set_mode(15) + vehicle:update_target_location(current_target, target_loc) + --vehicle:set_target_location(target_loc) + --gcs:send_text(0, string.format("New pos: Lat: %.2f, Lon: %.2f", target_loc:lat(),target_loc:lng())) + gcs:send_text(4, "Updating target") + flag=1 + end + gcs:send_text(0, "Mission not clear") + + end + + + end + end + + return update, 100 +end + +return update,100