Coverage for pySDC/projects/Monodomain/problem_classes/TestODE.py: 91%
76 statements
« prev ^ index » next coverage.py v7.6.1, created at 2024-09-19 09:13 +0000
« prev ^ index » next coverage.py v7.6.1, created at 2024-09-19 09:13 +0000
1import logging
2import numpy as np
3from pySDC.core.problem import Problem
4from pySDC.core.common import RegisterParams
5from pySDC.implementations.datatype_classes.mesh import mesh
6from pySDC.projects.Monodomain.datatype_classes.my_mesh import imexexp_mesh
9"""
10Here we define the problems classes for the multirate Dahlquist test equation y'=lambda_I*y + lambda_E*y + lambda_e*y
11Things are done so that it is compatible witht the sweepers.
12"""
15class Parabolic(RegisterParams):
16 def __init__(self, **problem_params):
17 self._makeAttributeAndRegister(*problem_params.keys(), localVars=problem_params, readOnly=True)
18 self.shape = (1,)
19 self.init = ((1,), None, np.dtype("float64"))
22class TestODE(Problem):
23 def __init__(self, **problem_params):
24 self.logger = logging.getLogger("step")
26 self.parabolic = Parabolic(**problem_params)
27 self.size = 1 # one state variable
28 self.init = ((self.size, *self.parabolic.init[0]), self.parabolic.init[1], self.parabolic.init[2])
30 # invoke super init
31 super(TestODE, self).__init__(self.init)
32 # store all problem params dictionary values as attributes
33 self._makeAttributeAndRegister(*problem_params.keys(), localVars=problem_params, readOnly=True)
35 # initial and end time
36 self.t0 = 0.0
37 self.Tend = 1.0 if self.end_time < 0.0 else self.end_time
39 # set lambdas, if not provided by user
40 if not hasattr(self, 'lmbda_laplacian'):
41 self.lmbda_laplacian = -5.0
42 if not hasattr(self, 'lmbda_gating'):
43 self.lmbda_gating = -10.0
44 if not hasattr(self, 'lmbda_others'):
45 self.lmbda_others = -1.0
47 self.dtype_u = mesh
48 self.dtype_f = mesh
50 def init_exp_extruded(self, new_dim_shape):
51 return ((*new_dim_shape, 1, self.init[0][1]), self.init[1], self.init[2])
53 def initial_value(self):
54 u0 = self.dtype_u(self.init, val=1.0)
56 return u0
58 def eval_f(self, u, t, fh=None):
59 if fh is None:
60 fh = self.dtype_f(init=self.init, val=0.0)
62 fh[0] = (self.lmbda_laplacian + self.lmbda_gating + self.lmbda_others) * u[0]
64 return fh
67class MultiscaleTestODE(TestODE):
68 def __init__(self, **problem_params):
69 super(MultiscaleTestODE, self).__init__(**problem_params)
71 self.dtype_f = imexexp_mesh
73 self.rhs_stiff_indeces = [0]
74 self.rhs_stiff_args = [0]
75 self.rhs_nonstiff_indeces = [0]
76 self.rhs_nonstiff_args = [0]
77 self.rhs_exp_args = [0]
78 self.rhs_exp_indeces = [0]
79 self.rhs_non_exp_indeces = []
81 self.constant_lambda_and_phi = True
83 self.one = self.dtype_u(init=self.init, val=1.0)
85 def solve_system(self, rhs, factor, u0, t, u_sol=None):
86 if u_sol is None:
87 u_sol = self.dtype_u(init=self.init, val=0.0)
89 u_sol[0] = rhs[0] / (1 - factor * self.lmbda_laplacian)
91 return u_sol
93 def eval_f(self, u, t, eval_impl=True, eval_expl=True, eval_exp=True, fh=None, zero_untouched_indeces=True):
95 if fh is None:
96 fh = self.dtype_f(init=self.init, val=0.0)
98 if eval_expl:
99 fh.expl[0] = self.lmbda_others * u[0]
101 if eval_impl:
102 fh.impl[0] = self.lmbda_laplacian * u[0]
104 if eval_exp:
105 fh.exp[0] = self.lmbda_gating * u[0]
107 return fh
109 def eval_lmbda_yinf_exp(self, u, lmbda, yinf):
110 lmbda[0] = self.lmbda_gating
111 yinf[0] = 0.0
113 def lmbda_eval(self, u, t, lmbda=None):
114 if lmbda is None:
115 lmbda = self.dtype_u(init=self.init, val=0.0)
117 lmbda[0] = self.lmbda_gating
119 return lmbda