Skip to content

Commit

Permalink
Added proper timeout calculation when triggering the ion chamber device.
Browse files Browse the repository at this point in the history
  • Loading branch information
canismarko committed Nov 22, 2024
1 parent 0b66267 commit 73fd25b
Show file tree
Hide file tree
Showing 2 changed files with 45 additions and 1 deletion.
20 changes: 19 additions & 1 deletion src/haven/devices/ion_chamber.py
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@ class IonChamber(StandardReadable, Triggerable):

_ophyd_labels_ = {"ion_chambers", "detectors"}
_trigger_statuses = {}
_clock_register_width = 32 # bits in the register

def __init__(
self,
Expand Down Expand Up @@ -270,6 +271,21 @@ async def connect(
# Update the labjack's input's .DESC field to match the scaler channel
await self.voltmeter_channel.description.set(desc)

async def default_timeout(self):
"""Calculate the expected timeout for triggering this ion chamber."""
aws = [
self.mcs.scaler.channels[0].is_gate.get_value(),
self.mcs.scaler.preset_time.get_value(),
self.mcs.scaler.clock_frequency.get_value(),
]
is_time_limited, count_time, clock_freq = await asyncio.gather(*aws)
if is_time_limited:
# We're using the preset time to decide when to stop
return count_time + DEFAULT_TIMEOUT
else:
max_count = 2**self._clock_register_width / clock_freq
return max_count + DEFAULT_TIMEOUT

@AsyncStatus.wrap
async def trigger(self, record_dark_current=False):
"""Instruct the ion chamber's scaler to capture one data point.
Expand All @@ -294,8 +310,10 @@ async def trigger(self, record_dark_current=False):
if last_status is not None and not last_status.done:
await last_status
return
# Calculate expected timeout value
timeout = await self.default_timeout()
# Nothing to wait on yet, so trigger the scaler and stash the result
st = signal.set(True)
st = signal.set(True, timeout=timeout)
self._trigger_statuses[signal.source] = st
await st

Expand Down
26 changes: 26 additions & 0 deletions src/haven/tests/test_ion_chamber.py
Original file line number Diff line number Diff line change
Expand Up @@ -123,6 +123,7 @@ async def test_readables(ion_chamber):
@pytest.mark.asyncio
async def test_trigger(ion_chamber):
await ion_chamber.connect(mock=True)
set_mock_value(ion_chamber.mcs.scaler.clock_frequency, 50e6)
assert ion_chamber._trigger_statuses == {}
# Does the same trigger twice return the same
status1 = ion_chamber.trigger()
Expand All @@ -146,6 +147,31 @@ async def test_trigger_dark_current(ion_chamber, monkeypatch):
assert ion_chamber.mcs.scaler.record_dark_current.trigger.called


async def test_default_timeout_with_time(ion_chamber):
await ion_chamber.connect(mock=True)
await ion_chamber.mcs.scaler.channels[0].is_gate.set(True)
await ion_chamber.mcs.scaler.preset_time.set(17)
timeout = await ion_chamber.default_timeout()
assert timeout == pytest.approx(27)


async def test_default_timeout_with_gate(ion_chamber):
"""If another channel is gated, then use the maximum timeout.
Assume the internal register is 32-bit, then we can count for at most:
2**32 / clock_freq
"""
clock_freq = 50e6
await ion_chamber.connect(mock=True)
await ion_chamber.mcs.scaler.channels[0].is_gate.set(False)
await ion_chamber.mcs.scaler.clock_frequency.set(clock_freq)
timeout = await ion_chamber.default_timeout()
buff_size = 2**32
assert timeout == pytest.approx(buff_size / clock_freq + 10)


@pytest.mark.asyncio
async def test_net_current_signal(ion_chamber):
"""Test that scaler tick counts get properly converted to ion chamber current."""
Expand Down

0 comments on commit 73fd25b

Please sign in to comment.