forked from ecell/epdp
-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathGreensFunction3DSym.cpp
147 lines (110 loc) · 3.38 KB
/
GreensFunction3DSym.cpp
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
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif /* HAVE_CONFIG_H */
#include <stdexcept>
#include <sstream>
#include <boost/format.hpp>
#include <gsl/gsl_errno.h>
#include <gsl/gsl_roots.h>
#include "GreensFunction3DSym.hpp"
Real GreensFunction3DSym::p_r(Real r, Real t) const
{
const Real D( getD() );
const Real Dt( D * t );
const Real Dt4( 4.0 * Dt );
const Real Dt4Pi( Dt4 * M_PI );
const Real term1( 1.0 / sqrt( gsl_pow_3( Dt4Pi ) ) );
const Real term2( exp( - r * r / Dt4 ) );
const Real jacobian( 4.0 * r * r * M_PI );
return jacobian * term1 * term2;
}
Real GreensFunction3DSym::ip_r(Real r, Real t) const
{
const Real D( getD() );
const Real Dt( D * t );
const Real sqrtDt_r( 1.0 / sqrt( D * t ) );
const Real sqrtPi_r( 1.0 / sqrt( M_PI ) );
const Real term1( exp( - r * r / ( 4.0 * Dt ) ) *
r * sqrtDt_r * sqrtPi_r );
const Real term2( erf( r * 0.5 * sqrtDt_r ) );
return term2 - term1;
}
struct ip_r_params
{
GreensFunction3DSym const* const gf;
const Real t;
const Real value;
};
static Real ip_r_F(Real r, ip_r_params const* params)
{
const GreensFunction3DSym* const gf( params->gf );
const Real t( params->t );
const Real value( params->value );
return gf->ip_r( r, t ) - value;
}
Real GreensFunction3DSym::drawR(Real rnd, Real t) const
{
// input parameter range checks.
if ( !(rnd <= 1.0 && rnd >= 0.0 ) )
{
throw std::invalid_argument( ( boost::format( "GreensFunction3DSym: rnd <= 1.0 && rnd >= 0.0 : rnd=%.16g" ) % rnd ).str() );
}
if ( !(t >= 0.0 ) )
{
throw std::invalid_argument( ( boost::format( "GreensFunction3DSym: t >= 0.0 : t=%.16g" ) % t ).str() );
}
// t == 0 or D == 0 means no move.
if( t == 0.0 || getD() == 0.0 )
{
return 0.0;
}
ip_r_params params = { this, t, rnd };
gsl_function F =
{
reinterpret_cast<typeof(F.function)>( &ip_r_F ),
¶ms
};
Real max_r( 4.0 * sqrt( 6.0 * getD() * t ) );
while( GSL_FN_EVAL( &F, max_r ) < 0.0 )
{
max_r *= 10;
}
const gsl_root_fsolver_type* solverType( gsl_root_fsolver_brent );
gsl_root_fsolver* solver( gsl_root_fsolver_alloc( solverType ) );
gsl_root_fsolver_set( solver, &F, 0.0, max_r );
const unsigned int maxIter( 100 );
unsigned int i( 0 );
while( true )
{
gsl_root_fsolver_iterate( solver );
const Real low( gsl_root_fsolver_x_lower( solver ) );
const Real high( gsl_root_fsolver_x_upper( solver ) );
const int status( gsl_root_test_interval( low, high, 1e-15,
this->TOLERANCE ) );
if( status == GSL_CONTINUE )
{
if( i >= maxIter )
{
gsl_root_fsolver_free( solver );
throw std::runtime_error("GreensFunction3DSym: drawR: failed to converge");
}
}
else
{
break;
}
++i;
}
//printf("%d\n", i );
const Real r( gsl_root_fsolver_root( solver ) );
gsl_root_fsolver_free( solver );
return r;
}
std::string GreensFunction3DSym::dump() const
{
std::ostringstream ss;
ss << "D = " << this->getD() << std::endl;
return ss.str();
}
Logger& GreensFunction3DSym::log_(
Logger::get_logger("GreensFunction3DSym"));