2018-06-05 04:56:47 -05:00
|
|
|
#include "RigGeoMechBoreHoleStressCalculator.h"
|
|
|
|
|
|
|
|
//==================================================================================================
|
|
|
|
/// Internal root finding class to find a Well Pressure that gives:
|
|
|
|
/// a) a zero SigmaT for estimating the fracture gradient.
|
|
|
|
/// b) a solution to the Stassi-d'Alia failure criterion for estimating the shear failure gradient.
|
|
|
|
//==================================================================================================
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2019-09-06 03:40:57 -05:00
|
|
|
RigGeoMechBoreHoleStressCalculator::RigGeoMechBoreHoleStressCalculator( const caf::Ten3d& tensor,
|
|
|
|
double porePressure,
|
|
|
|
double poissonRatio,
|
|
|
|
double uniaxialCompressiveStrength,
|
|
|
|
int nThetaSubSamples )
|
|
|
|
: m_tensor( tensor )
|
|
|
|
, m_porePressure( porePressure )
|
|
|
|
, m_poissonRatio( poissonRatio )
|
|
|
|
, m_uniaxialCompressiveStrength( uniaxialCompressiveStrength )
|
|
|
|
, m_nThetaSubSamples( nThetaSubSamples )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
|
|
|
calculateStressComponents();
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/// Simple bisection method for now
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2019-09-06 03:40:57 -05:00
|
|
|
double RigGeoMechBoreHoleStressCalculator::solveFractureGradient( double* thetaOut )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
|
|
|
MemberFunc fn = &RigGeoMechBoreHoleStressCalculator::sigmaTMinOfMin;
|
2019-09-06 03:40:57 -05:00
|
|
|
return solveSecant( fn, thetaOut );
|
2018-06-05 04:56:47 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2019-09-06 03:40:57 -05:00
|
|
|
double RigGeoMechBoreHoleStressCalculator::solveStassiDalia( double* thetaOut )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
|
|
|
MemberFunc fn = &RigGeoMechBoreHoleStressCalculator::stassiDalia;
|
2019-09-06 03:40:57 -05:00
|
|
|
return solveSecant( fn, thetaOut );
|
2018-06-05 04:56:47 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/// Bi-section root finding method: https://en.wikipedia.org/wiki/Bisection_method
|
|
|
|
/// Used as fall-back in case the secant method doesn't converge.
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2019-09-06 03:40:57 -05:00
|
|
|
double RigGeoMechBoreHoleStressCalculator::solveBisection( double minPw, double maxPw, MemberFunc fn, double* thetaOut )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
const int N = 50;
|
2018-06-05 04:56:47 -05:00
|
|
|
const double epsilon = 1.0e-10;
|
|
|
|
|
|
|
|
double theta = 0.0;
|
|
|
|
|
2019-09-06 03:40:57 -05:00
|
|
|
std::pair<double, double> largestNegativeValue( 0.0, -std::numeric_limits<double>::infinity() );
|
|
|
|
std::pair<double, double> smallestPositiveValue( 0.0, std::numeric_limits<double>::infinity() );
|
|
|
|
|
|
|
|
for ( int i = 0; i <= N; ++i )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
double pw = minPw + ( maxPw - minPw ) * i / static_cast<double>( N );
|
|
|
|
double f_pw = ( this->*fn )( pw, &theta );
|
|
|
|
if ( f_pw >= 0.0 && f_pw < smallestPositiveValue.second )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
smallestPositiveValue = std::make_pair( pw, f_pw );
|
2018-06-05 04:56:47 -05:00
|
|
|
}
|
2019-09-06 03:40:57 -05:00
|
|
|
if ( f_pw < 0.0 && f_pw > largestNegativeValue.second )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
largestNegativeValue = std::make_pair( pw, f_pw );
|
|
|
|
}
|
2018-06-05 04:56:47 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Provide a warning if there was no solution to the equation
|
2019-09-06 03:40:57 -05:00
|
|
|
if ( largestNegativeValue.second == -std::numeric_limits<double>::infinity() )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
|
|
|
// No solution. Function is always positive. Pick smallest value.
|
|
|
|
return smallestPositiveValue.first;
|
|
|
|
}
|
2019-09-06 03:40:57 -05:00
|
|
|
if ( smallestPositiveValue.second == std::numeric_limits<double>::infinity() )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
|
|
|
// No solution. Function is always negative. Pick largest value.
|
2019-09-06 03:40:57 -05:00
|
|
|
return largestNegativeValue.first;
|
2018-06-05 04:56:47 -05:00
|
|
|
}
|
2019-09-06 03:40:57 -05:00
|
|
|
minPw = largestNegativeValue.first;
|
2018-09-03 03:37:29 -05:00
|
|
|
double minPwFuncVal = largestNegativeValue.second;
|
2019-09-06 03:40:57 -05:00
|
|
|
maxPw = smallestPositiveValue.first;
|
2018-09-03 03:37:29 -05:00
|
|
|
double maxPwFuncVal = smallestPositiveValue.second;
|
2018-06-05 04:56:47 -05:00
|
|
|
|
2019-09-06 03:40:57 -05:00
|
|
|
double range = std::abs( maxPw - minPw );
|
|
|
|
|
2018-06-05 04:56:47 -05:00
|
|
|
int i = 0;
|
2019-09-06 03:40:57 -05:00
|
|
|
for ( ; i <= N && range > m_porePressure * epsilon; ++i )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
double midPw = ( minPw + maxPw ) * 0.5;
|
|
|
|
double midPwFuncVal = ( this->*fn )( midPw, &theta );
|
|
|
|
if ( midPwFuncVal * minPwFuncVal < 0.0 )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
maxPw = midPw;
|
2018-06-05 04:56:47 -05:00
|
|
|
maxPwFuncVal = midPwFuncVal;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
minPw = midPw;
|
2018-06-05 04:56:47 -05:00
|
|
|
minPwFuncVal = midPwFuncVal;
|
|
|
|
}
|
2019-09-06 03:40:57 -05:00
|
|
|
range = std::abs( maxPw - minPw );
|
2018-06-05 04:56:47 -05:00
|
|
|
}
|
2019-09-06 03:40:57 -05:00
|
|
|
CVF_ASSERT( i < N ); // Otherwise it hasn't converged
|
2018-06-05 04:56:47 -05:00
|
|
|
|
2019-09-06 03:40:57 -05:00
|
|
|
if ( thetaOut )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
|
|
|
*thetaOut = theta;
|
|
|
|
}
|
2019-09-06 03:40:57 -05:00
|
|
|
|
2018-06-05 04:56:47 -05:00
|
|
|
// Return average of minPw and maxPw.
|
2019-09-06 03:40:57 -05:00
|
|
|
return 0.5 * ( maxPw + minPw );
|
2018-06-05 04:56:47 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
/// Secant root finding method: https://en.wikipedia.org/wiki/Secant_method
|
|
|
|
/// Basically a Newton's method using finite differences for the derivative.
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2019-09-06 03:40:57 -05:00
|
|
|
double RigGeoMechBoreHoleStressCalculator::solveSecant( MemberFunc fn, double* thetaOut )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
|
|
|
const double epsilon = 1.0e-10;
|
2019-09-06 03:40:57 -05:00
|
|
|
const int N = 50;
|
|
|
|
double theta = 0.0;
|
|
|
|
|
|
|
|
double x_0 = 0.0;
|
|
|
|
double f_x0 = ( this->*fn )( x_0, &theta );
|
|
|
|
double x_1 = m_porePressure;
|
|
|
|
double f_x1 = ( this->*fn )( x_1, &theta );
|
|
|
|
double x = 0.0;
|
|
|
|
double f_x = 0.0;
|
|
|
|
int i = 0;
|
|
|
|
for ( ; i <= N && std::abs( f_x1 - f_x0 ) > epsilon; ++i )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
x = x_1 - f_x1 * ( x_1 - x_0 ) / ( f_x1 - f_x0 );
|
|
|
|
f_x = ( this->*fn )( x, &theta );
|
|
|
|
if ( std::abs( f_x ) < epsilon * m_porePressure ) break;
|
2018-06-05 04:56:47 -05:00
|
|
|
|
|
|
|
// Update iteration variables
|
2019-09-06 03:40:57 -05:00
|
|
|
x_0 = x_1;
|
2018-06-05 04:56:47 -05:00
|
|
|
f_x0 = f_x1;
|
2019-09-06 03:40:57 -05:00
|
|
|
x_1 = x;
|
|
|
|
f_x1 = f_x;
|
2018-06-05 04:56:47 -05:00
|
|
|
}
|
|
|
|
|
2019-09-06 03:40:57 -05:00
|
|
|
if ( i == N || std::abs( f_x ) > epsilon * m_porePressure )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
2018-09-03 03:37:29 -05:00
|
|
|
// Fallback to bisection if secant doesn't converge or converged to a wrong solution.
|
2019-09-06 03:40:57 -05:00
|
|
|
return solveBisection( 0.0, m_porePressure * 2.0, fn, thetaOut );
|
2018-06-05 04:56:47 -05:00
|
|
|
}
|
|
|
|
|
2019-09-06 03:40:57 -05:00
|
|
|
if ( thetaOut )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
|
|
|
*thetaOut = theta;
|
|
|
|
}
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2019-09-06 03:40:57 -05:00
|
|
|
double RigGeoMechBoreHoleStressCalculator::sigmaTMinOfMin( double wellPressure, double* thetaAtMin ) const
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
CVF_ASSERT( thetaAtMin );
|
2018-06-05 04:56:47 -05:00
|
|
|
double sigma_t_min_min = std::numeric_limits<double>::max();
|
2019-09-06 03:40:57 -05:00
|
|
|
for ( const cvf::Vec4d& stressComponentsForAngle : m_stressComponents )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
|
|
|
// Perform all these internal calculations in double to reduce significance errors
|
2019-09-06 03:40:57 -05:00
|
|
|
double sigma_theta = stressComponentsForAngle[1] - wellPressure;
|
|
|
|
const double& sigma_z = stressComponentsForAngle[2];
|
|
|
|
double tauSqrx4 = std::pow( stressComponentsForAngle[3], 2 ) * 4.0;
|
|
|
|
double sigma_t_min = 0.5 * ( ( sigma_z + sigma_theta ) -
|
|
|
|
std::sqrt( std::pow( sigma_z - sigma_theta, 2 ) + tauSqrx4 ) ) -
|
|
|
|
m_porePressure;
|
|
|
|
if ( sigma_t_min < sigma_t_min_min )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
|
|
|
sigma_t_min_min = sigma_t_min;
|
2019-09-06 03:40:57 -05:00
|
|
|
*thetaAtMin = stressComponentsForAngle[0];
|
2018-06-05 04:56:47 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return sigma_t_min_min;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2019-09-06 03:40:57 -05:00
|
|
|
double RigGeoMechBoreHoleStressCalculator::stassiDalia( double wellPressure, double* thetaAtMin ) const
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
CVF_ASSERT( thetaAtMin );
|
2018-06-05 04:56:47 -05:00
|
|
|
double minStassiDalia = std::numeric_limits<double>::max();
|
2019-09-06 03:40:57 -05:00
|
|
|
for ( const cvf::Vec4d& stressComponentsForAngle : m_stressComponents )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
double sigma_theta = stressComponentsForAngle[1] - wellPressure;
|
|
|
|
const double& sigma_z = stressComponentsForAngle[2];
|
|
|
|
double tauSqrx4 = std::pow( stressComponentsForAngle[3], 2 ) * 4.0;
|
2018-06-05 04:56:47 -05:00
|
|
|
|
|
|
|
double sigma_1 = wellPressure - m_porePressure;
|
2019-09-06 03:40:57 -05:00
|
|
|
double sigma_2 = 0.5 * ( ( sigma_z + sigma_theta ) +
|
|
|
|
std::sqrt( std::pow( sigma_z - sigma_theta, 2 ) + tauSqrx4 ) ) -
|
|
|
|
m_porePressure;
|
|
|
|
double sigma_3 = 0.5 * ( ( sigma_z + sigma_theta ) -
|
|
|
|
std::sqrt( std::pow( sigma_z - sigma_theta, 2 ) + tauSqrx4 ) ) -
|
|
|
|
m_porePressure;
|
|
|
|
|
|
|
|
double stassiDalia = std::pow( sigma_1 - sigma_2, 2 ) + std::pow( sigma_2 - sigma_3, 2 ) +
|
|
|
|
std::pow( sigma_1 - sigma_3, 2 ) -
|
|
|
|
2 * m_uniaxialCompressiveStrength * ( sigma_1 + sigma_2 + sigma_3 );
|
|
|
|
|
|
|
|
if ( stassiDalia < minStassiDalia )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
|
|
|
minStassiDalia = stassiDalia;
|
2019-09-06 03:40:57 -05:00
|
|
|
*thetaAtMin = stressComponentsForAngle[0];
|
2018-06-05 04:56:47 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return minStassiDalia;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
void RigGeoMechBoreHoleStressCalculator::calculateStressComponents()
|
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
m_stressComponents.reserve( m_nThetaSubSamples );
|
2018-06-05 04:56:47 -05:00
|
|
|
|
2019-09-06 03:40:57 -05:00
|
|
|
for ( int i = 0; i < m_nThetaSubSamples; ++i )
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
2019-09-06 03:40:57 -05:00
|
|
|
double theta = ( i * cvf::PI_F ) / ( m_nThetaSubSamples - 1.0 );
|
|
|
|
cvf::Vec4d stressComponentsForAngle = calculateStressComponentsForSegmentAngle( theta );
|
|
|
|
m_stressComponents.push_back( stressComponentsForAngle );
|
2018-06-05 04:56:47 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
|
|
|
///
|
|
|
|
//--------------------------------------------------------------------------------------------------
|
2019-09-06 03:40:57 -05:00
|
|
|
cvf::Vec4d RigGeoMechBoreHoleStressCalculator::calculateStressComponentsForSegmentAngle( double theta ) const
|
2018-06-05 04:56:47 -05:00
|
|
|
{
|
|
|
|
cvf::Vec4d stressComponents;
|
|
|
|
|
2019-09-06 03:40:57 -05:00
|
|
|
const double& sx = m_tensor[caf::Ten3d::SXX];
|
|
|
|
const double& sy = m_tensor[caf::Ten3d::SYY];
|
|
|
|
const double& sz = m_tensor[caf::Ten3d::SZZ];
|
2018-06-05 04:56:47 -05:00
|
|
|
const double& txy = m_tensor[caf::Ten3d::SXY];
|
|
|
|
const double& txz = m_tensor[caf::Ten3d::SZX];
|
|
|
|
const double& tyz = m_tensor[caf::Ten3d::SYZ];
|
|
|
|
|
|
|
|
stressComponents[0] = theta;
|
2019-09-06 03:40:57 -05:00
|
|
|
stressComponents[1] = sx + sy - 2 * ( sx - sy ) * cos( 2 * theta ) - 4 * txy * sin( 2 * theta );
|
|
|
|
stressComponents[2] = sz - m_poissonRatio * ( 2 * ( sx - sy ) * cos( 2 * theta ) + 4 * txy * sin( 2 * theta ) );
|
|
|
|
stressComponents[3] = 2 * ( tyz * cos( theta ) - txz * sin( theta ) );
|
2018-06-05 04:56:47 -05:00
|
|
|
|
|
|
|
return stressComponents;
|
|
|
|
}
|