github.com/jingcheng-WU/gonum@v0.9.1-0.20210323123734-f1a2a11a8f7b/lapack/gonum/dlaqr1.go (about)

     1  // Copyright ©2016 The Gonum Authors. All rights reserved.
     2  // Use of this source code is governed by a BSD-style
     3  // license that can be found in the LICENSE file.
     4  
     5  package gonum
     6  
     7  import "math"
     8  
     9  // Dlaqr1 sets v to a scalar multiple of the first column of the product
    10  //  (H - (sr1 + i*si1)*I)*(H - (sr2 + i*si2)*I)
    11  // where H is a 2×2 or 3×3 matrix, I is the identity matrix of the same size,
    12  // and i is the imaginary unit. Scaling is done to avoid overflows and most
    13  // underflows.
    14  //
    15  // n is the order of H and must be either 2 or 3. It must hold that either sr1 =
    16  // sr2 and si1 = -si2, or si1 = si2 = 0. The length of v must be equal to n. If
    17  // any of these conditions is not met, Dlaqr1 will panic.
    18  //
    19  // Dlaqr1 is an internal routine. It is exported for testing purposes.
    20  func (impl Implementation) Dlaqr1(n int, h []float64, ldh int, sr1, si1, sr2, si2 float64, v []float64) {
    21  	switch {
    22  	case n != 2 && n != 3:
    23  		panic("lapack: n must be 2 or 3")
    24  	case ldh < n:
    25  		panic(badLdH)
    26  	case len(h) < (n-1)*ldh+n:
    27  		panic(shortH)
    28  	case !((sr1 == sr2 && si1 == -si2) || (si1 == 0 && si2 == 0)):
    29  		panic(badShifts)
    30  	case len(v) != n:
    31  		panic(shortV)
    32  	}
    33  
    34  	if n == 2 {
    35  		s := math.Abs(h[0]-sr2) + math.Abs(si2) + math.Abs(h[ldh])
    36  		if s == 0 {
    37  			v[0] = 0
    38  			v[1] = 0
    39  		} else {
    40  			h21s := h[ldh] / s
    41  			v[0] = h21s*h[1] + (h[0]-sr1)*((h[0]-sr2)/s) - si1*(si2/s)
    42  			v[1] = h21s * (h[0] + h[ldh+1] - sr1 - sr2)
    43  		}
    44  		return
    45  	}
    46  
    47  	s := math.Abs(h[0]-sr2) + math.Abs(si2) + math.Abs(h[ldh]) + math.Abs(h[2*ldh])
    48  	if s == 0 {
    49  		v[0] = 0
    50  		v[1] = 0
    51  		v[2] = 0
    52  	} else {
    53  		h21s := h[ldh] / s
    54  		h31s := h[2*ldh] / s
    55  		v[0] = (h[0]-sr1)*((h[0]-sr2)/s) - si1*(si2/s) + h[1]*h21s + h[2]*h31s
    56  		v[1] = h21s*(h[0]+h[ldh+1]-sr1-sr2) + h[ldh+2]*h31s
    57  		v[2] = h31s*(h[0]+h[2*ldh+2]-sr1-sr2) + h21s*h[2*ldh+1]
    58  	}
    59  }