CoreNEURON
eion.cpp
Go to the documentation of this file.
1 /*
2 # =============================================================================
3 # Copyright (c) 2016 - 2022 Blue Brain Project/EPFL
4 #
5 # See top-level LICENSE file for details.
6 # =============================================================================.
7 */
8 
9 /// THIS FILE IS AUTO GENERATED DONT MODIFY IT.
10 
11 #include <math.h>
12 #include <string.h>
13 
15 #include "coreneuron/mpi/nrnmpi.h"
19 
20 #define _STRIDE _cntml_padded + _iml
21 
22 namespace coreneuron {
23 
24 // for each ion it refers to internal concentration, external concentration, and charge,
25 const int ion_global_map_member_size = 3;
26 
27 
28 #define nparm 5
29 static const char* mechanism[] = {/*just a template*/
30  "0",
31  "na_ion",
32  "ena",
33  "nao",
34  "nai",
35  0,
36  "ina",
37  "dina_dv_",
38  0,
39  0};
40 
41 void nrn_init_ion(NrnThread*, Memb_list*, int);
42 void nrn_alloc_ion(double*, Datum*, int);
43 
44 static int na_ion, k_ion, ca_ion; /* will get type for these special ions */
45 
46 int nrn_is_ion(int type) {
47  // Old: commented to remove dependency on memb_func and alloc function
48  // return (memb_func[type].alloc == ion_alloc);
49  return (type < nrn_ion_global_map_size // type smaller than largest ion's
50  && nrn_ion_global_map[type] != nullptr); // allocated ion charge variables
51 }
52 
54 double** nrn_ion_global_map;
55 #define global_conci(type) nrn_ion_global_map[type][0]
56 #define global_conco(type) nrn_ion_global_map[type][1]
57 #define global_charge(type) nrn_ion_global_map[type][2]
58 
59 double nrn_ion_charge(int type) {
60  return global_charge(type);
61 }
62 
63 void ion_reg(const char* name, double valence) {
64  char buf[7][50];
65 #define VAL_SENTINAL -10000.
66 
67  sprintf(buf[0], "%s_ion", name);
68  sprintf(buf[1], "e%s", name);
69  sprintf(buf[2], "%si", name);
70  sprintf(buf[3], "%so", name);
71  sprintf(buf[5], "i%s", name);
72  sprintf(buf[6], "di%s_dv_", name);
73  for (int i = 0; i < 7; i++) {
74  mechanism[i + 1] = buf[i];
75  }
76  mechanism[5] = nullptr; /* buf[4] not used above */
77  int mechtype = nrn_get_mechtype(buf[0]);
78  if (mechtype >= nrn_ion_global_map_size ||
79  nrn_ion_global_map[mechtype] == nullptr) { // if hasn't yet been allocated
80 
81  // allocates mem for ion in ion_map and sets null all non-ion types
82  if (nrn_ion_global_map_size <= mechtype) {
83  int size = mechtype + 1;
84  nrn_ion_global_map = (double**) erealloc(nrn_ion_global_map, sizeof(double*) * size);
85 
86  for (int i = nrn_ion_global_map_size; i < mechtype; i++) {
87  nrn_ion_global_map[i] = nullptr;
88  }
89  nrn_ion_global_map_size = mechtype + 1;
90  }
92  sizeof(double));
93 
94  register_mech((const char**) mechanism,
97  nullptr,
98  nullptr,
100  nullptr,
101  nullptr,
102  -1,
103  1);
104  mechtype = nrn_get_mechtype(mechanism[1]);
105  _nrn_layout_reg(mechtype, SOA_LAYOUT);
106  hoc_register_prop_size(mechtype, nparm, 1);
107  hoc_register_dparam_semantics(mechtype, 0, "iontype");
108  nrn_writes_conc(mechtype, 1);
109 
110  {
111  // See https://en.cppreference.com/w/cpp/io/c/fprintf: If a call to
112  // sprintf or snprintf causes copying to take place between objects
113  // that overlap, the behavior is undefined.
114  std::string const old_buf_0{buf[0]};
115  sprintf(buf[0], "%si0_%s", name, old_buf_0.c_str());
116  }
117  sprintf(buf[1], "%so0_%s", name, buf[0]);
118  if (strcmp("na", name) == 0) {
119  na_ion = mechtype;
120  global_conci(mechtype) = DEF_nai;
121  global_conco(mechtype) = DEF_nao;
122  global_charge(mechtype) = 1.;
123  } else if (strcmp("k", name) == 0) {
124  k_ion = mechtype;
125  global_conci(mechtype) = DEF_ki;
126  global_conco(mechtype) = DEF_ko;
127  global_charge(mechtype) = 1.;
128  } else if (strcmp("ca", name) == 0) {
129  ca_ion = mechtype;
130  global_conci(mechtype) = DEF_cai;
131  global_conco(mechtype) = DEF_cao;
132  global_charge(mechtype) = 2.;
133  } else {
134  global_conci(mechtype) = DEF_ioni;
135  global_conco(mechtype) = DEF_iono;
136  global_charge(mechtype) = VAL_SENTINAL;
137  }
138  }
139  double val = global_charge(mechtype);
140  if (valence != VAL_SENTINAL && val != VAL_SENTINAL && valence != val) {
141  fprintf(stderr,
142  "%s ion valence defined differently in\n\
143 two USEION statements (%g and %g)\n",
144  buf[0],
145  valence,
146  global_charge(mechtype));
147  nrn_exit(1);
148  } else if (valence == VAL_SENTINAL && val == VAL_SENTINAL) {
149  fprintf(stderr,
150  "%s ion valence must be defined in\n\
151 the USEION statement of any model using this ion\n",
152  buf[0]);
153  nrn_exit(1);
154  } else if (valence != VAL_SENTINAL) {
155  global_charge(mechtype) = valence;
156  }
157 }
158 
159 #if VECTORIZE
160 #define erev pd[0 * _STRIDE] /* From Eion */
161 #define conci pd[1 * _STRIDE]
162 #define conco pd[2 * _STRIDE]
163 #define cur pd[3 * _STRIDE]
164 #define dcurdv pd[4 * _STRIDE]
165 
166 /*
167  handle erev, conci, conc0 "in the right way" according to ion_style
168  default. See nrn/lib/help/nrnoc.help.
169 ion_style("name_ion", [c_style, e_style, einit, eadvance, cinit])
170 
171  ica is assigned
172  eca is parameter but if conc exists then eca is assigned
173  if conc is nrnocCONST then eca calculated on finitialize
174  if conc is STATE then eca calculated on fadvance and conc finitialize
175  with global nai0, nao0
176 
177  nernst(ci, co, charge) and ghk(v, ci, co, charge) available to hoc
178  and models.
179 */
180 
181 #define iontype ppd[_iml] /* how _AMBIGUOUS is to be handled */
182 /*the bitmap is
183 03 concentration unused, nrnocCONST, DEP, STATE
184 04 initialize concentrations
185 030 reversal potential unused, nrnocCONST, DEP, STATE
186 040 initialize reversal potential
187 0100 calc reversal during fadvance
188 0200 ci being written by a model
189 0400 co being written by a model
190 */
191 
192 #define charge global_charge(type)
193 #define conci0 global_conci(type)
194 #define conco0 global_conco(type)
195 
196 double nrn_nernst_coef(int type) {
197  /* for computing jacobian element dconc'/dconc */
198  return ktf(celsius) / charge;
199 }
200 
201 /* Must be called prior to any channels which update the currents */
202 void nrn_cur_ion(NrnThread* nt, Memb_list* ml, int type) {
203  int _cntml_actual = ml->nodecount;
204  double* pd;
205  Datum* ppd;
206  (void) nt; /* unused */
207  /*printf("ion_cur %s\n", memb_func[type].sym->name);*/
208  int _cntml_padded = ml->_nodecount_padded;
209  pd = ml->data;
210  ppd = ml->pdata;
211  // clang-format off
212  nrn_pragma_acc(parallel loop present(pd[0:_cntml_padded * 5],
213  ppd[0:_cntml_actual],
216  if (nt->compute_gpu)
217  async(nt->stream_id))
218  // clang-format on
219  nrn_pragma_omp(target teams distribute parallel for simd if(nt->compute_gpu))
220  for (int _iml = 0; _iml < _cntml_actual; ++_iml) {
221  dcurdv = 0.;
222  cur = 0.;
223  if (iontype & 0100) {
224  erev = nrn_nernst(conci, conco, charge, celsius);
225  }
226  };
227 }
228 
229 /* Must be called prior to other models which possibly also initialize
230  concentrations based on their own states
231 */
232 void nrn_init_ion(NrnThread* nt, Memb_list* ml, int type) {
233  int _cntml_actual = ml->nodecount;
234  double* pd;
235  Datum* ppd;
236  (void) nt; /* unused */
237 
238  // skip initialization if restoring from checkpoint
239  if (_nrn_skip_initmodel == 1) {
240  return;
241  }
242 
243  /*printf("ion_init %s\n", memb_func[type].sym->name);*/
244  int _cntml_padded = ml->_nodecount_padded;
245  pd = ml->data;
246  ppd = ml->pdata;
247  // There was no async(...) clause in the initial OpenACC implementation, so
248  // no `nowait` clause has been added to the OpenMP implementation. TODO:
249  // verify if this can be made asynchronous or if there is a strong reason it
250  // needs to be like this.
251  // clang-format off
252  nrn_pragma_acc(parallel loop present(pd[0:_cntml_padded * 5],
253  ppd[0:_cntml_actual],
256  if (nt->compute_gpu))
257  // clang-format on
258  nrn_pragma_omp(target teams distribute parallel for simd if(nt->compute_gpu))
259  for (int _iml = 0; _iml < _cntml_actual; ++_iml) {
260  if (iontype & 04) {
261  conci = conci0;
262  conco = conco0;
263  }
264  if (iontype & 040) {
265  erev = nrn_nernst(conci, conco, charge, celsius);
266  }
267  }
268 }
269 
270 void nrn_alloc_ion(double* p, Datum* ppvar, int _type) {
271  assert(0);
272 }
273 
274 void second_order_cur(NrnThread* _nt, int secondorder) {
275  int _cntml_padded;
276  double* pd;
277  (void) _nt; /* unused */
278  double* _vec_rhs = _nt->_actual_rhs;
279 
280  if (secondorder == 2) {
281  for (NrnThreadMembList* tml = _nt->tml; tml; tml = tml->next)
282  if (nrn_is_ion(tml->index)) {
283  Memb_list* ml = tml->ml;
284  int _cntml_actual = ml->nodecount;
285  int* ni = ml->nodeindices;
286  _cntml_padded = ml->_nodecount_padded;
287  pd = ml->data;
288  nrn_pragma_acc(parallel loop present(pd [0:_cntml_padded * 5],
289  ni [0:_cntml_actual],
290  _vec_rhs [0:_nt->end]) if (_nt->compute_gpu)
291  async(_nt->stream_id))
292  nrn_pragma_omp(target teams distribute parallel for simd if(_nt->compute_gpu))
293  for (int _iml = 0; _iml < _cntml_actual; ++_iml) {
294  cur += dcurdv * (_vec_rhs[ni[_iml]]);
295  }
296  }
297  }
298 }
299 } // namespace coreneuron
300 #endif
VAL_SENTINAL
#define VAL_SENTINAL
coreneuron::nrn_get_mechtype
int nrn_get_mechtype(const char *name)
Get mechanism type by the mechanism name.
Definition: mk_mech.cpp:138
membfunc.hpp
DEF_iono
#define DEF_iono
Definition: membrane_definitions.h:61
coreneuron::nrn_cur_ion
void nrn_cur_ion(NrnThread *_nt, Memb_list *ml, int type)
coreneuron::Datum
int Datum
Definition: nrnconf.h:23
coreneuron::nrn_nernst
constexpr double nrn_nernst(double ci, double co, double z, double celsius)
Definition: membfunc.hpp:129
SOA_LAYOUT
#define SOA_LAYOUT
Definition: data_layout.hpp:11
nrnoc_aux.hpp
coreneuron::ion_global_map_member_size
const int ion_global_map_member_size
coreneuron::nrn_init_ion
void nrn_init_ion(NrnThread *, Memb_list *, int)
global_conci
#define global_conci(type)
coreneuron::ion_reg
void ion_reg(const char *, double)
global_conco
#define global_conco(type)
nrn_pragma_omp
nrn_pragma_acc(routine seq) nrn_pragma_omp(declare target) philox4x32_ctr_t coreneuron_random123_philox4x32_helper(coreneuron nrn_pragma_omp(end declare target) namespace coreneuron
Provide a helper function in global namespace that is declared target for OpenMP offloading to functi...
Definition: nrnran123.h:69
coreneuron::register_mech
int register_mech(const char **m, mod_alloc_t alloc, mod_f_t cur, mod_f_t jacob, mod_f_t stat, mod_f_t initialize, mod_f_t private_constructor, mod_f_t private_destructor, int nrnpointerindex, int vectorized)
Definition: register_mech.cpp:112
DEF_ioni
#define DEF_ioni
Definition: membrane_definitions.h:60
coreneuron::mechanism
static const char * mechanism[]
Definition: capac.cpp:22
coreneuron.hpp
coreneuron
THIS FILE IS AUTO GENERATED DONT MODIFY IT.
Definition: corenrn_parameters.cpp:12
coreneuron::_nrn_layout_reg
void _nrn_layout_reg(int, int)
Definition: register_mech.cpp:176
DEF_nai
#define DEF_nai
Definition: membrane_definitions.h:44
nparm
#define nparm
coreneuron::i
int i
Definition: cellorder.cpp:485
coreneuron::nrn_is_ion
int nrn_is_ion(int)
coreneuron::_nrn_skip_initmodel
bool _nrn_skip_initmodel
Definition: finitialize.cpp:19
coreneuron::nrn_ion_global_map
double ** nrn_ion_global_map
coreneuron::nrn_ion_global_map_size
int nrn_ion_global_map_size
coreneuron::nrn_exit
void nrn_exit(int err)
Definition: nrnoc_aux.cpp:30
coreneuron::nrn_writes_conc
void nrn_writes_conc(int, int)
Definition: register_mech.cpp:163
coreneuron::second_order_cur
void second_order_cur(NrnThread *_nt, int secondorder)
global_charge
#define global_charge(type)
coreneuron::hoc_register_dparam_semantics
void hoc_register_dparam_semantics(int type, int, const char *name)
Definition: register_mech.cpp:207
coreneuron::emalloc
static void * emalloc(size_t size)
Definition: mpispike.cpp:30
data_layout.hpp
coreneuron::celsius
double celsius
Definition: register_mech.cpp:22
coreneuron::secondorder
int secondorder
Definition: register_mech.cpp:21
coreneuron::ktf
constexpr double ktf(double celsius)
Definition: membfunc.hpp:124
coreneuron::erealloc
void * erealloc(void *ptr, size_t size)
Definition: nrnoc_aux.cpp:94
coreneuron::nrn_alloc_ion
void nrn_alloc_ion(double *data, Datum *pdata, int type)
coreneuron::nrn_pragma_acc
nrn_pragma_acc(routine vector) static void triang_interleaved2(NrnThread *nt
Definition: ivocvect.cpp:30
coreneuron::hoc_register_prop_size
void hoc_register_prop_size(int, int, int)
Definition: register_mech.cpp:192
coreneuron::if
if(ncell==0)
Definition: cellorder.cpp:637
DEF_ko
#define DEF_ko
Definition: membrane_definitions.h:50
DEF_cao
#define DEF_cao
Definition: membrane_definitions.h:55
DEF_cai
#define DEF_cai
Definition: membrane_definitions.h:54
nrnmpi.h
DEF_ki
#define DEF_ki
Definition: membrane_definitions.h:49
DEF_nao
#define DEF_nao
Definition: membrane_definitions.h:45