This repository has been archived on 2019-08-15. You can view files and clone it, but cannot push or open issues or pull requests.
TheThingsNetworkGateway/lora_single_chan_gateway/board_config.py

270 lines
No EOL
8.6 KiB
Python

import datetime
import logging
import time
import RPi.GPIO as GPIO
import spidev
class LoraBoardDraguino():
def __init__(self, frequency, sf):
assert 7 <= sf <= 12
self.frequency = frequency
self.sf = sf
# pin are given in BCM schema
self._pin_ss = 25 # GPIO 6
self._pin_dio0 = 4 # GPIO 7
self._pin_rst = 17 # GPIO 0
self._spi_bus = 0
self._spi_cs = 0 # TODO: Draguino has it's own non-standard CS pin
self.spi = None
self._is_sx1272 = None # is set in setup
def __enter__(self):
self.setup_device()
return self
def __exit__(self, exc_type, exc_val, exc_tb):
self.teardown_device()
def _init_pins(self):
GPIO.setmode(GPIO.BCM)
GPIO.setup(self._pin_ss, GPIO.OUT)
GPIO.setup(self._pin_dio0, GPIO.IN)
GPIO.setup(self._pin_rst, GPIO.OUT)
# save default values
#GPIO.output(self._pin_ss, 1)
#GPIO.output(self._pin_rst, 0)
def _init_spi(self):
self.spi = spidev.SpiDev()
self.spi.open(self._spi_bus, self._spi_cs)
self.spi.max_speed_hz = 5000000 # TODO: what value works good?
def setup_device(self):
self._init_pins()
self._init_spi()
GPIO.output(self._pin_rst, 1)
time.sleep(0.10)
GPIO.output(self._pin_rst, 0)
time.sleep(0.10)
version = self.read_register(SX127x.REG_VERSION)
if version == 0x22:
logging.info("SX1272 detected, starting...")
self._is_sx1272 = True
else:
# sx1276?
GPIO.output(self._pin_rst, 0)
time.sleep(0.10)
GPIO.output(self._pin_rst, 1)
time.sleep(0.10)
version = self.read_register(SX127x.REG_VERSION)
if version == 0x12:
logging.info("SX1276 detected, starting...")
self._is_sx1272 = False
else:
logging.critical("Unrecognized transceiver")
raise RuntimeError("Unrecognized transceiver")
self.write_register(SX127x.REG_OPMODE, SX127x.SX72_MODE_LONG_RANGE | SX127x.SX72_MODE_SLEEP)
frf = int((self.frequency << 19) / 32000000)
self.write_register(SX127x.REG_FRF_MSB, (frf >> 16) & 0xFF)
self.write_register(SX127x.REG_FRF_MID, (frf >> 8) & 0xFF)
self.write_register(SX127x.REG_FRF_LSB, frf & 0xFF)
#self.write_register(SX127x.REG_SYNC_WORD, 0x34); # LoRaWAN public sync word
if self._is_sx1272:
if self.sf == 11 or self.sf == 12:
self.write_register(SX127x.REG_MODEM_CONFIG, 0x0B)
else:
self.write_register(SX127x.REG_MODEM_CONFIG, 0x0A)
self.write_register(SX127x.REG_MODEM_CONFIG2, (self.sf << 4) | 0x04)
else:
if self.sf == 11 or self.sf == 12:
self.write_register(SX127x.REG_MODEM_CONFIG3, 0x0C)
else:
self.write_register(SX127x.REG_MODEM_CONFIG3, 0x04)
self.write_register(SX127x.REG_MODEM_CONFIG, 0x72) # 125kHz 4/5
self.write_register(SX127x.REG_MODEM_CONFIG2, (self.sf << 4) | 0x04) # enable CTC
if self.sf == 10 or self.sf == 11 or self.sf == 12:
self.write_register(SX127x.REG_SYMB_TIMEOUT_LSB, 0x05)
else:
self.write_register(SX127x.REG_SYMB_TIMEOUT_LSB, 0x08)
self.write_register(SX127x.REG_MAX_PAYLOAD_LENGTH, 0x80)
self.write_register(SX127x.REG_PAYLOAD_LENGTH, 0x80)
#self.write_register(SX127x.REG_HOP_PERIOD, 0xFF)
self.write_register(SX127x.REG_FIFO_ADDR_PTR, self.read_register(SX127x.REG_FIFO_RX_BASE_AD))
self.write_register(SX127x.REG_PA_CONFIG, 0) # set to a very low value
# Set Continuous Receive Mode
self.write_register(SX127x.REG_LNA, SX127x.LNA_MAX_GAIN)
self.write_register(SX127x.REG_OPMODE, SX127x.SX72_MODE_LONG_RANGE | SX127x.SX72_MODE_RX_CONTINUOS)
def teardown_device(self):
logging.info("tear down transceiver...")
GPIO.cleanup()
self.spi.close()
def read_register(self, register):
GPIO.output(self._pin_ss, 0)
value = self.spi.xfer([register & 0x7F, 0])[1]
GPIO.output(self._pin_ss, 1)
return value
def write_register(self, register, value):
GPIO.output(self._pin_ss, 0)
self.spi.xfer([register | 0x80, value])
GPIO.output(self._pin_ss, 1)
def write_bulk_register(self, register, values):
GPIO.output(self._pin_ss, 0)
self.spi.xfer([register | 0x80] + list(values))
GPIO.output(self._pin_ss, 1)
def set_mode_rx(self):
self.write_register(SX127x.REG_OPMODE, SX127x.SX72_MODE_LONG_RANGE | SX127x.SX72_MODE_RX_CONTINUOS)
def receive_package(self):
self.write_register(SX127x.REG_IRQ_FLAGS, 0x40) # clear rxDone
irqflags = self.read_register(SX127x.REG_IRQ_FLAGS)
if irqflags & 0x20:
logging.warning("CRC error")
self.write_register(SX127x.REG_IRQ_FLAGS, 0x20)
crc = False
#return {'datetime': datetime.datetime.now(),
# 'crc': False} # TODO: still get payload?
else:
crc = True
current_addr = self.read_register(SX127x.REG_FIFO_RX_CURRENT_ADDR)
received_count = self.read_register(SX127x.REG_RX_NB_BYTES)
self.write_register(SX127x.REG_FIFO_ADDR_PTR, current_addr)
payload = bytearray()
for _ in range(received_count):
payload.append(self.read_register(SX127x.REG_FIFO))
return {'datetime': datetime.datetime.utcnow().replace(tzinfo=datetime.timezone.utc),
'crc': crc,
'pkt_snr': self.pkt_snr,
'pkt_rssi': self.pkt_rssi,
'rssi': self.rssi,
'payload': bytes(payload)}
def send_package(self, msg):
assert type(msg) == bytes
assert len(msg) <= 255
# TODO: wait until send (done at end)
self.write_register(SX127x.REG_OPMODE, SX127x.SX72_MODE_STANDBY)
self.write_register(SX127x.REG_FIFO_ADDR_PTR, self.read_register(SX127x.REG_FIFO_TX_BASE_AD))
self.write_register(SX127x.REG_PAYLOAD_LENGTH, len(msg))
self.write_bulk_register(SX127x.REG_FIFO, msg)
self.write_register(SX127x.REG_OPMODE, SX127x.SX72_MODE_TX)
self.wait_until_send()
def wait_until_send(self):
while self.read_register(SX127x.REG_IRQ_FLAGS) & 0x08 == 0:
pass # busy waiting
self.write_register(SX127x.REG_IRQ_FLAGS, 0x08) # Clear TX DONE
@property
def pkt_snr(self):
snr_value = self.read_register(SX127x.REG_PKT_SNR_VALUE)
if snr_value & 0x80:
snr_value = ((~snr_value + 1) & 0xFF) >> 2
return -snr_value
else:
return (snr_value & 0xFF) >> 2
@property
def pkt_rssi(self):
if self._is_sx1272:
rssicorr = 139
else:
rssicorr = 157
return self.read_register(0x1A) - rssicorr
@property
def rssi(self):
if self._is_sx1272:
rssicorr = 139
else:
rssicorr = 157
return self.read_register(0x1B) - rssicorr
class SX127x:
REG_FIFO = 0x00
REG_FRF_MSB = 0x06
REG_FRF_MID = 0x07
REG_FRF_LSB = 0x08
REG_PA_CONFIG = 0x09
REG_LNA = 0x0C
REG_FIFO_ADDR_PTR = 0x0D
REG_FIFO_TX_BASE_AD = 0x0E
REG_FIFO_RX_BASE_AD = 0x0F
REG_RX_NB_BYTES = 0x13
REG_OPMODE = 0x01
REG_FIFO_RX_CURRENT_ADDR = 0x10
REG_IRQ_FLAGS = 0x12
REG_DIO_MAPPING_1 = 0x40
REG_DIO_MAPPING_2 = 0x41
REG_MODEM_CONFIG = 0x1D
REG_MODEM_CONFIG2 = 0x1E
REG_MODEM_CONFIG3 = 0x26
REG_SYMB_TIMEOUT_LSB = 0x1F
REG_PKT_SNR_VALUE = 0x19
REG_PAYLOAD_LENGTH = 0x22
REG_IRQ_FLAGS_MASK = 0x11
REG_MAX_PAYLOAD_LENGTH = 0x23
REG_HOP_PERIOD = 0x24
REG_SYNC_WORD = 0x39
REG_VERSION = 0x42
SX72_MODE_LONG_RANGE = 0x80
SX72_MODE_SLEEP = 0x00
SX72_MODE_STANDBY = 0x01
SX72_MODE_FSTX = 0x02
SX72_MODE_TX = 0x03
SX72_MODE_FSRX = 0x04
SX72_MODE_RX_CONTINUOS = 0x05
SX72_MODE_RX_SINGLE = 0x06
LNA_MAX_GAIN = 0x23
LNA_OFF_GAIN = 0x00
LNA_LOW_GAIN = 0x20