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/periph_manager/base.py8
-rw-r--r--mpm/python/usrp_mpm/periph_manager/n310.py16
-rw-r--r--mpm/python/usrp_mpm/rpc_server.py91
3 files changed, 102 insertions, 13 deletions
diff --git a/mpm/python/usrp_mpm/periph_manager/base.py b/mpm/python/usrp_mpm/periph_manager/base.py
index 87544e451..2be8c7570 100644
--- a/mpm/python/usrp_mpm/periph_manager/base.py
+++ b/mpm/python/usrp_mpm/periph_manager/base.py
@@ -352,6 +352,13 @@ class PeriphManagerBase(object):
self.log.trace("Resetting SID pool...")
self._available_endpoints = list(range(256))
+ def tear_down(self):
+ """
+ Tear down all members that need to be specially handled before
+ deconstruction.
+ """
+ self.log.debug("Teardown called for Peripheral Manager base.")
+
@no_claim
def list_updateable_components(self):
"""
@@ -456,7 +463,6 @@ class PeriphManagerBase(object):
update_func(filepath, metadata)
return True
-
def load_fpga_image(self, target=None):
"""
load a new fpga image
diff --git a/mpm/python/usrp_mpm/periph_manager/n310.py b/mpm/python/usrp_mpm/periph_manager/n310.py
index 7adbc85f0..79c866aa4 100644
--- a/mpm/python/usrp_mpm/periph_manager/n310.py
+++ b/mpm/python/usrp_mpm/periph_manager/n310.py
@@ -31,6 +31,7 @@ from ..net import get_mac_addr
from ..mpmtypes import SID
from usrp_mpm.rpc_server import no_rpc
from usrp_mpm import net
+from usrp_mpm import dtoverlay
from ..sysfs_gpio import SysFSGPIO
from ..ethtable import EthDispatcherTable
from ..liberiotable import LiberioDispatcherTable
@@ -476,10 +477,12 @@ class n310(PeriphManagerBase):
'fpga': {
'callback': "update_fpga",
'path': '/lib/firmware/n3xx.bin',
+ 'reset': True,
},
'dts': {
'callback': "update_dts",
'path': '/lib/firmware/n3xx.dts',
+ 'reset': False,
},
}
@@ -586,6 +589,19 @@ class n310(PeriphManagerBase):
for xport_mgr in itervalues(self._xport_mgrs):
xport_mgr.deinit()
+ def tear_down(self):
+ """
+ Tear down all members that need to be specially handled before
+ deconstruction.
+ For N310, this means the overlay.
+ """
+ active_overlays = self.list_active_overlays()
+ self.log.trace("N310 has active device tree overlays: {}".format(
+ active_overlays
+ ))
+ for overlay in active_overlays:
+ dtoverlay.rm_overlay(overlay)
+
###########################################################################
# Transport API
###########################################################################
diff --git a/mpm/python/usrp_mpm/rpc_server.py b/mpm/python/usrp_mpm/rpc_server.py
index 2cd44dfe5..38c9107ee 100644
--- a/mpm/python/usrp_mpm/rpc_server.py
+++ b/mpm/python/usrp_mpm/rpc_server.py
@@ -54,22 +54,20 @@ class MPMServer(RPCServer):
RPC calls to appropiate calls in the periph_manager and dboard_managers.
"""
# This is a list of methods in this class which require a claim
- default_claimed_methods = ['init', 'reclaim', 'unclaim']
+ default_claimed_methods = ['init', 'update_component', 'reclaim', 'unclaim']
- def __init__(self, state, mgr, *args, **kwargs):
+ def __init__(self, state, mgr, mgr_generator=None, *args, **kwargs):
self.log = get_main_logger().getChild('RPCServer')
self._state = state
self._timer = Greenlet()
self.session_id = None
self.periph_manager = mgr
+ self._mgr_generator = mgr_generator
self._db_methods = []
self._mb_methods = []
self.claimed_methods = copy.copy(self.default_claimed_methods)
self._last_error = ""
- self._update_component_commands(mgr, '', '_mb_methods')
- for db_slot, dboard in enumerate(mgr.dboards):
- cmd_prefix = 'db_' + str(db_slot) + '_'
- self._update_component_commands(dboard, cmd_prefix, '_db_methods')
+ self._init_rpc_calls(mgr)
# We call the server __init__ function here, and not earlier, because
# first the commands need to be registered
super(MPMServer, self).__init__(
@@ -78,6 +76,15 @@ class MPMServer(RPCServer):
**kwargs
)
+ def _init_rpc_calls(self, mgr):
+ """
+ Register all RPC calls for the motherboard and daughterboards
+ """
+ self._update_component_commands(mgr, '', '_mb_methods')
+ for db_slot, dboard in enumerate(mgr.dboards):
+ cmd_prefix = 'db_' + str(db_slot) + '_'
+ self._update_component_commands(dboard, cmd_prefix, '_db_methods')
+
def _check_token_valid(self, token):
"""
Returns True iff:
@@ -93,7 +100,6 @@ class MPMServer(RPCServer):
len(token) == TOKEN_LEN and \
self._state.claim_token.value == token
-
def _update_component_commands(self, component, namespace, storage):
"""
Detect available methods for an object and add them to the RPC server.
@@ -137,6 +143,7 @@ class MPMServer(RPCServer):
raise RuntimeError("Invalid token!")
try:
return function(*args)
+
except Exception as ex:
self.log.error(
"Uncaught exception in method %s: %s",
@@ -251,6 +258,66 @@ class MPMServer(RPCServer):
self._reset_timer()
return result
+ def reset_mgr(self):
+ """
+ Reset the Peripheral Manager for this RPC server.
+ """
+ # reassign
+ self.periph_manager.tear_down()
+ self.periph_manager = None
+ if self._mgr_generator is None:
+ raise RuntimeError("Can't reset peripheral manager- no generator function.")
+ self.periph_manager = self._mgr_generator()
+ self._init_rpc_calls(self.periph_manager)
+
+ def update_component(self, token, file_metadata_l, data_l):
+ """"
+ Updates the device component files specified by the metadata and data
+ :param file_metadata_l: List of dictionary of strings containing metadata
+ :param data_l: List of binary string with the file contents to be written
+ """
+ self._timer.kill() # Stop the timer, update_component can take some time.
+ # Check the claimed status
+ if not self._check_token_valid(token):
+ self._last_error =\
+ "Attempt to update component without valid claim from {}".format(
+ self.client_host
+ )
+ self.log.error(self._last_error)
+ raise RuntimeError("Attempt to update component without valid claim.")
+ result = self.periph_manager.update_component(file_metadata_l, data_l)
+ if not result:
+ component_ids = [metadata['id'] for metadata in file_metadata_l]
+ raise RuntimeError("Failed to update components: {}".format(component_ids))
+
+ # Check if we need to reset the peripheral manager
+ reset_now = False
+ for metadata, data in zip(file_metadata_l, data_l):
+ # Make sure the component is in the updateable_components
+ component_id = metadata['id']
+ if component_id in self.periph_manager.updateable_components:
+ # Check if that updating that component means the PM should be reset
+ if self.periph_manager.updateable_components[component_id]['reset']:
+ reset_now = True
+ else:
+ self.log.debug("ID {} not in updateable components ({})".format(
+ component_id, self.periph_manager.updateable_components))
+
+ try:
+ self.log.trace("Reset after updating component? {}".format(reset_now))
+ if reset_now:
+ self.reset_mgr()
+ self.log.debug("Reset the periph manager")
+ except Exception as ex:
+ self.log.error(
+ "Error in update_component while resetting: {}".format(
+ ex
+ ))
+ self._last_error = str(ex)
+
+ self.log.debug("End of update_component")
+ self._reset_timer()
+
def reclaim(self, token):
"""
reclaim a MPM device with a token. This operation will fail
@@ -286,8 +353,8 @@ class MPMServer(RPCServer):
self._state.claim_status.value = False
self._state.claim_token.value = b''
self.session_id = None
- self.periph_manager.claimed = False
try:
+ self.periph_manager.claimed = False
self.periph_manager.set_connection_type(None)
self.periph_manager.deinit()
except Exception as ex:
@@ -334,14 +401,14 @@ class MPMServer(RPCServer):
-def _rpc_server_process(shared_state, port, mgr):
+def _rpc_server_process(shared_state, port, mgr, mgr_generator):
"""
This is the actual process that's running the RPC server.
"""
connections = Pool(1000)
server = StreamServer(
('0.0.0.0', port),
- handle=MPMServer(shared_state, mgr),
+ handle=MPMServer(shared_state, mgr, mgr_generator),
spawn=connections)
# catch signals and stop the stream server
signal(signal.SIGTERM, lambda *args: server.stop())
@@ -349,12 +416,12 @@ def _rpc_server_process(shared_state, port, mgr):
server.serve_forever()
-def spawn_rpc_process(state, udp_port, mgr):
+def spawn_rpc_process(state, udp_port, mgr, mgr_generator):
"""
Returns a process that contains the RPC server
"""
- proc_args = [udp_port, state, mgr]
+ proc_args = [udp_port, state, mgr, mgr_generator]
proc = Process(target=_rpc_server_process, args=proc_args)
proc.start()
return proc