CoreNEURON
patternstim.cpp
Go to the documentation of this file.
1 /*
2 # =============================================================================
3 # Copyright (c) 2016 - 2021 Blue Brain Project/EPFL
4 #
5 # See top-level LICENSE file for details.
6 # =============================================================================.
7 */
8 
9 // Want to have the classical NEURON PatternStim functionality available
10 // in coreneuron to allow debugging and trajectory verification on
11 // desktop single process tests. Since pattern.mod provides most of what
12 // we need even in the coreneuron context, we placed a minimally modified
13 // version of that in coreneuron/mechanism/mech/modfile/pattern.mod and this file
14 // provides an interface that creates an instance of the
15 // PatternStim ARTIFICIAL_CELL in thread 0 and attaches the spike raster
16 // data to it.
17 
18 #include <algorithm>
19 
20 #include "coreneuron/nrnconf.h"
27 
28 namespace coreneuron {
29 // from translated patstim.mod
30 void _pattern_reg(void);
31 // from patstim.mod
32 extern void pattern_stim_setup_helper(int size,
33  double* tvec,
34  int* gidvec,
35  int icnt,
36  int cnt,
37  double* _p,
38  Datum* _ppvar,
39  ThreadDatum* _thread,
40  NrnThread* _nt,
41  Memb_list* ml,
42  double v);
43 
44 static size_t read_raster_file(const char* fname, double** tvec, int** gidvec, double tstop);
45 
47 
49  // limited to PatternStim for now.
50  // if called, must be called before nrn_setup and after mk_mech.
51  int type = nrn_get_mechtype("PatternStim");
52  if (!corenrn.get_memb_func(type).initialize) {
53  _pattern_reg();
54  }
56 }
57 
58 // fname is the filename of an output_spikes.h format raster file.
59 // todo : add function for memory cleanup (to be called at the end of simulation)
60 void nrn_mkPatternStim(const char* fname, double tstop) {
61  int type = nrn_get_mechtype("PatternStim");
62  if (!corenrn.get_memb_func(type).sym) {
63  printf("nrn_set_extra_thread_vdata must be called (after mk_mech, and before nrn_setup\n");
64  assert(0);
65  }
66 
67  // if there is empty thread then return, don't need patternstim
68  if (nrn_threads == nullptr || nrn_threads->ncell == 0) {
69  return;
70  }
71 
72  double* tvec;
73  int* gidvec;
74 
75  // todo : handle when spike raster will be very large (int < size_t)
76  size_t size = read_raster_file(fname, &tvec, &gidvec, tstop);
77 
78  Point_process* pnt = nrn_artcell_instantiate("PatternStim");
79  NrnThread* nt = nrn_threads + pnt->_tid;
80 
81  Memb_list* ml = nt->_ml_list[type];
82  int layout = corenrn.get_mech_data_layout()[type];
83  int sz = corenrn.get_prop_param_size()[type];
84  int psz = corenrn.get_prop_dparam_size()[type];
85  int _cntml = ml->nodecount;
86  int _iml = pnt->_i_instance;
87  double* _p = ml->data;
88  Datum* _ppvar = ml->pdata;
89  if (layout == Layout::AoS) {
90  _p += _iml * sz;
91  _ppvar += _iml * psz;
92  } else if (layout == Layout::SoA) {
93  ;
94  } else {
95  assert(0);
96  }
97  pattern_stim_setup_helper(size, tvec, gidvec, _iml, _cntml, _p, _ppvar, nullptr, nt, ml, 0.0);
98 }
99 
100 size_t read_raster_file(const char* fname, double** tvec, int** gidvec, double tstop) {
101  FILE* f = fopen(fname, "r");
102  nrn_assert(f);
103 
104  // skip first line containing "scatter" string
105  char dummy[100];
106  nrn_assert(fgets(dummy, 100, f));
107 
108  std::vector<std::pair<double, int>> spikes;
109  spikes.reserve(10000);
110 
111  double stime;
112  int gid;
113 
114  while (fscanf(f, "%lf %d\n", &stime, &gid) == 2) {
115  if (stime >= t && stime <= tstop) {
116  spikes.push_back(std::make_pair(stime, gid));
117  }
118  }
119 
120  fclose(f);
121 
122  // pattern.mod expects sorted spike raster (this is to avoid
123  // injecting all events at the begining of the simulation).
124  // sort spikes according to time
125  std::sort(spikes.begin(), spikes.end());
126 
127  // fill gid and time vectors
128  *tvec = (double*) emalloc(spikes.size() * sizeof(double));
129  *gidvec = (int*) emalloc(spikes.size() * sizeof(int));
130 
131  for (size_t i = 0; i < spikes.size(); i++) {
132  (*tvec)[i] = spikes[i].first;
133  (*gidvec)[i] = spikes[i].second;
134  }
135 
136  return spikes.size();
137 }
138 
139 // see nrn_setup.cpp:read_phase2 for how it creates NrnThreadMembList instances.
142  tml->dependencies = nullptr;
143  tml->ndependencies = 0;
144  tml->index = type;
145  tml->next = nullptr;
146 
147  // fill in tml->ml info. The data is not in the cache efficient
148  // NrnThread arrays but there should not be many of these instances.
149  int psize = corenrn.get_prop_param_size()[type];
150  int dsize = corenrn.get_prop_dparam_size()[type];
151  int layout = corenrn.get_mech_data_layout()[type];
152  tml->ml = (Memb_list*) ecalloc(1, sizeof(Memb_list));
153  tml->ml->nodecount = 1;
154  tml->ml->_nodecount_padded = tml->ml->nodecount;
155  tml->ml->nodeindices = nullptr;
156  tml->ml->data = (double*) ecalloc(tml->ml->nodecount * psize, sizeof(double));
157  tml->ml->pdata = (Datum*) ecalloc(nrn_soa_padded_size(tml->ml->nodecount, layout) * dsize,
158  sizeof(Datum));
159  tml->ml->_thread = nullptr;
160  tml->ml->_net_receive_buffer = nullptr;
161  tml->ml->_net_send_buffer = nullptr;
162  tml->ml->_permute = nullptr;
163 
164  if (auto* const priv_ctor = corenrn.get_memb_func(tml->index).private_constructor) {
165  priv_ctor(nt, tml->ml, tml->index);
166  }
167 
168  return tml;
169 }
170 
171 // Opportunistically implemented to create a single PatternStim.
172 // So only does enough to get that functionally incorporated into the model
173 // and other types may require additional work. In particular, we
174 // append a new NrnThreadMembList with one item to the thread 0 tml list
175 // in order for the artificial cell to get its INITIAL block called but
176 // we do not modify any of the other thread 0 data arrays or counts.
177 
178 Point_process* nrn_artcell_instantiate(const char* mechname) {
179  int type = nrn_get_mechtype(mechname);
180  NrnThread* nt = nrn_threads + 0;
181 
182  // printf("nrn_artcell_instantiate %s type=%d\n", mechname, type);
183 
184  // create and append to nt.tml
185  auto tml = alloc_nrn_thread_memb(nt, type);
186 
187  assert(nt->_ml_list[type] == nullptr); // FIXME
188  nt->_ml_list[type] = tml->ml;
189 
190  if (!nt->tml) {
191  nt->tml = tml;
192  } else {
193  for (NrnThreadMembList* i = nt->tml; i; i = i->next) {
194  if (!i->next) {
195  i->next = tml;
196  break;
197  }
198  }
199  }
200 
201  // Here we have a problem with no easy general solution. ml->pdata are
202  // integer indexes into the nt->_data nt->_idata and nt->_vdata array
203  // depending on context,
204  // but nrn_setup.cpp allocated these to exactly have the size needed by
205  // the file defined model (at least for _vdata) and so there are no slots
206  // for pdata to index into for this new instance.
207  // So nrn_setup.cpp:phase2 needs to
208  // be notified that some extra space will be required. For now, defer
209  // the general situation of several instances for several types and
210  // demand that this method is never called more than once. We introduce
211  // a int nrn_extra_thread0_vdata (only that is needed by PatternStim)
212  // which will be used by
213  // nrn_setup.cpp:phase2 to allocate the appropriately larger
214  // _vdata arrays for thread 0 (without changing _nvdata so
215  // that we can fill in the indices here)
216  static int cnt = 0;
217  if (++cnt > 1) {
218  printf("nrn_artcell_instantiate cannot be called more than once\n");
219  assert(0);
220  }
221  // note that PatternStim internal usage for the 4 ppvar values is:
222  // #define _nd_area _nt->_data[_ppvar[0]] (not used since ARTIFICIAL_CELL)
223  // #define _p_ptr _nt->_vdata[_ppvar[2]] (the BBCORE_POINTER)
224  // #define _tqitem &(_nt->_vdata[_ppvar[3]]) (for net_send)
225  // and general external usage is:
226  // _nt->_vdata[_ppvar[1]] = Point_process*
227  //
228 
229  Point_process* pnt = new Point_process;
230  pnt->_type = type;
231  pnt->_tid = nt->id;
232  pnt->_i_instance = 0;
233  // as though all dparam index into _vdata
234  int dsize = corenrn.get_prop_dparam_size()[type];
235  assert(dsize <= nrn_extra_thread0_vdata);
236  for (int i = 0; i < dsize; ++i) {
237  tml->ml->pdata[i] = nt->_nvdata + i;
238  }
239  nt->_vdata[nt->_nvdata + 1] = (void*) pnt;
240 
241  return pnt;
242 }
243 } // namespace coreneuron
coreneuron::CoreNeuron::get_mech_data_layout
auto & get_mech_data_layout()
Definition: coreneuron.hpp:174
coreneuron::NrnThreadMembList::dependencies
int * dependencies
Definition: multicore.hpp:36
coreneuron::nrn_get_mechtype
int nrn_get_mechtype(const char *name)
Get mechanism type by the mechanism name.
Definition: mk_mech.cpp:138
coreneuron::NrnThread::_vdata
void ** _vdata
Definition: multicore.hpp:108
coreneuron::ecalloc
void * ecalloc(size_t n, size_t size)
Definition: nrnoc_aux.cpp:85
coreneuron::Point_process
Definition: mechanism.hpp:35
coreneuron::Datum
int Datum
Definition: nrnconf.h:23
coreneuron::nrn_artcell_instantiate
Point_process * nrn_artcell_instantiate(const char *mechname)
Definition: patternstim.cpp:178
nrnoc_aux.hpp
coreneuron::CoreNeuron::get_memb_func
auto & get_memb_func(size_t idx)
Definition: coreneuron.hpp:138
output_spikes.hpp
coreneuron::NrnThread::id
int id
Definition: multicore.hpp:99
coreneuron::NrnThreadMembList::ndependencies
int ndependencies
Definition: multicore.hpp:37
coreneuron::Memb_list
Definition: mechanism.hpp:131
coreneuron::NrnThread::tml
NrnThreadMembList * tml
Definition: multicore.hpp:80
coreneuron::nrn_soa_padded_size
int nrn_soa_padded_size(int cnt, int layout)
calculate size after padding for specific memory layout
Definition: mem_layout_util.cpp:15
coreneuron::NrnThreadMembList::index
int index
Definition: multicore.hpp:35
coreneuron.hpp
coreneuron::NrnThread::_nvdata
size_t _nvdata
Definition: multicore.hpp:104
coreneuron::nrn_set_extra_thread0_vdata
void nrn_set_extra_thread0_vdata()
Definition: patternstim.cpp:48
coreneuron
THIS FILE IS AUTO GENERATED DONT MODIFY IT.
Definition: corenrn_parameters.cpp:12
coreneuron::t
double t
Definition: register_mech.cpp:22
coreneuron::i
int i
Definition: cellorder.cpp:485
coreneuron::NrnThread::_ml_list
Memb_list ** _ml_list
Definition: multicore.hpp:81
nrniv_decl.h
coreneuron::_pattern_reg
void _pattern_reg(void)
coreneuron::CoreNeuron::get_prop_dparam_size
auto & get_prop_dparam_size()
Definition: coreneuron.hpp:170
coreneuron::NrnThread
Definition: multicore.hpp:75
coreneuron::NrnThreadMembList
Definition: multicore.hpp:32
cnt
#define cnt
Definition: tqueue.hpp:44
coreneuron::pattern_stim_setup_helper
void pattern_stim_setup_helper(int size, double *tvec, int *gidvec, int icnt, int cnt, double *_p, Datum *_ppvar, ThreadDatum *_thread, NrnThread *_nt, Memb_list *ml, double v)
coreneuron::AoS
@ AoS
Definition: nrniv_decl.h:69
coreneuron::nrn_threads
NrnThread * nrn_threads
Definition: multicore.cpp:56
coreneuron::corenrn
CoreNeuron corenrn
Definition: multicore.cpp:53
coreneuron::Memb_list::_net_send_buffer
NetSendBuffer_t * _net_send_buffer
Definition: mechanism.hpp:143
coreneuron::Memb_list::nodecount
int nodecount
Definition: mechanism.hpp:144
coreneuron::read_raster_file
static size_t read_raster_file(const char *fname, double **tvec, int **gidvec, double tstop)
Definition: patternstim.cpp:100
coreneuron::Memb_list::_net_receive_buffer
NetReceiveBuffer_t * _net_receive_buffer
Definition: mechanism.hpp:142
coreneuron::alloc_nrn_thread_memb
static NrnThreadMembList * alloc_nrn_thread_memb(NrnThread *nt, int type)
Definition: patternstim.cpp:140
coreneuron::Memb_list::pdata
Datum * pdata
Definition: mechanism.hpp:140
nrnconf.h
coreneuron::NrnThreadMembList::next
NrnThreadMembList * next
Definition: multicore.hpp:33
coreneuron::Point_process::_type
short _type
Definition: mechanism.hpp:37
coreneuron::nrn_mkPatternStim
void nrn_mkPatternStim(const char *fname, double tstop)
Definition: patternstim.cpp:60
coreneuron::nrn_extra_thread0_vdata
int nrn_extra_thread0_vdata
Definition: patternstim.cpp:46
coreneuron::emalloc
static void * emalloc(size_t size)
Definition: mpispike.cpp:30
multicore.hpp
coreneuron::CoreNeuron::get_prop_param_size
auto & get_prop_param_size()
Definition: coreneuron.hpp:166
v
#define v
Definition: md1redef.h:11
coreneuron::Point_process::_tid
short _tid
Definition: mechanism.hpp:38
coreneuron::Memb_list::_thread
ThreadDatum * _thread
Definition: mechanism.hpp:141
coreneuron::Memb_list::_nodecount_padded
int _nodecount_padded
Definition: mechanism.hpp:145
coreneuron::Memb_list::data
double * data
Definition: mechanism.hpp:139
nrn_assert
#define nrn_assert(x)
assert()-like macro, independent of NDEBUG status
Definition: nrn_assert.h:33
coreneuron::SoA
@ SoA
Definition: nrniv_decl.h:69
coreneuron::NrnThreadMembList::ml
Memb_list * ml
Definition: multicore.hpp:34
nrn_assert.h
coreneuron::NrnThread::ncell
int ncell
Definition: multicore.hpp:97
coreneuron::Memb_list::nodeindices
int * nodeindices
Definition: mechanism.hpp:137
coreneuron::Memb_list::_permute
int * _permute
Definition: mechanism.hpp:138
coreneuron::Point_process::_i_instance
int _i_instance
Definition: mechanism.hpp:36