You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
249 lines
8.5 KiB
249 lines
8.5 KiB
2 years ago
|
"""
|
||
|
test_sgc_input_phaselocking.py
|
||
|
|
||
|
Test phase locking from an input sgc to a target cell type. Runs simulations
|
||
|
with AN input, and plots the results, including PSTH and phase histogram.
|
||
|
|
||
|
Usage: python test_sgc_input_phaselocking.py celltype stimulus species
|
||
|
where:
|
||
|
celltype is one of [bushy, tstellate, octopus, dstellate] (default: bushy)
|
||
|
stimulus is one of [tone, SAM, clicks] (default: tone)
|
||
|
species is one of [guineapig, mouse] (default: guineapig)
|
||
|
|
||
|
"""
|
||
|
|
||
|
import sys
|
||
|
import numpy as np
|
||
|
import pyqtgraph as pg
|
||
|
from neuron import h
|
||
|
from cnmodel.protocols import Protocol
|
||
|
from cnmodel import cells
|
||
|
from cnmodel.util import sound
|
||
|
import cnmodel.util.pynrnutilities as PU
|
||
|
from cnmodel import data
|
||
|
|
||
|
|
||
|
class SGCInputTestPL(Protocol):
|
||
|
def set_cell(self, cell="bushy"):
|
||
|
self.cell = cell
|
||
|
|
||
|
def run(
|
||
|
self, temp=34.0, dt=0.025, seed=575982035, stimulus="tone", species="mouse"
|
||
|
):
|
||
|
assert stimulus in ["tone", "SAM", "clicks"] # cases available
|
||
|
if self.cell == "bushy":
|
||
|
postCell = cells.Bushy.create(species=species)
|
||
|
elif self.cell == "tstellate":
|
||
|
postCell = cells.TStellate.create(species=species)
|
||
|
elif self.cell == "octopus":
|
||
|
postCell = cells.Octopus.create(species=species)
|
||
|
elif self.cell == "dstellate":
|
||
|
postCell = cells.DStellate.create(species=species)
|
||
|
else:
|
||
|
raise ValueError(
|
||
|
"cell %s is not yet implemented for phaselocking" % self.cell
|
||
|
)
|
||
|
self.post_cell = postCell
|
||
|
self.stimulus = stimulus
|
||
|
self.run_duration = 1.0 # in seconds
|
||
|
self.pip_duration = 0.8 # in seconds
|
||
|
self.pip_start = [0.02] # in seconds
|
||
|
self.Fs = 100e3 # in Hz
|
||
|
self.f0 = 4000.0 # stimulus in Hz
|
||
|
self.cf = 4000.0 # SGCs in Hz
|
||
|
self.fMod = 100.0 # mod freq, Hz
|
||
|
self.dMod = 50.0 # % mod depth, Hz
|
||
|
self.dbspl = 60.0
|
||
|
if self.stimulus == "SAM":
|
||
|
self.stim = sound.SAMTone(
|
||
|
rate=self.Fs,
|
||
|
duration=self.run_duration,
|
||
|
f0=self.f0,
|
||
|
fmod=self.fMod,
|
||
|
dmod=self.dMod,
|
||
|
dbspl=self.dbspl,
|
||
|
ramp_duration=2.5e-3,
|
||
|
pip_duration=self.pip_duration,
|
||
|
pip_start=self.pip_start,
|
||
|
)
|
||
|
if self.stimulus == "tone":
|
||
|
self.f0 = 1000.0
|
||
|
self.cf = 1000.0
|
||
|
self.stim = sound.TonePip(
|
||
|
rate=self.Fs,
|
||
|
duration=self.run_duration,
|
||
|
f0=self.f0,
|
||
|
dbspl=self.dbspl,
|
||
|
ramp_duration=2.5e-3,
|
||
|
pip_duration=self.pip_duration,
|
||
|
pip_start=self.pip_start,
|
||
|
)
|
||
|
|
||
|
if self.stimulus == "clicks":
|
||
|
self.click_rate = 0.020 # msec
|
||
|
self.stim = sound.ClickTrain(
|
||
|
rate=self.Fs,
|
||
|
duration=self.run_duration,
|
||
|
f0=self.f0,
|
||
|
dbspl=self.dbspl,
|
||
|
click_start=0.010,
|
||
|
click_duration=100.0e-6,
|
||
|
click_interval=self.click_rate,
|
||
|
nclicks=int((self.run_duration - 0.01) / self.click_rate),
|
||
|
ramp_duration=2.5e-3,
|
||
|
)
|
||
|
|
||
|
n_sgc = data.get(
|
||
|
"convergence", species=species, post_type=postCell.type, pre_type="sgc"
|
||
|
)[0]
|
||
|
self.n_sgc = int(np.round(n_sgc))
|
||
|
|
||
|
self.pre_cells = []
|
||
|
self.synapses = []
|
||
|
j = 0
|
||
|
for k in range(self.n_sgc):
|
||
|
seed = seed + k
|
||
|
preCell = cells.DummySGC(cf=self.cf, sr=2)
|
||
|
synapse = preCell.connect(postCell)
|
||
|
for i in range(synapse.terminal.n_rzones):
|
||
|
self["xmtr%03d" % j] = synapse.terminal.relsite._ref_XMTR[i]
|
||
|
j = j + 1
|
||
|
synapse.terminal.relsite.Dep_Flag = False
|
||
|
preCell.set_sound_stim(self.stim, seed=seed)
|
||
|
self.pre_cells.append(preCell)
|
||
|
self.synapses.append(synapse)
|
||
|
|
||
|
self["vm"] = postCell.soma(0.5)._ref_v
|
||
|
# self['prevm'] = preCell.soma(0.5)._ref_v
|
||
|
self["t"] = h._ref_t
|
||
|
postCell.cell_initialize()
|
||
|
h.tstop = 1e3 * self.run_duration # duration of a run
|
||
|
h.celsius = temp
|
||
|
h.dt = dt
|
||
|
|
||
|
self.custom_init()
|
||
|
h.run()
|
||
|
|
||
|
def show(self):
|
||
|
self.win = pg.GraphicsWindow()
|
||
|
p1 = self.win.addPlot(title="stim", row=0, col=0)
|
||
|
p1.plot(self.stim.time * 1000, self.stim.sound)
|
||
|
p1.setXLink(p1)
|
||
|
|
||
|
p2 = self.win.addPlot(title="AN spikes", row=1, col=0)
|
||
|
vt = pg.VTickGroup(self.pre_cells[0]._spiketrain)
|
||
|
p2.addItem(vt)
|
||
|
p2.setXLink(p1)
|
||
|
|
||
|
p3 = self.win.addPlot(title="%s Spikes" % self.cell, row=2, col=0)
|
||
|
bspk = PU.findspikes(self["t"], self["vm"], -30.0)
|
||
|
bspktick = pg.VTickGroup(bspk)
|
||
|
p3.addItem(bspktick)
|
||
|
p3.setXLink(p1)
|
||
|
|
||
|
p4 = self.win.addPlot(title="%s Vm" % self.cell, row=3, col=0)
|
||
|
p4.plot(self["t"], self["vm"])
|
||
|
p4.setXLink(p1)
|
||
|
|
||
|
p5 = self.win.addPlot(title="xmtr", row=0, col=1)
|
||
|
j = 0
|
||
|
for k in range(self.n_sgc):
|
||
|
synapse = self.synapses[k]
|
||
|
for i in range(synapse.terminal.n_rzones):
|
||
|
p5.plot(self["t"], self["xmtr%03d" % j], pen=(i, 15))
|
||
|
j = j + 1
|
||
|
p5.setXLink(p1)
|
||
|
|
||
|
p6 = self.win.addPlot(title="AN phase", row=1, col=1)
|
||
|
phasewin = [
|
||
|
self.pip_start[0] + 0.25 * self.pip_duration,
|
||
|
self.pip_start[0] + self.pip_duration,
|
||
|
]
|
||
|
prespk = self.pre_cells[0]._spiketrain # just sample one...
|
||
|
spkin = prespk[np.where(prespk > phasewin[0] * 1e3)]
|
||
|
spikesinwin = spkin[np.where(spkin <= phasewin[1] * 1e3)]
|
||
|
print("\nCell type: %s" % self.cell)
|
||
|
print("Stimulus: ")
|
||
|
|
||
|
# set freq for VS calculation
|
||
|
if self.stimulus == "tone":
|
||
|
f0 = self.f0
|
||
|
print(
|
||
|
"Tone: f0=%.3f at %3.1f dbSPL, cell CF=%.3f"
|
||
|
% (self.f0, self.dbspl, self.cf)
|
||
|
)
|
||
|
if self.stimulus == "SAM":
|
||
|
f0 = self.fMod
|
||
|
print(
|
||
|
(
|
||
|
"SAM Tone: f0=%.3f at %3.1f dbSPL, fMod=%3.1f dMod=%5.2f, cell CF=%.3f"
|
||
|
% (self.f0, self.dbspl, self.fMod, self.dMod, self.cf)
|
||
|
)
|
||
|
)
|
||
|
if self.stimulus == "clicks":
|
||
|
f0 = 1.0 / self.click_rate
|
||
|
print(
|
||
|
"Clicks: interval %.3f at %3.1f dbSPL, cell CF=%.3f "
|
||
|
% (self.click_rate, self.dbspl, self.cf)
|
||
|
)
|
||
|
vs = PU.vector_strength(spikesinwin, f0)
|
||
|
|
||
|
print(
|
||
|
"AN Vector Strength at %.1f: %7.3f, d=%.2f (us) Rayleigh: %7.3f p = %.3e n = %d"
|
||
|
% (f0, vs["r"], vs["d"] * 1e6, vs["R"], vs["p"], vs["n"])
|
||
|
)
|
||
|
(hist, binedges) = np.histogram(vs["ph"])
|
||
|
p6.plot(
|
||
|
binedges, hist, stepMode=True, fillBrush=(100, 100, 255, 150), fillLevel=0
|
||
|
)
|
||
|
p6.setXRange(0.0, 2 * np.pi)
|
||
|
|
||
|
p7 = self.win.addPlot(title="%s phase" % self.cell, row=2, col=1)
|
||
|
spkin = bspk[np.where(bspk > phasewin[0] * 1e3)]
|
||
|
spikesinwin = spkin[np.where(spkin <= phasewin[1] * 1e3)]
|
||
|
vs = PU.vector_strength(spikesinwin, f0)
|
||
|
print(
|
||
|
"%s Vector Strength: %7.3f, d=%.2f (us) Rayleigh: %7.3f p = %.3e n = %d"
|
||
|
% (self.cell, vs["r"], vs["d"] * 1e6, vs["R"], vs["p"], vs["n"])
|
||
|
)
|
||
|
(hist, binedges) = np.histogram(vs["ph"])
|
||
|
p7.plot(
|
||
|
binedges, hist, stepMode=True, fillBrush=(100, 100, 255, 150), fillLevel=0
|
||
|
)
|
||
|
p7.setXRange(0.0, 2 * np.pi)
|
||
|
p7.setXLink(p6)
|
||
|
|
||
|
self.win.show()
|
||
|
|
||
|
|
||
|
if __name__ == "__main__":
|
||
|
if len(sys.argv) > 1 and sys.argv[1] in ["help", "-h", "--help"]:
|
||
|
print(
|
||
|
"\nUsage: python test_sgc_input_phaselocking.py\n celltype [bushy, tstellate, octopus, dstellate] (default: bushy)"
|
||
|
)
|
||
|
print(" stimulus [tone, SAM, clicks] (default: tone)")
|
||
|
print(" species [guineapig mouse] (default: guineapig)\n")
|
||
|
exit(0)
|
||
|
if len(sys.argv) > 1:
|
||
|
cell = sys.argv[1]
|
||
|
else:
|
||
|
cell = "bushy"
|
||
|
if len(sys.argv) > 2:
|
||
|
stimulus = sys.argv[2]
|
||
|
else:
|
||
|
stimulus = "tone"
|
||
|
if len(sys.argv) > 3:
|
||
|
species = sys.argv[3]
|
||
|
else:
|
||
|
species = "guineapig"
|
||
|
print("cell type: ", cell)
|
||
|
prot = SGCInputTestPL()
|
||
|
prot.set_cell(cell)
|
||
|
prot.run(stimulus=stimulus, species=species)
|
||
|
prot.show()
|
||
|
|
||
|
import sys
|
||
|
|
||
|
if sys.flags.interactive == 0:
|
||
|
pg.QtGui.QApplication.exec_()
|