-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathEngineGuid.m
80 lines (68 loc) · 2.08 KB
/
EngineGuid.m
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
classdef EngineGuid < EngineAero
properties (Access = protected)
% New state scalars
bank;
count;
% New state vectors
Qir;
MRP;
end
properties (Access = protected, Constant)
tolMRP = 0.01;
offtime = 10; % seconds
end
methods (Access = protected)
function Mc = controls(self, ~, ~, sc)
% Rate
Wref = zeros(3, 1);
dWref = zeros(3, 1);
deltaW = self.W - Wref;
% Attitude - body w.r.t. reference
q0 = self.Q(1); q1 = self.Q(2); q2 = self.Q(3); q3 = self.Q(4);
Qref = [q0, -q1, -q2, -q3 ;
q1, q0, q3, -q2 ;
q2, -q3, q0, q1 ;
q3, q2, -q1, q0];
Qbr = self.bank * Qref * self.Qir;
self.MRP = Qbr(2:4) / (1 + Qbr(1)); % Quaternions -> Modified Rodrigues Parameters
% Lyapunov-derived control law
Mc = sc.I * (dWref - cross(self.W, Wref)) + cross(self.W, sc.I * self.W) - diag(sc.hold) * self.MRP - diag(sc.damp) * deltaW;
end
function [val, ter, sgn] = event(self, t, sc, pl)
% Call superclass method
[val, ter, sgn] = event@EngineAero(self, t, sc, pl);
if ~self.bank && t >= sc.tref(self.count)
% Reference attitude
PRA = (mod(self.count, 2) * 2 - 1) * pi;
PRV = self.Urel / norm(self.Urel);
q0 = cos(PRA/2); q1 = PRV(1) * sin(PRA/2); q2 = PRV(2) * sin(PRA/2); q3 = PRV(3) * sin(PRA/2);
Qref = [q0, -q1, -q2, -q3;
q1, q0, q3, -q2;
q2, -q3, q0, q1;
q3, q2, -q1, q0];
Qri = Qref * self.Q; % quaternion addition
self.Qir = [Qri(1); -Qri(2:4)];
% Update counters
self.MRP = ones(3, 1); % to fool the conditional check that follows
self.bank = true;
elseif self.bank && (max(abs(self.MRP)) < self.tolMRP || t > sc.tref(self.count) + self.offtime)
self.bank = false;
self.count = self.count + 1;
end
% Append new event outputs
val = [val; t - sc.tref(self.count)];
ter = [ter; false];
sgn = [sgn; +1];
end
function initreset(self)
% Call superclass method
initreset@EngineAero(self);
% New state scalars
self.bank = false;
self.count = 1;
% New state vectors
self.Qir = zeros(4, 1);
self.MRP = zeros(3, 1);
end
end
end