2017-03-08 17:28:55 +00:00
|
|
|
#
|
2018-02-20 00:53:15 +00:00
|
|
|
# Copyright 2017 Ettus Research, a National Instruments Company
|
2017-03-08 17:28:55 +00:00
|
|
|
#
|
2018-02-20 00:53:15 +00:00
|
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
2017-03-08 17:28:55 +00:00
|
|
|
#
|
|
|
|
|
"""
|
|
|
|
|
Code to run the discovery port
|
|
|
|
|
"""
|
|
|
|
|
|
2017-03-28 01:03:52 +00:00
|
|
|
from multiprocessing import Process
|
2017-03-08 17:28:55 +00:00
|
|
|
import socket
|
2017-05-05 21:59:35 +00:00
|
|
|
from usrp_mpm.mpmtypes import MPM_DISCOVERY_PORT
|
2017-12-12 17:59:50 +00:00
|
|
|
from usrp_mpm.mpmlog import get_main_logger
|
2018-01-10 01:49:34 +00:00
|
|
|
from usrp_mpm.mpmutils import to_binary_str
|
2017-03-08 17:28:55 +00:00
|
|
|
|
2018-01-10 01:49:34 +00:00
|
|
|
RESPONSE_PREAMBLE = b"USRP-MPM"
|
|
|
|
|
RESPONSE_SEP = b";"
|
|
|
|
|
RESPONSE_CLAIMED_KEY = b"claimed"
|
2021-10-19 16:09:31 +00:00
|
|
|
# A buffer size large enough to capture any UDP packet we receive on the
|
|
|
|
|
# discovery socket
|
|
|
|
|
MAX_SOCK_BUFSIZ = 9000
|
2017-12-08 19:17:30 +00:00
|
|
|
# For setsockopt
|
|
|
|
|
IP_MTU_DISCOVER = 10
|
|
|
|
|
IP_PMTUDISC_DO = 2
|
2017-03-08 17:28:55 +00:00
|
|
|
|
2018-01-10 01:49:34 +00:00
|
|
|
def spawn_discovery_process(shared_state, discovery_addr):
|
2017-03-08 17:28:55 +00:00
|
|
|
"""
|
|
|
|
|
Returns a process that contains the device discovery.
|
|
|
|
|
|
|
|
|
|
Arguments:
|
2017-06-07 05:04:08 +00:00
|
|
|
shared_state -- Shared state of device (is it claimed, etc.). Is a
|
|
|
|
|
SharedState() object.
|
|
|
|
|
discovery_addr -- Discovery will listen on this address(es)
|
2017-03-08 17:28:55 +00:00
|
|
|
"""
|
2017-06-07 05:04:08 +00:00
|
|
|
proc = Process(
|
|
|
|
|
target=_discovery_process,
|
2018-01-10 01:49:34 +00:00
|
|
|
args=(shared_state, discovery_addr)
|
2017-06-07 05:04:08 +00:00
|
|
|
)
|
2017-03-28 01:03:52 +00:00
|
|
|
proc.start()
|
|
|
|
|
return proc
|
2017-03-08 17:28:55 +00:00
|
|
|
|
|
|
|
|
|
2018-01-10 01:49:34 +00:00
|
|
|
def _discovery_process(state, discovery_addr):
|
2017-03-08 17:28:55 +00:00
|
|
|
"""
|
|
|
|
|
The actual process for device discovery. Is spawned by
|
|
|
|
|
spawn_discovery_process().
|
|
|
|
|
"""
|
2018-01-10 01:49:34 +00:00
|
|
|
log = get_main_logger().getChild('discovery')
|
|
|
|
|
def create_response_string(state):
|
2017-03-08 17:28:55 +00:00
|
|
|
" Generate the string that gets sent back to the requester. "
|
|
|
|
|
return RESPONSE_SEP.join(
|
|
|
|
|
[RESPONSE_PREAMBLE] + \
|
2018-01-10 01:49:34 +00:00
|
|
|
[b"type="+state.dev_type.value] + \
|
|
|
|
|
[b"product="+state.dev_product.value] + \
|
|
|
|
|
[b"serial="+state.dev_serial.value] + \
|
2022-02-04 07:37:28 +00:00
|
|
|
[b"name="+state.dev_name.value] + \
|
2021-05-14 15:15:34 +00:00
|
|
|
[b"fpga="+state.dev_fpga_type.value] + \
|
2018-01-10 01:49:34 +00:00
|
|
|
[RESPONSE_CLAIMED_KEY+to_binary_str("={}".format(state.claim_status.value))]
|
2017-03-08 17:28:55 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
|
2017-06-07 05:04:08 +00:00
|
|
|
# FIXME really, we should only bind to the subnet but I haven't gotten that
|
|
|
|
|
# working yet
|
2017-03-08 17:28:55 +00:00
|
|
|
sock.bind((("0.0.0.0", MPM_DISCOVERY_PORT)))
|
2022-02-04 09:26:14 +00:00
|
|
|
sock.setsockopt(socket.IPPROTO_IP, IP_MTU_DISCOVER, IP_PMTUDISC_DO)
|
2017-03-08 17:28:55 +00:00
|
|
|
|
2017-06-07 05:04:08 +00:00
|
|
|
# TODO yeah I know that's not how you do this
|
2017-12-08 19:17:30 +00:00
|
|
|
discovery_addr_prefix = discovery_addr.replace('.255', '')
|
2017-06-07 05:04:08 +00:00
|
|
|
if discovery_addr == '0.0.0.0':
|
|
|
|
|
discovery_addr_prefix = ''
|
2017-03-28 01:03:52 +00:00
|
|
|
|
2017-06-07 05:04:08 +00:00
|
|
|
try:
|
|
|
|
|
while True:
|
2021-10-19 16:09:31 +00:00
|
|
|
data, sender = sock.recvfrom(MAX_SOCK_BUFSIZ)
|
2018-03-06 23:16:22 +00:00
|
|
|
log.debug("Got poked by: %s", sender[0])
|
2017-06-07 05:04:08 +00:00
|
|
|
# TODO this is still part of the awful subnet identification
|
|
|
|
|
if not sender[0].startswith(discovery_addr_prefix):
|
|
|
|
|
continue
|
|
|
|
|
if data.strip(b"\0") == b"MPM-DISC":
|
2018-03-06 23:16:22 +00:00
|
|
|
log.debug("Sending discovery response to %s port: %d",
|
2018-03-06 23:45:15 +00:00
|
|
|
sender[0], sender[1])
|
2018-01-10 01:49:34 +00:00
|
|
|
resp_str = create_response_string(state)
|
|
|
|
|
send_data = resp_str
|
2018-03-06 23:16:22 +00:00
|
|
|
log.trace("Return data: %s", send_data)
|
2022-02-04 09:26:14 +00:00
|
|
|
sock.sendto(send_data, sender)
|
2017-06-07 05:04:08 +00:00
|
|
|
elif data.strip(b"\0").startswith(b"MPM-ECHO"):
|
2018-03-06 23:16:22 +00:00
|
|
|
log.debug("Received echo request from {sender}"
|
2018-03-06 23:45:15 +00:00
|
|
|
.format(sender=sender[0]))
|
2017-06-07 05:04:08 +00:00
|
|
|
send_data = data
|
2017-12-08 19:17:30 +00:00
|
|
|
try:
|
2022-02-04 09:26:14 +00:00
|
|
|
sock.sendto(send_data, sender)
|
2017-12-08 19:17:30 +00:00
|
|
|
except OSError as ex:
|
2018-03-06 23:16:22 +00:00
|
|
|
log.warning("ECHO send error: %s", str(ex))
|
2017-06-07 05:04:08 +00:00
|
|
|
except Exception as err:
|
2017-12-08 19:17:30 +00:00
|
|
|
log.error("Unexpected error: `%s' Type: `%s'", str(err), type(err))
|
2017-06-07 05:04:08 +00:00
|
|
|
sock.close()
|
|
|
|
|
exit(1)
|