31 #include "petsc_utils.h"
41 std::map<int,std::string>
units;
44 namespace user_globals {
112 Stim & s = param_globals::stim[
idx];
113 Stimulus & curstim = param_globals::stimulus[
idx];
115 s.ptcl.bcl = curstim.bcl;
116 s.ptcl.npls = curstim.npls;
117 s.ptcl.start = curstim.start;
118 s.ptcl.duration = curstim.duration;
122 std::string pulse_name;
123 pulse_name =
"pulse_" + std::to_string(
idx);
124 s.pulse.name = (
char*) calloc(pulse_name.length()+1,
sizeof(char));
125 strcpy(s.pulse.name, pulse_name.c_str());
129 s.pulse.file = strdup(curstim.pulse_file);
130 s.pulse.bias = curstim.bias;
131 s.pulse.tau_plateau = curstim.tau_plateau;
132 s.pulse.tau_edge = curstim.tau_edge;
133 s.pulse.s2 = curstim.s2;
134 s.pulse.d1 = curstim.d1;
135 s.pulse.strength = curstim.strength;
138 s.elec.vtx_fcn = curstim.vtx_fcn;
141 s.crct.type = curstim.stimtype;
142 s.crct.balance = curstim.balance;
143 s.crct.total_current = curstim.total_current;
146 s.elec.geomID = curstim.geometry;
148 s.elec.vtx_file = strdup(curstim.vtx_file);
149 s.elec.dump_vtx_file = curstim.dump_vtx_file;
152 s.elec.geom_type = 2;
153 s.elec.p0[0] = curstim.x0 - (curstim.ctr_def?curstim.xd/2.:0.);
154 s.elec.p0[1] = curstim.y0 - (curstim.ctr_def?curstim.yd/2.:0.);
155 s.elec.p0[2] = curstim.z0 - (curstim.ctr_def?curstim.zd/2.:0.);
156 s.elec.p1[0] = s.elec.p0[0] + curstim.xd;
157 s.elec.p1[1] = s.elec.p0[1] + curstim.yd;
158 s.elec.p1[2] = s.elec.p0[2] + curstim.zd;
161 s.name = strdup(curstim.name);
171 const Stim & cur_stim = param_globals::stim[
idx];
174 if(cur_stim.name != std::string(
""))
175 name = cur_stim.name;
177 name =
"Stimulus_" + std::to_string(
idx);
182 #ifndef WITH_LEADFIELD
184 log_msg(0, 5, 0,
"Error: LF_I stimulus requested but leadfield support not compiled in. "
185 "Rebuild with -DENABLE_LEADFIELD=ON.");
214 const Stim & cur_stim = param_globals::stim[idx];
215 const Pulse & pulse = cur_stim.pulse;
220 switch (cur_stim.crct.type) {
228 if (pulse.file && strlen(pulse.file))
235 assign(pulse.strength, cur_stim.ptcl.duration, param_globals::dt, wvt);
240 std::string tUnit =
"ms";
241 this->wave.setUnits(tUnit,sUnit);
247 std::string tstim = str_list;
248 std::vector<std::string> stims;
251 stlist.resize(stims.size());
252 size_t widx = 0, err = 0;
254 for(
size_t i=0; i<stims.size(); i++) {
255 std::string & stim = stims[i];
259 double last = widx == 0 ? 0.0 : stlist[widx-1];
265 size_t nread = sscanf(stim.c_str(),
"%lf", &cur);
267 stlist[widx++] = cur + last;
270 size_t nread = sscanf(stim.c_str(),
"%lf", &cur);
272 stlist[widx++] = cur;
279 if(widx != stims.size())
280 log_msg(0, 4, 0,
"%s warning: Some values of %s could not be converted in stim times!",
289 const Stim & cur_stim = param_globals::stim[idx];
290 const Protocol & ptcl = cur_stim.ptcl;
292 if(strlen(ptcl.stimlist) == 0) {
297 param_globals::tend,
npls,
pcl, ptcl.duration, name.c_str());
299 std::vector<double> stims;
303 for(
double t : stims)
306 log_msg(0,0,0,
"stim %d: .bcl and .npls values will be ignored.", idx);
317 const Stim & cur_stim = param_globals::stim[idx];
337 default:
scale = 1.0;
break;
351 const Stim & cur_stim = param_globals::stim[idx];
352 const Pulse & pulse = cur_stim.pulse;
358 stepPars.
trig = pulse.trig;
359 stepPars.
rise =
true;
371 truncExp.
tau_plat = pulse.tau_plateau;
372 truncExp.
s2r = pulse.s2;
373 truncExp.
bias = pulse.bias;
376 if (pulse.d1 == 1.) {
390 sinePars.
frq = pulse.freq;
391 sinePars.
phase = pulse.phase;
405 _err = file_wave.
read_trace(pulse.file, unitize);
412 "Failed reading pulse for stimulus[%d] from file %s.\n", idx, pulse.file);
477 mesh.
pl.globalize(glob_idx);
489 char dump_name[1024];
491 if (strlen(
name.c_str())) {
492 snprintf(dump_name,
sizeof dump_name,
"%s.vtx",
name.c_str());
494 snprintf(dump_name,
sizeof dump_name,
"ELECTRODE_%d.vtx",
idx);
497 f =
f_open(dump_name,
"w");
501 fprintf(f->
fd,
"%zd\nextra\n", num_vtx);
508 log_msg(0, 4, 0,
"error: stimulus[%d] cannot be dumped!");
517 const Stim & curstim = param_globals::stim[idx];
519 dump_vtx = curstim.elec.dump_vtx_file;
522 switch(curstim.crct.type) {
528 grid = extra_mesh;
break;
536 grid = intra_mesh;
break;
541 log_msg(0,5,0,
"stim_electrode::setup error: Can't determine domain from stim type! Aborting!", __func__);
549 bool vertex_file_given = strlen(curstim.elec.vtx_file) > 0;
550 bool tag_index_given = curstim.elec.geomID > -1;
554 if(vertex_file_given && tag_index_given)
555 log_msg(0,3,0,
"%s warning: More than one stimulus electrode definintions set in electrode %d", __func__, idx);
557 if(vertex_file_given) {
568 if(curstim.elec.vtx_fcn)
575 log_msg(0, 5, 0,
"Stimulus %d: Specified vertices are not in stimulus domain! Aborting!", idx);
579 else if(tag_index_given) {
582 int tag = curstim.elec.geomID;
583 log_msg(logger, 0, 0,
"Stimulus %d: Selecting vertices from tag %d", idx, tag);
591 log_msg(logger, 0, 0,
"Stimulus %d: Selecting vertices from shape.", idx);
595 shape.
p0 = curstim.elec.p0;
596 shape.
p1 = curstim.elec.p1;
597 shape.
radius = curstim.elec.radius;
606 log_msg(0,5,0,
"error: Empty stimulus[%d] electrode def! Aborting!", idx);
617 if(
is_dbc(s.phys.type) && s.is_active()) {
630 for(
size_t i=0; i<dbc_idx.
size(); i++)
631 (*dbc_buff->
nod)[widx++] = petsc_nbr[dbc_idx[i]];
636 if(s.electrode.scaling.size()) {
638 dbc->
set(*dbc_buff->
nod, s.electrode.scaling);
641 dbc->
set(*dbc_buff->
nod, 1.0);
655 if(
stimuli[d.first].is_active() ==
false)
676 for (
auto dbc_idx : dbc_nod) {
681 dbc->
set(dbc_nod, 0.0);
686 dbc->
set(dbc_nod, 1.0);
694 const int & dbc_idx = it->first;
696 double strength = 0.0;
697 bool is_active =
stimuli[dbc_idx].value(strength);
706 if(
stimuli[dbc_idx].electrode.scaling.size()) {
711 scaled_strength[widx++] = s * strength;
713 rhs.
set(*dbc.
nod, scaled_strength);
715 rhs.
set(*dbc.
nod, strength);
opencarp::real_t SF_real
Global scalar type.
opencarp::global_index_t SF_int
Global algebraic index type.
const meshdata< mesh_int_t, mesh_real_t > * mesh_ptr() const
virtual void mult(const abstract_vector< T, S > &x, abstract_vector< T, S > &b) const =0
std::set< T > boundary_idxs
virtual void mult_LR(const abstract_vector< T, S > &L, const abstract_vector< T, S > &R)=0
virtual void diag_add(const abstract_vector< T, S > &diag)=0
virtual void add_scaled(const abstract_vector< T, S > &vec, S k)=0
virtual void set(const vector< T > &idx, const vector< S > &vals, const bool additive=false, const bool local=false)=0
overlapping_layout< T > pl
nodal parallel layout
MPI_Comm comm
the parallel mesh is defined on a MPI world
vector< T > & get_numbering(SF_nbr nbr_type)
Get the vector defining a certain numbering.
size_t size() const
The current size of the vector.
std::map< int, dbc_data * > active_dbc
the DBCs that are currently active
sf_mat & mat
the matrix we link the dbc_manager to
void enforce_dbc_rhs(sf_vec &rhs)
void recompute_dbcs()
recompute the dbc data.
const SF::vector< stimulus > & stimuli
the stimuli we link the dbc_manager to
bool dbc_update()
check if dbcs have updated
class to store shape definitions
float s2r
strength of subpulse relative to leading pulse (biphasic pulse)
sReal duration
pulse duration, default is 1 ms
float tau_plat
time constant governing plateau of pulse
float tau_edge
time constant for leading/trailing edges
float bias
constant term to add to stimulus waveform
double d1
duration of first sub-pulse in [ms] (zero with monophasic pulse)
biphasic truncated exponentials (capacitive discharge)
virtual SF::vector< sReal > & sample(time_trace &trc)
virtual SF::vector< sReal > & sample(time_trace &trc)
monophasic truncated exponentials (capacitive discharge)
virtual SF::vector< sReal > & sample(time_trace &trc)
virtual SF::vector< sReal > & sample(time_trace &trc)
float phase
phase in degree
virtual SF::vector< sReal > & sample(time_trace &trc)
SF::vector< sReal > t
time axis
void resample(time_trace &trc)
SF::vector< sReal > f
store function values of trace
int read_trace(const std::string fname)
determine duration of a signal stored in file
SF::vector< mesh_int_t > vertices
void setup(int idx, mesh_t intra_mesh, mesh_t extra_mesh)
std::string input_filename
SF::vector< SF_real > scaling
double scale
internal unit conversion scaling
bool total_current
whether we apply total current scaling
stim_t type
type of stimulus
std::string unit
physical units of stimulus
void setup(int idx, mesh_t intra_mesh, mesh_t extra_mesh)
assign stimulus physics parameters
int timer_id
timer for stimulus
int npls
number of stimulus pulses
double pcl
pacing cycle length
double start
start time of protocol
void setup(int idx, std::string name)
Setup from a param stimulus index.
define the wave form of a stimulation pulse
sig::time_trace wave
wave form of stimulus pulse
void assign(double _strength, double _duration, double _dt, waveform_t _wform)
void setup(int id)
Setup from a param stimulus index.
double duration
duration of stimulus
waveform_t wform
wave form of stimulus
double strength
strength of stimulus
stim_protocol ptcl
applied stimulation protocol used
int idx
index in global input stimulus array
stim_electrode electrode
electrode geometry
mesh_t associated_intra_mesh
stim_pulse pulse
stimulus wave form
mesh_t associated_extra_mesh
void translate(int id)
convert legacy definitions to new format
bool is_active() const
Return whether stim is active.
void setup(int idx)
Setup from a param stimulus index.
void dump_vtx_file(int idx)
Export the vertices to vtx file.
stim_physics phys
physics of stimulus
bool value(double &v) const
Get the current value if the stimulus is active.
std::string name
label stimulus
centralize time managment and output triggering
bool trigger(int ID) const
double time_step
global reference time step
int add_neq_timer(const std::vector< double > &itrig, double idur, const char *iname, const char *poolname=nullptr)
int add_eq_timer(double istart, double iend, int ntrig, double iintv, double idur, const char *iname, const char *poolname=nullptr)
Add a equidistant step timer to the array of timers.
int trigger_elapse(int ID) const
Tissue level electrics, main Electrics physics class.
void print_vector(MPI_Comm comm, const vector< T > &vec, const short dpn, FILE *fd)
void sort_parallel(MPI_Comm comm, const vector< T > &idx, vector< T > &out_idx)
Sort index values parallel ascending across the ranks.
void restrict_to_set(vector< T > &v, const hashmap::unordered_set< T > &set)
void init_vector(SF::abstract_vector< T, S > **vec)
@ NBR_PETSC
PETSc numbering of nodes.
@ NBR_REF
The nodal numbering of the reference mesh (the one stored on HD).
std::set< int > current_stim
std::map< int, std::string > units
std::set< int > potential_stim
timer_manager * tm_manager
a manager for the various physics timers
std::map< physic_t, Basic_physic * > physics_reg
the physics
void sample_wave_form(stim_pulse &sp, int idx)
sample a signal given in analytic form
sf_mesh & get_mesh(const mesh_t gt)
Get a mesh by specifying the gridID.
bool is_dbc(stim_t type)
whether stimulus is a dirichlet type. implies boundary conditions on matrix
int get_rank(MPI_Comm comm=PETSC_COMM_WORLD)
short get_mesh_dim(mesh_t id)
get (lowest) dimension of the mesh used in the experiment
T get_global(T in, MPI_Op OP, MPI_Comm comm=PETSC_COMM_WORLD)
Do a global reduction on a variable.
bool is_potential(stim_t type)
uses current for stimulation
void init_stim_info(void)
uses potential for stimulation
bool is_extra(stim_t type)
whether stimulus is on extra grid (or on intra)
FILE_SPEC f_open(const char *fname, const char *mode)
Open a FILE_SPEC.
bool is_current(stim_t type)
uses current as stimulation
void warn_when_passing_intra_vtx(const std::string filename)
void split_string(const std::string &input, const char s, STRVEC &list)
Split a string holding a character-seperated list into a vector of strings.
void indices_from_region_tag(SF::vector< mesh_int_t > &idx, const sf_mesh &mesh, const int tag)
Populate vertex data with the vertices of a given tag region.
void indices_from_geom_shape(SF::vector< mesh_int_t > &idx, const sf_mesh &mesh, const geom_shape shape, const bool nodal)
Populate vertex data with the vertices inside a defined box shape.
void log_msg(FILE_SPEC out, int level, unsigned char flag, const char *fmt,...)
mesh_t
The enum identifying the different meshes we might want to load.
void read_indices_with_data(SF::vector< T > &idx, SF::vector< S > &dat, const std::string filename, const hashmap::unordered_map< mesh_int_t, mesh_int_t > &dd_map, const int dpn, MPI_Comm comm)
like read_indices, but with associated data for each index
void read_indices(SF::vector< T > &idx, const std::string filename, const hashmap::unordered_map< mesh_int_t, mesh_int_t > &dd_map, MPI_Comm comm)
Read indices from a file.
void get_stim_list(const char *str_list, std::vector< double > &stlist)
void update_cwd()
save the current working directory to curdir so that we can switch back to it if needed.
void f_close(FILE_SPEC &f)
Close a FILE_SPEC.
#define UM_to_CM
convert um to cm
#define UM2_to_CM2
convert um^2 to cm^2
Electrical stimulation functions.
SF::vector< SF_int > * nod