CoreNEURON
lfp.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 #include "coreneuron/io/lfp.hpp"
10 
11 #define BOOST_TEST_MODULE LFPTest
12 #include <boost/test/included/unit_test.hpp>
13 
14 #include <iostream>
15 
16 using namespace coreneuron;
17 using namespace coreneuron::lfputils;
18 
19 template <typename F>
20 double integral(F f, double a, double b, int n) {
21  double step = (b - a) / n; // width of each small rectangle
22  double area = 0.0; // signed area
23  for (int i = 0; i < n; i++) {
24  area += f(a + (i + 0.5) * step) * step; // sum up each small rectangle
25  }
26  return area;
27 }
28 
29 
30 BOOST_AUTO_TEST_CASE(LFP_PointSource_LineSource) {
31 #if NRNMPI
32  nrnmpi_init(nullptr, nullptr, false);
33 #endif
34  double segment_length{1.0e-6};
35  double segment_start_val{1.0e-6};
36  std::array<double, 3> segment_start = std::array<double, 3>{0.0, 0.0, segment_start_val};
37  std::array<double, 3> segment_end =
38  paxpy(segment_start, 1.0, std::array<double, 3>{0.0, 0.0, segment_length});
39  double floor{1.0e-6};
40  pi = 3.141592653589;
41 
42  std::array<double, 10> vals;
43  double circling_radius{1.0e-6};
44  std::array<double, 3> segment_middle{0.0, 0.0, 1.5e-6};
45  double medium_resistivity_fac{1.0};
46  for (auto k = 0; k < 10; k++) {
47  std::array<double, 3> approaching_elec =
48  paxpy(segment_middle, 1.0, std::array<double, 3>{0.0, 1.0e-5 - k * 1.0e-6, 0.0});
49  std::array<double, 3> circling_elec =
50  paxpy(segment_middle,
51  1.0,
52  std::array<double, 3>{0.0,
53  circling_radius * std::cos(2.0 * pi * k / 10),
54  circling_radius * std::sin(2.0 * pi * k / 10)});
55 
56  double analytic_approaching_lfp = line_source_lfp_factor(
57  approaching_elec, segment_start, segment_end, floor, medium_resistivity_fac);
58  double analytic_circling_lfp = line_source_lfp_factor(
59  circling_elec, segment_start, segment_end, floor, medium_resistivity_fac);
60  double numeric_circling_lfp = integral(
61  [&](double x) {
62  return 1.0 / std::max(floor,
63  norm(paxpy(circling_elec,
64  -1.0,
65  paxpy(segment_end,
66  x,
67  paxpy(segment_start, -1.0, segment_end)))));
68  },
69  0.0,
70  1.0,
71  10000);
72  // TEST of analytic vs numerical integration
73  std::clog << "ANALYTIC line source " << analytic_circling_lfp
74  << " vs NUMERIC line source LFP " << numeric_circling_lfp << "\n";
75  BOOST_REQUIRE_CLOSE(analytic_circling_lfp, numeric_circling_lfp, 1.0e-6);
76  // TEST of LFP Flooring
77  BOOST_REQUIRE((approaching_elec[1] < 0.866e-6) ? analytic_approaching_lfp == 1.0e6 : true);
78  vals[k] = analytic_circling_lfp;
79  }
80  // TEST of SYMMETRY of LFP FORMULA
81  for (size_t k = 0; k < 5; k++) {
82  BOOST_REQUIRE(std::abs((vals[k] - vals[k + 5]) /
83  std::max(std::abs(vals[k]), std::abs(vals[k + 5]))) < 1.0e-12);
84  }
85  std::vector<std::array<double, 3>> segments_starts = {{0., 0., 1.},
86  {0., 0., 0.5},
87  {0.0, 0.0, 0.0},
88  {0.0, 0.0, -0.5}};
89  std::vector<std::array<double, 3>> segments_ends = {{0., 0., 0.},
90  {0., 0., 1.},
91  {0., 0., 0.5},
92  {0.0, 0.0, 0.0}};
93  std::vector<double> radii{0.1, 0.1, 0.1, 0.1};
94  std::vector<std::array<double, 3>> electrodes = {{0.0, 0.3, 0.0}, {0.0, 0.7, 0.8}};
95  std::vector<int> indices = {0, 1, 2, 3};
96  LFPCalculator<LineSource> lfp(segments_starts, segments_ends, radii, indices, electrodes, 1.0);
97  lfp.template lfp<std::vector<double>>({0.0, 1.0, 2.0, 3.0});
98  std::vector<double> res_line_source = lfp.lfp_values();
100  segments_starts, segments_ends, radii, indices, electrodes, 1.0);
101  lfpp.template lfp<std::vector<double>>({0.0, 1.0, 2.0, 3.0});
102  std::vector<double> res_point_source = lfpp.lfp_values();
103  BOOST_REQUIRE_CLOSE(res_line_source[0], res_point_source[0], 1.0);
104  BOOST_REQUIRE_CLOSE(res_line_source[1], res_point_source[1], 1.0);
105 #if NRNMPI
106  nrnmpi_finalize();
107 #endif
108 }
coreneuron::nrnmpi_finalize
mpi_function< cnrn_make_integral_constant_t(nrnmpi_finalize_impl)> nrnmpi_finalize
Definition: nrnmpidec.cpp:16
coreneuron
THIS FILE IS AUTO GENERATED DONT MODIFY IT.
Definition: corenrn_parameters.cpp:12
coreneuron::lfputils::norm
double norm(const Point3D &p1)
Definition: lfp.hpp:22
coreneuron::i
int i
Definition: cellorder.cpp:485
coreneuron::lfputils
Definition: lfp.cpp:10
coreneuron::LFPCalculator::lfp_values
const std::vector< double > & lfp_values() const noexcept
Definition: lfp.hpp:93
coreneuron::lfputils::line_source_lfp_factor
double line_source_lfp_factor(const Point3D &e_pos, const Point3D &seg_0, const Point3D &seg_1, const double radius, const double f)
Definition: lfp.cpp:12
coreneuron::nrnmpi_init
mpi_function< cnrn_make_integral_constant_t(nrnmpi_init_impl)> nrnmpi_init
Definition: nrnmpidec.cpp:15
area
#define area
Definition: md1redef.h:12
coreneuron::LFPCalculator
LFPCalculator allows calculation of LFP given membrane currents.
Definition: lfp.hpp:72
coreneuron::lfputils::paxpy
Point3D paxpy(const Point3D &p1, const double alpha, const Point3D &p2)
Definition: lfp.hpp:30
integral
double integral(F f, double a, double b, int n)
Definition: lfp.cpp:20
BOOST_AUTO_TEST_CASE
BOOST_AUTO_TEST_CASE(LFP_PointSource_LineSource)
Definition: lfp.cpp:30
lfp.hpp
coreneuron::pi
double pi
Definition: register_mech.cpp:22
nrnmpi.h