slarfg.c

来自「InsightToolkit-1.4.0(有大量的优化算法程序)」· C语言 代码 · 共 123 行

C
123
字号
#include "f2c.h"
#include "netlib.h"

/* Subroutine */ void slarfg_(const integer *n, real *alpha, real *x, const integer *incx, real *tau)
{
    /* System generated locals */
    const integer nm1 = *n - 1;
    real r__1;

    /* Local variables */
    static real beta;
    static integer j;
    static real xnorm;
    static real safmin, rsafmn;
    static integer knt;

/*  -- LAPACK auxiliary routine (version 2.0) -- */
/*     Univ. of Tennessee, Univ. of California Berkeley, NAG Ltd., */
/*     Courant Institute, Argonne National Lab, and Rice University */
/*     September 30, 1994 */

/*  Purpose                                                               */
/*  =======                                                               */
/*                                                                        */
/*  SLARFG generates a real elementary reflector H of order n, such       */
/*  that                                                                  */
/*                                                                        */
/*        H * ( alpha ) = ( beta ),   H' * H = I.                         */
/*            (   x   )   (   0  )                                        */
/*                                                                        */
/*  where alpha and beta are scalars, and x is an (n-1)-element real      */
/*  vector. H is represented in the form                                  */
/*                                                                        */
/*        H = I - tau * ( 1 ) * ( 1 v' ) ,                                */
/*                      ( v )                                             */
/*                                                                        */
/*  where tau is a real scalar and v is a real (n-1)-element              */
/*  vector.                                                               */
/*                                                                        */
/*  If the elements of x are all zero, then tau = 0 and H is taken to be  */
/*  the unit matrix.                                                      */
/*                                                                        */
/*  Otherwise  1 <= tau <= 2.                                             */
/*                                                                        */
/*  Arguments                                                             */
/*  =========                                                             */
/*                                                                        */
/*  N       (input) INTEGER                                               */
/*          The order of the elementary reflector.                        */
/*                                                                        */
/*  ALPHA   (input/output) REAL                                           */
/*          On entry, the value alpha.                                    */
/*          On exit, it is overwritten with the value beta.               */
/*                                                                        */
/*  X       (input/output) REAL array, dimension                          */
/*                         (1+(N-2)*abs(INCX))                            */
/*          On entry, the vector x.                                       */
/*          On exit, it is overwritten with the vector v.                 */
/*                                                                        */
/*  INCX    (input) INTEGER                                               */
/*          The increment between elements of X. INCX > 0.                */
/*                                                                        */
/*  TAU     (output) REAL                                                 */
/*          The value tau.                                                */
/*                                                                        */
/*  ===================================================================== */

    if (*n <= 1) {
        *tau = 0.f;
        return;
    }

    xnorm = snrm2_(&nm1, x, incx);

    if (xnorm == 0.f) {

/*        H  =  I */

        *tau = 0.f;
    } else {

/*        general case */

        r__1 = slapy2_(alpha, &xnorm);
        beta = -r_sign(&r__1, alpha);
        safmin = slamch_("S") / slamch_("E");
        if (abs(beta) < safmin) {

/*           XNORM, BETA may be inaccurate; scale X and recompute them */

            rsafmn = 1.f / safmin;
            knt = 0;
            do {
                ++knt;
                sscal_(&nm1, &rsafmn, x, incx);
                beta *= rsafmn;
                *alpha *= rsafmn;
            } while (abs(beta) < safmin);

/*           New BETA is at most 1, at least SAFMIN */

            xnorm = snrm2_(&nm1, x, incx);
            r__1 = slapy2_(alpha, &xnorm);
            beta = -r_sign(&r__1, alpha);
            *tau = (beta - *alpha) / beta;
            r__1 = 1.f / (*alpha - beta);
            sscal_(&nm1, &r__1, x, incx);

/*           If ALPHA is subnormal, it may lose relative accuracy */

            *alpha = beta;
            for (j = 0; j < knt; ++j) {
                *alpha *= safmin;
            }
        } else {
            *tau = (beta - *alpha) / beta;
            r__1 = 1.f / (*alpha - beta);
            sscal_(&nm1, &r__1, x, incx);
            *alpha = beta;
        }
    }
} /* slarfg_ */

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?