2022-04-01 11:18:18 -04:00
# MIT License
#
# Copyright (c) 2016-2022 Mark Qvist / unsigned.io
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in all
# copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
# SOFTWARE.
2020-04-22 06:07:13 -04:00
from . Interface import Interface
2018-04-04 08:14:22 -04:00
from time import sleep
import sys
import threading
import time
import RNS
class KISS ( ) :
2020-08-13 06:15:56 -04:00
FEND = 0xC0
FESC = 0xDB
TFEND = 0xDC
TFESC = 0xDD
CMD_UNKNOWN = 0xFE
CMD_DATA = 0x00
CMD_TXDELAY = 0x01
CMD_P = 0x02
CMD_SLOTTIME = 0x03
CMD_TXTAIL = 0x04
CMD_FULLDUPLEX = 0x05
CMD_SETHARDWARE = 0x06
CMD_READY = 0x0F
CMD_RETURN = 0xFF
@staticmethod
def escape ( data ) :
data = data . replace ( bytes ( [ 0xdb ] ) , bytes ( [ 0xdb , 0xdd ] ) )
data = data . replace ( bytes ( [ 0xc0 ] ) , bytes ( [ 0xdb , 0xdc ] ) )
return data
2020-05-03 17:06:53 -04:00
2018-04-04 09:26:34 -04:00
class KISSInterface ( Interface ) :
2020-08-13 06:15:56 -04:00
MAX_CHUNK = 32768
2022-04-17 13:35:31 -04:00
BITRATE_GUESS = 1200
2024-11-21 06:25:59 -05:00
DEFAULT_IFAC_SIZE = 8
2020-08-13 06:15:56 -04:00
owner = None
port = None
speed = None
databits = None
parity = None
stopbits = None
serial = None
2024-11-21 06:25:59 -05:00
def __init__ ( self , owner , configuration ) :
2021-12-01 07:57:40 -05:00
import importlib
2021-12-01 07:39:51 -05:00
if importlib . util . find_spec ( ' serial ' ) != None :
import serial
else :
RNS . log ( " Using the KISS interface requires a serial communication module to be installed. " , RNS . LOG_CRITICAL )
RNS . log ( " You can install one with the command: python3 -m pip install pyserial " , RNS . LOG_CRITICAL )
RNS . panic ( )
2023-09-30 13:11:10 -04:00
super ( ) . __init__ ( )
2024-11-21 06:25:59 -05:00
c = configuration
name = c [ " name " ]
preamble = int ( c [ " preamble " ] ) if " preamble " in c else None
txtail = int ( c [ " txtail " ] ) if " txtail " in c else None
persistence = int ( c [ " persistence " ] ) if " persistence " in c else None
slottime = int ( c [ " slottime " ] ) if " slottime " in c else None
flow_control = c . as_bool ( " flow_control " ) if " flow_control " in c else False
port = c [ " port " ] if " port " in c else None
speed = int ( c [ " speed " ] ) if " speed " in c else 9600
databits = int ( c [ " databits " ] ) if " databits " in c else 8
parity = c [ " parity " ] if " parity " in c else " N "
stopbits = int ( c [ " stopbits " ] ) if " stopbits " in c else 1
beacon_interval = int ( c [ " id_interval " ] ) if " id_interval " in c else None
beacon_data = c [ " id_callsign " ] if " id_callsign " in c else None
if port == None :
raise ValueError ( " No port specified for serial interface " )
2021-09-24 14:10:04 -04:00
2022-05-29 09:43:50 -04:00
self . HW_MTU = 564
2021-05-17 16:01:56 -04:00
if beacon_data == None :
beacon_data = " "
2021-12-05 05:44:30 -05:00
self . pyserial = serial
2020-08-13 06:15:56 -04:00
self . serial = None
self . owner = owner
self . name = name
self . port = port
self . speed = speed
self . databits = databits
self . parity = serial . PARITY_NONE
self . stopbits = stopbits
self . timeout = 100
self . online = False
2020-08-13 09:06:39 -04:00
self . beacon_i = beacon_interval
self . beacon_d = beacon_data . encode ( " utf-8 " )
self . first_tx = None
2022-04-17 13:35:31 -04:00
self . bitrate = KISSInterface . BITRATE_GUESS
2020-08-13 06:15:56 -04:00
self . packet_queue = [ ]
self . flow_control = flow_control
self . interface_ready = False
2021-09-03 04:56:49 -04:00
self . flow_control_timeout = 5
2020-08-13 09:06:39 -04:00
self . flow_control_locked = time . time ( )
2020-08-13 06:15:56 -04:00
self . preamble = preamble if preamble != None else 350 ;
self . txtail = txtail if txtail != None else 20 ;
self . persistence = persistence if persistence != None else 64 ;
self . slottime = slottime if slottime != None else 20 ;
if parity . lower ( ) == " e " or parity . lower ( ) == " even " :
self . parity = serial . PARITY_EVEN
if parity . lower ( ) == " o " or parity . lower ( ) == " odd " :
self . parity = serial . PARITY_ODD
try :
2021-12-05 05:44:30 -05:00
self . open_port ( )
2020-08-13 06:15:56 -04:00
except Exception as e :
RNS . log ( " Could not open serial port " + self . port , RNS . LOG_ERROR )
raise e
if self . serial . is_open :
2021-12-05 05:44:30 -05:00
self . configure_device ( )
2020-08-13 06:15:56 -04:00
else :
raise IOError ( " Could not open serial port " )
2021-12-05 05:44:30 -05:00
def open_port ( self ) :
2021-12-05 08:35:25 -05:00
RNS . log ( " Opening serial port " + self . port + " ... " , RNS . LOG_VERBOSE )
2021-12-05 05:44:30 -05:00
self . serial = self . pyserial . Serial (
port = self . port ,
baudrate = self . speed ,
bytesize = self . databits ,
parity = self . parity ,
stopbits = self . stopbits ,
xonxoff = False ,
rtscts = False ,
timeout = 0 ,
inter_byte_timeout = None ,
write_timeout = None ,
dsrdtr = False ,
)
def configure_device ( self ) :
# Allow time for interface to initialise before config
sleep ( 2.0 )
thread = threading . Thread ( target = self . readLoop )
2022-09-30 13:02:25 -04:00
thread . daemon = True
2021-12-05 05:44:30 -05:00
thread . start ( )
self . online = True
RNS . log ( " Serial port " + self . port + " is now open " )
RNS . log ( " Configuring KISS interface parameters... " )
self . setPreamble ( self . preamble )
self . setTxTail ( self . txtail )
self . setPersistence ( self . persistence )
self . setSlotTime ( self . slottime )
self . setFlowControl ( self . flow_control )
self . interface_ready = True
RNS . log ( " KISS interface configured " )
2020-08-13 06:15:56 -04:00
def setPreamble ( self , preamble ) :
preamble_ms = preamble
preamble = int ( preamble_ms / 10 )
if preamble < 0 :
preamble = 0
if preamble > 255 :
preamble = 255
kiss_command = bytes ( [ KISS . FEND ] ) + bytes ( [ KISS . CMD_TXDELAY ] ) + bytes ( [ preamble ] ) + bytes ( [ KISS . FEND ] )
written = self . serial . write ( kiss_command )
if written != len ( kiss_command ) :
raise IOError ( " Could not configure KISS interface preamble to " + str ( preamble_ms ) + " (command value " + str ( preamble ) + " ) " )
def setTxTail ( self , txtail ) :
txtail_ms = txtail
txtail = int ( txtail_ms / 10 )
if txtail < 0 :
txtail = 0
if txtail > 255 :
txtail = 255
kiss_command = bytes ( [ KISS . FEND ] ) + bytes ( [ KISS . CMD_TXTAIL ] ) + bytes ( [ txtail ] ) + bytes ( [ KISS . FEND ] )
written = self . serial . write ( kiss_command )
if written != len ( kiss_command ) :
raise IOError ( " Could not configure KISS interface TX tail to " + str ( txtail_ms ) + " (command value " + str ( txtail ) + " ) " )
def setPersistence ( self , persistence ) :
if persistence < 0 :
persistence = 0
if persistence > 255 :
persistence = 255
kiss_command = bytes ( [ KISS . FEND ] ) + bytes ( [ KISS . CMD_P ] ) + bytes ( [ persistence ] ) + bytes ( [ KISS . FEND ] )
written = self . serial . write ( kiss_command )
if written != len ( kiss_command ) :
raise IOError ( " Could not configure KISS interface persistence to " + str ( persistence ) )
def setSlotTime ( self , slottime ) :
slottime_ms = slottime
slottime = int ( slottime_ms / 10 )
if slottime < 0 :
slottime = 0
if slottime > 255 :
slottime = 255
kiss_command = bytes ( [ KISS . FEND ] ) + bytes ( [ KISS . CMD_SLOTTIME ] ) + bytes ( [ slottime ] ) + bytes ( [ KISS . FEND ] )
written = self . serial . write ( kiss_command )
if written != len ( kiss_command ) :
raise IOError ( " Could not configure KISS interface slot time to " + str ( slottime_ms ) + " (command value " + str ( slottime ) + " ) " )
def setFlowControl ( self , flow_control ) :
kiss_command = bytes ( [ KISS . FEND ] ) + bytes ( [ KISS . CMD_READY ] ) + bytes ( [ 0x01 ] ) + bytes ( [ KISS . FEND ] )
written = self . serial . write ( kiss_command )
if written != len ( kiss_command ) :
if ( flow_control ) :
raise IOError ( " Could not enable KISS interface flow control " )
else :
raise IOError ( " Could not enable KISS interface flow control " )
def processIncoming ( self , data ) :
2021-09-24 14:10:04 -04:00
self . rxb + = len ( data )
2020-08-13 06:15:56 -04:00
self . owner . inbound ( data , self )
def processOutgoing ( self , data ) :
2021-09-24 14:10:04 -04:00
datalen = len ( data )
2020-08-13 06:15:56 -04:00
if self . online :
if self . interface_ready :
if self . flow_control :
self . interface_ready = False
2020-08-13 09:06:39 -04:00
self . flow_control_locked = time . time ( )
2020-08-13 06:15:56 -04:00
data = data . replace ( bytes ( [ 0xdb ] ) , bytes ( [ 0xdb ] ) + bytes ( [ 0xdd ] ) )
data = data . replace ( bytes ( [ 0xc0 ] ) , bytes ( [ 0xdb ] ) + bytes ( [ 0xdc ] ) )
frame = bytes ( [ KISS . FEND ] ) + bytes ( [ 0x00 ] ) + data + bytes ( [ KISS . FEND ] )
written = self . serial . write ( frame )
2021-09-24 14:10:04 -04:00
self . txb + = datalen
2020-08-13 09:06:39 -04:00
if data == self . beacon_d :
self . first_tx = None
else :
if self . first_tx == None :
self . first_tx = time . time ( )
2020-08-13 06:15:56 -04:00
if written != len ( frame ) :
raise IOError ( " Serial interface only wrote " + str ( written ) + " bytes of " + str ( len ( data ) ) )
else :
self . queue ( data )
def queue ( self , data ) :
self . packet_queue . append ( data )
def process_queue ( self ) :
if len ( self . packet_queue ) > 0 :
data = self . packet_queue . pop ( 0 )
self . interface_ready = True
self . processOutgoing ( data )
elif len ( self . packet_queue ) == 0 :
self . interface_ready = True
def readLoop ( self ) :
try :
in_frame = False
escape = False
command = KISS . CMD_UNKNOWN
data_buffer = b " "
last_read_ms = int ( time . time ( ) * 1000 )
while self . serial . is_open :
if self . serial . in_waiting :
byte = ord ( self . serial . read ( 1 ) )
last_read_ms = int ( time . time ( ) * 1000 )
if ( in_frame and byte == KISS . FEND and command == KISS . CMD_DATA ) :
in_frame = False
self . processIncoming ( data_buffer )
elif ( byte == KISS . FEND ) :
in_frame = True
command = KISS . CMD_UNKNOWN
data_buffer = b " "
2022-05-29 09:43:50 -04:00
elif ( in_frame and len ( data_buffer ) < self . HW_MTU ) :
2020-08-13 06:15:56 -04:00
if ( len ( data_buffer ) == 0 and command == KISS . CMD_UNKNOWN ) :
# We only support one HDLC port for now, so
# strip off the port nibble
byte = byte & 0x0F
command = byte
elif ( command == KISS . CMD_DATA ) :
if ( byte == KISS . FESC ) :
escape = True
else :
if ( escape ) :
if ( byte == KISS . TFEND ) :
byte = KISS . FEND
if ( byte == KISS . TFESC ) :
byte = KISS . FESC
escape = False
data_buffer = data_buffer + bytes ( [ byte ] )
elif ( command == KISS . CMD_READY ) :
self . process_queue ( )
else :
time_since_last = int ( time . time ( ) * 1000 ) - last_read_ms
if len ( data_buffer ) > 0 and time_since_last > self . timeout :
data_buffer = b " "
in_frame = False
command = KISS . CMD_UNKNOWN
escape = False
2021-09-03 04:56:49 -04:00
sleep ( 0.05 )
2020-08-13 06:15:56 -04:00
2020-08-13 09:06:39 -04:00
if self . flow_control :
if not self . interface_ready :
if time . time ( ) > self . flow_control_locked + self . flow_control_timeout :
2021-09-03 04:56:49 -04:00
RNS . log ( " Interface " + str ( self ) + " is unlocking flow control due to time-out. This should not happen. Your hardware might have missed a flow-control READY command, or maybe it does not support flow-control. " , RNS . LOG_WARNING )
2020-08-13 09:06:39 -04:00
self . process_queue ( )
if self . beacon_i != None and self . beacon_d != None :
if self . first_tx != None :
if time . time ( ) > self . first_tx + self . beacon_i :
RNS . log ( " Interface " + str ( self ) + " is transmitting beacon data: " + str ( self . beacon_d . decode ( " utf-8 " ) ) , RNS . LOG_DEBUG )
self . first_tx = None
self . processOutgoing ( self . beacon_d )
2020-08-13 06:15:56 -04:00
except Exception as e :
self . online = False
RNS . log ( " A serial port error occurred, the contained exception was: " + str ( e ) , RNS . LOG_ERROR )
2021-12-05 08:35:25 -05:00
RNS . log ( " The interface " + str ( self ) + " experienced an unrecoverable error and is now offline. " , RNS . LOG_ERROR )
2021-09-18 16:49:04 -04:00
if RNS . Reticulum . panic_on_interface_error :
RNS . panic ( )
2020-08-13 06:15:56 -04:00
2021-12-05 08:35:25 -05:00
RNS . log ( " Reticulum will attempt to reconnect the interface periodically. " , RNS . LOG_ERROR )
2021-12-05 05:44:30 -05:00
self . online = False
self . serial . close ( )
self . reconnect_port ( )
def reconnect_port ( self ) :
while not self . online :
try :
time . sleep ( 5 )
2021-12-05 08:35:25 -05:00
RNS . log ( " Attempting to reconnect serial port " + str ( self . port ) + " for " + str ( self ) + " ... " , RNS . LOG_VERBOSE )
2021-12-05 05:44:30 -05:00
self . open_port ( )
if self . serial . is_open :
self . configure_device ( )
except Exception as e :
RNS . log ( " Error while reconnecting port, the contained exception was: " + str ( e ) , RNS . LOG_ERROR )
2021-12-05 08:35:25 -05:00
RNS . log ( " Reconnected serial port for " + str ( self ) )
2023-10-01 05:39:24 -04:00
def should_ingress_limit ( self ) :
return False
2020-08-13 06:15:56 -04:00
def __str__ ( self ) :
return " KISSInterface[ " + self . name + " ] "