18 std::vector<int> thread_gids;
22 std::vector<T> intersection;
24 std::sort(thread_gids.begin(), thread_gids.end());
25 std::sort(target_gids.begin(), target_gids.end());
27 std::set_intersection(thread_gids.begin(),
31 back_inserter(intersection));
40 #if defined(ENABLE_BIN_REPORTS) || defined(ENABLE_SONATA_REPORTS)
41 if (report_config.
start <
t) {
44 report_config.
stop = std::min(report_config.
stop, tstop);
46 for (
const auto& mech: report_config.
mech_names) {
50 std::cerr <<
"[ERROR] mechanism to report: " << report_config.
mech_names[0]
51 <<
" is not mapped in this simulation, cannot report on it \n";
60 const std::vector<int>& nodes_to_gid = map_gids(nt);
62 VarsToReport vars_to_report;
64 switch (report_config.
type) {
68 vars_to_report = get_section_vars_to_report(nt,
75 register_section_report(nt, report_config, vars_to_report, is_soma_target);
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);
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);
87 if (!vars_to_report.empty()) {
88 auto report_event = std::make_unique<ReportEvent>(
91 m_report_events.push_back(std::move(report_event));
96 std::cerr <<
"[WARNING] : Reporting is disabled. Please recompile with either libsonata or "
99 #endif // defined(ENABLE_BIN_REPORTS) || defined(ENABLE_SONATA_REPORTS)
102 #if defined(ENABLE_BIN_REPORTS) || defined(ENABLE_SONATA_REPORTS)
103 void ReportHandler::register_section_report(
const NrnThread& nt,
105 const VarsToReport& vars_to_report,
106 bool is_soma_target) {
108 std::cerr <<
"[WARNING] : Format '" << config.
format <<
"' in report '"
112 void ReportHandler::register_custom_report(
const NrnThread& nt,
113 const ReportConfiguration& config,
114 const VarsToReport& vars_to_report) {
116 std::cerr <<
"[WARNING] : Format '" << config.format <<
"' in report '"
117 << config.output_path <<
"' not supported.\n";
135 std::cerr <<
"SectionType not handled in getSectionTypeStr" << std::endl;
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) {
146 int section_id = section.first;
147 const auto& segment_ids = section.second;
150 if (all_compartments) {
151 for (
const auto& segment_id: segment_ids) {
153 double* variable = report_variable + segment_id;
154 to_report.emplace_back(VarWithMapping(section_id, variable));
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));
166 VarsToReport ReportHandler::get_section_vars_to_report(
const NrnThread& nt,
167 const std::vector<int>& gids_to_report,
168 double* report_variable,
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);
175 std::cerr <<
"[COMPARTMENTS] Error : mapping information is missing for a Cell group "
180 for (
const auto& gid: gids_to_report) {
181 const auto& cell_mapping = mapinfo->get_cell_mapping(gid);
182 if (cell_mapping ==
nullptr) {
184 <<
"[COMPARTMENTS] Error : Compartment mapping information is missing for gid "
188 std::vector<VarWithMapping> to_report;
189 to_report.reserve(cell_mapping->size());
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);
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);
203 vars_to_report[gid] = to_report;
205 return vars_to_report;
208 VarsToReport ReportHandler::get_summation_vars_to_report(
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];
217 std::cerr <<
"[COMPARTMENTS] Error : mapping information is missing for a Cell group "
222 for (
const auto& gid: gids_to_report) {
223 bool has_imembrane =
false;
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") {
232 if (mech_name ==
"IClamp" || mech_name ==
"SEClamp") {
235 Memb_list* ml = nt._ml_list[mech_id];
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)) {
245 summation_report.currents_[segment_id].push_back(
246 std::make_pair(var_value, scale));
250 has_imembrane =
true;
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 "
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);
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,
270 report.section_all_compartments);
273 const auto& section_mapping = cell_mapping->secmapvec;
274 for (
const auto& sections: section_mapping) {
275 for (
auto& section: sections->secmap) {
277 int section_id = section.first;
278 auto& segment_ids = section.second;
279 for (
const auto& segment_id: segment_ids) {
282 summation_report.currents_[segment_id].push_back(
283 std::make_pair(nt.nrn_fast_imem->nrn_sav_rhs + segment_id, 1));
286 double* variable = report_variable + segment_id;
287 to_report.emplace_back(VarWithMapping(section_id, variable));
289 summation_report.gid_segments_[gid].push_back(segment_id);
294 vars_to_report[gid] = to_report;
296 return vars_to_report;
299 VarsToReport ReportHandler::get_synapse_vars_to_report(
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) {
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];
314 std::vector<VarWithMapping> to_report;
315 to_report.reserve(ml->nodecount);
317 for (
int j = 0; j < ml->nodecount; j++) {
318 double* is_selected =
320 bool report_variable =
false;
324 if (is_selected ==
nullptr) {
325 report_variable =
true;
327 report_variable = *is_selected != 0.;
329 if ((nodes_to_gids[ml->nodeindices[j]] == gid) && report_variable) {
334 to_report.emplace_back(
static_cast<int>(*synapse_id), var_value);
337 if (!to_report.empty()) {
338 vars_to_report[gid] = to_report;
341 return vars_to_report;
345 std::vector<int> ReportHandler::map_gids(
const NrnThread& nt)
const {
346 std::vector<int> nodes_gid(nt.end, -1);
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_;
352 if (thvar_index >= 0) {
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;
365 for (
int i = nt.ncell + 1;
i < nt.end;
i++) {
366 nodes_gid[
i] = nodes_gid[nt._v_parent_index[
i]];
370 #endif // defined(ENABLE_BIN_REPORTS) || defined(ENABLE_SONATA_REPORTS)