File: SchurDecompEig.cc

package info (click to toggle)
dynare 6.3-1
  • links: PTS, VCS
  • area: main
  • in suites: trixie
  • size: 67,632 kB
  • sloc: cpp: 79,090; ansic: 28,916; objc: 12,430; yacc: 4,528; pascal: 1,993; lex: 1,441; sh: 1,121; python: 634; makefile: 626; lisp: 163; xml: 18
file content (116 lines) | stat: -rw-r--r-- 3,655 bytes parent folder | download | duplicates (2)
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
/*
 * Copyright © 2004-2011 Ondra Kamenik
 * Copyright © 2019-2023 Dynare Team
 *
 * This file is part of Dynare.
 *
 * Dynare 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 3 of the License, or
 * (at your option) any later version.
 *
 * Dynare 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 Dynare.  If not, see <https://www.gnu.org/licenses/>.
 */

#include "SchurDecompEig.hh"
#include "SylvException.hh"

#include <dynlapack.h>

#include <vector>

/* Bubble diagonal 1×1 or 2×2 block from position ‘from’ to position
   ‘to’. If an eigenvalue cannot be swapped with its neighbour, the
   neighbour is bubbled also in front. The method returns a new
   position ‘to’, where the original block pointed by ‘to’ happens to
   appear at the end. ‘from’ must be greater than ‘to’.
*/
SchurDecompEig::diag_iter
SchurDecompEig::bubbleEigen(diag_iter from, diag_iter to)
{
  auto run = from;
  while (run != to)
    {
      auto runm = run;
      if (!tryToSwap(run, runm) && runm == to)
        ++to;
      else
        {
          /* Bubble all eigenvalues from runm(incl.) to run(excl.),
             this includes either bubbling generated eigenvalues due
             to split, or an eigenvalue which couldn't be swapped */
          while (runm != run)
            {
              to = bubbleEigen(runm, to);
              ++runm;
            }
        }
    }
  return to;
}

/* This tries to swap two neighbouring eigenvalues, ‘it’ and ‘--it’,
   and returns ‘itadd’. If the blocks can be swapped, new eigenvalues
   can emerge due to possible 2×2 block splits. ‘it’ then points to
   the last eigenvalue coming from block pointed by ‘it’ at the
   begining, and ‘itadd’ points to the first. On swap failure, ‘it’ is
   not changed, and ‘itadd’ points to previous eignevalue (which must
   be moved backwards before). In either case, it is necessary to
   resolve eigenvalues from ‘itadd’ to ‘it’, before the ‘it’ can be
   resolved.
   The success is signaled by returned true.
*/
bool
SchurDecompEig::tryToSwap(diag_iter& it, diag_iter& itadd)
{
  itadd = it;
  --itadd;

  lapack_int n = getDim(), ldt = getT().getLD(), ldq = getQ().getLD();
  lapack_int ifst = it->getIndex() + 1;
  lapack_int ilst = itadd->getIndex() + 1;
  std::vector<double> work(n);
  lapack_int info;
  dtrexc("V", &n, getT().base(), &ldt, getQ().base(), &ldq, &ifst, &ilst, work.data(), &info);
  if (info < 0)
    throw SYLV_MES_EXCEPTION("Wrong argument to dtrexc.");

  if (info == 0)
    {
      // swap successful
      getT().swapDiagLogically(itadd);
      // check for 2×2 block splits
      getT().checkDiagConsistency(it);
      getT().checkDiagConsistency(itadd);
      // and go back by ‘it’ in NEW eigenvalue set
      --it;
      return true;
    }
  return false;
}

void
SchurDecompEig::orderEigen()
{
  auto run = getT().diag_begin();
  auto runp = run;
  ++runp;
  double last_size = 0.0;
  while (runp != getT().diag_end())
    {
      auto least = getT().findNextLargerBlock(run, getT().diag_end(), last_size);
      last_size = least->getSize();
      if (run == least)
        ++run;
      else
        run = bubbleEigen(least, run);
      runp = run;
      ++runp;
    }
}