Files
StarPilot/selfdrive/controls/lib/latcontrol_pid.py
T
firestar5683 617aa0a5db honda I6
2026-05-01 14:19:28 -05:00

107 lines
4.8 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)
if abs_angle < 8.0:
return 1.0
speed_weight = min(max((v_ego - 2.0) / 8.0, 0.0), 1.0)
angle_weight = min(max((abs_angle - 8.0) / 32.0, 0.0), 1.0)
phase = desired_angle_deg * desired_angle_delta_deg
is_left = desired_angle_deg > 0.0
base_scale = 0.02 if is_left else 0.05
turn_in_scale = 0.02 if is_left else 0.03
unwind_scale = 0.00 if is_left else 0.02
scale = 1.0 + (speed_weight * angle_weight * base_scale)
if phase > 0.2:
scale += speed_weight * angle_weight * turn_in_scale
elif phase < -0.2:
scale += speed_weight * angle_weight * unwind_scale
return scale
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_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