Important changes to repositories hosted on mbed.com
Mbed hosted mercurial repositories are deprecated and are due to be permanently deleted in July 2026.
To keep a copy of this software download the repository Zip archive or clone locally using Mercurial.
It is also possible to export all your personal repositories from the account settings page.
Fork of SX127x by
sx127x_fsk.cpp
- Committer:
- dudmuck
- Date:
- 2014-05-02
- Revision:
- 4:d987ac2836bf
- Parent:
- 3:3bf2515b1eed
File content as of revision 4:d987ac2836bf:
#include "sx127x_fsk.h"
/* SX127x driver
* Copyright (c) 2013 Semtech
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
SX127x_fsk::SX127x_fsk(SX127x& r) : m_xcvr(r)
{
}
SX127x_fsk::~SX127x_fsk()
{
}
void SX127x_fsk::write_fifo(uint8_t len)
{
int i;
m_xcvr.m_cs = 0;
m_xcvr.m_spi.write(REG_FIFO | 0x80); // bit7 is high for writing to radio
if (!m_xcvr.RegOpMode.bits.LongRangeMode && RegPktConfig1.bits.PacketFormatVariable) {
m_xcvr.m_spi.write(len);
}
for (i = 0; i < len; i++) {
m_xcvr.m_spi.write(m_xcvr.tx_buf[i]);
}
m_xcvr.m_cs = 1;
}
void SX127x_fsk::enable()
{
m_xcvr.set_opmode(RF_OPMODE_SLEEP);
m_xcvr.RegOpMode.bits.LongRangeMode = 0;
m_xcvr.write_reg(REG_OPMODE, m_xcvr.RegOpMode.octet);
wait(0.01);
RegPktConfig1.octet = m_xcvr.read_reg(REG_FSK_PACKETCONFIG1);
RegPktConfig2.word = m_xcvr.read_u16(REG_FSK_PACKETCONFIG2);
RegRxConfig.octet = m_xcvr.read_u16(REG_FSK_RXCONFIG);
RegPreambleDetect.octet = m_xcvr.read_reg(REG_FSK_PREAMBLEDETECT);
RegSyncConfig.octet = m_xcvr.read_reg(REG_FSK_SYNCCONFIG);
RegFifoThreshold.octet = m_xcvr.read_reg(REG_FSK_FIFOTHRESH);
RegAfcFei.octet = m_xcvr.read_reg(REG_FSK_AFCFEI);
if (!RegFifoThreshold.bits.TxStartCondition) {
RegFifoThreshold.bits.TxStartCondition = 1; // start TX on fifoEmpty==0
m_xcvr.write_reg(REG_FSK_FIFOTHRESH, RegFifoThreshold.octet);
}
if (RegSyncConfig.bits.AutoRestartRxMode != 1) {
RegSyncConfig.bits.AutoRestartRxMode = 1;
m_xcvr.write_reg(REG_FSK_SYNCCONFIG, RegSyncConfig.octet);
}
if (RegPreambleDetect.bits.PreambleDetectorTol != 10) {
RegPreambleDetect.bits.PreambleDetectorTol = 10;
m_xcvr.write_reg(REG_FSK_PREAMBLEDETECT, RegPreambleDetect.octet);
}
m_xcvr.set_opmode(RF_OPMODE_STANDBY);
}
void SX127x_fsk::init()
{
m_xcvr.set_opmode(RF_OPMODE_STANDBY);
RegRxConfig.bits.RxTrigger = 6; // have RX restart (trigger) on preamble detection
RegRxConfig.bits.AfcAutoOn = 1; // have AFC performed on RX restart (RX trigger)
m_xcvr.write_reg(REG_FSK_RXCONFIG, RegRxConfig.octet);
RegPreambleDetect.bits.PreambleDetectorOn = 1; // enable preamble detector
m_xcvr.write_reg(REG_FSK_PREAMBLEDETECT, RegPreambleDetect.octet);
m_xcvr.write_reg(REG_FSK_SYNCVALUE1, 0x55);
m_xcvr.write_reg(REG_FSK_SYNCVALUE2, 0x6f);
m_xcvr.write_reg(REG_FSK_SYNCVALUE3, 0x4e);
RegSyncConfig.bits.SyncSize = 2;
m_xcvr.write_reg(REG_FSK_SYNCCONFIG, RegSyncConfig.octet);
// in case these were changed from default:
set_bitrate(4800);
set_tx_fdev_hz(5050);
set_rx_dcc_bw_hz(10500, 0); // rxbw
set_rx_dcc_bw_hz(50000, 1); // afcbw
}
uint32_t SX127x_fsk::get_bitrate()
{
uint16_t br = m_xcvr.read_u16(REG_FSK_BITRATEMSB);
if (br == 0)
return 0;
else
return XTAL_FREQ / br;
}
void SX127x_fsk::set_bitrate(uint32_t bps)
{
uint16_t tmpBitrate = XTAL_FREQ / bps;
//printf("tmpBitrate:%d = %d / %d\r\n", tmpBitrate, XTAL_FREQ, bps);
m_xcvr.write_u16(REG_FSK_BITRATEMSB, tmpBitrate);
}
void SX127x_fsk::set_tx_fdev_hz(uint32_t hz)
{
float tmpFdev = hz / FREQ_STEP_HZ;
uint16_t v;
//printf("tmpFdev:%f = %d / %f\r\n", tmpFdev, hz, FREQ_STEP_HZ);
v = (uint16_t)tmpFdev;
m_xcvr.write_u16(REG_FSK_FDEVMSB, v);
}
uint32_t SX127x_fsk::get_tx_fdev_hz(void)
{
uint16_t fdev = m_xcvr.read_u16(REG_FSK_FDEVMSB);
return fdev * FREQ_STEP_HZ;
}
uint32_t SX127x_fsk::ComputeRxBw( uint8_t mantisse, uint8_t exponent )
{
// rxBw
if (m_xcvr.RegOpMode.bits.ModulationType == 0)
return XTAL_FREQ / (mantisse * (1 << (exponent+2)));
else
return XTAL_FREQ / (mantisse * (1 << (exponent+3)));
}
void SX127x_fsk::ComputeRxBwMantExp( uint32_t rxBwValue, uint8_t* mantisse, uint8_t* exponent )
{
uint8_t tmpExp, tmpMant;
double tmpRxBw;
double rxBwMin = 10e6;
for( tmpExp = 0; tmpExp < 8; tmpExp++ ) {
for( tmpMant = 16; tmpMant <= 24; tmpMant += 4 ) {
tmpRxBw = ComputeRxBw(tmpMant, tmpExp);
if( fabs( tmpRxBw - rxBwValue ) < rxBwMin ) {
rxBwMin = fabs( tmpRxBw - rxBwValue );
*mantisse = tmpMant;
*exponent = tmpExp;
}
}
}
}
uint32_t SX127x_fsk::get_rx_bw_hz(uint8_t addr)
{
RegRxBw_t reg_bw;
uint8_t mantissa;
if (m_xcvr.RegOpMode.bits.LongRangeMode)
return 0;
reg_bw.octet = m_xcvr.read_reg(addr);
switch (reg_bw.bits.Mantissa) {
case 0: mantissa = 16; break;
case 1: mantissa = 20; break;
case 2: mantissa = 24; break;
default: mantissa = 0; break;
}
if (addr == REG_FSK_RXBW)
RegRxBw.octet = reg_bw.octet;
else if (addr == REG_FSK_AFCBW)
RegAfcBw.octet = reg_bw.octet;
return ComputeRxBw(mantissa, reg_bw.bits.Exponent);
}
void SX127x_fsk::set_rx_dcc_bw_hz(uint32_t bw_hz, char afc)
{
uint8_t mantisse = 0;
uint8_t exponent = 0;
if (m_xcvr.RegOpMode.bits.LongRangeMode)
return;
ComputeRxBwMantExp( bw_hz, &mantisse, &exponent );
//printf("bw_hz:%d, mantisse:%d, exponent:%d\n", bw_hz, mantisse, exponent );
switch( mantisse ) {
case 16:
RegRxBw.bits.Mantissa = 0;
break;
case 20:
RegRxBw.bits.Mantissa = 1;
break;
case 24:
RegRxBw.bits.Mantissa = 2;
break;
default:
// Something went terribely wrong
printf("maintisse:%d\n", mantisse);
break;
}
RegRxBw.bits.Exponent = exponent;
if (afc)
m_xcvr.write_reg(REG_FSK_AFCBW, RegRxBw.octet);
else
m_xcvr.write_reg(REG_FSK_RXBW, RegRxBw.octet);
}
void SX127x_fsk::start_tx(uint16_t arg_len)
{
uint16_t pkt_buf_len;
RegIrqFlags2_t RegIrqFlags2;
int maxlen = FSK_FIFO_SIZE-1;
if (m_xcvr.RegOpMode.bits.Mode == RF_OPMODE_RECEIVER) {
m_xcvr.set_opmode(RF_OPMODE_STANDBY);
}
if (m_xcvr.RegDioMapping1.bits.Dio0Mapping != 0) {
m_xcvr.RegDioMapping1.bits.Dio0Mapping = 0;
m_xcvr.write_reg(REG_DIOMAPPING1, m_xcvr.RegDioMapping1.octet);
}
if (RegPktConfig1.bits.PacketFormatVariable) {
pkt_buf_len = arg_len;
} else {
pkt_buf_len = RegPktConfig2.bits.PayloadLength;
printf("fixed-fmt %d\r\n", pkt_buf_len);
}
RegIrqFlags2.octet = m_xcvr.read_reg(REG_FSK_IRQFLAGS2);
if (RegIrqFlags2.bits.FifoEmpty) {
if (RegPktConfig1.bits.PacketFormatVariable) {
--maxlen; // space for length byte
if (pkt_buf_len > maxlen) {
/*setup_FifoLevel(FALLING_EDGE);
radio_write_fifo_(pkt_buf, maxlen, pkt_buf_len);
remaining_ = pkt_buf_len - maxlen;
printf("var-oversized %d R=%d\r\n", pkt_buf_len, remaining_);*/
printf("var-oversized %d\r\n", pkt_buf_len);
} else {
//setup_FifoLevel(NO_EDGE); // disable
write_fifo(pkt_buf_len);
//remaining_ = 0; // all was sent
}
} else { // fixed-length pkt format...
pkt_buf_len = RegPktConfig2.bits.PayloadLength;
if (RegPktConfig2.bits.PayloadLength > maxlen) {
/*setup_FifoLevel(FALLING_EDGE);
radio_write_fifo_(pkt_buf, maxlen, -1);
remaining_ = SX1272FSK->RegPktConfig2.bits.PayloadLength - maxlen;*/
printf("todo: fsk large packet\r\n");
} else {
//setup_FifoLevel(NO_EDGE); // disable
printf("fixed-write-fifo %d\r\n", RegPktConfig2.bits.PayloadLength);
write_fifo(RegPktConfig2.bits.PayloadLength);
}
}
} else
printf("fifo not empty %02x\r\n", RegIrqFlags2.octet);
m_xcvr.set_opmode(RF_OPMODE_TRANSMITTER);
}
void SX127x_fsk::config_dio0_for_pktmode_rx()
{
if (RegPktConfig1.bits.CrcOn) {
if (m_xcvr.RegDioMapping1.bits.Dio0Mapping != 1) {
m_xcvr.RegDioMapping1.bits.Dio0Mapping = 1; // to CrcOk
m_xcvr.write_reg(REG_DIOMAPPING1, m_xcvr.RegDioMapping1.octet);
}
} else { // Crc Off, use PayloadReady
if (m_xcvr.RegDioMapping1.bits.Dio0Mapping != 0) {
m_xcvr.RegDioMapping1.bits.Dio0Mapping = 0; // to PayloadReady
m_xcvr.write_reg(REG_DIOMAPPING1, m_xcvr.RegDioMapping1.octet);
}
}
}
void SX127x_fsk::start_rx()
{
if (m_xcvr.RegOpMode.bits.LongRangeMode)
return;
if (m_xcvr.RegOpMode.bits.Mode == RF_OPMODE_RECEIVER) {
// was already receiving, restart it
m_xcvr.set_opmode(RF_OPMODE_STANDBY);
wait(0.01);
}
config_dio0_for_pktmode_rx();
m_xcvr.set_opmode(RF_OPMODE_RECEIVER);
}
service_action_e SX127x_fsk::service()
{
int i;
if (m_xcvr.RegOpMode.bits.Mode == RF_OPMODE_TRANSMITTER) {
if (m_xcvr.dio0) {
m_xcvr.set_opmode(RF_OPMODE_STANDBY);
return SERVICE_TX_DONE;
}
} else if (m_xcvr.RegOpMode.bits.Mode == RF_OPMODE_RECEIVER && m_xcvr.dio0) {
if (RegRxConfig.bits.AfcAutoOn)
RegAfcValue = m_xcvr.read_s16(REG_FSK_AFCMSB);
if (RegPktConfig1.bits.PacketFormatVariable) {
rx_buf_length = m_xcvr.read_reg(REG_FIFO);
} else {
rx_buf_length = RegPktConfig2.bits.PayloadLength;
}
m_xcvr.m_cs = 0;
m_xcvr.m_spi.write(REG_FIFO); // bit7 is low for reading from radio
for (i = 0; i < rx_buf_length; i++) {
m_xcvr.rx_buf[i] = m_xcvr.m_spi.write(0);
}
m_xcvr.m_cs = 1;
return SERVICE_READ_FIFO;
}
return SERVICE_NONE;
}
