You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
			
				
					83 lines
				
				2.9 KiB
			
		
		
			
		
	
	
					83 lines
				
				2.9 KiB
			| 
											3 days ago
										 | import numpy as np
 | ||
|  | 
 | ||
|  | from opendbc.can import CANPacker
 | ||
|  | from opendbc.car import Bus, DT_CTRL
 | ||
|  | from opendbc.car.common.pid import PIDController
 | ||
|  | from opendbc.car.body import bodycan
 | ||
|  | from opendbc.car.body.values import SPEED_FROM_RPM
 | ||
|  | from opendbc.car.interfaces import CarControllerBase
 | ||
|  | 
 | ||
|  | MAX_TORQUE = 500
 | ||
|  | MAX_TORQUE_RATE = 50
 | ||
|  | MAX_ANGLE_ERROR = np.radians(7)
 | ||
|  | MAX_POS_INTEGRATOR = 0.2   # meters
 | ||
|  | MAX_TURN_INTEGRATOR = 0.1  # meters
 | ||
|  | 
 | ||
|  | 
 | ||
|  | class CarController(CarControllerBase):
 | ||
|  |   def __init__(self, dbc_names, CP):
 | ||
|  |     super().__init__(dbc_names, CP)
 | ||
|  |     self.packer = CANPacker(dbc_names[Bus.main])
 | ||
|  | 
 | ||
|  |     # PIDs
 | ||
|  |     self.turn_pid = PIDController(110, k_i=11.5, rate=1 / DT_CTRL)
 | ||
|  |     self.wheeled_speed_pid = PIDController(110, k_i=11.5, rate=1 / DT_CTRL)
 | ||
|  | 
 | ||
|  |     self.torque_r_filtered = 0.
 | ||
|  |     self.torque_l_filtered = 0.
 | ||
|  | 
 | ||
|  |   @staticmethod
 | ||
|  |   def deadband_filter(torque, deadband):
 | ||
|  |     if torque > 0:
 | ||
|  |       torque += deadband
 | ||
|  |     else:
 | ||
|  |       torque -= deadband
 | ||
|  |     return torque
 | ||
|  | 
 | ||
|  |   def update(self, CC, CS, now_nanos):
 | ||
|  | 
 | ||
|  |     torque_l = 0
 | ||
|  |     torque_r = 0
 | ||
|  | 
 | ||
|  |     if CC.enabled:
 | ||
|  |       # Read these from the joystick
 | ||
|  |       # TODO: this isn't acceleration, okay?
 | ||
|  |       speed_desired = CC.actuators.accel / 5.
 | ||
|  |       speed_diff_desired = -CC.actuators.torque / 2.
 | ||
|  | 
 | ||
|  |       speed_measured = SPEED_FROM_RPM * (CS.out.wheelSpeeds.fl + CS.out.wheelSpeeds.fr) / 2.
 | ||
|  |       speed_error = speed_desired - speed_measured
 | ||
|  | 
 | ||
|  |       torque = self.wheeled_speed_pid.update(speed_error, freeze_integrator=False)
 | ||
|  | 
 | ||
|  |       speed_diff_measured = SPEED_FROM_RPM * (CS.out.wheelSpeeds.fl - CS.out.wheelSpeeds.fr)
 | ||
|  |       turn_error = speed_diff_measured - speed_diff_desired
 | ||
|  |       freeze_integrator = ((turn_error < 0 and self.turn_pid.error_integral <= -MAX_TURN_INTEGRATOR) or
 | ||
|  |                            (turn_error > 0 and self.turn_pid.error_integral >= MAX_TURN_INTEGRATOR))
 | ||
|  |       torque_diff = self.turn_pid.update(turn_error, freeze_integrator=freeze_integrator)
 | ||
|  | 
 | ||
|  |       # Combine 2 PIDs outputs
 | ||
|  |       torque_r = torque + torque_diff
 | ||
|  |       torque_l = torque - torque_diff
 | ||
|  | 
 | ||
|  |       # Torque rate limits
 | ||
|  |       self.torque_r_filtered = np.clip(self.deadband_filter(torque_r, 10),
 | ||
|  |                                        self.torque_r_filtered - MAX_TORQUE_RATE,
 | ||
|  |                                        self.torque_r_filtered + MAX_TORQUE_RATE)
 | ||
|  |       self.torque_l_filtered = np.clip(self.deadband_filter(torque_l, 10),
 | ||
|  |                                        self.torque_l_filtered - MAX_TORQUE_RATE,
 | ||
|  |                                        self.torque_l_filtered + MAX_TORQUE_RATE)
 | ||
|  |       torque_r = int(np.clip(self.torque_r_filtered, -MAX_TORQUE, MAX_TORQUE))
 | ||
|  |       torque_l = int(np.clip(self.torque_l_filtered, -MAX_TORQUE, MAX_TORQUE))
 | ||
|  | 
 | ||
|  |     can_sends = []
 | ||
|  |     can_sends.append(bodycan.create_control(self.packer, torque_l, torque_r))
 | ||
|  | 
 | ||
|  |     new_actuators = CC.actuators.as_builder()
 | ||
|  |     new_actuators.accel = torque_l
 | ||
|  |     new_actuators.torque = torque_r
 | ||
|  |     new_actuators.torqueOutputCan = torque_r
 | ||
|  | 
 | ||
|  |     self.frame += 1
 | ||
|  |     return new_actuators, can_sends
 |