-
Notifications
You must be signed in to change notification settings - Fork 62
/
Copy pathcontroller.py
128 lines (119 loc) · 6.44 KB
/
controller.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
import numpy as np
import math
import time
import threading
class Controller_PID_Point2Point():
def __init__(self, get_state, get_time, actuate_motors, params, quad_identifier):
self.quad_identifier = quad_identifier
self.actuate_motors = actuate_motors
self.get_state = get_state
self.get_time = get_time
self.MOTOR_LIMITS = params['Motor_limits']
self.TILT_LIMITS = [(params['Tilt_limits'][0]/180.0)*3.14,(params['Tilt_limits'][1]/180.0)*3.14]
self.YAW_CONTROL_LIMITS = params['Yaw_Control_Limits']
self.Z_LIMITS = [self.MOTOR_LIMITS[0]+params['Z_XY_offset'],self.MOTOR_LIMITS[1]-params['Z_XY_offset']]
self.LINEAR_P = params['Linear_PID']['P']
self.LINEAR_I = params['Linear_PID']['I']
self.LINEAR_D = params['Linear_PID']['D']
self.LINEAR_TO_ANGULAR_SCALER = params['Linear_To_Angular_Scaler']
self.YAW_RATE_SCALER = params['Yaw_Rate_Scaler']
self.ANGULAR_P = params['Angular_PID']['P']
self.ANGULAR_I = params['Angular_PID']['I']
self.ANGULAR_D = params['Angular_PID']['D']
self.xi_term = 0
self.yi_term = 0
self.zi_term = 0
self.thetai_term = 0
self.phii_term = 0
self.gammai_term = 0
self.thread_object = None
self.target = [0,0,0]
self.yaw_target = 0.0
self.run = True
def wrap_angle(self,val):
return( ( val + np.pi) % (2 * np.pi ) - np.pi )
def update(self):
[dest_x,dest_y,dest_z] = self.target
[x,y,z,x_dot,y_dot,z_dot,theta,phi,gamma,theta_dot,phi_dot,gamma_dot] = self.get_state(self.quad_identifier)
x_error = dest_x-x
y_error = dest_y-y
z_error = dest_z-z
self.xi_term += self.LINEAR_I[0]*x_error
self.yi_term += self.LINEAR_I[1]*y_error
self.zi_term += self.LINEAR_I[2]*z_error
dest_x_dot = self.LINEAR_P[0]*(x_error) + self.LINEAR_D[0]*(-x_dot) + self.xi_term
dest_y_dot = self.LINEAR_P[1]*(y_error) + self.LINEAR_D[1]*(-y_dot) + self.yi_term
dest_z_dot = self.LINEAR_P[2]*(z_error) + self.LINEAR_D[2]*(-z_dot) + self.zi_term
throttle = np.clip(dest_z_dot,self.Z_LIMITS[0],self.Z_LIMITS[1])
dest_theta = self.LINEAR_TO_ANGULAR_SCALER[0]*(dest_x_dot*math.sin(gamma)-dest_y_dot*math.cos(gamma))
dest_phi = self.LINEAR_TO_ANGULAR_SCALER[1]*(dest_x_dot*math.cos(gamma)+dest_y_dot*math.sin(gamma))
dest_gamma = self.yaw_target
dest_theta,dest_phi = np.clip(dest_theta,self.TILT_LIMITS[0],self.TILT_LIMITS[1]),np.clip(dest_phi,self.TILT_LIMITS[0],self.TILT_LIMITS[1])
theta_error = dest_theta-theta
phi_error = dest_phi-phi
gamma_dot_error = (self.YAW_RATE_SCALER*self.wrap_angle(dest_gamma-gamma)) - gamma_dot
self.thetai_term += self.ANGULAR_I[0]*theta_error
self.phii_term += self.ANGULAR_I[1]*phi_error
self.gammai_term += self.ANGULAR_I[2]*gamma_dot_error
x_val = self.ANGULAR_P[0]*(theta_error) + self.ANGULAR_D[0]*(-theta_dot) + self.thetai_term
y_val = self.ANGULAR_P[1]*(phi_error) + self.ANGULAR_D[1]*(-phi_dot) + self.phii_term
z_val = self.ANGULAR_P[2]*(gamma_dot_error) + self.gammai_term
z_val = np.clip(z_val,self.YAW_CONTROL_LIMITS[0],self.YAW_CONTROL_LIMITS[1])
m1 = throttle + x_val + z_val
m2 = throttle + y_val - z_val
m3 = throttle - x_val + z_val
m4 = throttle - y_val - z_val
M = np.clip([m1,m2,m3,m4],self.MOTOR_LIMITS[0],self.MOTOR_LIMITS[1])
self.actuate_motors(self.quad_identifier,M)
def update_target(self,target):
self.target = target
def update_yaw_target(self,target):
self.yaw_target = self.wrap_angle(target)
def thread_run(self,update_rate,time_scaling):
update_rate = update_rate*time_scaling
last_update = self.get_time()
while(self.run==True):
time.sleep(0)
self.time = self.get_time()
if (self.time - last_update).total_seconds() > update_rate:
self.update()
last_update = self.time
def start_thread(self,update_rate=0.005,time_scaling=1):
self.thread_object = threading.Thread(target=self.thread_run,args=(update_rate,time_scaling))
self.thread_object.start()
def stop_thread(self):
self.run = False
class Controller_PID_Velocity(Controller_PID_Point2Point):
def update(self):
[dest_x,dest_y,dest_z] = self.target
[x,y,z,x_dot,y_dot,z_dot,theta,phi,gamma,theta_dot,phi_dot,gamma_dot] = self.get_state(self.quad_identifier)
x_error = dest_x-x_dot
y_error = dest_y-y_dot
z_error = dest_z-z
self.xi_term += self.LINEAR_I[0]*x_error
self.yi_term += self.LINEAR_I[1]*y_error
self.zi_term += self.LINEAR_I[2]*z_error
dest_x_dot = self.LINEAR_P[0]*(x_error) + self.LINEAR_D[0]*(-x_dot) + self.xi_term
dest_y_dot = self.LINEAR_P[1]*(y_error) + self.LINEAR_D[1]*(-y_dot) + self.yi_term
dest_z_dot = self.LINEAR_P[2]*(z_error) + self.LINEAR_D[2]*(-z_dot) + self.zi_term
throttle = np.clip(dest_z_dot,self.Z_LIMITS[0],self.Z_LIMITS[1])
dest_theta = self.LINEAR_TO_ANGULAR_SCALER[0]*(dest_x_dot*math.sin(gamma)-dest_y_dot*math.cos(gamma))
dest_phi = self.LINEAR_TO_ANGULAR_SCALER[1]*(dest_x_dot*math.cos(gamma)+dest_y_dot*math.sin(gamma))
dest_gamma = self.yaw_target
dest_theta,dest_phi = np.clip(dest_theta,self.TILT_LIMITS[0],self.TILT_LIMITS[1]),np.clip(dest_phi,self.TILT_LIMITS[0],self.TILT_LIMITS[1])
theta_error = dest_theta-theta
phi_error = dest_phi-phi
gamma_dot_error = (self.YAW_RATE_SCALER*self.wrap_angle(dest_gamma-gamma)) - gamma_dot
self.thetai_term += self.ANGULAR_I[0]*theta_error
self.phii_term += self.ANGULAR_I[1]*phi_error
self.gammai_term += self.ANGULAR_I[2]*gamma_dot_error
x_val = self.ANGULAR_P[0]*(theta_error) + self.ANGULAR_D[0]*(-theta_dot) + self.thetai_term
y_val = self.ANGULAR_P[1]*(phi_error) + self.ANGULAR_D[1]*(-phi_dot) + self.phii_term
z_val = self.ANGULAR_P[2]*(gamma_dot_error) + self.gammai_term
z_val = np.clip(z_val,self.YAW_CONTROL_LIMITS[0],self.YAW_CONTROL_LIMITS[1])
m1 = throttle + x_val + z_val
m2 = throttle + y_val - z_val
m3 = throttle - x_val + z_val
m4 = throttle - y_val - z_val
M = np.clip([m1,m2,m3,m4],self.MOTOR_LIMITS[0],self.MOTOR_LIMITS[1])
self.actuate_motors(self.quad_identifier,M)