aboutsummaryrefslogtreecommitdiffstats
path: root/mpm/python/usrp_mpm
diff options
context:
space:
mode:
Diffstat (limited to 'mpm/python/usrp_mpm')
-rw-r--r--mpm/python/usrp_mpm/CMakeLists.txt1
-rw-r--r--mpm/python/usrp_mpm/periph_manager/base.py12
-rw-r--r--mpm/python/usrp_mpm/periph_manager/n310.py88
-rw-r--r--mpm/python/usrp_mpm/periph_manager/udev.py35
-rw-r--r--mpm/python/usrp_mpm/rpc_server.py12
-rw-r--r--mpm/python/usrp_mpm/types.py51
6 files changed, 142 insertions, 57 deletions
diff --git a/mpm/python/usrp_mpm/CMakeLists.txt b/mpm/python/usrp_mpm/CMakeLists.txt
index 1ab48de0a..b401ac6c4 100644
--- a/mpm/python/usrp_mpm/CMakeLists.txt
+++ b/mpm/python/usrp_mpm/CMakeLists.txt
@@ -24,6 +24,7 @@ SET(USRP_MPM_TOP_FILES
${CMAKE_CURRENT_SOURCE_DIR}/types.py
${CMAKE_CURRENT_SOURCE_DIR}/discovery.py
${CMAKE_CURRENT_SOURCE_DIR}/rpc_server.py
+ ${CMAKE_CURRENT_SOURCE_DIR}/uio.py
)
LIST(APPEND USRP_MPM_FILES ${USRP_MPM_TOP_FILES})
ADD_SUBDIRECTORY(periph_manager)
diff --git a/mpm/python/usrp_mpm/periph_manager/base.py b/mpm/python/usrp_mpm/periph_manager/base.py
index c84205a76..77414c034 100644
--- a/mpm/python/usrp_mpm/periph_manager/base.py
+++ b/mpm/python/usrp_mpm/periph_manager/base.py
@@ -21,7 +21,7 @@ Mboard implementation base class
import os
from ..types import EEPROM
from .. import dboard_manager
-from .udev import get_eeprom
+from .udev import get_eeprom_path
from .udev import get_spidev_nodes
from six import iteritems
@@ -44,17 +44,20 @@ class PeriphManagerBase(object):
dboard_eeprom_addrs = {}
dboard_spimaster_addrs = {}
updateable_components = []
+ sid_endpoints = {}
+ available_endpoints = range(256)
def __init__(self):
# I know my EEPROM address, lets use it
self.overlays = ""
(self._eeprom_head, self._eeprom_rawdata) = EEPROM().read_eeprom(
- get_eeprom(self.mboard_eeprom_addr))
+ get_eeprom_path(self.mboard_eeprom_addr))
+ print self._eeprom_head
self._dboard_eeproms = {}
for dboard_slot, eeprom_addr in self.dboard_eeprom_addrs.iteritems():
spi_devices = []
# I know EEPROM adresses for my dboard slots
- eeprom_data = EEPROM().read_eeprom(get_eeprom(eeprom_addr))
+ eeprom_data = EEPROM().read_eeprom(get_eeprom_path(eeprom_addr))
# I know spidev masters on the dboard slots
hw_pid = eeprom_data[0].get("hw_pid", 0)
if hw_pid in dboard_manager.HW_PIDS:
@@ -116,7 +119,7 @@ class PeriphManagerBase(object):
# Init dboards
pass
- def _probe_interface(self, sender_addr):
+ def _allocate_sid(self, sender_addr, sid, xbar_src_addr, xbar_src_port):
"""
Overload this method in actual device implementation
"""
@@ -127,4 +130,3 @@ class PeriphManagerBase(object):
Overload this method in actual device implementation
"""
return []
-
diff --git a/mpm/python/usrp_mpm/periph_manager/n310.py b/mpm/python/usrp_mpm/periph_manager/n310.py
index 1b01ac066..4f193b54b 100644
--- a/mpm/python/usrp_mpm/periph_manager/n310.py
+++ b/mpm/python/usrp_mpm/periph_manager/n310.py
@@ -23,7 +23,13 @@ from .base import PeriphManagerBase
from .net import get_iface_addrs
from .net import byte_to_mac
from .net import get_mac_addr
+from .udev import get_uio_node
+from ..types import SID
+from ..uio import uio
+from .. import libpyusrp_periphs as lib
from logging import getLogger
+import netaddr
+import socket
LOG = getLogger(__name__)
@@ -34,9 +40,9 @@ class n310(PeriphManagerBase):
"""
hw_pids = "1"
mboard_type = "n310"
- mboard_eeprom_addr = "e0007000.spi:ec@0:i2c-tunnel"
- dboard_eeprom_addrs = {"A": "something", "B": "else"}
- dboard_spimaster_addrs = {"A": "something", "B": "else"}
+ mboard_eeprom_addr = "e0005000.i2c"
+ # dboard_eeprom_addrs = {"A": "something", "B": "else"}
+ # dboard_spimaster_addrs = {"A": "something", "B": "else"}
interfaces = {}
def __init__(self, *args, **kwargs):
@@ -44,21 +50,22 @@ class n310(PeriphManagerBase):
super(n310, self).__init__(*args, **kwargs)
data = self._read_eeprom_v1(self._eeprom_rawdata)
# mac 0: mgmt port, mac1: sfp0, mac2: sfp1
- self.interfaces["mgmt"] = {
- "mac_addr": byte_to_mac(data[0]),
- "addrs": get_iface_addrs(byte_to_mac(data[0]))
- }
- self.interfaces["sfp0"] = {
- "mac_addr": byte_to_mac(data[1]),
- "addrs": get_iface_addrs(byte_to_mac(data[1]))
- }
- self.interfaces["sfp1"] = {
- "mac_addr": byte_to_mac(data[2]),
- "addrs": get_iface_addrs(byte_to_mac(data[2]))
- }
- self.mboard_info["serial"] = data[3] # some format
+ # self.interfaces["mgmt"] = {
+ # "mac_addr": byte_to_mac(data[0]),
+ # "addrs": get_iface_addrs(byte_to_mac(data[0]))
+ # }
+ # self.interfaces["sfp0"] = {
+ # "mac_addr": byte_to_mac(data[1]),
+ # "addrs": get_iface_addrs(byte_to_mac(data[1]))
+ # }
+ # self.interfaces["sfp1"] = {
+ # "mac_addr": byte_to_mac(data[2]),
+ # "addrs": get_iface_addrs(byte_to_mac(data[2]))
+ # }
+ self.mboard_info["serial"] = data[0] # some format
+ with open("/sys/class/rfnoc_crossbar/crossbar0/local_addr", "w") as xbar:
+ xbar.write("0x2")
- print(data)
# if header.get("dataversion", 0) == 1:
@@ -66,15 +73,17 @@ class n310(PeriphManagerBase):
"""
read eeprom with data version 1
"""
- # data_version contains
+ # data contains
+ # 24 bytes header -> ignore them here
+ # 8 bytes serial
# 6 bytes mac_addr0
# 2 bytes pad
# 6 bytes mac_addr1
# 2 bytes pad
# 6 bytes mac_addr2
# 2 bytes pad
- # 8 bytes serial
- return struct.unpack_from("6s 2x 6s 2x 6s 2x 8s", data)
+ # 4 bytes CRC
+ return struct.unpack_from("!28x 8s 6s 2x 6s 2x 6s 2x I", data)
def get_interfaces(self):
"""
@@ -89,15 +98,40 @@ class n310(PeriphManagerBase):
"""
return self.interfaces.get(interface, {}).get("addrs", [])
- def _probe_interface(self, sender_addr):
+ def _allocate_sid(self, sender_addr, port, sid, xbar_src_addr, xbar_src_port):
"""
Get the MAC address of the sender and store it in the FPGA ARP table
"""
mac_addr = get_mac_addr(sender_addr)
+ new_ep = self.available_endpoints.pop(0)
if mac_addr is not None:
- # Do something with mac_address
- return True
- return False
-
-
-
+ if sender_addr not in self.sid_endpoints:
+ self.sid_endpoints.update({sender_addr: (new_ep,)})
+ else:
+ current_allocation = self.sid_endpoints.get(sender_addr)
+ new_allocation = current_allocation + (new_ep,)
+ self.sid_endpoints.update({sender_addr: new_allocation})
+ sid = SID(sid)
+ sid.set_src_addr(xbar_src_addr)
+ sid.set_src_ep(new_ep)
+ my_xbar = lib.xbar.xbar.make("/dev/crossbar0") # TODO
+ my_xbar.set_route(xbar_src_addr, 0) # TODO
+ # uio_path, uio_size = get_uio_node("misc-enet-regs0")
+ uio_path = "/dev/uio0"
+ uio_size = 0x2000
+ LOG.debug("got uio_path and size")
+ uio_obj = uio(uio_path, uio_size, read_only=False)
+ LOG.info("got my uio")
+ LOG.info("ip_addr: %s", sender_addr)
+ # LOG.info("mac_addr: %s", mac_addr)
+ ip_addr = int(netaddr.IPAddress(sender_addr))
+ mac_addr = int(netaddr.EUI(mac_addr))
+ uio_obj.poke32(0x1000 + 4*new_ep, ip_addr)
+ print("sid: %x" % (sid.get()))
+ print("gonna poke: %x %x" % (0x1000+4*new_ep, ip_addr))
+ uio_obj.poke32(0x1800 + 4*new_ep, mac_addr & 0xFFFFFFFF)
+ print("gonna poke: %x %x" % (0x1800+4*new_ep, mac_addr))
+ port = int(port)
+ uio_obj.poke32(0x1400 + 4*new_ep, ((int(port) << 16) | (mac_addr >> 32)))
+ print("gonna poke: %x %x" % (0x1400+4*new_ep, ((port << 16) | (mac_addr >> 32))))
+ return sid.get()
diff --git a/mpm/python/usrp_mpm/periph_manager/udev.py b/mpm/python/usrp_mpm/periph_manager/udev.py
index 014e18ede..a42c95ef5 100644
--- a/mpm/python/usrp_mpm/periph_manager/udev.py
+++ b/mpm/python/usrp_mpm/periph_manager/udev.py
@@ -16,19 +16,22 @@
#
import pyudev
+import os
+from logging import getLogger
+LOG = getLogger(__name__)
-def get_eeprom(address):
+def get_eeprom_path(address):
"""
Return EEPROM device path for a given I2C address
"""
context = pyudev.Context()
parent = pyudev.Device.from_name(context, "platform", address)
- paths = [device.dev_node if device.dev_node is not None else device.sys_path
+ paths = [device.device_node if device.device_node is not None else device.sys_path
for device in context.list_devices(parent=parent, subsystem="nvmem")]
if len(paths) != 1:
raise Exception("{0} paths to EEPROM found!".format(len(paths)))
- return paths[0]
+ return paths[0] + "/nvmem"
def get_spidev_nodes(spi_master):
@@ -37,6 +40,30 @@ def get_spidev_nodes(spi_master):
"""
context = pyudev.Context()
parent = pyudev.Device.from_name(context, "platform", spi_master)
- paths = [device.dev_node if device.dev_node is not None else device.sys_path
+ paths = [device.sys_path
for device in context.list_devices(parent=parent, subsystem="spidev")]
return paths
+
+def get_uio_node(uio_name):
+ """
+ Return found uio device path for a give parent name
+ """
+ context = pyudev.Context()
+ paths = [device.sys_path
+ for device in context.list_devices(subsystem="uio")]
+ LOG.debug("get_uio_node")
+ LOG.debug("got paths: %s", paths)
+ for path in paths:
+ with open(os.path.join(path, "maps", "map0", "name"), "r") as uio_file:
+ name = uio_file.read()
+ LOG.debug("uio_node name: %s", name.strip())
+ if name.strip() == uio_name:
+ with open(os.path.join(path, "maps", "map0", "size"), "r") as uio_file:
+ size = uio_file.read()
+ LOG.debug("uio_node size: %s", size.strip())
+ LOG.debug("uio_node syspath: %s", path)
+ # device = pyudev.Device.from_sys_path(context, path)
+ LOG.debug("got udev device")
+ LOG.debug("device_node: %s size: %s", "/dev/uio0", size.strip())
+ return ("/dev/uio0", int(size.strip()))
+ return ("", 0)
diff --git a/mpm/python/usrp_mpm/rpc_server.py b/mpm/python/usrp_mpm/rpc_server.py
index f712c5c87..df73b1ab0 100644
--- a/mpm/python/usrp_mpm/rpc_server.py
+++ b/mpm/python/usrp_mpm/rpc_server.py
@@ -185,20 +185,18 @@ class MPMServer(RPCServer):
This is as safe method which can be called without a claim on the device
"""
info = self.periph_manager._get_device_info()
- if self.host in ["127.0.0.1", "::1"]:
+ if self.client_host in ["127.0.0.1", "::1"]:
info["connection"] = "local"
else:
info["connection"] = "remote"
return info
- def probe_interface(self, token):
+ def allocate_sid(self, token, *args):
"""
- Forwards the call to periph_manager._probe_interface with the client ip addresss
- as argument. Should be used to probe the data interfaces on the device
+ Forwards the call to periph_manager._allocate_sid with the client ip addresss
+ as argument. Should be used to setup interfaces
"""
- if token[:256] != self._state.claim_token.value:
- return False
- return self.periph_manager._probe_interface(self.host)
+ return self.periph_manager._allocate_sid(self.client_host, *args)
diff --git a/mpm/python/usrp_mpm/types.py b/mpm/python/usrp_mpm/types.py
index cc8fe8b8d..1fbddb934 100644
--- a/mpm/python/usrp_mpm/types.py
+++ b/mpm/python/usrp_mpm/types.py
@@ -45,35 +45,58 @@ class SharedState(object):
lock=self.lock) # String with max length of 256
+class SID(object):
+ def __init__(self, sid=0):
+ self.src_addr = sid >> 24
+ self.src_ep = (sid >> 16) & 0xFF
+ self.dst_addr = (sid >> 8) & 0xFF
+ self.dst_ep = sid & 0xFF
+
+ def set_src_addr(self, new_addr):
+ self.src_addr = new_addr & 0xFF
+
+ def set_dst_addr(self, new_addr):
+ self.dst_addr = new_addr & 0xFF
+
+ def set_src_ep(self, new_addr):
+ self.src_ep = new_addr & 0xFF
+
+ def set_dst_ep(self, new_addr):
+ self.dst_ep = new_addr & 0xFF
+
+ def get(self):
+ return (self.src_addr << 24) | (self.src_ep << 16) | (self.dst_addr << 8) | self.dst_ep
+
+
class EEPROM(object):
"""
Reads out common properties and rawdata out of a nvmem path
"""
# eeprom_header contains:
# 4 bytes magic
- # 4 bytes CRC
- # 2 bytes data_version
+ # 4 bytes version
+ # 4x4 bytes mcu_flags -> throw them away
# 2 bytes hw_pid
# 2 bytes hw_rev
- # 2 bytes pad
#
- # 16 bytes in total
- eeprom_header = struct.Struct("I I H H H 2x")
+ # 28 bytes in total
+ eeprom_header = struct.Struct("!I I 16x H H")
def read_eeprom(self, nvmem_path):
"""
- Read the EEPROM located at nvmem_path and return a tuple (header, body)
+ Read the EEPROM located at nvmem_path and return a tuple (header, data)
Header is already parsed in the common header fields
+ Data contains the full eeprom data structure
"""
with open(nvmem_path, "rb") as nvmem_file:
- header = nvmem_file.read(16)
- data = nvmem_file.read(240)
- header = self.eeprom_header.unpack(header)
+ data = nvmem_file.read(256)
+ _header = self.eeprom_header.unpack_from(data)
+ print hex(_header[0]), hex(_header[1]), hex(_header[2]), hex(_header[3])
header = {
- "magic": header[0],
- "crc": header[1],
- "data_version": header[2],
- "hw_pid": header[3],
- "hw_rev": header[4],
+ "magic": _header[0],
+ "version": _header[1],
+ "hw_pid": _header[2],
+ "hw_rev": _header[3],
}
+ print header
return (header, data)