diff options
Diffstat (limited to 'mpm/python')
-rw-r--r-- | mpm/python/CMakeLists.txt | 13 | ||||
-rw-r--r-- | mpm/python/copy_python_module.cmake | 2 | ||||
-rwxr-xr-x | mpm/python/test_rpc.py | 63 | ||||
-rwxr-xr-x | mpm/python/usrp_hwd.py | 57 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/CMakeLists.txt | 13 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/__init__.py | 2 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/dboard_manager/CMakeLists.txt | 30 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/dboard_manager/__init__.py | 1 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/dboard_manager/base.py | 1 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/dboard_manager/magnesium.py | 2 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/dboard_manager/test.py | 57 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/discovery.py | 6 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/periph_manager/CMakeLists.txt | 28 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/periph_manager/__init__.py.in (renamed from mpm/python/usrp_mpm/periph_manager/__init__.py) | 4 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/periph_manager/n310.py | 2 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/periph_manager/test.py | 63 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/rpc_process.py | 48 | ||||
-rw-r--r-- | mpm/python/usrp_mpm/rpc_server.py | 91 |
18 files changed, 308 insertions, 175 deletions
diff --git a/mpm/python/CMakeLists.txt b/mpm/python/CMakeLists.txt index 382f244c9..7daa3bbf5 100644 --- a/mpm/python/CMakeLists.txt +++ b/mpm/python/CMakeLists.txt @@ -73,17 +73,21 @@ TARGET_INCLUDE_DIRECTORIES(pyusrp_periphs PUBLIC TARGET_LINK_LIBRARIES(pyusrp_periphs ${Boost_PYTHON_LIBRARY} ${Boost_LIBRARIES} usrp-periphs) ADD_CUSTOM_COMMAND(TARGET pyusrp_periphs POST_BUILD COMMAND ${CMAKE_COMMAND} -E copy ${CMAKE_CURRENT_BINARY_DIR}/libpyusrp_periphs.so ${CMAKE_CURRENT_BINARY_DIR}/usrp_mpm/libpyusrp_periphs.so) +SET(USRP_MPM_FILES "") ADD_SUBDIRECTORY(usrp_mpm) +SET(OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/build/timestamp") SET(SETUP_PY_IN "${CMAKE_CURRENT_SOURCE_DIR}/setup.py.in") SET(SETUP_PY "${CMAKE_CURRENT_BINARY_DIR}/setup.py") -SET(OUTPUT "${CMAKE_CURRENT_BINARY_DIR}/build/timestamp") +SET(PERIPH_MGR_INIT_IN "${CMAKE_CURRENT_SOURCE_DIR}/usrp_mpm/periph_manager/__init__.py.in") +SET(PERIPH_MGR_INIT "${CMAKE_CURRENT_BINARY_DIR}/usrp_mpm/periph_manager/__init__.py") CONFIGURE_FILE(${SETUP_PY_IN} ${SETUP_PY}) +CONFIGURE_FILE(${PERIPH_MGR_INIT_IN} ${PERIPH_MGR_INIT}) ADD_CUSTOM_COMMAND(OUTPUT ${OUTPUT} - COMMAND ${CMAKE_COMMAND} -DSOURCE_DIR="${CMAKE_CURRENT_SOURCE_DIR}" -DBINARY_DIR="${CMAKE_CURRENT_BINARY_DIR}" -P ${CMAKE_CURRENT_SOURCE_DIR}/copy_python_module.cmake - COMMAND ${PYTHON} ${SETUP_PY} -q build + COMMAND ${CMAKE_COMMAND} -DSOURCE_DIR="${CMAKE_CURRENT_SOURCE_DIR}" -DBINARY_DIR="${CMAKE_CURRENT_BINARY_DIR}" -P ${CMAKE_CURRENT_SOURCE_DIR}/copy_python_module.cmake + COMMAND ${PYTHON_EXECUTABLE} ${SETUP_PY} -q build COMMAND ${CMAKE_COMMAND} -E touch ${OUTPUT} DEPENDS ${USRP_MPM_FILES}) ADD_CUSTOM_TARGET(usrp_mpm ALL DEPENDS ${OUTPUT} pyusrp_periphs) @@ -93,3 +97,6 @@ EXECUTE_PROCESS(COMMAND ${PYTHON_EXECUTABLE} -c OUTPUT_VARIABLE USRP_MPM_PYTHON_DIR OUTPUT_STRIP_TRAILING_WHITESPACE ) INSTALL(DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}/build/lib/usrp_mpm DESTINATION ${CMAKE_INSTALL_PREFIX}/${USRP_MPM_PYTHON_DIR}) +INSTALL(PROGRAMS + usrp_hwd.py + DESTINATION ${RUNTIME_DIR}) diff --git a/mpm/python/copy_python_module.cmake b/mpm/python/copy_python_module.cmake index 1b36d4fab..501793419 100644 --- a/mpm/python/copy_python_module.cmake +++ b/mpm/python/copy_python_module.cmake @@ -1,4 +1,4 @@ SET(BINARY_DIR "" CACHE STRING "") SET(SOURCE_DIR "" CACHE STRING "") -FILE(COPY ${SOURCE_DIR}/usrp_mpm/ DESTINATION ${BINARY_DIR}/usrp_mpm +FILE(COPY "${SOURCE_DIR}/usrp_mpm/" DESTINATION ${BINARY_DIR}/usrp_mpm FILES_MATCHING PATTERN *.py) diff --git a/mpm/python/test_rpc.py b/mpm/python/test_rpc.py deleted file mode 100755 index 1a97a103c..000000000 --- a/mpm/python/test_rpc.py +++ /dev/null @@ -1,63 +0,0 @@ -#!/usr/bin/env python -import socket -from mprpc import RPCClient -import usrp_mpm as mpm -import argparse - - -def parse_args(): - parser = argparse.ArgumentParser() - parser.add_argument("-a", "--address", default="0.0.0.0", type=str, help="Destination address") - parser.add_argument("-p", "--port", default=0, type=int, help="Destination port") - sub_parsers = parser.add_subparsers(dest="command") - - rpc_parser = sub_parsers.add_parser("rpc", help="Issue RPC") - rpc_parser.add_argument("-c", "--call", required=True, help="command to issue") - rpc_parser.add_argument("arguments", nargs="*") - - disc_parser = sub_parsers.add_parser("disc", help="Issue discovery") - return parser.parse_args() - - -def rpc(address, port, command, *args): - if not port: - port = mpm.types.MPM_RPC_PORT - client = RPCClient(address, port) - if args: - result = client.call(command, *args) - else: - result = client.call(command) - return result - - -def discovery(address, port): - if not port: - port = mpm.types.MPM_DISCOVERY_PORT - sock = socket.socket( - socket.AF_INET, - socket.SOCK_DGRAM) - sock.sendto(mpm.types.MPM_DISCOVERY_MESSAGE, (address, port)) - sock.settimeout(1.0) # wait max 1 second - while True: - try: - data, sender = sock.recvfrom(4096) - print("Received respons from: {}".format(sender[0])) - print("Dicovery data: {}".format(data)) - except: - break - - -def main(): - args = parse_args() - if args.command == "rpc": - if args.arguments: - result = rpc(args.address, args.port, args.call, *args.arguments) - else: - result = rpc(args.address, args.port, args.call) - print(result) - elif args.command == "disc": - discovery(args.address, args.port) - - -if __name__ == "__main__": - exit(not(main())) diff --git a/mpm/python/usrp_hwd.py b/mpm/python/usrp_hwd.py index dc72ce3b9..a0c66f62b 100755 --- a/mpm/python/usrp_hwd.py +++ b/mpm/python/usrp_hwd.py @@ -18,19 +18,33 @@ """ Main executable for the USRP Hardware Daemon """ - from __future__ import print_function -from multiprocessing import Value -import ctypes +from logging import getLogger +from logging import StreamHandler +from logging import DEBUG +from logging import Formatter import signal -import time +import sys import usrp_mpm as mpm from usrp_mpm.types import shared_state -from usrp_mpm.types import graceful_exit +from usrp_mpm.periph_manager import periph_manager + +log = getLogger("usrp_mpm") +_PROCESSES = [] -def signal_handler(signum, frame): - raise graceful_exit() +def kill_time(signal, frame): + """ + kill all processes + to be used in a signal handler + """ + for proc in _PROCESSES: + proc.terminate() + log.info("Terminating pid: {0}".format(proc.pid)) + for proc in _PROCESSES: + proc.join() + log.info("System exiting") + sys.exit(0) def main(): @@ -39,19 +53,24 @@ def main(): Main process loop. """ - procs = [] - signal.signal(signal.SIGTERM, signal_handler) - signal.signal(signal.SIGINT, signal_handler) + # Setup logging + log.setLevel(DEBUG) + ch = StreamHandler() + ch.setLevel(DEBUG) + formatter = Formatter('[%(asctime)s] [%(levelname)s] [%(name)s] [%(message)s]') + ch.setFormatter(formatter) + log.addHandler(ch) + + shared = shared_state() - procs.append(mpm.spawn_discovery_process({'a': "foo"}, shared)) - # procs.append(mpm.spawn_rpc_process("Echo", 5000)) - procs.append(mpm.spawn_rpc_process("MPM", mpm.types.MPM_RPC_PORT, shared)) - try: - for proc in procs: - proc.join() - except mpm.types.graceful_exit: - pass + mgr = periph_manager() + _PROCESSES.append( + mpm.spawn_discovery_process({'serial': mgr.get_serial()}, shared)) + _PROCESSES.append( + mpm.spawn_rpc_process(mpm.types.MPM_RPC_PORT, shared, mgr)) + signal.signal(signal.SIGTERM, kill_time) + signal.signal(signal.SIGINT, kill_time) + signal.pause() if __name__ == '__main__': exit(not main()) - diff --git a/mpm/python/usrp_mpm/CMakeLists.txt b/mpm/python/usrp_mpm/CMakeLists.txt index f8f94bb5d..12bcdac11 100644 --- a/mpm/python/usrp_mpm/CMakeLists.txt +++ b/mpm/python/usrp_mpm/CMakeLists.txt @@ -18,15 +18,16 @@ ######################################################################## # This file included, use CMake directory variables ######################################################################## - -SET(USRP_MPM_FILES +SET(USRP_MPM_FILES ${USRP_MPM_FILES}) +SET(USRP_MPM_TOP_FILES ${CMAKE_CURRENT_SOURCE_DIR}/__init__.py ${CMAKE_CURRENT_SOURCE_DIR}/types.py ${CMAKE_CURRENT_SOURCE_DIR}/discovery.py - ${CMAKE_CURRENT_SOURCE_DIR}/rpc_process.py ${CMAKE_CURRENT_SOURCE_DIR}/rpc_server.py ${CMAKE_CURRENT_SOURCE_DIR}/periphs.py - ${CMAKE_CURRENT_SOURCE_DIR}/periph_manager - ${CMAKE_CURRENT_SOURCE_DIR}/dboard_manager - PARENT_SCOPE ) +LIST(APPEND USRP_MPM_FILES ${USRP_MPM_TOP_FILES}) +ADD_SUBDIRECTORY(periph_manager) +ADD_SUBDIRECTORY(dboard_manager) + +SET(USRP_MPM_FILES ${USRP_MPM_FILES} PARENT_SCOPE) diff --git a/mpm/python/usrp_mpm/__init__.py b/mpm/python/usrp_mpm/__init__.py index c349d3d77..c7043c70c 100644 --- a/mpm/python/usrp_mpm/__init__.py +++ b/mpm/python/usrp_mpm/__init__.py @@ -19,7 +19,7 @@ MPM Module """ from discovery import spawn_discovery_process -from rpc_process import spawn_rpc_process +from rpc_server import spawn_rpc_process import types import periph_manager import dboard_manager diff --git a/mpm/python/usrp_mpm/dboard_manager/CMakeLists.txt b/mpm/python/usrp_mpm/dboard_manager/CMakeLists.txt new file mode 100644 index 000000000..b642d506e --- /dev/null +++ b/mpm/python/usrp_mpm/dboard_manager/CMakeLists.txt @@ -0,0 +1,30 @@ +# +# Copyright 2017 Ettus Research (National Instruments) +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see <http://www.gnu.org/licenses/>. +# + +######################################################################## +# This file included, use CMake directory variables +######################################################################## +SET(USRP_MPM_FILES ${USRP_MPM_FILES}) +SET(USRP_MPM_DBMGR_FILES + ${CMAKE_CURRENT_SOURCE_DIR}/__init__.py + ${CMAKE_CURRENT_SOURCE_DIR}/base.py + ${CMAKE_CURRENT_SOURCE_DIR}/magnesium.py + ${CMAKE_CURRENT_SOURCE_DIR}/eiscat.py + ${CMAKE_CURRENT_SOURCE_DIR}/unknown.py + ) +LIST(APPEND USRP_MPM_FILES ${USRP_MPM_DBMGR_FILES}) +SET(USRP_MPM_FILES ${USRP_MPM_FILES} PARENT_SCOPE) diff --git a/mpm/python/usrp_mpm/dboard_manager/__init__.py b/mpm/python/usrp_mpm/dboard_manager/__init__.py index 774f348d4..02746e78f 100644 --- a/mpm/python/usrp_mpm/dboard_manager/__init__.py +++ b/mpm/python/usrp_mpm/dboard_manager/__init__.py @@ -20,6 +20,7 @@ dboards module __init__.py from .. import libpyusrp_periphs as lib from magnesium import magnesium from eiscat import eiscat +from test import test from unknown import unknown hw_pids = { diff --git a/mpm/python/usrp_mpm/dboard_manager/base.py b/mpm/python/usrp_mpm/dboard_manager/base.py index bf0a115cc..257a2424d 100644 --- a/mpm/python/usrp_mpm/dboard_manager/base.py +++ b/mpm/python/usrp_mpm/dboard_manager/base.py @@ -30,6 +30,7 @@ class dboard_manager(object): Sanitizes arguments before calling C++ functions. Ties various constants to specific daughterboard class """ + _eeprom = {} def __init__(self, eeprom={}): self._eeprom = eeprom diff --git a/mpm/python/usrp_mpm/dboard_manager/magnesium.py b/mpm/python/usrp_mpm/dboard_manager/magnesium.py index 2201064ad..d48768208 100644 --- a/mpm/python/usrp_mpm/dboard_manager/magnesium.py +++ b/mpm/python/usrp_mpm/dboard_manager/magnesium.py @@ -20,6 +20,7 @@ magnesium dboard implementation module from base import dboard_manager from base import lib from base import log +import struct class magnesium(dboard_manager): hw_pid = 2 @@ -47,4 +48,3 @@ class magnesium(dboard_manager): # magnesium eeprom contains # nothing return struct.unpack_from("x", data) - diff --git a/mpm/python/usrp_mpm/dboard_manager/test.py b/mpm/python/usrp_mpm/dboard_manager/test.py new file mode 100644 index 000000000..b2e422bb4 --- /dev/null +++ b/mpm/python/usrp_mpm/dboard_manager/test.py @@ -0,0 +1,57 @@ +# +# Copyright 2017 Ettus Research (National Instruments) +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see <http://www.gnu.org/licenses/>. +# +""" +magnesium dboard implementation module +""" +from base import dboard_manager +from base import log + + +class fake_spi(object): + def __init__(self, addr): + self.addr = addr + + +class test_device(object): + def __init__(self, dev1, dev2, dev3): + self.dev1 = fake_spi(dev1) + self.dev2 = fake_spi(dev2) + self.dev3 = fake_spi(dev3) + + def test_method1(self, argument): + return argument + + +class test(dboard_manager): + hw_pid = 234 + special_eeprom_addrs = {"special0": "something"} + spi_chipselect = {"0": "dev1", "1": "dev2", "2": "dev3"} + spidevs = {} + + def __init__(self, *args, **kwargs): + # eeprom_data is a tuple (head_dict, raw_data) + super(test, self).__init__(*args, **kwargs) + # I'm the test device, I can fake out my EEPROM + self.dev1 = "0" + self.dev2 = "1" + self.dev3 = "2" + + def init_device(self): + self._device = test_device(self.dev1, self.dev2, self.dev3) + + + diff --git a/mpm/python/usrp_mpm/discovery.py b/mpm/python/usrp_mpm/discovery.py index 727fe2b94..174866498 100644 --- a/mpm/python/usrp_mpm/discovery.py +++ b/mpm/python/usrp_mpm/discovery.py @@ -68,9 +68,9 @@ def _discovery_process(device_info, state): try: while True: data, sender = sock.recvfrom(4096) - if data == "MPM-DISC": + if data.strip("\0") == "MPM-DISC": send_data = create_response_string() send_sock.sendto(send_data, sender) - except graceful_exit: + except: sock.close() - print("I'm done") + send_sock.close() diff --git a/mpm/python/usrp_mpm/periph_manager/CMakeLists.txt b/mpm/python/usrp_mpm/periph_manager/CMakeLists.txt new file mode 100644 index 000000000..879ac20c1 --- /dev/null +++ b/mpm/python/usrp_mpm/periph_manager/CMakeLists.txt @@ -0,0 +1,28 @@ +# +# Copyright 2017 Ettus Research (National Instruments) +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see <http://www.gnu.org/licenses/>. +# + +######################################################################## +# This file included, use CMake directory variables +######################################################################## +SET(USRP_MPM_FILES ${USRP_MPM_FILES}) +SET(USRP_MPM_PERIPHMGR_FILES + ${CMAKE_CURRENT_SOURCE_DIR}/__init__.py.in + ${CMAKE_CURRENT_SOURCE_DIR}/base.py + ${CMAKE_CURRENT_SOURCE_DIR}/n310.py + ) +LIST(APPEND USRP_MPM_FILES ${USRP_MPM_TOP_FILES}) +SET(USRP_MPM_FILES ${USRP_MPM_FILES} PARENT_SCOPE) diff --git a/mpm/python/usrp_mpm/periph_manager/__init__.py b/mpm/python/usrp_mpm/periph_manager/__init__.py.in index 96eecf62e..0956b849e 100644 --- a/mpm/python/usrp_mpm/periph_manager/__init__.py +++ b/mpm/python/usrp_mpm/periph_manager/__init__.py.in @@ -24,6 +24,6 @@ from .. import dboard_manager from .. import types try: - from n310 import n310 as periph_manager + from ${MPM_DEVICE} import ${MPM_DEVICE} as periph_manager except ImportError: - raise("Could not import n310") + raise Exception("Could not import ${MPM_DEVICE}") diff --git a/mpm/python/usrp_mpm/periph_manager/n310.py b/mpm/python/usrp_mpm/periph_manager/n310.py index e270387f6..d1c31540b 100644 --- a/mpm/python/usrp_mpm/periph_manager/n310.py +++ b/mpm/python/usrp_mpm/periph_manager/n310.py @@ -27,7 +27,7 @@ class n310(periph_manager): dboard_eeprom_addrs = {"A": "something", "B": "else"} dboard_spimaster_addrs = {"A": "something", "B": "else"} - def __init__(self, eeprom_device, *args, **kwargs): + def __init__(self, *args, **kwargs): # First initialize parent class - will populate self._eeprom_head and self._eeprom_rawdata super(n310, self).__init__(*args, **kwargs) data = self.read_eeprom_v1(self._eeprom_rawdata) diff --git a/mpm/python/usrp_mpm/periph_manager/test.py b/mpm/python/usrp_mpm/periph_manager/test.py new file mode 100644 index 000000000..c9cbc1f3f --- /dev/null +++ b/mpm/python/usrp_mpm/periph_manager/test.py @@ -0,0 +1,63 @@ +# +# Copyright 2017 Ettus Research (National Instruments) +# +# This program is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. +# +# This program is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. +# +# You should have received a copy of the GNU General Public License +# along with this program. If not, see <http://www.gnu.org/licenses/>. +# +""" +test periph_manager implementation module +""" +from base import periph_manager +from . import dboard_manager +import random +import string +import struct + + +class test(periph_manager): + hw_pids = "42" + mboard_eeprom_addr = None + dboard_eeprom_addrs = {"A": "something", "B": "else"} + dboard_spimaster_addrs = {"A": "something", "B": "else"} + + def __init__(self, *args, **kwargs): + # First initialize parent class - will populate self._eeprom_head and self._eeprom_rawdata + # super(n310, self).__init__(*args, **kwargs) + # if header.get("dataversion", 0) == 1: + self._eeprom = self.read_eeprom_fake() + self._serial = "AABBCCDDEEFF" + + # I'm the test periph_manager, I know I have test dboards attached + self.dboards = { + "A": dboard_manager.test(self.read_db_eeprom_random()), + "B": dboard_manager.test(self.read_db_eeprom_random()) + } + + def read_eeprom_fake(self): + fake_eeprom = { + "magic": 42, + "crc": 4242, + "data_version": 42, + "hw_pid": 42, + "hw_rev": 5 + } + + return fake_eeprom + + def read_db_eeprom_random(self): + fake_eeprom = { + "serial": ''.join( + random.choice("ABCDEF" + string.digits) + for _ in range(16)) + } + return fake_eeprom diff --git a/mpm/python/usrp_mpm/rpc_process.py b/mpm/python/usrp_mpm/rpc_process.py deleted file mode 100644 index b8783f325..000000000 --- a/mpm/python/usrp_mpm/rpc_process.py +++ /dev/null @@ -1,48 +0,0 @@ -# -# Copyright 2017 Ettus Research (National Instruments) -# -# This program is free software: you can redistribute it and/or modify -# it under the terms of the GNU General Public License as published by -# the Free Software Foundation, either version 3 of the License, or -# (at your option) any later version. -# -# This program is distributed in the hope that it will be useful, -# but WITHOUT ANY WARRANTY; without even the implied warranty of -# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the -# GNU General Public License for more details. -# -# You should have received a copy of the GNU General Public License -# along with this program. If not, see <http://www.gnu.org/licenses/>. -# -""" -Code to run the discovery port -""" - -from __future__ import print_function -from multiprocessing import Process - -from gevent.server import StreamServer -from mprpc import RPCServer -from types import graceful_exit, MPM_RPC_PORT -import rpc_server - - - -def spawn_rpc_process(server, state, udp_port=MPM_RPC_PORT): - """ - Returns a process that contains the RPC server - """ - - p_args = [server, udp_port, state] - p = Process(target=_rpc_server_process, args=p_args) - p.start() - return p - - -def _rpc_server_process(server, shared_state, port): - try: - rpc_class = getattr(rpc_server, server+"Server") - server = StreamServer(('0.0.0.0', port), handle=rpc_class(shared_state)) - server.serve_forever() - except graceful_exit: - server.close() diff --git a/mpm/python/usrp_mpm/rpc_server.py b/mpm/python/usrp_mpm/rpc_server.py index a5dda0826..ddb588aa6 100644 --- a/mpm/python/usrp_mpm/rpc_server.py +++ b/mpm/python/usrp_mpm/rpc_server.py @@ -17,25 +17,57 @@ """ Implemented RPC Servers """ - from __future__ import print_function +from gevent.server import StreamServer +from types import graceful_exit, MPM_RPC_PORT from mprpc import RPCServer -from usrp_mpm import periphs +from six import iteritems +import time + +from multiprocessing import Process +class MPMServer(RPCServer): + _db_methods = {} + def __init__(self, state, mgr): + self._state = state + # Instead do self.mboard = periphs.init_periph_manager(args...) + self.periph_manager = mgr + for db_slot, db in iteritems(mgr.dboards): + methods = (m for m in dir(db) if not m.startswith('_') and callable(getattr(db, m))) + for method in methods: + command_name = 'db_'+ db_slot + '_' + method + self._add_command(getattr(db,method), command_name) + db_methods = self._db_methods.get(db_slot, []) + db_methods.append(command_name) + self._db_methods.update({db_slot: db_methods}) -class EchoServer(RPCServer): - def echo(self, arg): - print(arg) - return arg + # When we do init we can just add dboard/periph_manager methods with setattr(self, method) + # Maybe using partial + # To remove methods again we also have to remove them from self._methods dict (they're cached) + super(MPMServer, self).__init__() + def _add_command(self, function, command): + setattr(self, command, function) -class ClaimServer(RPCServer): - def __init__(self, state): - self._state = state - super(ClaimServer, self).__init__() + + def list_methods(self): + """ + Returns all public methods of this RPC server + """ + methods = filter(lambda entry: not entry.startswith('_'), dir(self)) # Return public methods + methods_with_docs = map(lambda m: (m, getattr(self, m).__doc__), methods) + return methods_with_docs + + def ping(self, data=None): + """ + Take in data as argument and send it back + """ + return data def claim(self, token): - 'claim `token` - claims the MPM device with given token' + """ + claim `token` - claims the MPM device with given token + """ if self._state.claim_status.value: if self._state.claim_token.value == token: return True @@ -45,28 +77,33 @@ class ClaimServer(RPCServer): return True def unclaim(self, token): - 'unclaim `token` - unclaims the MPM device if it is claimed with this token' + """ + unclaim `token` - unclaims the MPM device if it is claimed with this token + """ if self._state.claim_status.value and self._state.claim_token.value == token: self._state.claim_status.value = False self._state.claim_token.value = "" return True return False - def list_methods(self): - methods = filter(lambda entry: not entry.startswith('_'), dir(self)) # Return public methods - methods_with_docs = map(lambda m: (m, getattr(self,m).__doc__), methods) - return methods_with_docs +def _rpc_server_process(shared_state, port, mgr): + """ + Start the RPC server + """ + server = StreamServer(('0.0.0.0', port), handle=MPMServer(shared_state, mgr)) + try: + server.serve_forever() + except: + server.close() -class MPMServer(RPCServer): - def __init__(self, state): - # Instead do self.mboard = periphs.init_periph_manager(args...) - self.periph_manager = periphs.init_periph_manager() - # When we do init we can just add dboard/periph_manager methods with setattr(self, method) - # Maybe using partial - # To remove methods again we also have to remove them from self._methods dict (they're cached) - def get_clock_id(self, dboard): - dboard = getattr(self.mboard, "get_dboard_"+dboard) - clk = dboard.get_clock_gen() - return clk.get_chip_id() +def spawn_rpc_process(state, udp_port, mgr): + """ + Returns a process that contains the RPC server + """ + + p_args = [udp_port, state, mgr] + p = Process(target=_rpc_server_process, args=p_args) + p.start() + return p |