mirror of
https://github.com/firestar5683/StarPilot.git
synced 2026-06-28 01:52:06 +08:00
139 lines
6.7 KiB
Python
139 lines
6.7 KiB
Python
import math
|
|
|
|
from cereal import log
|
|
from opendbc.car.honda.carcontroller import get_civic_bosch_modified_steering_pressed
|
|
from opendbc.car.honda.values import CAR as HONDA, HondaFlags
|
|
from openpilot.starpilot.common.testing_grounds import testing_ground
|
|
from openpilot.selfdrive.controls.lib.latcontrol import LatControl
|
|
from openpilot.common.pid import PIDController
|
|
|
|
|
|
def civic_bosch_modified_lateral_testing_ground_active() -> bool:
|
|
return testing_ground.use("8", "B")
|
|
|
|
|
|
def get_civic_bosch_modified_pid_output_scale(desired_angle_deg: float, desired_angle_delta_deg: float, v_ego: float) -> float:
|
|
abs_angle = abs(desired_angle_deg)
|
|
speed_weight = min(max((v_ego - 4.0) / 10.0, 0.0), 1.0)
|
|
center_speed_weight = 0.70 + (0.30 * speed_weight)
|
|
center_weight = min(max((18.0 - abs_angle) / 18.0, 0.0), 1.0)
|
|
mid_turn_weight = min(max((abs_angle - 10.0) / 10.0, 0.0), 1.0)
|
|
angle_weight = min(max((abs_angle - 18.0) / 10.0, 0.0), 1.0)
|
|
phase = desired_angle_deg * desired_angle_delta_deg
|
|
|
|
is_left = desired_angle_deg > 0.0
|
|
center_taper = 0.32
|
|
mid_turn_scale = 0.12 if is_left else -0.10
|
|
mid_turn_turn_in_scale = 0.08 if is_left else -0.08
|
|
mid_turn_unwind_scale = -0.05 if is_left else -0.08
|
|
base_scale = 0.12 if is_left else 0.10
|
|
turn_in_scale = 0.12 if is_left else 0.12
|
|
unwind_scale = 0.14 if is_left else 0.18
|
|
|
|
scale = 1.0 - (center_speed_weight * center_weight * center_taper)
|
|
scale += speed_weight * mid_turn_weight * mid_turn_scale
|
|
scale += speed_weight * angle_weight * base_scale
|
|
if phase > 0.2:
|
|
scale += speed_weight * mid_turn_weight * mid_turn_turn_in_scale
|
|
scale += speed_weight * angle_weight * turn_in_scale
|
|
elif phase < -0.2:
|
|
scale += speed_weight * mid_turn_weight * mid_turn_unwind_scale
|
|
scale -= speed_weight * angle_weight * unwind_scale
|
|
|
|
return max(scale, 0.70)
|
|
|
|
|
|
def get_civic_bosch_modified_pid_output_alpha(desired_angle_deg: float, desired_angle_delta_deg: float,
|
|
v_ego: float, output_torque: float, prev_output_torque: float) -> float:
|
|
abs_angle = abs(desired_angle_deg)
|
|
if abs_angle < 0.75 or abs_angle > 22.0:
|
|
return 1.0
|
|
|
|
speed_weight = min(max((v_ego - 4.0) / 10.0, 0.0), 1.0)
|
|
onset = min(max((abs_angle - 0.75) / 5.25, 0.0), 1.0)
|
|
cutoff = min(max((22.0 - abs_angle) / 8.0, 0.0), 1.0)
|
|
band_weight = onset * cutoff
|
|
small_curve_weight = min(max((12.0 - abs_angle) / 6.0, 0.0), 1.0)
|
|
large_turn_weight = min(max((abs_angle - 16.0) / 6.0, 0.0), 1.0)
|
|
transition_weight = min(abs(desired_angle_delta_deg) / 0.35, 1.0)
|
|
sign_change_weight = 1.0 if (output_torque * prev_output_torque) < 0.0 else 0.0
|
|
|
|
smoothing = band_weight * (0.36 + (0.22 * speed_weight) + (0.12 * transition_weight) + (0.12 * sign_change_weight))
|
|
smoothing *= 1.0 + (0.35 * small_curve_weight)
|
|
smoothing *= 1.0 - (0.50 * large_turn_weight)
|
|
return min(max(1.0 - smoothing, 0.14), 1.0)
|
|
|
|
|
|
class LatControlPID(LatControl):
|
|
def __init__(self, CP, CI, dt):
|
|
super().__init__(CP, CI, dt)
|
|
self.pid = PIDController((CP.lateralTuning.pid.kpBP, CP.lateralTuning.pid.kpV),
|
|
(CP.lateralTuning.pid.kiBP, CP.lateralTuning.pid.kiV),
|
|
pos_limit=self.steer_max, neg_limit=-self.steer_max)
|
|
self.ff_factor = CP.lateralTuning.pid.kf
|
|
self.get_steer_feedforward = CI.get_steer_feedforward_function()
|
|
self.is_civic_bosch_modified = CP.carFingerprint == HONDA.HONDA_CIVIC_BOSCH and bool(CP.flags & HondaFlags.EPS_MODIFIED)
|
|
self.prev_angle_steers_des_no_offset = 0.0
|
|
self.modified_civic_steering_pressed_filter_s = 0.0
|
|
self.modified_civic_steering_pressed_prev = False
|
|
self.prev_output_torque = 0.0
|
|
|
|
def update(self, active, CS, VM, params, steer_limited_by_safety, desired_curvature, curvature_limited, lat_delay, calibrated_pose, model_data, starpilot_toggles):
|
|
pid_log = log.ControlsState.LateralPIDState.new_message()
|
|
pid_log.steeringAngleDeg = float(CS.steeringAngleDeg)
|
|
pid_log.steeringRateDeg = float(CS.steeringRateDeg)
|
|
|
|
angle_steers_des_no_offset = math.degrees(VM.get_steer_from_curvature(-desired_curvature, CS.vEgo, params.roll))
|
|
angle_steers_des = angle_steers_des_no_offset + params.angleOffsetDeg
|
|
error = angle_steers_des - CS.steeringAngleDeg
|
|
|
|
pid_log.steeringAngleDesiredDeg = angle_steers_des
|
|
pid_log.angleError = error
|
|
if not active:
|
|
output_torque = 0.0
|
|
pid_log.active = False
|
|
self.prev_angle_steers_des_no_offset = angle_steers_des_no_offset
|
|
self.modified_civic_steering_pressed_filter_s = 0.0
|
|
self.modified_civic_steering_pressed_prev = False
|
|
self.prev_output_torque = 0.0
|
|
|
|
else:
|
|
# offset does not contribute to resistive torque
|
|
ff = self.ff_factor * self.get_steer_feedforward(angle_steers_des_no_offset, CS.vEgo)
|
|
steering_pressed = CS.steeringPressed
|
|
if self.is_civic_bosch_modified:
|
|
self.modified_civic_steering_pressed_filter_s, steering_pressed = get_civic_bosch_modified_steering_pressed(
|
|
bool(CS.steeringPressed),
|
|
float(getattr(CS, "steeringTorque", 0.0)),
|
|
float(self.prev_output_torque),
|
|
self.modified_civic_steering_pressed_filter_s,
|
|
self.modified_civic_steering_pressed_prev,
|
|
)
|
|
self.modified_civic_steering_pressed_prev = steering_pressed
|
|
|
|
freeze_integrator = steer_limited_by_safety or steering_pressed or CS.vEgo < 5
|
|
|
|
output_torque = self.pid.update(error,
|
|
feedforward=ff,
|
|
speed=CS.vEgo,
|
|
freeze_integrator=freeze_integrator)
|
|
|
|
if self.is_civic_bosch_modified and civic_bosch_modified_lateral_testing_ground_active():
|
|
desired_angle_delta = angle_steers_des_no_offset - self.prev_angle_steers_des_no_offset
|
|
output_torque *= get_civic_bosch_modified_pid_output_scale(angle_steers_des_no_offset, desired_angle_delta, CS.vEgo)
|
|
output_alpha = get_civic_bosch_modified_pid_output_alpha(angle_steers_des_no_offset, desired_angle_delta, CS.vEgo,
|
|
output_torque, self.prev_output_torque)
|
|
output_torque = self.prev_output_torque + (output_alpha * (output_torque - self.prev_output_torque))
|
|
output_torque = float(max(min(output_torque, self.steer_max), -self.steer_max))
|
|
|
|
pid_log.active = True
|
|
pid_log.p = float(self.pid.p)
|
|
pid_log.i = float(self.pid.i)
|
|
pid_log.f = float(self.pid.f)
|
|
pid_log.output = float(output_torque)
|
|
pid_log.saturated = bool(self._check_saturation(self.steer_max - abs(output_torque) < 1e-3, CS, steer_limited_by_safety, curvature_limited))
|
|
self.prev_angle_steers_des_no_offset = angle_steers_des_no_offset
|
|
self.prev_output_torque = float(output_torque)
|
|
|
|
return output_torque, angle_steers_des, pid_log
|