Ice Sheet System Model  4.18
Code documentation
cubic.cpp
Go to the documentation of this file.
1 #include <math.h>
2 
3 #include "./numerics.h"
4 
6 
7  IssmDouble ret;
8 
9  if (Z> 0.0){
10  ret = fabs(pow(fabs(Z),1./3.));
11  }
12  else if(Z< 0.0){
13  ret = - fabs(pow(fabs(Z),1./3.));
14  }
15  else{
16  ret = 0.;
17  }
18  return ret;
19 }
20 
22  /* Find the real roots of linear/quadratic and cubic equations:
23  *
24  * a x^3 + bx^2 + cx + d = 0
25  *
26  * returns the roots in x
27  * num is the number of roots */
28 
29  /*Some useful constants*/
30  const IssmDouble pi = 3.1415926535897932;
31  const IssmDouble third = 1./3.;
32 
33  /*Intermediaries*/
34  IssmDouble U[3],W,P,Q,delta,phi;
35 
36  /* determine the degree of the polynomial */
37  if (a != 0.0){
38  //cubic problem
39  W = b/a *third;
40  P = pow((c/a *third - pow(W,2)),3);
41  Q = -.5 *(2.0*pow(W,3)-(c*W-d)/a );
42  delta = pow(Q,2)+P;
43  if ( delta < 0.0 ){
44  //three real solutions!
45  //Confine the argument of coeffCOS to the interval [-1;1]!
46  phi = acos(min(1.0,max(-1.0,Q/sqrt(-P))));
47  P = 2.0*pow((-P),(5.e-1*third));
48  for(int i=0;i<3;i++) U[i] = P*cos((phi+2*((IssmDouble)i)*pi)*third)-W;
49  x[0] = min(U[0], min(U[1], U[2]));
50  x[1] = max(min(U[0], U[1]),max( min(U[0], U[2]), min(U[1], U[2])));
51  x[2] = max(U[0], max(U[1], U[2]));
52  *num = 3;
53  }
54  else{
55  // only one real solution!
56  delta = sqrt(delta);
57  x[0] = CBRT(Q+delta)+CBRT(Q-delta)-W;
58  *num=1;
59  }
60  }
61  else if (b != 0.0){
62  // quadratic problem
63  P = 0.5*c/b;
64  delta = pow(P,2)-d/b;
65  if (delta > 0.0){
66  // 2 real solutions
67  x[0] = -P - sqrt(delta);
68  x[1] = -P + sqrt(delta);
69  *num = 2;
70  }
71  else{
72  // no real solution
73  *num = 0;
74  }
75  }
76  else if (c != 0.0){
77  //linear equation
78  x[0] = d/c;
79  *num = 1;
80  }
81  else{
82  //no equation
83  *num = 0;
84  }
85 
86  /* perform one step of a newton iteration in order to minimize round-off errors */
87  for(int i=0;i<*num;i++){
88  x[i] = x[i] - (d+x[i]*(c+x[i]*(b+x[i]*a)))/(c+x[i]*(2.0*b+x[i]*3.0*a));
89  }
90 
91  return 0;
92 }
IssmDouble
double IssmDouble
Definition: types.h:37
cubic
int cubic(IssmDouble a, IssmDouble b, IssmDouble c, IssmDouble d, IssmDouble x[3], int *num)
Definition: cubic.cpp:21
min
IssmDouble min(IssmDouble a, IssmDouble b)
Definition: extrema.cpp:14
numerics.h
prototypes for numerics.h
CBRT
IssmDouble CBRT(IssmDouble Z)
Definition: cubic.cpp:5
max
IssmDouble max(IssmDouble a, IssmDouble b)
Definition: extrema.cpp:24