diff options
Diffstat (limited to 'mpm/python/usrp_mpm/sys_utils')
-rw-r--r-- | mpm/python/usrp_mpm/sys_utils/uio.py | 68 |
1 files changed, 36 insertions, 32 deletions
diff --git a/mpm/python/usrp_mpm/sys_utils/uio.py b/mpm/python/usrp_mpm/sys_utils/uio.py index c07227776..acbe2cc4f 100644 --- a/mpm/python/usrp_mpm/sys_utils/uio.py +++ b/mpm/python/usrp_mpm/sys_utils/uio.py @@ -18,17 +18,27 @@ Access to UIO mapped memory. """ -import struct import os -import mmap -from builtins import hex from builtins import object +from contextlib import contextmanager import pyudev +import usrp_mpm.libpyusrp_periphs as lib from usrp_mpm.mpmlog import get_logger UIO_SYSFS_BASE_DIR = '/sys/class/uio' UIO_DEV_BASE_DIR = '/dev' + +@contextmanager +def open_uio(label=None, path=None, length=None, read_only=True, offset=None): + """Convenience function for creating a UIO object. + Use this like you would open() for a file""" + uio_obj = UIO(label, path, length, read_only, offset) + uio_obj.open() + yield uio_obj + uio_obj.close() + + def get_all_uio_devs(): """ Return a list of all uio devices. Will look something like @@ -43,6 +53,7 @@ def get_all_uio_devs(): # Typically means UIO devices return [] + def get_uio_map_info(uio_dev, map_num): """ Returns all the map info for a given UIO device and map number. @@ -64,6 +75,7 @@ def get_uio_map_info(uio_dev, map_num): map_info[info_file] = map_info_value return map_info + def find_uio_device(label, logger=None): """ Given a label, returns a tuple (uio_device, map_info). @@ -85,6 +97,7 @@ def find_uio_device(label, logger=None): logger.warning("Found no matching UIO device for label `{0}'".format(label)) return None, None + class UIO(object): """ Provides peek/poke interfaces for uio-mapped memory. @@ -123,35 +136,30 @@ class UIO(object): self.log.error("Could not find a UIO device for label {0}".format(label)) raise RuntimeError("Could not find a UIO device for label {0}".format(label)) self._read_only = read_only - self.log.trace("Opening UIO device file {}...".format(self._path)) - self._fd = os.open(self._path, os.O_RDONLY if read_only else os.O_RDWR) - self.log.trace("Calling mmap({fd}, length={length}, offset={offset})".format( - fd=self._fd, length=hex(length), offset=hex(offset) - )) - self._mm = mmap.mmap( - self._fd, - length, - flags=mmap.MAP_SHARED, - prot=mmap.PROT_READ | (0 if read_only else mmap.PROT_WRITE), - offset=offset, - ) - - def __del__(self): - """ - Destructor needs to close the uio-mapped memory - """ - try: - self._mm.close() - os.close(self._fd) - except: - self.log.warning("Failed to properly destruct UIO object.") - pass + # Our UIO objects are managed in C++ land, which gives us more granular control over + # opening and closing + self._uio = lib.types.mmap_regs_iface(self._path, length, offset, self._read_only, False) + + def __enter__(self): + self.open() + return self + + def __exit__(self, exc_type, exc_value, traceback): + self.close() + return exc_type is None + + def open(self): + self._uio.open() + return self + + def close(self): + self._uio.close() def peek32(self, addr): """ Returns the 32-bit value starting at address addr as an integer """ - return struct.unpack('@I', self._mm[addr:addr+4])[0] + return self._uio.peek32(addr) def poke32(self, addr, val): """ @@ -160,8 +168,4 @@ class UIO(object): A value that exceeds 32 bits will be truncated to 32 bits. """ assert not self._read_only - self._mm[addr:addr+4] = struct.pack( - '@I', - (val & 0xFFFFFFFF), - ) - + return self._uio.poke32(addr, val) |