view scripts/control/dre.m @ 3279:6dd06d525de6

[project @ 1999-10-12 16:52:40 by jwe]
author jwe
date Tue, 12 Oct 1999 16:53:06 +0000
parents 041ea33fbbf4
children f7e4a95916f2
line wrap: on
line source

# Copyright (C) 1998 Auburn University.  All Rights Reserved
#
# This file is part of Octave.
#
# Octave is free software; you can redistribute it and/or modify it
# under the terms of the GNU General Public License as published by the
# Free Software Foundation; either version 2, or (at your option) any
# later version.
#
# Octave is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
# FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
# for more details.
#
# You should have received a copy of the GNU General Public License
# along with Octave; see the file COPYING.  If not, write to the Free
# Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.

function [tvals,Plist] = dre(sys,Q,R,Qf,t0,tf,Ptol,maxits)
# [tvals,Plist] = dre(sys,Q,R,Qf,t0,tf{,Ptol,maxits});
# Solve the differential Riccati equation
#   -d P/dt = A'P + P A - P B inv(R) B' P + Q
#   P(tf) = Qf
# for the LTI system sys.  Solution of standard LTI
# state feedback optimization
#   min \int_{t_0}^{t_f} x' Q x + u' R u dt + x(t_f)' Qf x(t_f)
# optimal input is
#   u = - inv(R) B' P(t) x
# inputs:
#   sys: continuous time system data structure
#   Q: state integral penalty
#   R: input integral penalty
#   Qf: state terminal penalty
#   t0,tf: limits on the integral
#   Ptol: tolerance (used to select time samples; see below); default = 0.1
#   max number of refinement iterations (default=10)
# outputs:
#   tvals: time values at which P(t) is computed
#   Plist: list values of P(t); nth(Plist,ii) is P(tvals(ii)).
#
#   tvals is selected so that || nth(Plist,ii) - nth(Plist,ii-1) || < Ptol
#     for ii=2:length(tvals)
#
# Reference:

if(nargin < 6 | nargin > 8 | nargout != 2)
  usage("[tvals,Plist] = dre(sys,Q,R,Qf,t0,tf{,Ptol})");
elseif(!is_struct(sys))
  error("sys must be a system data structure")
elseif(is_digital(sys))
  error("sys must be a continuous time system")
elseif(!is_matrix(Q) | !is_matrix(R) | !is_matrix(Qf))
  error("Q, R, and Qf must be matrices.");
elseif(!is_scalar(t0) | !is_scalar(tf))
  error("t0 and tf must be scalars")
elseif(t0 >= tf)		error("t0=%e >= tf=%e",t0,tf);
elseif(nargin == 6)		Ptol = 0.1;
elseif(!is_scalar(Ptol))	error("Ptol must be a scalar");
elseif(Ptol <= 0)		error("Ptol must be positive");
endif

if(nargin < 8) maxits = 10;
elseif(!is_scalar(maxits))	error("maxits must be a scalar");
elseif(maxits <= 0)		error("maxits must be positive");
endif
maxits = ceil(maxits);

[aa,bb] = sys2ss(sys);
nn = sysdimensions(sys,"cst");
mm = sysdimensions(sys,"in");
pp = sysdimensions(sys,"out");

if(size(Q) != [nn, nn])
  error("Q(%dx%d); sys has %d states",rows(Q),columns(Q),nn);
elseif(size(Qf) != [nn, nn])
  error("Qf(%dx%d); sys has %d states",rows(Qf),columns(Qf),nn);
elseif(size(R) != [mm, mm])
  error("R(%dx%d); sys has %d inputs",rows(R),columns(R),mm);
endif

# construct Hamiltonian matrix
H = [aa , -(bb/R)*bb' ; -Q, -aa'];

# select time step to avoid numerical overflow
fast_eig = max(abs(eig(H)));
tc = log(10)/fast_eig;
nst = ceil((tf-t0)/tc);
tvals = -linspace(-tf,-t0,nst);
Plist = list(Qf);
In = eye(nn);
n1 = nn+1;
n2 = nn+nn;
done = 0;
while(!done)
  done = 1;      # assume this pass will do the job
  # sort time values in reverse order
  tvals = -sort(-tvals);
  tvlen = length(tvals);
  maxerr = 0;
  # compute new values of P(t); recompute old values just in case
  for ii=2:tvlen
    uv_i_minus_1 = [ In ; nth(Plist,ii-1) ];
    delta_t = tvals(ii-1) - tvals(ii);
    uv = expm(-H*delta_t)*uv_i_minus_1;
    Qi = uv(n1:n2,1:nn)/uv(1:nn,1:nn);
    Plist(ii) = (Qi+Qi')/2;
    # check error
    Perr = norm(nth(Plist,ii) - nth(Plist,ii-1))/norm(nth(Plist,ii));
    maxerr = max(maxerr,Perr);
    if(Perr > Ptol)
      new_t = mean(tvals([ii,ii-1]));
      tvals = [tvals, new_t];
      done = 0;
    endif
  endfor

  # check number of iterations
  maxits = maxits - 1;
  done = done+(maxits==0);
endwhile
if(maxerr > Ptol)
  warning("dre: \n\texiting with%4d points, max rel chg. =%e, Ptol=%e\n", ...
	tvlen,maxerr,Ptol);
  tvals = tvals(1:length(Plist));
endif
endfunction