Skip to content

Commit

Permalink
add code for calibrating ODE models in EMIx simulations
Browse files Browse the repository at this point in the history
  • Loading branch information
adajel committed Aug 8, 2024
1 parent 0af8171 commit a418fd3
Show file tree
Hide file tree
Showing 8 changed files with 1,814 additions and 0 deletions.
634 changes: 634 additions & 0 deletions emix-simulations/make_figures_calibration.py

Large diffs are not rendered by default.

117 changes: 117 additions & 0 deletions emix-simulations/make_mesh_3D.py
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
170 changes: 170 additions & 0 deletions emix-simulations/mm_glial.py
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]
Loading

0 comments on commit a418fd3

Please sign in to comment.