1
0
forked from M-Labs/artiq

dma: tentative port to NAC3

will not work due to missing context manager and possibly memory management issues
This commit is contained in:
Sebastien Bourdeauducq 2022-03-01 17:44:48 +08:00
parent 156cf42f76
commit a3b55b6164

View File

@ -5,34 +5,36 @@ the core device's SDRAM, and playing them back at higher speeds than the CPU
alone could achieve. alone could achieve.
""" """
from artiq.language.core import syscall, kernel from numpy import int32, int64
from artiq.language.types import TInt32, TInt64, TStr, TNone, TTuple
from artiq.language.core import nac3, extern, kernel, Kernel, KernelInvariant
from artiq.coredevice.exceptions import DMAError from artiq.coredevice.exceptions import DMAError
from artiq.coredevice.core import Core
from numpy import int64
@syscall
def dma_record_start(name: TStr) -> TNone: @extern
def dma_record_start(name: str):
raise NotImplementedError("syscall not simulated") raise NotImplementedError("syscall not simulated")
@syscall @extern
def dma_record_stop(duration: TInt64) -> TNone: def dma_record_stop(duration: int64):
raise NotImplementedError("syscall not simulated") raise NotImplementedError("syscall not simulated")
@syscall @extern
def dma_erase(name: TStr) -> TNone: def dma_erase(name: str):
raise NotImplementedError("syscall not simulated") raise NotImplementedError("syscall not simulated")
@syscall @extern
def dma_retrieve(name: TStr) -> TTuple([TInt64, TInt32]): def dma_retrieve(name: str) -> tuple[int64, int32]:
raise NotImplementedError("syscall not simulated") raise NotImplementedError("syscall not simulated")
@syscall @extern
def dma_playback(timestamp: TInt64, ptr: TInt32) -> TNone: def dma_playback(timestamp: int64, ptr: int32):
raise NotImplementedError("syscall not simulated") raise NotImplementedError("syscall not simulated")
@nac3
class DMARecordContextManager: class DMARecordContextManager:
"""Context manager returned by :meth:`CoreDMA.record()`. """Context manager returned by :meth:`CoreDMA.record()`.
@ -44,6 +46,9 @@ class DMARecordContextManager:
are stored in a newly created trace, and ``now`` is restored to the value are stored in a newly created trace, and ``now`` is restored to the value
it had before the context manager was entered. it had before the context manager was entered.
""" """
name: Kernel[str]
saved_now_mu: Kernel[int64]
def __init__(self): def __init__(self):
self.name = "" self.name = ""
self.saved_now_mu = int64(0) self.saved_now_mu = int64(0)
@ -52,21 +57,24 @@ class DMARecordContextManager:
def __enter__(self): def __enter__(self):
dma_record_start(self.name) # this may raise, so do it before altering now dma_record_start(self.name) # this may raise, so do it before altering now
self.saved_now_mu = now_mu() self.saved_now_mu = now_mu()
at_mu(0) at_mu(int64(0))
@kernel @kernel
def __exit__(self, type, value, traceback): def __exit__(self):
dma_record_stop(now_mu()) # see above dma_record_stop(now_mu()) # see above
at_mu(self.saved_now_mu) at_mu(self.saved_now_mu)
@nac3
class CoreDMA: class CoreDMA:
"""Core device Direct Memory Access (DMA) driver. """Core device Direct Memory Access (DMA) driver.
Gives access to the DMA functionality of the core device. Gives access to the DMA functionality of the core device.
""" """
kernel_invariants = {"core", "recorder"} core: KernelInvariant[Core]
recorder: KernelInvariant[DMARecordContextManager]
epoch: Kernel[int32]
def __init__(self, dmgr, core_device="core"): def __init__(self, dmgr, core_device="core"):
self.core = dmgr.get(core_device) self.core = dmgr.get(core_device)
@ -74,7 +82,7 @@ class CoreDMA:
self.epoch = 0 self.epoch = 0
@kernel @kernel
def record(self, name): def record(self, name: str) -> DMARecordContextManager:
"""Returns a context manager that will record a DMA trace called ``name``. """Returns a context manager that will record a DMA trace called ``name``.
Any previously recorded trace with the same name is overwritten. Any previously recorded trace with the same name is overwritten.
The trace will persist across kernel switches.""" The trace will persist across kernel switches."""
@ -83,13 +91,13 @@ class CoreDMA:
return self.recorder return self.recorder
@kernel @kernel
def erase(self, name): def erase(self, name: str):
"""Removes the DMA trace with the given name from storage.""" """Removes the DMA trace with the given name from storage."""
self.epoch += 1 self.epoch += 1
dma_erase(name) dma_erase(name)
@kernel @kernel
def playback(self, name): def playback(self, name: str):
"""Replays a previously recorded DMA trace. This function blocks until """Replays a previously recorded DMA trace. This function blocks until
the entire trace is submitted to the RTIO FIFOs.""" the entire trace is submitted to the RTIO FIFOs."""
(advance_mu, ptr) = dma_retrieve(name) (advance_mu, ptr) = dma_retrieve(name)
@ -97,14 +105,14 @@ class CoreDMA:
delay_mu(advance_mu) delay_mu(advance_mu)
@kernel @kernel
def get_handle(self, name): def get_handle(self, name: str) -> tuple[int32, int64, int32]:
"""Returns a handle to a previously recorded DMA trace. The returned handle """Returns a handle to a previously recorded DMA trace. The returned handle
is only valid until the next call to :meth:`record` or :meth:`erase`.""" is only valid until the next call to :meth:`record` or :meth:`erase`."""
(advance_mu, ptr) = dma_retrieve(name) (advance_mu, ptr) = dma_retrieve(name)
return (self.epoch, advance_mu, ptr) return (self.epoch, advance_mu, ptr)
@kernel @kernel
def playback_handle(self, handle): def playback_handle(self, handle: tuple[int32, int64, int32]):
"""Replays a handle obtained with :meth:`get_handle`. Using this function """Replays a handle obtained with :meth:`get_handle`. Using this function
is much faster than :meth:`playback` for replaying a set of traces repeatedly, is much faster than :meth:`playback` for replaying a set of traces repeatedly,
but incurs the overhead of managing the handles onto the programmer.""" but incurs the overhead of managing the handles onto the programmer."""