CoreNEURON
report_handler.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 #include "report_handler.hpp"
13 
14 namespace coreneuron {
15 
16 template <typename T>
17 std::vector<T> intersection_gids(const NrnThread& nt, std::vector<T>& target_gids) {
18  std::vector<int> thread_gids;
19  for (int i = 0; i < nt.ncell; i++) {
20  thread_gids.push_back(nt.presyns[i].gid_);
21  }
22  std::vector<T> intersection;
23 
24  std::sort(thread_gids.begin(), thread_gids.end());
25  std::sort(target_gids.begin(), target_gids.end());
26 
27  std::set_intersection(thread_gids.begin(),
28  thread_gids.end(),
29  target_gids.begin(),
30  target_gids.end(),
31  back_inserter(intersection));
32 
33  return intersection;
34 }
35 
37  double dt,
38  double tstop,
39  double delay) {
40 #if defined(ENABLE_BIN_REPORTS) || defined(ENABLE_SONATA_REPORTS)
41  if (report_config.start < t) {
42  report_config.start = t;
43  }
44  report_config.stop = std::min(report_config.stop, tstop);
45 
46  for (const auto& mech: report_config.mech_names) {
47  report_config.mech_ids.emplace_back(nrn_get_mechtype(mech.data()));
48  }
49  if (report_config.type == SynapseReport && report_config.mech_ids.empty()) {
50  std::cerr << "[ERROR] mechanism to report: " << report_config.mech_names[0]
51  << " is not mapped in this simulation, cannot report on it \n";
52  nrn_abort(1);
53  }
54  for (int ith = 0; ith < nrn_nthread; ++ith) {
55  NrnThread& nt = nrn_threads[ith];
56  double* report_variable = nt._actual_v;
57  if (!nt.ncell) {
58  continue;
59  }
60  const std::vector<int>& nodes_to_gid = map_gids(nt);
61  const std::vector<int> gids_to_report = intersection_gids(nt, report_config.target);
62  VarsToReport vars_to_report;
63  bool is_soma_target;
64  switch (report_config.type) {
65  case IMembraneReport:
66  report_variable = nt.nrn_fast_imem->nrn_sav_rhs;
67  case SectionReport:
68  vars_to_report = get_section_vars_to_report(nt,
69  gids_to_report,
70  report_variable,
71  report_config.section_type,
72  report_config.section_all_compartments);
73  is_soma_target = report_config.section_type == SectionType::Soma ||
74  report_config.section_type == SectionType::Cell;
75  register_section_report(nt, report_config, vars_to_report, is_soma_target);
76  break;
77  case SummationReport:
78  vars_to_report =
79  get_summation_vars_to_report(nt, gids_to_report, report_config, nodes_to_gid);
80  register_custom_report(nt, report_config, vars_to_report);
81  break;
82  default:
83  vars_to_report =
84  get_synapse_vars_to_report(nt, gids_to_report, report_config, nodes_to_gid);
85  register_custom_report(nt, report_config, vars_to_report);
86  }
87  if (!vars_to_report.empty()) {
88  auto report_event = std::make_unique<ReportEvent>(
89  dt, t, vars_to_report, report_config.output_path.data(), report_config.report_dt);
90  report_event->send(t, net_cvode_instance, &nt);
91  m_report_events.push_back(std::move(report_event));
92  }
93  }
94 #else
95  if (nrnmpi_myid == 0) {
96  std::cerr << "[WARNING] : Reporting is disabled. Please recompile with either libsonata or "
97  "reportinglib. \n";
98  }
99 #endif // defined(ENABLE_BIN_REPORTS) || defined(ENABLE_SONATA_REPORTS)
100 }
101 
102 #if defined(ENABLE_BIN_REPORTS) || defined(ENABLE_SONATA_REPORTS)
103 void ReportHandler::register_section_report(const NrnThread& nt,
104  const ReportConfiguration& config,
105  const VarsToReport& vars_to_report,
106  bool is_soma_target) {
107  if (nrnmpi_myid == 0) {
108  std::cerr << "[WARNING] : Format '" << config.format << "' in report '"
109  << config.output_path << "' not supported.\n";
110  }
111 }
112 void ReportHandler::register_custom_report(const NrnThread& nt,
113  const ReportConfiguration& config,
114  const VarsToReport& vars_to_report) {
115  if (nrnmpi_myid == 0) {
116  std::cerr << "[WARNING] : Format '" << config.format << "' in report '"
117  << config.output_path << "' not supported.\n";
118  }
119 }
120 
121 std::string getSectionTypeStr(SectionType type) {
122  switch (type) {
123  case All:
124  return "All";
125  case Cell:
126  case Soma:
127  return "soma";
128  case Axon:
129  return "axon";
130  case Dendrite:
131  return "dend";
132  case Apical:
133  return "apic";
134  default:
135  std::cerr << "SectionType not handled in getSectionTypeStr" << std::endl;
136  nrn_abort(1);
137  }
138 }
139 
140 void register_sections_to_report(const SecMapping* sections,
141  std::vector<VarWithMapping>& to_report,
142  double* report_variable,
143  bool all_compartments) {
144  for (const auto& section: sections->secmap) {
145  // compartment_id
146  int section_id = section.first;
147  const auto& segment_ids = section.second;
148 
149  // get all compartment values (otherwise, just middle point)
150  if (all_compartments) {
151  for (const auto& segment_id: segment_ids) {
152  // corresponding voltage in coreneuron voltage array
153  double* variable = report_variable + segment_id;
154  to_report.emplace_back(VarWithMapping(section_id, variable));
155  }
156  } else {
157  nrn_assert(segment_ids.size() % 2);
158  // corresponding voltage in coreneuron voltage array
159  const auto segment_id = segment_ids[segment_ids.size() / 2];
160  double* variable = report_variable + segment_id;
161  to_report.emplace_back(VarWithMapping(section_id, variable));
162  }
163  }
164 }
165 
166 VarsToReport ReportHandler::get_section_vars_to_report(const NrnThread& nt,
167  const std::vector<int>& gids_to_report,
168  double* report_variable,
169  SectionType section_type,
170  bool all_compartments) const {
171  VarsToReport vars_to_report;
172  const auto& section_type_str = getSectionTypeStr(section_type);
173  const auto* mapinfo = static_cast<NrnThreadMappingInfo*>(nt.mapping);
174  if (!mapinfo) {
175  std::cerr << "[COMPARTMENTS] Error : mapping information is missing for a Cell group "
176  << nt.ncell << '\n';
177  nrn_abort(1);
178  }
179 
180  for (const auto& gid: gids_to_report) {
181  const auto& cell_mapping = mapinfo->get_cell_mapping(gid);
182  if (cell_mapping == nullptr) {
183  std::cerr
184  << "[COMPARTMENTS] Error : Compartment mapping information is missing for gid "
185  << gid << '\n';
186  nrn_abort(1);
187  }
188  std::vector<VarWithMapping> to_report;
189  to_report.reserve(cell_mapping->size());
190 
191  if (section_type_str == "All") {
192  const auto& section_mapping = cell_mapping->secmapvec;
193  for (const auto& sections: section_mapping) {
194  register_sections_to_report(sections, to_report, report_variable, all_compartments);
195  }
196  } else {
197  /** get section list mapping for the type, if available */
198  if (cell_mapping->get_seclist_section_count(section_type_str) > 0) {
199  const auto& sections = cell_mapping->get_seclist_mapping(section_type_str);
200  register_sections_to_report(sections, to_report, report_variable, all_compartments);
201  }
202  }
203  vars_to_report[gid] = to_report;
204  }
205  return vars_to_report;
206 }
207 
208 VarsToReport ReportHandler::get_summation_vars_to_report(
209  const NrnThread& nt,
210  const std::vector<int>& gids_to_report,
211  const ReportConfiguration& report,
212  const std::vector<int>& nodes_to_gids) const {
213  VarsToReport vars_to_report;
214  const auto* mapinfo = static_cast<NrnThreadMappingInfo*>(nt.mapping);
215  auto& summation_report = nt.summation_report_handler_->summation_reports_[report.output_path];
216  if (!mapinfo) {
217  std::cerr << "[COMPARTMENTS] Error : mapping information is missing for a Cell group "
218  << nt.ncell << '\n';
219  nrn_abort(1);
220  }
221 
222  for (const auto& gid: gids_to_report) {
223  bool has_imembrane = false;
224  // In case we need convertion of units
225  int scale = 1;
226  for (auto i = 0; i < report.mech_ids.size(); ++i) {
227  auto mech_id = report.mech_ids[i];
228  auto var_name = report.var_names[i];
229  auto mech_name = report.mech_names[i];
230  if (mech_name != "i_membrane") {
231  // need special handling for Clamp processes to flip the current value
232  if (mech_name == "IClamp" || mech_name == "SEClamp") {
233  scale = -1;
234  }
235  Memb_list* ml = nt._ml_list[mech_id];
236  if (!ml) {
237  continue;
238  }
239 
240  for (int j = 0; j < ml->nodecount; j++) {
241  auto segment_id = ml->nodeindices[j];
242  if ((nodes_to_gids[ml->nodeindices[j]] == gid)) {
243  double* var_value =
244  get_var_location_from_var_name(mech_id, var_name.data(), ml, j);
245  summation_report.currents_[segment_id].push_back(
246  std::make_pair(var_value, scale));
247  }
248  }
249  } else {
250  has_imembrane = true;
251  }
252  }
253  const auto& cell_mapping = mapinfo->get_cell_mapping(gid);
254  if (cell_mapping == nullptr) {
255  std::cerr << "[SUMMATION] Error : Compartment mapping information is missing for gid "
256  << gid << '\n';
257  nrn_abort(1);
258  }
259  std::vector<VarWithMapping> to_report;
260  to_report.reserve(cell_mapping->size());
261  summation_report.summation_.resize(nt.end);
262  double* report_variable = summation_report.summation_.data();
263  const auto& section_type_str = getSectionTypeStr(report.section_type);
264  if (report.section_type != SectionType::All) {
265  if (cell_mapping->get_seclist_section_count(section_type_str) > 0) {
266  const auto& sections = cell_mapping->get_seclist_mapping(section_type_str);
267  register_sections_to_report(sections,
268  to_report,
269  report_variable,
270  report.section_all_compartments);
271  }
272  }
273  const auto& section_mapping = cell_mapping->secmapvec;
274  for (const auto& sections: section_mapping) {
275  for (auto& section: sections->secmap) {
276  // compartment_id
277  int section_id = section.first;
278  auto& segment_ids = section.second;
279  for (const auto& segment_id: segment_ids) {
280  // corresponding voltage in coreneuron voltage array
281  if (has_imembrane) {
282  summation_report.currents_[segment_id].push_back(
283  std::make_pair(nt.nrn_fast_imem->nrn_sav_rhs + segment_id, 1));
284  }
285  if (report.section_type == SectionType::All) {
286  double* variable = report_variable + segment_id;
287  to_report.emplace_back(VarWithMapping(section_id, variable));
288  } else if (report.section_type == SectionType::Cell) {
289  summation_report.gid_segments_[gid].push_back(segment_id);
290  }
291  }
292  }
293  }
294  vars_to_report[gid] = to_report;
295  }
296  return vars_to_report;
297 }
298 
299 VarsToReport ReportHandler::get_synapse_vars_to_report(
300  const NrnThread& nt,
301  const std::vector<int>& gids_to_report,
302  const ReportConfiguration& report,
303  const std::vector<int>& nodes_to_gids) const {
304  VarsToReport vars_to_report;
305  for (const auto& gid: gids_to_report) {
306  // There can only be 1 mechanism
307  nrn_assert(report.mech_ids.size() == 1);
308  auto mech_id = report.mech_ids[0];
309  auto var_name = report.var_names[0];
310  Memb_list* ml = nt._ml_list[mech_id];
311  if (!ml) {
312  continue;
313  }
314  std::vector<VarWithMapping> to_report;
315  to_report.reserve(ml->nodecount);
316 
317  for (int j = 0; j < ml->nodecount; j++) {
318  double* is_selected =
320  bool report_variable = false;
321 
322  /// if there is no variable in mod file then report on every compartment
323  /// otherwise check the flag set in mod file
324  if (is_selected == nullptr) {
325  report_variable = true;
326  } else {
327  report_variable = *is_selected != 0.;
328  }
329  if ((nodes_to_gids[ml->nodeindices[j]] == gid) && report_variable) {
330  double* var_value = get_var_location_from_var_name(mech_id, var_name.data(), ml, j);
331  double* synapse_id =
333  nrn_assert(synapse_id && var_value);
334  to_report.emplace_back(static_cast<int>(*synapse_id), var_value);
335  }
336  }
337  if (!to_report.empty()) {
338  vars_to_report[gid] = to_report;
339  }
340  }
341  return vars_to_report;
342 }
343 
344 // map GIDs of every compartment, it consist in a backward sweep then forward sweep algorithm
345 std::vector<int> ReportHandler::map_gids(const NrnThread& nt) const {
346  std::vector<int> nodes_gid(nt.end, -1);
347  // backward sweep: from presyn compartment propagate back GID to parent
348  for (int i = 0; i < nt.n_presyn; i++) {
349  const int gid = nt.presyns[i].gid_;
350  const int thvar_index = nt.presyns[i].thvar_index_;
351  // only for non artificial cells
352  if (thvar_index >= 0) {
353  // setting all roots gids of the presyns nodes,
354  // index 0 have parent set to 0, so we must stop at j > 0
355  // also 0 is the parent of all, so it is an error to attribute a GID to it.
356  nodes_gid[thvar_index] = gid;
357  for (int j = thvar_index; j > 0; j = nt._v_parent_index[j]) {
358  nodes_gid[nt._v_parent_index[j]] = gid;
359  }
360  }
361  }
362  // forward sweep: setting all compartements nodes to the GID of its root
363  // already sets on above loop. This is working only because compartments are stored in order
364  // parents follow by childrens
365  for (int i = nt.ncell + 1; i < nt.end; i++) {
366  nodes_gid[i] = nodes_gid[nt._v_parent_index[i]];
367  }
368  return nodes_gid;
369 }
370 #endif // defined(ENABLE_BIN_REPORTS) || defined(ENABLE_SONATA_REPORTS)
371 
372 } // Namespace coreneuron
coreneuron::NrnThread::nrn_fast_imem
NrnFastImem * nrn_fast_imem
Definition: multicore.hpp:124
coreneuron::nrn_get_mechtype
int nrn_get_mechtype(const char *name)
Get mechanism type by the mechanism name.
Definition: mk_mech.cpp:138
coreneuron::ReportConfiguration::stop
double stop
Definition: nrnreport.hpp:101
coreneuron::nrn_nthread
int nrn_nthread
Definition: multicore.cpp:55
coreneuron::ReportConfiguration::section_type
SectionType section_type
Definition: nrnreport.hpp:97
coreneuron::ReportConfiguration::start
double start
Definition: nrnreport.hpp:100
utils.hpp
coreneuron::ReportConfiguration::type
ReportType type
Definition: nrnreport.hpp:96
coreneuron::Axon
@ Axon
Definition: nrnreport.hpp:83
SELECTED_VAR_MOD_NAME
#define SELECTED_VAR_MOD_NAME
Definition: nrnreport.hpp:49
coreneuron::ReportConfiguration::target
std::vector< int > target
Definition: nrnreport.hpp:104
coreneuron::SynapseReport
@ SynapseReport
Definition: nrnreport.hpp:76
coreneuron::NrnThread::presyns
PreSyn * presyns
Definition: multicore.hpp:83
report_handler.hpp
coreneuron::Cell
@ Cell
Definition: nrnreport.hpp:83
coreneuron::get_var_location_from_var_name
double * get_var_location_from_var_name(int mech_id, const char *variable_name, Memb_list *ml, int node_index)
Definition: mech_mapping.cpp:39
coreneuron::SummationReport
@ SummationReport
Definition: nrnreport.hpp:79
coreneuron::ReportConfiguration::mech_ids
std::vector< int > mech_ids
Definition: nrnreport.hpp:91
coreneuron::ReportConfiguration::report_dt
double report_dt
Definition: nrnreport.hpp:99
coreneuron
THIS FILE IS AUTO GENERATED DONT MODIFY IT.
Definition: corenrn_parameters.cpp:12
coreneuron::Dendrite
@ Dendrite
Definition: nrnreport.hpp:83
coreneuron::t
double t
Definition: register_mech.cpp:22
coreneuron::All
@ All
Definition: nrnreport.hpp:83
coreneuron::i
int i
Definition: cellorder.cpp:485
coreneuron::IMembraneReport
@ IMembraneReport
Definition: nrnreport.hpp:77
coreneuron::ReportConfiguration
Definition: nrnreport.hpp:85
coreneuron::ReportConfiguration::section_all_compartments
bool section_all_compartments
Definition: nrnreport.hpp:98
coreneuron::dt
double dt
Definition: register_mech.cpp:22
coreneuron::ReportConfiguration::output_path
std::string output_path
Definition: nrnreport.hpp:87
coreneuron::intersection_gids
std::vector< T > intersection_gids(const NrnThread &nt, std::vector< T > &target_gids)
Definition: report_handler.cpp:17
mech_mapping.hpp
nrnsection_mapping.hpp
coreneuron::NrnThread
Definition: multicore.hpp:75
coreneuron::nrn_threads
NrnThread * nrn_threads
Definition: multicore.cpp:56
coreneuron::SectionReport
@ SectionReport
Definition: nrnreport.hpp:78
coreneuron::nrn_abort
void nrn_abort(int errcode)
Definition: utils.cpp:13
coreneuron::SectionType
SectionType
Definition: nrnreport.hpp:83
coreneuron::ReportConfiguration::mech_names
std::vector< std::string > mech_names
Definition: nrnreport.hpp:89
coreneuron::Soma
@ Soma
Definition: nrnreport.hpp:83
coreneuron::net_cvode_instance
NetCvode * net_cvode_instance
Definition: netcvode.cpp:35
coreneuron::NrnFastImem::nrn_sav_rhs
double * nrn_sav_rhs
Definition: multicore.hpp:53
coreneuron::NrnThread::_actual_v
double * _actual_v
Definition: multicore.hpp:115
SYNAPSE_ID_MOD_NAME
#define SYNAPSE_ID_MOD_NAME
name of the variable in mod file used for setting synapse id
Definition: nrnreport.hpp:52
coreneuron::ReportHandler::create_report
virtual void create_report(ReportConfiguration &config, double dt, double tstop, double delay)
Definition: report_handler.cpp:36
coreneuron::nrnmpi_myid
int nrnmpi_myid
Definition: nrnmpi_def_cinc.cpp:11
coreneuron::PreSyn::gid_
int gid_
Definition: netcon.hpp:112
coreneuron::ReportConfiguration::format
std::string format
Definition: nrnreport.hpp:93
nrn_assert
#define nrn_assert(x)
assert()-like macro, independent of NDEBUG status
Definition: nrn_assert.h:33
coreneuron::NrnThread::ncell
int ncell
Definition: multicore.hpp:97
coreneuron::Apical
@ Apical
Definition: nrnreport.hpp:83