38 seulex::stepFactor1_ = 0.6,
39 seulex::stepFactor2_ = 0.93,
40 seulex::stepFactor3_ = 0.1,
41 seulex::stepFactor4_ = 4,
42 seulex::stepFactor5_ = 0.5,
43 seulex::kFactor1_ = 0.7,
44 seulex::kFactor2_ = 0.9;
53 jacRedo_(
min(1
e-4,
min(relTol_))),
56 coeff_(iMaxx_, iMaxx_),
73 const scalar cpuFunc = 1, cpuJac = 5, cpuLU = 1, cpuSolve = 1;
78 for (
int i=2; i<iMaxx_; i++)
80 nSeq_[i] = 2*nSeq_[i-2];
82 cpu_[0] = cpuJac + cpuLU + nSeq_[0]*(cpuFunc + cpuSolve);
84 for (
int k=0;
k<kMaxx_;
k++)
86 cpu_[
k+1] = cpu_[
k] + (nSeq_[
k+1]-1)*(cpuFunc + cpuSolve) + cpuLU;
90 for (
int k=0;
k<iMaxx_;
k++)
92 for (
int l=0; l<
k; l++)
94 scalar ratio = scalar(nSeq_[k])/nSeq_[l];
95 coeff_(k, l) = 1/(ratio - 1);
103 bool Foam::seulex::seul
114 scalar dx = dxTot/nSteps;
120 a_(i, j) = -dfdy_(i, j);
128 scalar xnew = x0 + dx;
134 for (
label nn=1; nn<nSteps; nn++)
144 dy1 +=
sqr(dy_[i]/scale[i]);
151 dy_[i] = dydx_[i] - dy_[i]/dx;
161 const scalar denom =
min(1, dy1 + small);
167 if (
mag(dy_[i]) > scale[i]*denom)
173 dy2 +=
sqr(dy_[i]/scale[i]);
190 y[i] = yTemp_[i] + dy_[i];
197 void Foam::seulex::extrapolate
204 for (
int j=k-1; j>0; j--)
209 table(j, i) + coeff_(k, j)*(table(j, i) - table[j-1][i]);
213 for (
int i=0; i<
n_; i++)
215 y[i] = table(0, i) + coeff_(k, 0)*(table(0, i) - y[i]);
253 scalar dx = step.
dxTry;
255 dxOpt_[0] =
mag(0.1*dx);
266 kTarg_ =
max(1,
min(kMaxx_ - 1,
int(logTol)));
274 bool jacUpdated =
false;
276 if (theta_ > jacRedo_)
283 scalar dxNew =
mag(dx);
286 while (firstk || step.
reject)
288 dx = step.
forward ? dxNew : -dxNew;
295 <<
"step size underflow :" << dx <<
endl;
300 for (k=0; k<=kTarg_+1; k++)
302 bool success = seul(x, y0_, dx, k, ySequence_, scale_);
307 dxNew =
mag(dx)*stepFactor5_;
319 table_[k-1][i] = ySequence_[i];
325 extrapolate(k, table_, y);
330 err +=
sqr((y[i] - table_(0, i))/scale_[i]);
333 if (err > 1/small || (k > 1 && err >= errOld))
336 dxNew =
mag(dx)*stepFactor5_;
339 errOld =
min(4*err, 1);
340 scalar expo = 1.0/(k + 1);
341 scalar facmin =
pow(stepFactor3_, expo);
349 fac = stepFactor2_/
pow(err/stepFactor1_, expo);
350 fac =
max(facmin/stepFactor4_,
min(1/facmin, fac));
352 dxOpt_[
k] =
mag(dx*fac);
353 temp_[
k] = cpu_[
k]/dxOpt_[
k];
355 if ((step.
first || step.
last) && err <= 1)
371 else if (err > nSeq_[kTarg_]*nSeq_[kTarg_ + 1]*4)
375 if (kTarg_>1 && temp_[k-1] < kFactor1_*temp_[k])
379 dxNew = dxOpt_[kTarg_];
390 else if (err > nSeq_[k + 1]*2)
393 if (kTarg_>1 && temp_[k-1] < kFactor1_*temp_[k])
397 dxNew = dxOpt_[kTarg_];
410 && temp_[kTarg_-1] < kFactor1_*temp_[kTarg_]
415 dxNew = dxOpt_[kTarg_];
428 if (theta_ > jacRedo_ && !jacUpdated)
447 else if (k <= kTarg_)
450 if (temp_[k-1] < kFactor1_*temp_[k])
454 else if (temp_[k] < kFactor2_*temp_[k - 1])
456 kopt =
min(k + 1, kMaxx_ - 1);
462 if (k > 2 && temp_[k-2] < kFactor1_*temp_[k - 1])
466 if (temp_[k] < kFactor2_*temp_[kopt])
468 kopt =
min(k, kMaxx_ - 1);
474 kTarg_ =
min(kopt, k);
475 dxNew =
min(
mag(dx), dxOpt_[kTarg_]);
482 dxNew = dxOpt_[kopt];
486 if (k < kTarg_ && temp_[k] < kFactor2_*temp_[k - 1])
488 dxNew = dxOpt_[
k]*cpu_[kopt + 1]/cpu_[
k];
492 dxNew = dxOpt_[
k]*cpu_[kopt]/cpu_[
k];
void shallowResize(const label m, const label n)
Resize the matrix without reallocating storage (unsafe)
#define forAll(list, i)
Loop across all elements in list.
intWM_LABEL_SIZE_t label
A label is an int32_t or int64_t as specified by the pre-processor macro WM_LABEL_SIZE.
Abstract base class for the systems of ordinary differential equations.
virtual bool resize()=0
Resize the ODE solver.
A list of keyword definitions, which are a keyword followed by any number of values (e...
dimensioned< Type > max(const dimensioned< Type > &, const dimensioned< Type > &)
void LUDecompose(scalarSquareMatrix &matrix, labelList &pivotIndices)
LU decompose the matrix with pivoting.
dimensionedSymmTensor sqr(const dimensionedVector &dv)
dimensionedScalar sqrt(const dimensionedScalar &ds)
Ostream & endl(Ostream &os)
Add newline and flush stream.
dimensionedScalar y0(const dimensionedScalar &ds)
label k
Boltzmann constant.
virtual void solve(scalar &x, scalarField &y, stepState &step) const
Solve the ODE system and the update the state.
An ODE solver for chemistry.
Macros for easy insertion into run-time selection tables.
virtual void jacobian(const scalar x, const scalarField &y, scalarField &dfdx, scalarSquareMatrix &dfdy) const =0
Calculate the Jacobian of the system.
virtual bool resize()
Resize the ODE solver.
void resizeMatrix(scalarSquareMatrix &m) const
const ODESystem & odes_
Reference to ODESystem.
addToRunTimeSelectionTable(ensightPart, ensightPartCells, istream)
defineTypeNameAndDebug(combustionModel, 0)
dimensioned< Type > min(const dimensioned< Type > &, const dimensioned< Type > &)
dimensionedScalar pow(const dimensionedScalar &ds, const dimensionedScalar &expt)
#define WarningInFunction
Report a warning using Foam::Warning.
scalarField relTol_
Relative convergence tolerance per step.
Abstract base-class for ODE system solvers.
scalarField absTol_
Absolute convergence tolerance per step.
seulex(const ODESystem &ode, const dictionary &dict)
Construct from ODESystem.
dimensioned< scalar > mag(const dimensioned< Type > &)
static void resizeField(UList< Type > &f, const label n)
const doubleScalar e
Elementary charge.
label n_
Size of the ODESystem (adjustable)
dimensionedScalar log10(const dimensionedScalar &ds)
virtual void derivatives(const scalar x, const scalarField &y, scalarField &dydx) const =0
Calculate the derivatives in dydx.
void LUBacksubstitute(const scalarSquareMatrix &luMmatrix, const labelList &pivotIndices, List< Type > &source)
LU back-substitution with given source, returning the solution.