-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
add code for calibrating ODE models in EMIx simulations
- Loading branch information
adajel
committed
Aug 8, 2024
1 parent
0af8171
commit a418fd3
Showing
8 changed files
with
1,814 additions
and
0 deletions.
There are no files selected for viewing
Large diffs are not rendered by default.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,117 @@ | ||
#!/usr/bin/env python3 | ||
|
||
""" | ||
This script generates a 3D mesh representing 4 axons. | ||
""" | ||
|
||
from dolfin import * | ||
import sys | ||
|
||
class Boundary(SubDomain): | ||
# define exterior boundary | ||
def inside(self, x, on_boundary): | ||
return on_boundary | ||
|
||
def add_axon(mesh, subdomains, surfaces, a, b): | ||
# define interior domain | ||
in_interior = """ (x[0] >= %g && x[0] <= %g && | ||
x[1] >= %g && x[1] <= %g && | ||
x[2] >= %g && x[2] <= %g) """ % ( | ||
a[0], | ||
b[0], | ||
a[1], | ||
b[1], | ||
a[2], | ||
b[2], | ||
) | ||
|
||
interior = CompiledSubDomain(in_interior) | ||
|
||
# mark interior and exterior domain | ||
for cell in cells(mesh): | ||
x = cell.midpoint().array() | ||
if (int(interior.inside(x, False))): | ||
subdomains[cell] = 1 | ||
assert sum(1 for _ in SubsetIterator(subdomains, 1)) > 0 | ||
|
||
for facet in facets(mesh): | ||
x = [facet.midpoint().x(), facet.midpoint().y(), facet.midpoint().z()] | ||
side_1 = near(x[0], a[0]) and a[1] <= x[1] <= b[1] and a[2] <= x[2] <= b[2] | ||
side_2 = near(x[0], b[0]) and a[1] <= x[1] <= b[1] and a[2] <= x[2] <= b[2] | ||
side_3 = near(x[1], a[1]) and a[0] <= x[0] <= b[0] and a[2] <= x[2] <= b[2] | ||
side_4 = near(x[1], b[1]) and a[0] <= x[0] <= b[0] and a[2] <= x[2] <= b[2] | ||
side_5 = near(x[2], a[2]) and a[0] <= x[0] <= b[0] and a[1] <= x[1] <= b[1] | ||
side_6 = near(x[2], b[2]) and a[0] <= x[0] <= b[0] and a[1] <= x[1] <= b[1] | ||
surfaces[facet] += side_1 or side_2 or side_3 or side_4 or side_5 or side_6 | ||
|
||
return | ||
|
||
import argparse | ||
from pathlib import Path | ||
|
||
|
||
class CustomParser( | ||
argparse.ArgumentDefaultsHelpFormatter, argparse.RawDescriptionHelpFormatter | ||
): ... | ||
|
||
def main(argv=None): | ||
parser = argparse.ArgumentParser(formatter_class=CustomParser) | ||
parser.add_argument( | ||
"-r", | ||
"--resolution", | ||
dest="resolution_factor", | ||
default=0, | ||
type=int, | ||
help="Mesh resolution factor", | ||
) | ||
parser.add_argument( | ||
"-d", | ||
"--directory", | ||
dest="mesh_dir", | ||
type=Path, | ||
default=Path("meshes/3D"), | ||
help="Directory to save the mesh", | ||
) | ||
args = parser.parse_args(argv) | ||
resolution_factor = args.resolution_factor | ||
out_dir = args.mesh_dir | ||
|
||
l = 1 | ||
nx = l * 105 * 2 ** resolution_factor | ||
ny = 10 * 2 ** resolution_factor | ||
nz = 10 * 2 ** resolution_factor | ||
|
||
# box mesh | ||
mesh = BoxMesh(Point(0, 0.0, 0.0), Point(l * 105, 1.0, 1.0), nx, ny, nz) | ||
subdomains = MeshFunction("size_t", mesh, mesh.topology().dim(), 0) | ||
surfaces = MeshFunction("size_t", mesh, mesh.topology().dim() - 1, 0) | ||
|
||
a = Point(5, 0.2, 0.2) | ||
b = Point(l * 105 - 5, 0.8, 0.8) | ||
add_axon(mesh, subdomains, surfaces, a, b) | ||
|
||
# mark exterior boundary | ||
Boundary().mark(surfaces, 5) | ||
|
||
# convert mesh from um to cm | ||
mesh.coordinates()[:, :] *= 1e-4 | ||
|
||
# save .xml files | ||
mesh_file = File(str((out_dir / f"mesh_{resolution_factor}.xml").absolute())) | ||
mesh_file << mesh | ||
|
||
subdomains_file = File(str((out_dir / f"subdomains_{resolution_factor}.xml").absolute())) | ||
subdomains_file << subdomains | ||
|
||
surfaces_file = File(str((out_dir / f"surfaces_{resolution_factor}.xml").absolute())) | ||
surfaces_file << surfaces | ||
|
||
# save .pvd files | ||
mesh_file = File(str((out_dir / f"mesh_{resolution_factor}.pvd").absolute())) | ||
mesh_file << mesh | ||
|
||
subdomains_file = File(str((out_dir / f"subdomains_{resolution_factor}.pvd").absolute())) | ||
subdomains_file << subdomains | ||
|
||
surfaces_file = File(str((out_dir / f"surfaces_{resolution_factor}.pvd").absolute())) | ||
surfaces_file << surfaces |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,170 @@ | ||
# Gotran generated code for the "hodgkin_huxley_squid_axon_model_1952_original" model | ||
|
||
import numpy as np | ||
import math | ||
|
||
def init_state_values(**values): | ||
""" | ||
Initialize state values | ||
""" | ||
# Init values | ||
phi_M_init = -81.73542331243227 | ||
|
||
init_values = np.array([phi_M_init], dtype=np.float_) | ||
|
||
# State indices and limit checker | ||
state_ind = dict([("V", 0)]) | ||
|
||
for state_name, value in values.items(): | ||
if state_name not in state_ind: | ||
raise ValueError("{0} is not a state.".format(state_name)) | ||
ind = state_ind[state_name] | ||
|
||
# Assign value | ||
init_values[ind] = value | ||
|
||
return init_values | ||
|
||
def init_parameter_values(**values): | ||
""" | ||
Initialize parameter values | ||
""" | ||
|
||
# Membrane parameters | ||
g_Na_bar = 0 # Na max conductivity (mS/cm**2) | ||
g_K_bar = 0 # K max conductivity (mS/cm**2) | ||
g_leak_Na = 0.1 # Na leak conductivity (mS/cm**2) | ||
g_leak_K = 1.7 # K leak conductivity (mS/cm**2) | ||
|
||
m_K = 2.0 # threshold ECS K (mol/m^3) - yao 2011 | ||
m_Na = 7.7 # threshold ICS Na (mol/m^3) - yao 2011 | ||
I_max = 43 # max pump strength (muA/cm^2) | ||
|
||
K_i_init = 101.45093275680335 | ||
K_e_init = 3.369320617474813 | ||
|
||
# Set initial parameter values | ||
init_values = np.array([g_Na_bar, g_K_bar, \ | ||
g_leak_Na, g_leak_K, \ | ||
0, 0, 0, 0, \ | ||
0, 0, 0, 0, 0, | ||
m_K, m_Na, I_max, K_e_init, K_i_init], dtype=np.float_) | ||
|
||
# Parameter indices and limit checker | ||
param_ind = dict([("g_Na_bar", 0), ("g_K_bar", 1), \ | ||
("g_leak_Na", 2), ("g_leak_K", 3), \ | ||
("E_Na", 4), ("E_K", 5), \ | ||
("Cm", 6), ("stim_amplitude", 7), \ | ||
("I_ch_Na", 8), ("I_ch_K", 9), ("I_ch_Cl", 10), \ | ||
("K_e", 11), ("Na_i", 12), \ | ||
("m_K", 13), ("m_Na", 14), ("I_max", 15), \ | ||
("K_e_init", 16), ("K_i_init", 17)]) | ||
|
||
for param_name, value in values.items(): | ||
if param_name not in param_ind: | ||
raise ValueError("{0} is not a parameter.".format(param_name)) | ||
ind = param_ind[param_name] | ||
|
||
# Assign value | ||
init_values[ind] = value | ||
|
||
return init_values | ||
|
||
def state_indices(*states): | ||
""" | ||
State indices | ||
""" | ||
state_inds = dict([("V", 0)]) | ||
|
||
indices = [] | ||
for state in states: | ||
if state not in state_inds: | ||
raise ValueError("Unknown state: '{0}'".format(state)) | ||
indices.append(state_inds[state]) | ||
if len(indices)>1: | ||
return indices | ||
else: | ||
return indices[0] | ||
|
||
def parameter_indices(*params): | ||
""" | ||
Parameter indices | ||
""" | ||
param_inds = dict([("g_Na_bar", 0), ("g_K_bar", 1), \ | ||
("g_leak_Na", 2), ("g_leak_K", 3), \ | ||
("E_Na", 4), ("E_K", 5), ("Cm", 6), \ | ||
("stim_amplitude", 7), ("I_ch_Na", 8), \ | ||
("I_ch_K", 9), ("I_ch_Cl", 10), ("K_e", 11), \ | ||
("Na_i", 12), ("m_K", 13), ("m_Na", 14), ("I_max", 15), \ | ||
("K_e_init", 16), ("K_i_init", 17)]) | ||
|
||
indices = [] | ||
for param in params: | ||
if param not in param_inds: | ||
raise ValueError("Unknown param: '{0}'".format(param)) | ||
indices.append(param_inds[param]) | ||
if len(indices)>1: | ||
return indices | ||
else: | ||
return indices[0] | ||
|
||
from numbalsoda import lsoda_sig | ||
from numba import njit, cfunc, jit | ||
import numpy as np | ||
import timeit | ||
import math | ||
|
||
@cfunc(lsoda_sig, nopython=True) | ||
def rhs_numba(t, states, values, parameters): | ||
""" | ||
Compute the right hand side of the\ | ||
hodgkin_huxley_squid_axon_model_1952_original ODE | ||
""" | ||
|
||
# Assign states | ||
#assert(len(states)) == 4 | ||
|
||
# Assign parameters | ||
#assert(len(parameters)) == 11 | ||
|
||
# # Init return args | ||
# if values is None: | ||
# values = np.zeros((4,), dtype=np.float_) | ||
# else: | ||
# assert isinstance(values, np.ndarray) and values.shape == (4,) | ||
|
||
i_pump = parameters[15] / ((1 + parameters[13] / parameters[11]) ** 2 \ | ||
* (1 + parameters[14] / parameters[12]) ** 3) | ||
|
||
# Physical parameters (PDEs) | ||
temperature = 300e3 # temperature (m K) | ||
R = 8.314e3 # Gas Constant (m J/(K mol)) | ||
F = 96485e3 # Faraday's constant (mC/ mol) | ||
|
||
# set conductance | ||
E_K_init = R * temperature / F * np.log(parameters[16]/parameters[17]) #5 | ||
dphi = states[0] - parameters[5] | ||
A = 1 + np.exp(18.4/42.4) # shorthand | ||
B = 1 + np.exp(-(0.1186e3 + E_K_init)/0.0441e3) # shorthand | ||
C = 1 + np.exp((dphi + 0.0185e3)/0.0425e3) # shorthand | ||
D = 1 + np.exp(-(0.1186e3 + states[0])/0.0441e3) # shorthand | ||
g_Kir = np.sqrt(parameters[11]/parameters[16])*(A*B)/(C*D) | ||
|
||
# define and return current | ||
i_Kir = parameters[3]*g_Kir*(states[0] - parameters[5]) # umol/(cm^2*ms) | ||
|
||
# Expressions for the Sodium channel component | ||
i_Na = parameters[2] * (states[3] - parameters[4]) + 3 * i_pump | ||
|
||
# Expressions for the Potassium channel component | ||
i_K = i_Kir - 2 * i_pump | ||
|
||
# set I_ch_Na | ||
parameters[8] = i_Na | ||
# set I_ch_K | ||
parameters[9] = i_K | ||
# set I_ch_Cl | ||
parameters[10] = 0.0 | ||
|
||
# update membrane potential | ||
values[0] = (- i_K - i_Na)/parameters[6] |
Oops, something went wrong.