154f251a9SBarry Smith /* fnoise/snesdnest.F -- translated by f2c (version 20020314).
254f251a9SBarry Smith */
3c6db04a5SJed Brown #include <petscsys.h>
454f251a9SBarry Smith #define FALSE_ 0
554f251a9SBarry Smith #define TRUE_ 1
654f251a9SBarry Smith
754f251a9SBarry Smith /* Noise estimation routine, written by Jorge More'. Details are below. */
854f251a9SBarry Smith
925acbd8eSLisandro Dalcin PETSC_INTERN PetscErrorCode SNESNoise_dnest_(PetscInt *, PetscScalar *, PetscScalar *, PetscScalar *, PetscScalar *, PetscScalar *, PetscInt *, PetscScalar *);
1025acbd8eSLisandro Dalcin
SNESNoise_dnest_(PetscInt * nf,double * fval,double * h__,double * fnoise,double * fder2,double * hopt,PetscInt * info,double * eps)11d71ae5a4SJacob Faibussowitsch PetscErrorCode SNESNoise_dnest_(PetscInt *nf, double *fval, double *h__, double *fnoise, double *fder2, double *hopt, PetscInt *info, double *eps)
12d71ae5a4SJacob Faibussowitsch {
1354f251a9SBarry Smith /* Initialized data */
1454f251a9SBarry Smith
159371c9d4SSatish Balay static double const__[15] = {.71, .41, .23, .12, .063, .033, .018, .0089, .0046, .0024, .0012, 6.1e-4, 3.1e-4, 1.6e-4, 8e-5};
1654f251a9SBarry Smith
1754f251a9SBarry Smith /* System generated locals */
18a7cc72afSBarry Smith PetscInt i__1;
1954f251a9SBarry Smith double d__1, d__2, d__3, d__4;
2054f251a9SBarry Smith
2154f251a9SBarry Smith /* Local variables */
2254f251a9SBarry Smith static double emin, emax;
23a7cc72afSBarry Smith static PetscInt dsgn[6];
248e653690SSatish Balay static double f_max, f_min, stdv;
25a7cc72afSBarry Smith static PetscInt i__, j;
2654f251a9SBarry Smith static double scale;
27a7cc72afSBarry Smith static PetscInt mh;
28a7cc72afSBarry Smith static PetscInt cancel[6], dnoise;
2954f251a9SBarry Smith static double err2, est1, est2, est3, est4;
3054f251a9SBarry Smith
3154f251a9SBarry Smith /* ********** */
3254f251a9SBarry Smith
3354f251a9SBarry Smith /* Subroutine dnest */
3454f251a9SBarry Smith
3554f251a9SBarry Smith /* This subroutine estimates the noise in a function */
3654f251a9SBarry Smith /* and provides estimates of the optimal difference parameter */
3754f251a9SBarry Smith /* for a forward-difference approximation. */
3854f251a9SBarry Smith
3954f251a9SBarry Smith /* The user must provide a difference parameter h, and the */
4054f251a9SBarry Smith /* function value at nf points centered around the current point. */
4154f251a9SBarry Smith /* For example, if nf = 7, the user must provide */
4254f251a9SBarry Smith
4354f251a9SBarry Smith /* f(x-2*h), f(x-h), f(x), f(x+h), f(x+2*h), */
4454f251a9SBarry Smith
4554f251a9SBarry Smith /* in the array fval. The use of nf = 7 function evaluations is */
4654f251a9SBarry Smith /* recommended. */
4754f251a9SBarry Smith
4854f251a9SBarry Smith /* The noise in the function is roughly defined as the variance in */
4954f251a9SBarry Smith /* the computed value of the function. The noise in the function */
5054f251a9SBarry Smith /* provides valuable information. For example, function values */
5154f251a9SBarry Smith /* smaller than the noise should be considered to be zero. */
5254f251a9SBarry Smith
5354f251a9SBarry Smith /* This subroutine requires an initial estimate for h. Under estimates */
5454f251a9SBarry Smith /* are usually preferred. If noise is not detected, the user should */
5535cb6cd3SPierre Jolivet /* increase or decrease h according to the output value of info. */
5654f251a9SBarry Smith /* In most cases, the subroutine detects noise with the initial */
5754f251a9SBarry Smith /* value of h. */
5854f251a9SBarry Smith
5954f251a9SBarry Smith /* The subroutine statement is */
6054f251a9SBarry Smith
6154f251a9SBarry Smith /* subroutine dnest(nf,fval,h,hopt,fnoise,info,eps) */
6254f251a9SBarry Smith
6354f251a9SBarry Smith /* where */
6454f251a9SBarry Smith
655bd1e576SStefano Zampini /* nf is a PetscInt variable. */
6654f251a9SBarry Smith /* On entry nf is the number of function values. */
6754f251a9SBarry Smith /* On exit nf is unchanged. */
6854f251a9SBarry Smith
6954f251a9SBarry Smith /* f is a double precision array of dimension nf. */
7054f251a9SBarry Smith /* On entry f contains the function values. */
7154f251a9SBarry Smith /* On exit f is overwritten. */
7254f251a9SBarry Smith
7354f251a9SBarry Smith /* h is a double precision variable. */
7454f251a9SBarry Smith /* On entry h is an estimate of the optimal difference parameter. */
7554f251a9SBarry Smith /* On exit h is unchanged. */
7654f251a9SBarry Smith
7754f251a9SBarry Smith /* fnoise is a double precision variable. */
7854f251a9SBarry Smith /* On entry fnoise need not be specified. */
7954f251a9SBarry Smith /* On exit fnoise is set to an estimate of the function noise */
8054f251a9SBarry Smith /* if noise is detected; otherwise fnoise is set to zero. */
8154f251a9SBarry Smith
8254f251a9SBarry Smith /* hopt is a double precision variable. */
8354f251a9SBarry Smith /* On entry hopt need not be specified. */
8454f251a9SBarry Smith /* On exit hopt is set to an estimate of the optimal difference */
8554f251a9SBarry Smith /* parameter if noise is detected; otherwise hopt is set to zero. */
8654f251a9SBarry Smith
875bd1e576SStefano Zampini /* info is a PetscInt variable. */
8854f251a9SBarry Smith /* On entry info need not be specified. */
8954f251a9SBarry Smith /* On exit info is set as follows: */
9054f251a9SBarry Smith
9154f251a9SBarry Smith /* info = 1 Noise has been detected. */
9254f251a9SBarry Smith
9354f251a9SBarry Smith /* info = 2 Noise has not been detected; h is too small. */
9454f251a9SBarry Smith /* Try 100*h for the next value of h. */
9554f251a9SBarry Smith
9654f251a9SBarry Smith /* info = 3 Noise has not been detected; h is too large. */
9754f251a9SBarry Smith /* Try h/100 for the next value of h. */
9854f251a9SBarry Smith
9954f251a9SBarry Smith /* info = 4 Noise has been detected but the estimate of hopt */
10054f251a9SBarry Smith /* is not reliable; h is too small. */
10154f251a9SBarry Smith
10254f251a9SBarry Smith /* eps is a double precision work array of dimension nf. */
10354f251a9SBarry Smith
10454f251a9SBarry Smith /* MINPACK-2 Project. April 1997. */
10554f251a9SBarry Smith /* Argonne National Laboratory. */
10654f251a9SBarry Smith /* Jorge J. More'. */
10754f251a9SBarry Smith
10854f251a9SBarry Smith /* ********** */
10954f251a9SBarry Smith /* Parameter adjustments */
11054f251a9SBarry Smith --eps;
11154f251a9SBarry Smith --fval;
11254f251a9SBarry Smith
11354f251a9SBarry Smith /* Function Body */
11454f251a9SBarry Smith *fnoise = 0.;
11554f251a9SBarry Smith *fder2 = 0.;
11654f251a9SBarry Smith *hopt = 0.;
11754f251a9SBarry Smith /* Compute an estimate of the second derivative and */
11854f251a9SBarry Smith /* determine a bound on the error. */
11954f251a9SBarry Smith mh = (*nf + 1) / 2;
12054f251a9SBarry Smith est1 = (fval[mh + 1] - fval[mh] * 2 + fval[mh - 1]) / *h__ / *h__;
12154eb447fSKarl Rupp est2 = (fval[mh + 2] - fval[mh] * 2 + fval[mh - 2]) / (*h__ * 2) / (*h__ * 2);
12254eb447fSKarl Rupp est3 = (fval[mh + 3] - fval[mh] * 2 + fval[mh - 3]) / (*h__ * 3) / (*h__ * 3);
12354f251a9SBarry Smith est4 = (est1 + est2 + est3) / 3;
12454f251a9SBarry Smith /* Computing MAX */
12554f251a9SBarry Smith /* Computing PETSCMAX */
12654f251a9SBarry Smith d__3 = PetscMax(est1, est2);
12754f251a9SBarry Smith /* Computing MIN */
12854f251a9SBarry Smith d__4 = PetscMin(est1, est2);
1292a274a78SSatish Balay d__1 = PetscMax(d__3, est3) - est4;
1302a274a78SSatish Balay d__2 = est4 - PetscMin(d__4, est3);
13154f251a9SBarry Smith err2 = PetscMax(d__1, d__2);
13254f251a9SBarry Smith /* write (2,123) est1, est2, est3 */
13354f251a9SBarry Smith /* 123 format ('Second derivative estimates', 3d12.2) */
134f5af7f23SKarl Rupp if (err2 <= PetscAbsScalar(est4) * .1) *fder2 = est4;
135f5af7f23SKarl Rupp else if (err2 < PetscAbsScalar(est4)) *fder2 = est3;
136f5af7f23SKarl Rupp else *fder2 = 0.;
137f5af7f23SKarl Rupp
13854f251a9SBarry Smith /* Compute the range of function values. */
1398e653690SSatish Balay f_min = fval[1];
1408e653690SSatish Balay f_max = fval[1];
14154f251a9SBarry Smith i__1 = *nf;
14254f251a9SBarry Smith for (i__ = 2; i__ <= i__1; ++i__) {
14354f251a9SBarry Smith /* Computing MIN */
1442a274a78SSatish Balay d__1 = f_min;
1452a274a78SSatish Balay d__2 = fval[i__];
1468e653690SSatish Balay f_min = PetscMin(d__1, d__2);
147f5af7f23SKarl Rupp
14854f251a9SBarry Smith /* Computing MAX */
1492a274a78SSatish Balay d__1 = f_max;
1502a274a78SSatish Balay d__2 = fval[i__];
1518e653690SSatish Balay f_max = PetscMax(d__1, d__2);
15254f251a9SBarry Smith }
15354f251a9SBarry Smith /* Construct the difference table. */
15454f251a9SBarry Smith dnoise = FALSE_;
15554f251a9SBarry Smith for (j = 1; j <= 6; ++j) {
15654f251a9SBarry Smith dsgn[j - 1] = FALSE_;
15754f251a9SBarry Smith cancel[j - 1] = FALSE_;
15854f251a9SBarry Smith scale = 0.;
15954f251a9SBarry Smith i__1 = *nf - j;
16054f251a9SBarry Smith for (i__ = 1; i__ <= i__1; ++i__) {
16154f251a9SBarry Smith fval[i__] = fval[i__ + 1] - fval[i__];
162f5af7f23SKarl Rupp if (fval[i__] == 0.) cancel[j - 1] = TRUE_;
163f5af7f23SKarl Rupp
16454f251a9SBarry Smith /* Computing MAX */
1652a274a78SSatish Balay d__1 = fval[i__];
1662a274a78SSatish Balay d__2 = scale;
1672a274a78SSatish Balay d__3 = PetscAbsScalar(d__1);
16854f251a9SBarry Smith scale = PetscMax(d__2, d__3);
16954f251a9SBarry Smith }
170f5af7f23SKarl Rupp
17154f251a9SBarry Smith /* Compute the estimates for the noise level. */
172f5af7f23SKarl Rupp if (scale == 0.) stdv = 0.;
173f5af7f23SKarl Rupp else {
17454f251a9SBarry Smith stdv = 0.;
17554f251a9SBarry Smith i__1 = *nf - j;
17654f251a9SBarry Smith for (i__ = 1; i__ <= i__1; ++i__) {
17754f251a9SBarry Smith /* Computing 2nd power */
17854f251a9SBarry Smith d__1 = fval[i__] / scale;
17954f251a9SBarry Smith stdv += d__1 * d__1;
18054f251a9SBarry Smith }
18154f251a9SBarry Smith stdv = scale * PetscSqrtScalar(stdv / (*nf - j));
18254f251a9SBarry Smith }
18354f251a9SBarry Smith eps[j] = const__[j - 1] * stdv;
18454f251a9SBarry Smith /* Determine differences in sign. */
18554f251a9SBarry Smith i__1 = *nf - j - 1;
18654f251a9SBarry Smith for (i__ = 1; i__ <= i__1; ++i__) {
18754f251a9SBarry Smith /* Computing MIN */
1882a274a78SSatish Balay d__1 = fval[i__];
1892a274a78SSatish Balay d__2 = fval[i__ + 1];
19054f251a9SBarry Smith /* Computing MAX */
1912a274a78SSatish Balay d__3 = fval[i__];
1922a274a78SSatish Balay d__4 = fval[i__ + 1];
193f5af7f23SKarl Rupp if (PetscMin(d__1, d__2) < 0. && PetscMax(d__3, d__4) > 0.) dsgn[j - 1] = TRUE_;
19454f251a9SBarry Smith }
19554f251a9SBarry Smith }
19654f251a9SBarry Smith /* First requirement for detection of noise. */
19754f251a9SBarry Smith dnoise = dsgn[3];
19854f251a9SBarry Smith /* Check for h too small or too large. */
19954f251a9SBarry Smith *info = 0;
200f5af7f23SKarl Rupp if (f_max == f_min) *info = 2;
201f5af7f23SKarl Rupp else /* if (complicated condition) */ {
20254f251a9SBarry Smith /* Computing MIN */
2032a274a78SSatish Balay d__1 = PetscAbsScalar(f_max);
2042a274a78SSatish Balay d__2 = PetscAbsScalar(f_min);
205f5af7f23SKarl Rupp if (f_max - f_min > PetscMin(d__1, d__2) * .1) *info = 3;
20654f251a9SBarry Smith }
2073ba16761SJacob Faibussowitsch if (*info != 0) PetscFunctionReturn(PETSC_SUCCESS);
208f5af7f23SKarl Rupp
20954f251a9SBarry Smith /* Determine the noise level. */
21054f251a9SBarry Smith /* Computing MIN */
21154f251a9SBarry Smith d__1 = PetscMin(eps[4], eps[5]);
21254f251a9SBarry Smith emin = PetscMin(d__1, eps[6]);
213f5af7f23SKarl Rupp
21454f251a9SBarry Smith /* Computing MAX */
21554f251a9SBarry Smith d__1 = PetscMax(eps[4], eps[5]);
21654f251a9SBarry Smith emax = PetscMax(d__1, eps[6]);
217f5af7f23SKarl Rupp
21854f251a9SBarry Smith if (emax <= emin * 4 && dnoise) {
21954f251a9SBarry Smith *fnoise = (eps[4] + eps[5] + eps[6]) / 3;
22054f251a9SBarry Smith if (*fder2 != 0.) {
22154f251a9SBarry Smith *info = 1;
22254f251a9SBarry Smith *hopt = PetscSqrtScalar(*fnoise / PetscAbsScalar(*fder2)) * 1.68;
22354f251a9SBarry Smith } else {
22454f251a9SBarry Smith *info = 4;
22554f251a9SBarry Smith *hopt = *h__ * 10;
22654f251a9SBarry Smith }
2273ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
22854f251a9SBarry Smith }
229f5af7f23SKarl Rupp
23054f251a9SBarry Smith /* Computing MIN */
23154f251a9SBarry Smith d__1 = PetscMin(eps[3], eps[4]);
23254f251a9SBarry Smith emin = PetscMin(d__1, eps[5]);
233f5af7f23SKarl Rupp
23454f251a9SBarry Smith /* Computing MAX */
23554f251a9SBarry Smith d__1 = PetscMax(eps[3], eps[4]);
23654f251a9SBarry Smith emax = PetscMax(d__1, eps[5]);
237f5af7f23SKarl Rupp
23854f251a9SBarry Smith if (emax <= emin * 4 && dnoise) {
23954f251a9SBarry Smith *fnoise = (eps[3] + eps[4] + eps[5]) / 3;
24054f251a9SBarry Smith if (*fder2 != 0.) {
24154f251a9SBarry Smith *info = 1;
24254f251a9SBarry Smith *hopt = PetscSqrtScalar(*fnoise / PetscAbsScalar(*fder2)) * 1.68;
24354f251a9SBarry Smith } else {
24454f251a9SBarry Smith *info = 4;
24554f251a9SBarry Smith *hopt = *h__ * 10;
24654f251a9SBarry Smith }
2473ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
24854f251a9SBarry Smith }
24954f251a9SBarry Smith /* Noise not detected; decide if h is too small or too large. */
25054f251a9SBarry Smith if (!cancel[3]) {
251f5af7f23SKarl Rupp if (dsgn[3]) *info = 2;
252f5af7f23SKarl Rupp else *info = 3;
2533ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
25454f251a9SBarry Smith }
25554f251a9SBarry Smith if (!cancel[2]) {
256f5af7f23SKarl Rupp if (dsgn[2]) *info = 2;
257f5af7f23SKarl Rupp else *info = 3;
2583ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
25954f251a9SBarry Smith }
260*1cb3f120SPierre Jolivet /* If there is cancellation on the third and fourth column */
26154f251a9SBarry Smith /* then h is too small */
26254f251a9SBarry Smith *info = 2;
2633ba16761SJacob Faibussowitsch PetscFunctionReturn(PETSC_SUCCESS);
26454f251a9SBarry Smith /* if (cancel .or. dsgn(3)) then */
26554f251a9SBarry Smith /* info = 2 */
26654f251a9SBarry Smith /* else */
26754f251a9SBarry Smith /* info = 3 */
26854f251a9SBarry Smith /* end if */
26954f251a9SBarry Smith } /* dnest_ */
270