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.
		
		
		
		
			
				
					76 lines
				
				2.0 KiB
			
		
		
			
		
	
	
					76 lines
				
				2.0 KiB
			| 
											6 years ago
										 | import numpy as np
 | ||
| 
											4 years ago
										 | from numbers import Number
 | ||
|  | 
 | ||
| 
											2 years ago
										 | from openpilot.common.numpy_fast import clip, interp
 | ||
| 
											6 years ago
										 | 
 | ||
| 
											4 years ago
										 | 
 | ||
| 
											1 year ago
										 | class PIDController:
 | ||
| 
											4 years ago
										 |   def __init__(self, k_p, k_i, k_f=0., k_d=0., pos_limit=1e308, neg_limit=-1e308, rate=100):
 | ||
|  |     self._k_p = k_p
 | ||
|  |     self._k_i = k_i
 | ||
|  |     self._k_d = k_d
 | ||
| 
											4 years ago
										 |     self.k_f = k_f   # feedforward gain
 | ||
|  |     if isinstance(self._k_p, Number):
 | ||
|  |       self._k_p = [[0], [self._k_p]]
 | ||
|  |     if isinstance(self._k_i, Number):
 | ||
|  |       self._k_i = [[0], [self._k_i]]
 | ||
| 
											4 years ago
										 |     if isinstance(self._k_d, Number):
 | ||
|  |       self._k_d = [[0], [self._k_d]]
 | ||
| 
											6 years ago
										 | 
 | ||
|  |     self.pos_limit = pos_limit
 | ||
|  |     self.neg_limit = neg_limit
 | ||
|  | 
 | ||
|  |     self.i_unwind_rate = 0.3 / rate
 | ||
|  |     self.i_rate = 1.0 / rate
 | ||
| 
											4 years ago
										 |     self.speed = 0.0
 | ||
| 
											6 years ago
										 | 
 | ||
|  |     self.reset()
 | ||
|  | 
 | ||
|  |   @property
 | ||
|  |   def k_p(self):
 | ||
|  |     return interp(self.speed, self._k_p[0], self._k_p[1])
 | ||
|  | 
 | ||
|  |   @property
 | ||
|  |   def k_i(self):
 | ||
|  |     return interp(self.speed, self._k_i[0], self._k_i[1])
 | ||
|  | 
 | ||
| 
											4 years ago
										 |   @property
 | ||
|  |   def k_d(self):
 | ||
|  |     return interp(self.speed, self._k_d[0], self._k_d[1])
 | ||
|  | 
 | ||
| 
											4 years ago
										 |   @property
 | ||
|  |   def error_integral(self):
 | ||
|  |     return self.i/self.k_i
 | ||
|  | 
 | ||
| 
											6 years ago
										 |   def reset(self):
 | ||
|  |     self.p = 0.0
 | ||
|  |     self.i = 0.0
 | ||
| 
											4 years ago
										 |     self.d = 0.0
 | ||
| 
											6 years ago
										 |     self.f = 0.0
 | ||
|  |     self.control = 0
 | ||
|  | 
 | ||
| 
											4 years ago
										 |   def update(self, error, error_rate=0.0, speed=0.0, override=False, feedforward=0., freeze_integrator=False):
 | ||
| 
											6 years ago
										 |     self.speed = speed
 | ||
|  | 
 | ||
| 
											4 years ago
										 |     self.p = float(error) * self.k_p
 | ||
| 
											6 years ago
										 |     self.f = feedforward * self.k_f
 | ||
| 
											4 years ago
										 |     self.d = error_rate * self.k_d
 | ||
| 
											6 years ago
										 | 
 | ||
|  |     if override:
 | ||
|  |       self.i -= self.i_unwind_rate * float(np.sign(self.i))
 | ||
|  |     else:
 | ||
|  |       i = self.i + error * self.k_i * self.i_rate
 | ||
| 
											4 years ago
										 |       control = self.p + i + self.d + self.f
 | ||
| 
											6 years ago
										 | 
 | ||
|  |       # Update when changing i will move the control away from the limits
 | ||
|  |       # or when i will move towards the sign of the error
 | ||
| 
											5 years ago
										 |       if ((error >= 0 and (control <= self.pos_limit or i < 0.0)) or
 | ||
| 
											6 years ago
										 |           (error <= 0 and (control >= self.neg_limit or i > 0.0))) and \
 | ||
|  |          not freeze_integrator:
 | ||
|  |         self.i = i
 | ||
|  | 
 | ||
| 
											4 years ago
										 |     control = self.p + self.i + self.d + self.f
 | ||
| 
											6 years ago
										 | 
 | ||
|  |     self.control = clip(control, self.neg_limit, self.pos_limit)
 | ||
|  |     return self.control
 |