-
Notifications
You must be signed in to change notification settings - Fork 3
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Constrained pendulum problem #36
base: master
Are you sure you want to change the base?
Changes from 9 commits
9c47f78
6d54390
0caf7cc
afdfa7e
cd25494
f072d66
dfee988
13cf20d
0bc36d2
42d822f
9b2ee98
a114be8
c3463f3
1b997dd
3bd98e8
5fa4f92
90bc7af
c51c478
46a4436
d23d13d
8453493
7ec54c6
c5cce23
ff96669
fc51b18
3c01164
e04aa6c
5a2cef0
73a155a
026c815
0e45849
2d282cc
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,24 @@ | ||
classdef Canonical < otp.pendulumdae.PendulumDAEProblem | ||
%CANONICAL The constrained pendulum problem | ||
% | ||
% See | ||
% Hairer, E., Roche, M., Lubich, C. (1989). Description of differential-algebraic problems. | ||
% In: The Numerical Solution of Differential-Algebraic Systems by Runge-Kutta Methods. | ||
% Lecture Notes in Mathematics, vol 1409. Springer, Berlin, Heidelberg. | ||
% https://doi.org/10.1007/BFb0093948 | ||
|
||
methods | ||
function obj = Canonical | ||
tspan = [0; 10]; | ||
|
||
params = otp.pendulumdae.PendulumDAEParameters; | ||
params.Mass = 1; | ||
params.Length = 1; | ||
params.Gravity = otp.utils.PhysicalConstants.EarthGravity; | ||
|
||
y0 = [sqrt(2)/2; sqrt(2)/2; 0; 0; 0; 0; 0]; | ||
|
||
obj = [email protected](tspan, y0, params); | ||
end | ||
end | ||
end |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
classdef PendulumDAEParameters | ||
%CONSTRAINEDPENDULUMPARAMETERS | ||
properties | ||
%GRAVITY is acceleration due to gravity | ||
Gravity %MATLAB ONLY: (1,1) {mustBeReal, mustBeFinite, mustBePositive} = 9.8 | ||
%MASSE of the node of the pendulum | ||
Mass %MATLAB ONLY: (:,1) {mustBeReal, mustBeFinite, mustBePositive} = 1 | ||
%LENGTH to the node of the pendulum | ||
Length %MATLAB ONLY: (:,1) {mustBeReal, mustBeFinite, mustBePositive} = 1 | ||
end | ||
end |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,87 @@ | ||
classdef PendulumDAEProblem < otp.Problem | ||
%CONSTRAINEDPENDULUM PROBLEM This is a Hessenberg Index-2 DAE problem posed in terms of three constraints | ||
% | ||
|
||
properties (SetAccess=protected) | ||
RHSDifferential | ||
RHSAlgebraic | ||
end | ||
|
||
properties (Dependent) | ||
Y0Differential | ||
Z0Algebraic | ||
end | ||
|
||
methods | ||
function obj = PendulumDAEProblem(timeSpan, y0, parameters) | ||
[email protected]('Constrained Pendulum', 7, timeSpan, y0, parameters); | ||
end | ||
end | ||
|
||
methods (Access = protected) | ||
function onSettingsChanged(obj) | ||
m = obj.Parameters.Mass; | ||
l = obj.Parameters.Length; | ||
g = obj.Parameters.Gravity; | ||
|
||
% get initial energy | ||
initialconstraints = otp.pendulumdae.invariants([], obj.Y0Differential, g, m, l, 0); | ||
E0 = initialconstraints(3); | ||
|
||
% The right hand size in terms of x, y, x', y', and three control parameters z | ||
obj.RHS = otp.RHS(@(t, y) otp.pendulumdae.f(t, y, g, m, l, E0), ... | ||
'Mass', otp.pendulumdae.mass([], [], g, m, l, E0), ... | ||
'JacobianVectorProduct', ... | ||
@(t, y, v) otp.pendulumdae.jacobianVectorProduct(t, y, g, m, l, E0, v), ... | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
|
||
'Vectorized', 'on'); | ||
|
||
% Generate the constituent RHS for the differential part | ||
obj.RHSDifferential = otp.RHS(@(t, y) otp.pendulumdae.fDifferential(t, y, g, m, l, E0), ... | ||
'JacobianVectorProduct', ... | ||
@(t, y, v) otp.pendulumdae.jacobianDifferentialVectorProduct(t, y, g, m, l, E0, v), ... | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
|
||
'Vectorized', 'on'); | ||
|
||
% Generate the constituent RHS for the algebraic part | ||
obj.RHSAlgebraic = otp.RHS(@(t, y) otp.pendulumdae.invariants(t, y, g, m, l, E0), ... | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The terms "algebraic" and "invariant" are used inconsistently. I think "algebraic" is the more standard DAE term, so I say we should name the functions |
||
'Jacobian', @(t, y) otp.pendulumdae.invariantsJacobian(t, y, g, m, l, E0), ... | ||
'JacobianVectorProduct', ... | ||
@(t, y, v) otp.pendulumdae.invariantsJacobianVectorProduct(t, y, g, m, l, E0, v), ... | ||
'JacobianAdjointVectorProduct', ... | ||
@(t, y, v) otp.pendulumdae.invariantsJacobianAdjointVectorProduct(t, y, g, m, l, E0, v), ... | ||
'HessianAdjointVectorProduct', ... | ||
@(t, y, v, u) otp.pendulumdae.invariantsHessianAdjointVectorProduct(t, y, g, m, l, E0, v, u), ... | ||
'Vectorized', 'on'); | ||
end | ||
|
||
function label = internalIndex2label(~, index) | ||
switch index | ||
case 1 | ||
label = 'x position'; | ||
case 2 | ||
label = 'y position'; | ||
case 3 | ||
label = 'x velocity'; | ||
case 4 | ||
label = 'y velocity'; | ||
case 5 | ||
label = 'Control 1'; | ||
case 6 | ||
label = 'Control 2'; | ||
case 7 | ||
label = 'Control 3'; | ||
end | ||
end | ||
end | ||
|
||
methods | ||
function y0differential = get.Y0Differential(obj) | ||
y0differential = obj.Y0(1:4); | ||
end | ||
|
||
function z0algebraic = get.Z0Algebraic(obj) | ||
z0algebraic = obj.Y0(5:end); | ||
end | ||
end | ||
|
||
end | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
function dfull = f(t, statepluscontrol, g, m, l, E0) | ||
|
||
state = statepluscontrol(1:4, :); | ||
control = statepluscontrol(5:end, :); | ||
|
||
dstate = otp.pendulumdae.fDifferential(t, state, g, m, l, E0) ... | ||
- otp.pendulumdae.invariantsJacobianAdjointVectorProduct(t, state, g, m, l, E0, control); | ||
|
||
c = otp.pendulumdae.invariants(t, state, g, m, l, E0); | ||
|
||
dfull = [dstate; c]; | ||
|
||
end |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
function dstate = fDifferential(~, state, g, m, ~, ~) | ||
|
||
x = state(1, :); | ||
y = state(2, :); | ||
u = state(3, :); | ||
v = state(4, :); | ||
|
||
lxy2 = x.^2 + y.^2; | ||
|
||
lambda = (m./lxy2).*(u.^2 + v.^2) - (g./lxy2).*y; | ||
|
||
dx = u; | ||
dy = v; | ||
du = -(lambda/m).*x; | ||
dv = -(lambda/m).*y - g/m; | ||
|
||
dstate = [dx; dy; du; dv]; | ||
|
||
end |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
function c = invariants(~, state, g, m, l, E0) | ||
|
||
x = state(1, :); | ||
y = state(2, :); | ||
u = state(3, :); | ||
v = state(4, :); | ||
|
||
% the fisrt invariant is length preservation | ||
c1 = x.^2 + y.^2 - l^2; | ||
|
||
% the second invariant is the derivative of the above, to convert this into an index-2 DAE | ||
c2 = 2*(x.*u + y.*v); | ||
|
||
% the third invariant is energy preservation, for numerical stability | ||
c3 = m*(g*(y + l) + 0.5*(u.^2 + v.^2)) - E0; | ||
|
||
c = [c1; c2; c3]; | ||
|
||
end |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
function dvp = invariantsHessianAdjointVectorProduct(~, ~, g, m, ~, ~, control, vec) | ||
|
||
z1 = control(1, :); | ||
z2 = control(2, :); | ||
z3 = control(3, :); | ||
|
||
vec1 = vec(1, :); | ||
vec2 = vec(2, :); | ||
vec3 = vec(3, :); | ||
vec4 = vec(4, :); | ||
|
||
dvp1 = 2*(vec1.*z1 + vec3.*z2); | ||
dvp2 = 2*(vec2.*z1 + vec4.*z2); | ||
dvp3 = 2*vec1.*z2 + m*vec3.*z3; | ||
dvp4 = 2*vec2.*z2 + m*vec4.*z3; | ||
|
||
dvp = [dvp1; dvp2; dvp3; dvp4]; | ||
|
||
end |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,27 @@ | ||
function dc = invariantsJacobian(~, state, g, m, ~, ~) | ||
|
||
x = state(1); | ||
y = state(2); | ||
u = state(3); | ||
v = state(4); | ||
|
||
dc1dx = 2*x; | ||
dc1dy = 2*y; | ||
dc1du = 0; | ||
dc1dv = 0; | ||
|
||
dc2dx = 2*u; | ||
dc2dy = 2*v; | ||
dc2du = 2*x; | ||
dc2dv = 2*y; | ||
|
||
dc3dx = 0; | ||
dc3dy = m*g; | ||
dc3du = m*u; | ||
dc3dv = m*v; | ||
|
||
dc = [dc1dx, dc1dy, dc1du, dc1dv; ... | ||
dc2dx, dc2dy, dc2du, dc2dv; ... | ||
dc3dx, dc3dy, dc3du, dc3dv]; | ||
|
||
end |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
function dvp = invariantsJacobianAdjointVectorProduct(~, state, g, m, ~, ~, control) | ||
|
||
x = state(1, :); | ||
y = state(2, :); | ||
u = state(3, :); | ||
v = state(4, :); | ||
|
||
z1 = control(1, :); | ||
z2 = control(2, :); | ||
z3 = control(3, :); | ||
|
||
dvp1 = 2*(x.*z1 + u.*z2); | ||
dvp2 = 2*(y.*z1 + v.*z2) + g*m*z3; | ||
dvp3 = 2*x.*z2 + m*u.*z3; | ||
dvp4 = 2*y.*z2 + m*v.*z3; | ||
|
||
dvp = [dvp1; dvp2; dvp3; dvp4]; | ||
|
||
end |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,20 @@ | ||
function dvp = invariantsJacobianVectorProduct(~, state, g, m, ~, ~, vec) | ||
|
||
x = state(1, :); | ||
y = state(2, :); | ||
u = state(3, :); | ||
v = state(4, :); | ||
|
||
vec1 = vec(1, :); | ||
vec2 = vec(2, :); | ||
vec3 = vec(3, :); | ||
vec4 = vec(4, :); | ||
|
||
|
||
dvp1 = 2*(x.*vec1 + y.*vec2); | ||
dvp2 = 2*(u.*vec1 + v.*vec2 + x.*vec3 + y.*vec4); | ||
dvp3 = m*(g*vec2 + u.*vec3 + v.*vec4); | ||
|
||
dvp = [dvp1; dvp2; dvp3]; | ||
|
||
end |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,31 @@ | ||
function dstatevp = jacobianDifferentialVectorProduct(~, state, g, m, ~, ~, vec) | ||
|
||
x = state(1, :); | ||
y = state(2, :); | ||
u = state(3, :); | ||
v = state(4, :); | ||
|
||
vec1 = vec(1, :); | ||
vec2 = vec(2, :); | ||
vec3 = vec(3, :); | ||
vec4 = vec(4, :); | ||
|
||
lxy2 = x.^2 + y.^2; | ||
|
||
lambda = (m./lxy2).*(u.^2 + v.^2) - (g./lxy2).*y; | ||
|
||
dlambdadx = (-2*m*(u.^2 + v.^2).*x + 2*g*x.*y)./(lxy2.^2); | ||
dlambdady = (-2*m*(u.^2 + v.^2).*y + g*(y.^2 - x.^2))./(lxy2.^2); | ||
dlambdadu = (2*m./lxy2).*u; | ||
dlambdadv = (2*m./lxy2).*v; | ||
|
||
inner = (vec1.*dlambdadx + vec2.*dlambdady + vec3.*dlambdadu + vec4.*dlambdadv); | ||
|
||
dxvp = vec3; | ||
dyvp = vec4; | ||
duvp = -(1/m)*(vec1.*lambda + x.*inner); | ||
dvvp = -(1/m)*(vec2.*lambda + y.*inner); | ||
|
||
dstatevp = [dxvp; dyvp; duvp; dvvp]; | ||
|
||
end |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,17 @@ | ||
function dfull = jacobianVectorProduct(t, statepluscontrol, g, m, l, E0, v) | ||
|
||
state = statepluscontrol(1:4, :); | ||
control = statepluscontrol(5:end, :); | ||
|
||
vstate = v(1:4, :); | ||
vcontrol = v(5:end, :); | ||
|
||
dstate = otp.pendulumdae.jacobianDifferentialVectorProduct(t, state, g, m, l, E0, vstate) ... | ||
- otp.pendulumdae.invariantsHessianAdjointVectorProduct(t, state, g, m, l, E0, control, vstate) ... | ||
- otp.pendulumdae.invariantsJacobianAdjointVectorProduct(t, state, g, m, l, E0, vcontrol); | ||
|
||
c = otp.pendulumdae.invariantsJacobianVectorProduct(t, state, g, m, l, E0, vstate); | ||
|
||
dfull = [dstate; c]; | ||
|
||
end |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
function M = mass(~, ~, ~, ~, ~, ~) | ||
|
||
M = [eye(4), zeros(4, 3); zeros(3, 7)]; | ||
|
||
end |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm not opposed to having these, but if we do include this, I think all DAE problems should have these for consistency. Also, I think
Y0Algebraic
would be better thanZ0Algebraic
.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Doesn't really differentiate the two. Algebraic implies there is no direct differential equation thus it is a variable obtained from differentiation of the constraint. I think it should stay.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think this naming is pretty clear. It's mostly motivated by established naming conventions. For partitioned problems, we use
<property><partition name>
. E.g.RHSStiff and RHSNonstiff
jacobianDifferential and jacobianAlgebraic
Y0Position and Y0Velocity
This is nice for autocompletion as well.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
My bad I misinterpreted what was being proposed. I thought you wanted
Y0Differential -> Y0Algebraic
notZ0Algebraic -> Y0Algebraic
. I agree on the convention you proposed.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I was thinking of changing it to that already, glad to see we all think alike