mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-03 06:28:27 -04:00
parent
375510ecc2
commit
dcbbc86f34
@ -6,6 +6,7 @@
|
||||
#include "AP_Radio_backend.h"
|
||||
#include "AP_Radio_cypress.h"
|
||||
#include "AP_Radio_cc2500.h"
|
||||
#include "AP_Radio_bk2425.h"
|
||||
|
||||
extern const AP_HAL::HAL& hal;
|
||||
|
||||
@ -15,7 +16,8 @@ const AP_Param::GroupInfo AP_Radio::var_info[] = {
|
||||
// @Param: _TYPE
|
||||
// @DisplayName: Set type of direct attached radio
|
||||
// @Description: This enables support for direct attached radio receivers
|
||||
// @Values: 0:None,1:CYRF6936
|
||||
// @Values: 0:None,1:CYRF6936,2:CC2500,3:BK2425
|
||||
|
||||
// @User: Advanced
|
||||
AP_GROUPINFO("_TYPE", 1, AP_Radio, radio_type, 0),
|
||||
|
||||
@ -108,7 +110,7 @@ const AP_Param::GroupInfo AP_Radio::var_info[] = {
|
||||
// @Description: Set transmitter maximum transmit power (from 1 to 8)
|
||||
// @Range: 1 8
|
||||
// @User: Advanced
|
||||
AP_GROUPINFO("_TXMAX", 14, AP_Radio, tx_max_power, 4),
|
||||
AP_GROUPINFO("_TXMAX", 14, AP_Radio, tx_max_power, 8),
|
||||
|
||||
// @Param: _BZOFS
|
||||
// @DisplayName: Transmitter buzzer adjustment
|
||||
@ -149,13 +151,39 @@ AP_Radio::AP_Radio(void)
|
||||
bool AP_Radio::init(void)
|
||||
{
|
||||
switch (radio_type) {
|
||||
#if (not defined AP_RADIO_CYRF6936 || AP_RADIO_CYRF6936)
|
||||
case RADIO_TYPE_CYRF6936:
|
||||
driver = new AP_Radio_cypress(*this);
|
||||
break;
|
||||
#endif
|
||||
#if CONFIG_HAL_BOARD == HAL_BOARD_CHIBIOS
|
||||
#if (not defined AP_RADIO_CC2500 || AP_RADIO_CC2500)
|
||||
case RADIO_TYPE_CC2500:
|
||||
driver = new AP_Radio_cc2500(*this);
|
||||
break;
|
||||
#endif
|
||||
#endif
|
||||
#if CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_CHIBIOS_SKYVIPER_F412
|
||||
#if (not defined AP_RADIO_BK2425 || AP_RADIO_BK2425)
|
||||
case RADIO_TYPE_BK2425:
|
||||
driver = new AP_Radio_beken(*this);
|
||||
break;
|
||||
#endif
|
||||
#endif
|
||||
#if CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_CHIBIOS_SKYVIPER_F412
|
||||
case RADIO_TYPE_AUTO:
|
||||
// auto-detect between cc2500 and beken radios
|
||||
#if (not defined AP_RADIO_CC2500 || AP_RADIO_CC2500)
|
||||
if (AP_Radio_cc2500::probe()) {
|
||||
driver = new AP_Radio_cc2500(*this);
|
||||
}
|
||||
#endif
|
||||
#if (not defined AP_RADIO_BK2425 || AP_RADIO_BK2425)
|
||||
if (driver == nullptr) {
|
||||
driver = new AP_Radio_beken(*this);
|
||||
}
|
||||
#endif
|
||||
break;
|
||||
#endif
|
||||
default:
|
||||
break;
|
||||
@ -227,6 +255,17 @@ void AP_Radio::handle_data_packet(mavlink_channel_t chan, const mavlink_data96_t
|
||||
}
|
||||
}
|
||||
|
||||
// play a tune on the TX
|
||||
void AP_Radio::play_tune(const char *tune_str)
|
||||
{
|
||||
mavlink_data96_t pkt {};
|
||||
uint8_t len = MIN(strlen(tune_str), 92);
|
||||
pkt.len = len;
|
||||
pkt.type = 43;
|
||||
memcpy(&pkt.data[0], tune_str, len);
|
||||
handle_data_packet(MAVLINK_COMM_0, pkt);
|
||||
}
|
||||
|
||||
// update status, should be called from main thread
|
||||
void AP_Radio::update(void)
|
||||
{
|
||||
@ -252,5 +291,15 @@ void AP_Radio::set_wifi_channel(uint8_t channel)
|
||||
}
|
||||
}
|
||||
|
||||
// change TX mode, toggling between mode1 and mode2
|
||||
void AP_Radio::change_txmode(void)
|
||||
{
|
||||
if (stick_mode == 2) {
|
||||
stick_mode.set_and_save_ifchanged(1);
|
||||
} else {
|
||||
stick_mode.set_and_save_ifchanged(2);
|
||||
}
|
||||
}
|
||||
|
||||
#endif // HAL_RCINPUT_WITH_AP_RADIO
|
||||
|
||||
|
@ -71,6 +71,8 @@ public:
|
||||
RADIO_TYPE_NONE=0,
|
||||
RADIO_TYPE_CYRF6936=1,
|
||||
RADIO_TYPE_CC2500=2,
|
||||
RADIO_TYPE_BK2425=3,
|
||||
RADIO_TYPE_AUTO=100,
|
||||
};
|
||||
|
||||
enum ap_radio_protocol {
|
||||
@ -78,6 +80,7 @@ public:
|
||||
PROTOCOL_DSM2=1,
|
||||
PROTOCOL_DSMX=2,
|
||||
PROTOCOL_D16=3,
|
||||
PROTOCOL_CC2500_GFSK=4, // deviation 57kHz for update of cc2500 with GFSK
|
||||
};
|
||||
|
||||
// get packet statistics
|
||||
@ -86,7 +89,8 @@ public:
|
||||
static const struct AP_Param::GroupInfo var_info[];
|
||||
|
||||
// get singleton instance
|
||||
static AP_Radio *get_singleton(void) {
|
||||
static AP_Radio *get_singleton(void)
|
||||
{
|
||||
return _singleton;
|
||||
}
|
||||
|
||||
@ -96,6 +100,12 @@ public:
|
||||
// set the 2.4GHz wifi channel used by companion computer, so it can be avoided
|
||||
void set_wifi_channel(uint8_t channel);
|
||||
|
||||
// play a tune on the TX
|
||||
void play_tune(const char *tune_str);
|
||||
|
||||
// change TX mode
|
||||
void change_txmode(void);
|
||||
|
||||
private:
|
||||
AP_Radio_backend *driver;
|
||||
|
||||
|
@ -64,72 +64,91 @@ public:
|
||||
virtual void set_wifi_channel(uint8_t channel) = 0;
|
||||
|
||||
protected:
|
||||
// These functions are for the radio code to access the parameters
|
||||
// that the user can set using the web interface.
|
||||
|
||||
AP_Radio::ap_radio_protocol get_protocol(void) const {
|
||||
AP_Radio::ap_radio_protocol get_protocol(void) const
|
||||
{
|
||||
return (AP_Radio::ap_radio_protocol)radio.protocol.get();
|
||||
}
|
||||
|
||||
uint8_t get_debug_level(void) const {
|
||||
uint8_t get_debug_level(void) const
|
||||
{
|
||||
return (uint8_t)radio.debug_level.get();
|
||||
}
|
||||
|
||||
bool get_disable_crc(void) const {
|
||||
bool get_disable_crc(void) const
|
||||
{
|
||||
return (bool)radio.disable_crc.get();
|
||||
}
|
||||
|
||||
uint8_t get_rssi_chan(void) const {
|
||||
uint8_t get_rssi_chan(void) const
|
||||
{
|
||||
return (uint8_t)radio.rssi_chan.get();
|
||||
}
|
||||
|
||||
uint8_t get_pps_chan(void) const {
|
||||
uint8_t get_pps_chan(void) const
|
||||
{
|
||||
return (uint8_t)radio.pps_chan.get();
|
||||
}
|
||||
|
||||
uint8_t get_tx_rssi_chan(void) const {
|
||||
uint8_t get_tx_rssi_chan(void) const
|
||||
{
|
||||
return (uint8_t)radio.tx_rssi_chan.get();
|
||||
}
|
||||
|
||||
uint8_t get_tx_pps_chan(void) const {
|
||||
uint8_t get_tx_pps_chan(void) const
|
||||
{
|
||||
return (uint8_t)radio.tx_pps_chan.get();
|
||||
}
|
||||
|
||||
bool get_telem_enable(void) const {
|
||||
bool get_telem_enable(void) const
|
||||
{
|
||||
return radio.telem_enable.get() > 0;
|
||||
}
|
||||
|
||||
uint8_t get_transmit_power(void) const {
|
||||
uint8_t get_transmit_power(void) const
|
||||
{
|
||||
return constrain_int16(radio.transmit_power.get(), 1, 8);
|
||||
}
|
||||
|
||||
uint8_t get_tx_max_power(void) const {
|
||||
uint8_t get_tx_max_power(void) const
|
||||
{
|
||||
return constrain_int16(radio.tx_max_power.get(), 1, 8);
|
||||
}
|
||||
|
||||
void set_tx_max_power_default(uint8_t v) {
|
||||
void set_tx_max_power_default(uint8_t v)
|
||||
{
|
||||
return radio.tx_max_power.set_default(v);
|
||||
}
|
||||
|
||||
int8_t get_fcc_test(void) const {
|
||||
int8_t get_fcc_test(void) const
|
||||
{
|
||||
return radio.fcc_test.get();
|
||||
}
|
||||
|
||||
uint8_t get_stick_mode(void) const {
|
||||
uint8_t get_stick_mode(void) const
|
||||
{
|
||||
return radio.stick_mode.get();
|
||||
}
|
||||
|
||||
uint8_t get_factory_test(void) const {
|
||||
uint8_t get_factory_test(void) const
|
||||
{
|
||||
return radio.factory_test.get();
|
||||
}
|
||||
|
||||
uint8_t get_tx_buzzer_adjust(void) const {
|
||||
uint8_t get_tx_buzzer_adjust(void) const
|
||||
{
|
||||
return radio.tx_buzzer_adjust.get();
|
||||
}
|
||||
|
||||
uint8_t get_autobind_time(void) const {
|
||||
uint8_t get_autobind_time(void) const
|
||||
{
|
||||
return radio.auto_bind_time.get();
|
||||
}
|
||||
|
||||
uint8_t get_autobind_rssi(void) const {
|
||||
uint8_t get_autobind_rssi(void) const
|
||||
{
|
||||
return radio.auto_bind_rssi.get();
|
||||
}
|
||||
|
||||
|
1415
libraries/AP_Radio/AP_Radio_bk2425.cpp
Normal file
1415
libraries/AP_Radio/AP_Radio_bk2425.cpp
Normal file
File diff suppressed because it is too large
Load Diff
261
libraries/AP_Radio/AP_Radio_bk2425.h
Normal file
261
libraries/AP_Radio/AP_Radio_bk2425.h
Normal file
@ -0,0 +1,261 @@
|
||||
/*
|
||||
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/>.
|
||||
*/
|
||||
#pragma once
|
||||
|
||||
/*
|
||||
AP_Radio implementation for CC2500 2.4GHz radio.
|
||||
|
||||
With thanks to cleanflight and betaflight projects
|
||||
*/
|
||||
|
||||
#include "AP_Radio_backend.h"
|
||||
|
||||
#if defined(HAL_RCINPUT_WITH_AP_RADIO) && CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_CHIBIOS_SKYVIPER_F412
|
||||
#include "hal.h"
|
||||
#include "telem_structure.h"
|
||||
#include "driver_bk2425.h"
|
||||
|
||||
#define BEKEN_MAX_CHANNELS 16
|
||||
// Documentation of the expected RSSI values. These are determined by the Cypress chip.
|
||||
enum {
|
||||
BK_RSSI_MIN = 0, // Minimum value for RSSI
|
||||
BK_RSSI_DEFAULT = 16, // The default value for RSSI for chips that do not support it.
|
||||
BK_RSSI_MAX = 31 // Maximum value for RSSI
|
||||
};
|
||||
|
||||
// This helper struct estimates the times (in microseconds) between packets,
|
||||
// according to the STM32 clock which may well be 2% different from the STM8 clock.
|
||||
// For instance it may be 5108 instead of the nominal 5000 microseconds.
|
||||
struct SyncTiming {
|
||||
enum { TARGET_DELTA_RX = 5000, // Nominal 5ms between packets is expected
|
||||
SLOP_DELTA_RX = TARGET_DELTA_RX / 10, // +/- 500us i.e. 10% skew each way is accepted.
|
||||
DIFF_DELTA_RX = TARGET_DELTA_RX / 100
|
||||
}; // Two consequetive deltas must be very close together (50us)
|
||||
uint32_t packet_timer; // Time we last received a valid control packet
|
||||
uint32_t rx_time_us; // Time we last received a packet
|
||||
uint32_t tx_time_us; // Time we last finished transmitting a packet
|
||||
uint32_t delta_rx_time_us; // Time between last rx packets
|
||||
uint32_t last_delta_rx_time_us; // previous version of the delta
|
||||
uint32_t sync_time_us; // Estimate of base time in microseconds between packets. 5000 +/- 500
|
||||
SyncTiming() : // Constructor to setup sensible initial conditions
|
||||
delta_rx_time_us(TARGET_DELTA_RX),
|
||||
last_delta_rx_time_us(TARGET_DELTA_RX),
|
||||
sync_time_us(TARGET_DELTA_RX)
|
||||
{}
|
||||
void Rx(uint32_t when); // Adjust the timing based on a new packet
|
||||
};
|
||||
|
||||
// Helper struct for synchronising channels when we change hopping table (e.g. learn of a WiFi channel change).
|
||||
struct SyncChannel {
|
||||
enum { countdown_invalid = 0 }; // When countdown is this value, no change is pending
|
||||
uint8_t channel; // Index within the channel hopping sequence. Corresponds to txChannel on the button board
|
||||
uint8_t lastchan; // Last requested index, if it is a factory test channel.
|
||||
uint8_t countdown; // How many packet slots until a pending table change occurs?
|
||||
uint8_t countdown_chan; // Which channel do we jump to when the table change happens?
|
||||
uint8_t hopping_current; // Which alternative channels are we on now
|
||||
uint8_t hopping_wanted; // Which alternative channels will we be on when Tx changes over?
|
||||
uint8_t hopping_countdown; // How many packet slots until a pending table change occurs?
|
||||
SyncChannel() : // Constructor to setup sensible initial conditions
|
||||
channel(0),
|
||||
lastchan(0),
|
||||
countdown(countdown_invalid),
|
||||
countdown_chan(0),
|
||||
hopping_current(0),
|
||||
hopping_wanted(0),
|
||||
hopping_countdown(countdown_invalid)
|
||||
{}
|
||||
void SetChannelIfSafe(uint8_t chan); // Check if valid channel index; we have received a packet describing the current channel index
|
||||
void SetChannel(uint8_t chan) // Already safe. We have received a packet describing the current channel index
|
||||
{
|
||||
channel = chan;
|
||||
}
|
||||
void SetCountdown(uint8_t cnt, uint8_t nextCh) // We receive a countdown to a non-normal channel change in the future
|
||||
{
|
||||
countdown = cnt;
|
||||
countdown_chan = nextCh;
|
||||
}
|
||||
void SetHopping(uint8_t cnt, uint8_t nextHopping) // We receive a countdown to a change in the adaptive table in the future/now
|
||||
{
|
||||
hopping_countdown = cnt;
|
||||
hopping_wanted = nextHopping;
|
||||
if (cnt == 0) {
|
||||
hopping_current = nextHopping;
|
||||
}
|
||||
}
|
||||
void NextChannel(void); // Step through the channels normally (taking countdowns into account)
|
||||
void SafeTable(void); // Give up on this WiFi table as packets have not been received
|
||||
};
|
||||
|
||||
// This helper struct determines which physical channels are better
|
||||
struct SyncAdaptive {
|
||||
uint32_t missed[CHANNEL_FCC_HIGH+1]; // Missed
|
||||
uint32_t rx[CHANNEL_FCC_HIGH+1]; // Received
|
||||
uint8_t hopping; // Currently wanted hopping state. Send this to the tx.
|
||||
SyncAdaptive() : // Constructor to setup sensible initial conditions
|
||||
hopping(0)
|
||||
{}
|
||||
void Miss(uint8_t channel);
|
||||
void Get(uint8_t channel);
|
||||
void Invalidate()
|
||||
{
|
||||
hopping = 0; // e.g. if we have jumped tables
|
||||
}
|
||||
};
|
||||
|
||||
// Support OTA upload. Assumes that mavlink offsets go from zero upwards contiguously
|
||||
struct FwUpload {
|
||||
enum { SZ_BUFFER = 128 }; // Must be a power of two
|
||||
mavlink_channel_t chan; // Reference for talking to mavlink subsystem
|
||||
uint8_t counter; // Used to throttle the upload, to prevent starvation of telemetry
|
||||
enum telem_type fw_type; // Whether we are uploading program code or a test tune
|
||||
|
||||
// Data that is reset by reset()
|
||||
bool need_ack; // When true, we need to talk to mavlink subsystem (ask for more firmware)
|
||||
uint32_t added; // The number of bytes added to the queue
|
||||
uint32_t sent; // The number of bytes sent to the tx
|
||||
uint32_t acked; // The number of bytes acked by the tx
|
||||
bool rx_ack; // True each time we receive a non-zero ack from the tx
|
||||
bool rx_reboot; // True when we are in the rebooting process
|
||||
uint8_t pending_data[SZ_BUFFER]; // Pending data (from mavlink packets) circular buffer
|
||||
uint8_t pending_head; // Where mavlink packets are added (relative to pending_data[0])
|
||||
uint8_t pending_tail; // Where DFU packets are taken from (relative to pending_data[0])
|
||||
uint16_t file_length; // The length of the file, six more than the value stored in the first 16 bit word
|
||||
uint16_t file_length_round; // file_length rounded up to 0x80
|
||||
|
||||
// Helper functions
|
||||
uint8_t pending_length()
|
||||
{
|
||||
return (pending_head - pending_tail) & (SZ_BUFFER-1);
|
||||
}
|
||||
uint8_t free_length()
|
||||
{
|
||||
return SZ_BUFFER - 1 - pending_length(); // Do not fill in the last byte in the circular buffer
|
||||
}
|
||||
void queue(const uint8_t *pSrc, uint8_t len); // Assumes sufficient room has been checked for
|
||||
void dequeue(uint8_t *pDst, uint8_t len); // Assumes sufficient data has been checked for
|
||||
void reset()
|
||||
{
|
||||
file_length = file_length_round = 0;
|
||||
added = sent = acked = 0;
|
||||
pending_head = pending_tail = 0;
|
||||
rx_reboot = rx_ack = need_ack = false;
|
||||
}
|
||||
};
|
||||
|
||||
// Main class for receiving (and replying) to Beken radio packets
|
||||
class AP_Radio_beken : public AP_Radio_backend
|
||||
{
|
||||
public:
|
||||
// Override base class functions
|
||||
AP_Radio_beken(AP_Radio &radio); // Normal constructore
|
||||
bool init(void) override; // initialise the radio
|
||||
bool reset(void) override; // reset the radio
|
||||
bool send(const uint8_t *pkt, uint16_t len) override; // send a packet
|
||||
void start_recv_bind(void) override; // start bind process as a receiver
|
||||
uint32_t last_recv_us(void) override; // return time in microseconds of last received R/C packet
|
||||
uint8_t num_channels(void) override; // return number of input channels
|
||||
uint16_t read(uint8_t chan) override; // return current "PWM" (value) of a channel
|
||||
void handle_data_packet(mavlink_channel_t chan, const mavlink_data96_t &m) override; // handle a data96 mavlink packet for fw upload
|
||||
void update(void) override; // update status
|
||||
|
||||
uint32_t get_tx_version(void) override // get TX fw version
|
||||
{
|
||||
// pack date into 16 bits for vendor_id in AUTOPILOT_VERSION
|
||||
return (uint16_t(tx_date.firmware_year)<<12) + (uint16_t(tx_date.firmware_month)<<8) + tx_date.firmware_day;
|
||||
}
|
||||
const AP_Radio::stats &get_stats(void) override; // get radio statistics structure
|
||||
|
||||
// Extra public functions
|
||||
void set_wifi_channel(uint8_t channel) override
|
||||
{
|
||||
t_status.wifi_chan = channel; // set the 2.4GHz wifi channel used by companion computer, so it can be avoided
|
||||
}
|
||||
|
||||
private:
|
||||
// Static functions, for interrupt support
|
||||
static void irq_handler_thd(void* arg);
|
||||
static void trigger_irq_radio_event(void);
|
||||
static void trigger_timeout_event(void *arg);
|
||||
|
||||
// Private functions
|
||||
void radio_init(void);
|
||||
uint8_t ProcessPacket(const uint8_t* packet, uint8_t rxaddr);
|
||||
uint8_t ProcessBindPacket(const packetFormatRx * rx);
|
||||
void BadDroneId(void); // The tx we are listening to wants to talk to another drone
|
||||
void setChannel(uint8_t channel);
|
||||
void nextChannel(uint8_t skip);
|
||||
uint16_t calc_crc(uint8_t *data, uint8_t len);
|
||||
void irq_handler(uint32_t when);
|
||||
void irq_timeout(uint32_t when);
|
||||
void save_bind_info(void);
|
||||
bool load_bind_info(void);
|
||||
void UpdateFccScan(void);
|
||||
bool UpdateTxData(void);
|
||||
void map_stick_mode(void); // Support mode1,2,3,4 for stick mapping
|
||||
void update_SRT_telemetry(void);
|
||||
void check_fw_ack(void);
|
||||
|
||||
// Static data, for interrupt support
|
||||
friend class SyncChannel; // For DebugPrintf support
|
||||
static AP_Radio_beken *radio_singleton; // Singleton pointer to the Beken radio instance
|
||||
static thread_t *_irq_handler_ctx;
|
||||
static virtual_timer_t timeout_vt;
|
||||
static uint32_t isr_irq_time_us; // Time the Beken IRQ was last triggered, in the handler interrupts (in microseconds)
|
||||
static uint32_t isr_timeout_time_us; // Time the timeout was last triggered (copied from irq_time_us via irq_when_us) (in microseconds)
|
||||
static uint32_t next_switch_us; // Time when we next want to switch radio channels (in microseconds)
|
||||
static uint32_t bind_time_ms; // Rough time in ms (milliseconds) when the last BIND command was received
|
||||
|
||||
// Class data
|
||||
AP_HAL::OwnPtr<AP_HAL::SPIDevice> dev; // Low level support of SPI device
|
||||
HAL_Semaphore sem; // semaphore between ISR and main thread to protect fwupload
|
||||
|
||||
AP_Radio::stats stats; // Radio stats (live) for the current time-period
|
||||
AP_Radio::stats last_stats; // Radio stats (snapshot) for the previous time-period
|
||||
uint16_t pwm_channels[BEKEN_MAX_CHANNELS]; // Channel data
|
||||
uint8_t chan_count; // Number of valid channels
|
||||
|
||||
Radio_Beken beken; // The low level class for communicating to the Beken chip
|
||||
SyncChannel syncch; // Index within the channel hopping sequence. Corresponds to txChannel on the button board
|
||||
static SyncTiming synctm; // Timing between packets, according to the local clock (not the tx clock).
|
||||
uint32_t already_bound; // True when we have received packets from a tx after bootup. Prevent auto-binding to something else.
|
||||
FwUpload fwupload; // Support OTA upload
|
||||
SyncAdaptive adaptive; // Support adaptive hopping
|
||||
struct {
|
||||
uint8_t firmware_year;
|
||||
uint8_t firmware_month;
|
||||
uint8_t firmware_day;
|
||||
} tx_date;
|
||||
|
||||
// Bind structure saved to storage
|
||||
static const uint16_t bind_magic = 0x120a;
|
||||
struct PACKED bind_info {
|
||||
uint16_t magic;
|
||||
uint8_t bindTxId[5]; // The transmission address I last used
|
||||
};
|
||||
|
||||
// Received
|
||||
struct telem_status t_status; // Keep track of certain data that can be sent as telemetry to the tx.
|
||||
uint32_t last_pps_ms; // Timestamp of the last PPS (packets per second) calculation, in milliseconds.
|
||||
uint32_t tx_pps; // Last telemetry PPS received from Tx
|
||||
uint32_t have_tx_pps; // 0=never received, 1=received at least one, 2=received recently
|
||||
uint32_t valid_connection; // Take some time before admitting to ardupilot we have a connection
|
||||
uint32_t telem_send_count; // How many telemetry packets have i sent?
|
||||
|
||||
// Parameters
|
||||
ITX_SPEED spd; // Speed of radio modulation.
|
||||
uint8_t myDroneId[4]; // CRC of the flight boards UUID, to inform the tx
|
||||
};
|
||||
|
||||
#endif // HAL_RCINPUT_WITH_AP_RADIO
|
File diff suppressed because it is too large
Load Diff
@ -26,7 +26,7 @@
|
||||
#include "telem_structure.h"
|
||||
#include "driver_cc2500.h"
|
||||
|
||||
#define CC2500_MAX_CHANNELS 16
|
||||
#define CC2500_MAX_PWM_CHANNELS 16
|
||||
|
||||
class AP_Radio_cc2500 : public AP_Radio_backend
|
||||
{
|
||||
@ -61,7 +61,8 @@ public:
|
||||
void update(void) override;
|
||||
|
||||
// get TX fw version
|
||||
uint32_t get_tx_version(void) override {
|
||||
uint32_t get_tx_version(void) override
|
||||
{
|
||||
// pack date into 16 bits for vendor_id in AUTOPILOT_VERSION
|
||||
return (uint16_t(tx_date.firmware_year)<<12) + (uint16_t(tx_date.firmware_month)<<8) + tx_date.firmware_day;
|
||||
}
|
||||
@ -70,10 +71,14 @@ public:
|
||||
const AP_Radio::stats &get_stats(void) override;
|
||||
|
||||
// set the 2.4GHz wifi channel used by companion computer, so it can be avoided
|
||||
void set_wifi_channel(uint8_t channel) override {
|
||||
// t_status.wifi_chan = channel;
|
||||
void set_wifi_channel(uint8_t channel) override
|
||||
{
|
||||
t_status.wifi_chan = channel;
|
||||
}
|
||||
|
||||
// static probe function for device detection
|
||||
static bool probe(void);
|
||||
|
||||
private:
|
||||
AP_HAL::OwnPtr<AP_HAL::SPIDevice> dev;
|
||||
static AP_Radio_cc2500 *radio_singleton;
|
||||
@ -92,11 +97,10 @@ private:
|
||||
AP_Radio::stats stats;
|
||||
AP_Radio::stats last_stats;
|
||||
|
||||
uint16_t pwm_channels[CC2500_MAX_CHANNELS];
|
||||
uint16_t pwm_channels[CC2500_MAX_PWM_CHANNELS];
|
||||
|
||||
Radio_CC2500 cc2500;
|
||||
|
||||
uint8_t calData[255][3];
|
||||
uint8_t bindTxId[2];
|
||||
int8_t bindOffset;
|
||||
uint8_t bindHopData[47];
|
||||
@ -107,7 +111,6 @@ private:
|
||||
int8_t fcc_chan;
|
||||
uint32_t packet_timer;
|
||||
static uint32_t irq_time_us;
|
||||
const uint32_t sync_time_us = 9000;
|
||||
uint8_t chan_count;
|
||||
uint32_t lost;
|
||||
uint32_t timeouts;
|
||||
@ -118,8 +121,12 @@ private:
|
||||
uint64_t bind_mask;
|
||||
uint8_t best_lqi;
|
||||
int8_t best_bindOffset;
|
||||
int8_t auto_bindOffset;
|
||||
uint8_t search_count;
|
||||
uint8_t last_wifi_channel;
|
||||
|
||||
uint32_t timeTunedMs;
|
||||
uint32_t autobind_start_recv_ms;
|
||||
|
||||
void initTuneRx(void);
|
||||
void initialiseData(uint8_t adr);
|
||||
@ -128,6 +135,7 @@ private:
|
||||
bool getBindData(uint8_t ccLen, uint8_t *packet);
|
||||
bool check_best_LQI(void);
|
||||
void setChannel(uint8_t channel);
|
||||
void setChannelRX(uint8_t channel);
|
||||
void nextChannel(uint8_t skip);
|
||||
|
||||
void parse_frSkyX(const uint8_t *packet);
|
||||
@ -141,12 +149,12 @@ private:
|
||||
void irq_timeout(void);
|
||||
|
||||
// bind structure saved to storage
|
||||
static const uint16_t bind_magic = 0x120a;
|
||||
static const uint16_t bind_magic = 0x120c;
|
||||
struct PACKED bind_info {
|
||||
uint16_t magic;
|
||||
uint8_t bindTxId[2];
|
||||
int8_t bindOffset;
|
||||
uint8_t listLength;
|
||||
uint8_t wifi_chan;
|
||||
uint8_t bindHopData[47];
|
||||
};
|
||||
|
||||
@ -171,6 +179,7 @@ private:
|
||||
uint8_t reg;
|
||||
uint8_t value;
|
||||
};
|
||||
static const config radio_config_GFSK[];
|
||||
static const config radio_config[];
|
||||
|
||||
struct {
|
||||
@ -192,16 +201,28 @@ private:
|
||||
uint8_t firmware_day;
|
||||
} tx_date;
|
||||
|
||||
struct telem_status t_status;
|
||||
struct telem_status_cc2500 t_status;
|
||||
uint32_t last_pps_ms;
|
||||
uint8_t tx_rssi;
|
||||
uint8_t tx_pps;
|
||||
bool have_tx_pps;
|
||||
uint8_t last_fcc_chan;
|
||||
uint32_t telem_send_count;
|
||||
|
||||
bool handle_D16_packet(const uint8_t *packet);
|
||||
bool handle_SRT_packet(const uint8_t *packet);
|
||||
bool handle_autobind_packet(const uint8_t *packet, uint8_t lqi);
|
||||
bool have_channel(uint8_t channel, uint8_t count, uint8_t loop);
|
||||
void setup_hopping_table_SRT(void);
|
||||
uint8_t map_RSSI_to_dBm(uint8_t rssi_raw);
|
||||
|
||||
// check sending of fw upload ack
|
||||
void check_fw_ack(void);
|
||||
void map_stick_mode(uint16_t *channels);
|
||||
void set_fcc_channel(void);
|
||||
|
||||
// check for double binding
|
||||
void check_double_bind(void);
|
||||
};
|
||||
|
||||
|
||||
|
@ -21,7 +21,6 @@
|
||||
https://github.com/esden/superbitrf-firmware
|
||||
*/
|
||||
#if CONFIG_HAL_BOARD == HAL_BOARD_CHIBIOS
|
||||
static THD_WORKING_AREA(_irq_handler_wa, 512);
|
||||
#define TIMEOUT_PRIORITY 181
|
||||
#define EVT_TIMEOUT EVENT_MASK(0)
|
||||
#define EVT_IRQ EVENT_MASK(1)
|
||||
@ -41,7 +40,7 @@ static THD_WORKING_AREA(_irq_handler_wa, 512);
|
||||
|
||||
extern const AP_HAL::HAL& hal;
|
||||
|
||||
#define Debug(level, fmt, args...) do { if ((level) <= get_debug_level()) { hal.console->printf(fmt, ##args); }} while (0)
|
||||
#define Debug(level, fmt, args...) do { if ((level) <= get_debug_level()) { gcs().send_text(MAV_SEVERITY_INFO, fmt, ##args); }} while (0)
|
||||
|
||||
#define LP_FIFO_SIZE 16 // Physical data FIFO lengths in Radio
|
||||
|
||||
@ -262,15 +261,16 @@ bool AP_Radio_cypress::init(void)
|
||||
{
|
||||
dev = hal.spi->get_device(CYRF_SPI_DEVICE);
|
||||
#if CONFIG_HAL_BOARD == HAL_BOARD_CHIBIOS
|
||||
if(_irq_handler_ctx != nullptr) {
|
||||
if (_irq_handler_ctx != nullptr) {
|
||||
AP_HAL::panic("AP_Radio_cypress: double instantiation of irq_handler\n");
|
||||
}
|
||||
chVTObjectInit(&timeout_vt);
|
||||
_irq_handler_ctx = chThdCreateStatic(_irq_handler_wa,
|
||||
sizeof(_irq_handler_wa),
|
||||
TIMEOUT_PRIORITY, /* Initial priority. */
|
||||
irq_handler_thd, /* Thread function. */
|
||||
NULL); /* Thread parameter. */
|
||||
_irq_handler_ctx = chThdCreateFromHeap(NULL,
|
||||
THD_WORKING_AREA_SIZE(2048),
|
||||
"radio_cypress",
|
||||
TIMEOUT_PRIORITY,
|
||||
irq_handler_thd,
|
||||
NULL);
|
||||
#endif
|
||||
load_bind_info();
|
||||
|
||||
@ -437,7 +437,7 @@ bool AP_Radio_cypress::send(const uint8_t *pkt, uint16_t len)
|
||||
|
||||
/* The PN codes */
|
||||
const uint8_t AP_Radio_cypress::pn_codes[5][9][8] = {
|
||||
{ /* Row 0 */
|
||||
{ /* Row 0 */
|
||||
/* Col 0 */ {0x03, 0xBC, 0x6E, 0x8A, 0xEF, 0xBD, 0xFE, 0xF8},
|
||||
/* Col 1 */ {0x88, 0x17, 0x13, 0x3B, 0x2D, 0xBF, 0x06, 0xD6},
|
||||
/* Col 2 */ {0xF1, 0x94, 0x30, 0x21, 0xA1, 0x1C, 0x88, 0xA9},
|
||||
@ -447,8 +447,8 @@ const uint8_t AP_Radio_cypress::pn_codes[5][9][8] = {
|
||||
/* Col 6 */ {0xEF, 0x03, 0x95, 0x89, 0xB4, 0x71, 0x61, 0x9D},
|
||||
/* Col 7 */ {0x40, 0xBA, 0x97, 0xD5, 0x86, 0x4F, 0xCC, 0xD1},
|
||||
/* Col 8 */ {0xD7, 0xA1, 0x54, 0xB1, 0x5E, 0x89, 0xAE, 0x86}
|
||||
},
|
||||
{ /* Row 1 */
|
||||
},
|
||||
{ /* Row 1 */
|
||||
/* Col 0 */ {0x83, 0xF7, 0xA8, 0x2D, 0x7A, 0x44, 0x64, 0xD3},
|
||||
/* Col 1 */ {0x3F, 0x2C, 0x4E, 0xAA, 0x71, 0x48, 0x7A, 0xC9},
|
||||
/* Col 2 */ {0x17, 0xFF, 0x9E, 0x21, 0x36, 0x90, 0xC7, 0x82},
|
||||
@ -458,8 +458,8 @@ const uint8_t AP_Radio_cypress::pn_codes[5][9][8] = {
|
||||
/* Col 6 */ {0x1E, 0x6A, 0xF0, 0x37, 0x52, 0x7B, 0x11, 0xD4},
|
||||
/* Col 7 */ {0x62, 0xF5, 0x2B, 0xAA, 0xFC, 0x33, 0xBF, 0xAF},
|
||||
/* Col 8 */ {0x40, 0x56, 0x32, 0xD9, 0x0F, 0xD9, 0x5D, 0x97}
|
||||
},
|
||||
{ /* Row 2 */
|
||||
},
|
||||
{ /* Row 2 */
|
||||
/* Col 0 */ {0x40, 0x56, 0x32, 0xD9, 0x0F, 0xD9, 0x5D, 0x97},
|
||||
/* Col 1 */ {0x8E, 0x4A, 0xD0, 0xA9, 0xA7, 0xFF, 0x20, 0xCA},
|
||||
/* Col 2 */ {0x4C, 0x97, 0x9D, 0xBF, 0xB8, 0x3D, 0xB5, 0xBE},
|
||||
@ -469,8 +469,8 @@ const uint8_t AP_Radio_cypress::pn_codes[5][9][8] = {
|
||||
/* Col 6 */ {0xF4, 0xDA, 0x06, 0xDB, 0xBF, 0x4E, 0x6F, 0xB3},
|
||||
/* Col 7 */ {0x9E, 0x08, 0xD1, 0xAE, 0x59, 0x5E, 0xE8, 0xF0},
|
||||
/* Col 8 */ {0xC0, 0x90, 0x8F, 0xBB, 0x7C, 0x8E, 0x2B, 0x8E}
|
||||
},
|
||||
{ /* Row 3 */
|
||||
},
|
||||
{ /* Row 3 */
|
||||
/* Col 0 */ {0xC0, 0x90, 0x8F, 0xBB, 0x7C, 0x8E, 0x2B, 0x8E},
|
||||
/* Col 1 */ {0x80, 0x69, 0x26, 0x80, 0x08, 0xF8, 0x49, 0xE7},
|
||||
/* Col 2 */ {0x7D, 0x2D, 0x49, 0x54, 0xD0, 0x80, 0x40, 0xC1},
|
||||
@ -480,8 +480,8 @@ const uint8_t AP_Radio_cypress::pn_codes[5][9][8] = {
|
||||
/* Col 6 */ {0xBF, 0x54, 0x98, 0xB9, 0xB7, 0x30, 0x5A, 0x88},
|
||||
/* Col 7 */ {0x35, 0xD1, 0xFC, 0x97, 0x23, 0xD4, 0xC9, 0x88},
|
||||
/* Col 8 */ {0x88, 0xE1, 0xD6, 0x31, 0x26, 0x5F, 0xBD, 0x40}
|
||||
},
|
||||
{ /* Row 4 */
|
||||
},
|
||||
{ /* Row 4 */
|
||||
/* Col 0 */ {0xE1, 0xD6, 0x31, 0x26, 0x5F, 0xBD, 0x40, 0x93},
|
||||
/* Col 1 */ {0xDC, 0x68, 0x08, 0x99, 0x97, 0xAE, 0xAF, 0x8C},
|
||||
/* Col 2 */ {0xC3, 0x0E, 0x01, 0x16, 0x0E, 0x32, 0x06, 0xBA},
|
||||
@ -491,7 +491,7 @@ const uint8_t AP_Radio_cypress::pn_codes[5][9][8] = {
|
||||
/* Col 6 */ {0x58, 0xB5, 0xB3, 0xDD, 0x0E, 0x28, 0xF1, 0xB0},
|
||||
/* Col 7 */ {0x5F, 0x30, 0x3B, 0x56, 0x96, 0x45, 0xF4, 0xA1},
|
||||
/* Col 8 */ {0x03, 0xBC, 0x6E, 0x8A, 0xEF, 0xBD, 0xFE, 0xF8}
|
||||
},
|
||||
},
|
||||
};
|
||||
const uint8_t AP_Radio_cypress::pn_bind[] = { 0x98, 0x88, 0x1B, 0xE4, 0x30, 0x79, 0x03, 0x84 };
|
||||
|
||||
@ -632,9 +632,7 @@ void AP_Radio_cypress::radio_init(void)
|
||||
start_receive();
|
||||
|
||||
// setup handler for rising edge of IRQ pin
|
||||
#if CONFIG_HAL_BOARD == HAL_BOARD_CHIBIOS
|
||||
hal.gpio->attach_interrupt(HAL_GPIO_RADIO_IRQ, trigger_irq_radio_event, AP_HAL::GPIO::INTERRUPT_RISING);
|
||||
#endif
|
||||
}
|
||||
|
||||
void AP_Radio_cypress::dump_registers(uint8_t n)
|
||||
@ -932,9 +930,7 @@ void AP_Radio_cypress::dsm2_start_sync(void)
|
||||
*/
|
||||
void AP_Radio_cypress::setup_timeout(uint32_t timeout_ms)
|
||||
{
|
||||
#if CONFIG_HAL_BOARD == HAL_BOARD_CHIBIOS
|
||||
chVTSet(&timeout_vt, chTimeMS2I(timeout_ms), trigger_timeout_event, nullptr);
|
||||
#endif
|
||||
}
|
||||
|
||||
/*
|
||||
@ -1121,7 +1117,7 @@ void AP_Radio_cypress::irq_handler(void)
|
||||
|
||||
switch (state) {
|
||||
case STATE_AUTOBIND:
|
||||
FALLTHROUGH;
|
||||
// fallthrough
|
||||
case STATE_RECV:
|
||||
case STATE_BIND:
|
||||
irq_handler_recv(rx_status);
|
||||
@ -1174,7 +1170,7 @@ void AP_Radio_cypress::irq_timeout(void)
|
||||
case STATE_AUTOBIND:
|
||||
case STATE_SEND_TELEM_WAIT:
|
||||
state = STATE_RECV;
|
||||
FALLTHROUGH;
|
||||
// fall through
|
||||
default:
|
||||
write_register(CYRF_XACT_CFG, CYRF_MODE_SYNTH_RX | CYRF_FRC_END);
|
||||
write_register(CYRF_RX_ABORT, 0);
|
||||
@ -1189,12 +1185,11 @@ void AP_Radio_cypress::irq_timeout(void)
|
||||
/*
|
||||
called on HRT timeout
|
||||
*/
|
||||
#if CONFIG_HAL_BOARD == HAL_BOARD_CHIBIOS
|
||||
void AP_Radio_cypress::irq_handler_thd(void *arg)
|
||||
{
|
||||
_irq_handler_ctx = chThdGetSelfX();
|
||||
(void)arg;
|
||||
while(true) {
|
||||
while (true) {
|
||||
eventmask_t evt = chEvtWaitAny(ALL_EVENTS);
|
||||
if (evt & EVT_IRQ) {
|
||||
radio_singleton->irq_handler();
|
||||
@ -1225,7 +1220,7 @@ void AP_Radio_cypress::trigger_irq_radio_event()
|
||||
}
|
||||
chSysUnlockFromISR();
|
||||
}
|
||||
#endif
|
||||
|
||||
/*
|
||||
Set the current DSM channel with SOP, CRC and data code
|
||||
*/
|
||||
@ -1280,7 +1275,7 @@ void AP_Radio_cypress::dsm_generate_channels_dsmx(uint8_t mfg_id[4], uint8_t cha
|
||||
uint32_t id_tmp = id;
|
||||
|
||||
// While not all channels are set
|
||||
while(idx < 23) {
|
||||
while (idx < 23) {
|
||||
int i;
|
||||
int count_3_27 = 0, count_28_51 = 0, count_52_76 = 0;
|
||||
|
||||
@ -1298,7 +1293,7 @@ void AP_Radio_cypress::dsm_generate_channels_dsmx(uint8_t mfg_id[4], uint8_t cha
|
||||
}
|
||||
|
||||
// Count the channel groups
|
||||
if(channels[i] <= 27) {
|
||||
if (channels[i] <= 27) {
|
||||
count_3_27++;
|
||||
} else if (channels[i] <= 51) {
|
||||
count_28_51++;
|
||||
|
@ -14,6 +14,8 @@
|
||||
*/
|
||||
#pragma once
|
||||
|
||||
#if HAL_RCINPUT_WITH_AP_RADIO
|
||||
|
||||
/*
|
||||
AP_Radio implementation for Cypress 2.4GHz radio.
|
||||
|
||||
@ -24,9 +26,7 @@
|
||||
*/
|
||||
|
||||
#include "AP_Radio_backend.h"
|
||||
#if CONFIG_HAL_BOARD == HAL_BOARD_CHIBIOS
|
||||
#include "hal.h"
|
||||
#endif
|
||||
#include "telem_structure.h"
|
||||
|
||||
class AP_Radio_cypress : public AP_Radio_backend
|
||||
@ -62,7 +62,8 @@ public:
|
||||
void update(void) override;
|
||||
|
||||
// get TX fw version
|
||||
uint32_t get_tx_version(void) override {
|
||||
uint32_t get_tx_version(void) override
|
||||
{
|
||||
// pack date into 16 bits for vendor_id in AUTOPILOT_VERSION
|
||||
return (uint16_t(dsm.tx_firmware_year)<<12) + (uint16_t(dsm.tx_firmware_month)<<8) + dsm.tx_firmware_day;
|
||||
}
|
||||
@ -71,7 +72,8 @@ public:
|
||||
const AP_Radio::stats &get_stats(void) override;
|
||||
|
||||
// set the 2.4GHz wifi channel used by companion computer, so it can be avoided
|
||||
void set_wifi_channel(uint8_t channel) override {
|
||||
void set_wifi_channel(uint8_t channel) override
|
||||
{
|
||||
t_status.wifi_chan = channel;
|
||||
}
|
||||
|
||||
@ -110,10 +112,8 @@ private:
|
||||
static const config cyrf_bind_config[];
|
||||
static const config cyrf_transfer_config[];
|
||||
|
||||
#if CONFIG_HAL_BOARD == HAL_BOARD_CHIBIOS
|
||||
virtual_timer_t timeout_vt;
|
||||
static thread_t *_irq_handler_ctx;
|
||||
#endif
|
||||
void radio_set_config(const struct config *config, uint8_t size);
|
||||
|
||||
void start_receive(void);
|
||||
@ -129,11 +129,9 @@ private:
|
||||
|
||||
// trampoline functions to take us from static IRQ function to class functions
|
||||
|
||||
#if CONFIG_HAL_BOARD == HAL_BOARD_CHIBIOS
|
||||
static void irq_handler_thd(void* arg);
|
||||
static void trigger_irq_radio_event(void);
|
||||
static void trigger_timeout_event(void *arg);
|
||||
#endif
|
||||
|
||||
static const uint8_t max_channels = 16;
|
||||
|
||||
@ -281,3 +279,4 @@ private:
|
||||
void setup_timeout(uint32_t timeout_ms);
|
||||
};
|
||||
|
||||
#endif
|
||||
|
698
libraries/AP_Radio/driver_bk2425.cpp
Normal file
698
libraries/AP_Radio/driver_bk2425.cpp
Normal file
@ -0,0 +1,698 @@
|
||||
// --------------------------------------------------------------------
|
||||
// low level driver for the beken radio on SPI
|
||||
// --------------------------------------------------------------------
|
||||
|
||||
#include "driver_bk2425.h"
|
||||
|
||||
#if defined(HAL_RCINPUT_WITH_AP_RADIO) && CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_CHIBIOS_SKYVIPER_F412
|
||||
|
||||
#include <utility>
|
||||
#include <AP_HAL_ChibiOS/AP_HAL_ChibiOS.h>
|
||||
using namespace ChibiOS;
|
||||
|
||||
//#pragma GCC optimize("O0")
|
||||
|
||||
extern const AP_HAL::HAL& hal;
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// Radio initialisation tables
|
||||
// --------------------------------------------------------------------
|
||||
|
||||
#if (TX_SPEED==250)
|
||||
ITX_SPEED Radio_Beken::gTxSpeed = ITX_250;
|
||||
#elif (TX_SPEED==100)
|
||||
ITX_SPEED Radio_Beken::gTxSpeed = ITX_1000;
|
||||
#elif (TX_SPEED==2000)
|
||||
ITX_SPEED Radio_Beken::gTxSpeed = ITX_2000;
|
||||
#endif
|
||||
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
static const uint8_t Bank1_RegTable[ITX_MAX][IREG_MAX][5]= {
|
||||
// (TX_SPEED == 250u) // [ITX_250]
|
||||
{
|
||||
{ BK2425_R1_4, 0xf9,0x96,0x8a,0xdb }, // 0xDB8A96f9ul, // [IREG1_4] REG4 250kbps
|
||||
{ BK2425_R1_5, 0x24,0x06,0x0f,0xb6 }, // 0xB60F0624ul, // [IREG1_5] REG5 250kbps
|
||||
PLL_SPEED, // [IREG1_12] REG12
|
||||
{ BK2425_R1_13, 0x36,0xb4,0x80,0x00 }, // 0x36B48000ul, // [IREG1_13] REG13
|
||||
{ BK2425_R1_4, 0xff,0x96,0x8a,0xdb }, // 0xDB8A96f9ul, // [IREG1_4A] REG4 250kbps
|
||||
},
|
||||
// (TX_SPEED == 1000u) [ITX_1000]
|
||||
{
|
||||
{ BK2425_R1_4, 0xf9,0x96,0x82,0x1b }, // 0x1B8296f9ul, // [IREG1_4] REG4 1Mbps
|
||||
{ BK2425_R1_5, 0x24,0x06,0x0f,0xa6 }, // 0xA60F0624ul, // [IREG1_5] REG5 1Mbps
|
||||
PLL_SPEED, // [IREG1_12] REG12
|
||||
{ BK2425_R1_13, 0x36,0xb4,0x80,0x00 }, // 0x36B48000ul, // [IREG1_13] REG13
|
||||
{ BK2425_R1_4, 0xff,0x96,0x82,0x1b }, // 0x1B8296f9ul, // [IREG1_4A] REG4 1Mbps
|
||||
},
|
||||
// (TX_SPEED == 2000u) [ITX_2000]
|
||||
{
|
||||
{ BK2425_R1_4, 0xf9,0x96,0x82,0xdb }, // 0xdb8296f9ul, // [IREG1_4] REG4 2Mbps
|
||||
{ BK2425_R1_5, 0x24,0x06,0x0f,0xb6 }, // 0xb60f0624ul, // [IREG1_5] REG5 2Mbps
|
||||
PLL_SPEED, // [IREG1_12] REG12
|
||||
{ BK2425_R1_13, 0x36,0xb4,0x80,0x00 }, // 0x36B48000ul, // [IREG1_13] REG13
|
||||
{ BK2425_R1_4, 0xff,0x96,0x82,0xdb }, // 0xdb8296f9ul, // [IREG1_4A] REG4 2Mbps
|
||||
},
|
||||
// (TX_SPEED == 0u) // [ITX_CARRIER]
|
||||
{
|
||||
{ BK2425_R1_4, 0xf9,0x96,0x82,0x21 }, // 0xF9968221ul, // [IREG1_4] REG4 carrier
|
||||
{ BK2425_R1_5, 0x24,0x06,0x0f,0xb6 }, // 0xB60F0624ul, // [IREG1_5] REG5 250kbps
|
||||
PLL_SPEED, // [IREG1_12] REG12
|
||||
{ BK2425_R1_13, 0x36,0xb4,0x80,0x00 }, // 0x36B48000ul, // [IREG1_13] REG13
|
||||
{ BK2425_R1_4, 0xff,0x96,0x82,0x21 }, // 0xDB8A96f9ul, // [IREG1_4A] REG4 250kbps
|
||||
}
|
||||
};
|
||||
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
static const uint8_t Bank0_Reg6[ITX_MAX][2] = {
|
||||
{BK_RF_SETUP, 0x27}, // 250kbps (6) 0x27=250kbps
|
||||
{BK_RF_SETUP, 0x07}, // 1000kbps (6) 0x07=1Mbps, high gain, high txpower
|
||||
{BK_RF_SETUP, 0x2F}, // 2000kbps (6) 0x2F=2Mbps, high gain, high txpower
|
||||
{BK_RF_SETUP, 0x37}, // 250kbps (6) 0x10=carrier
|
||||
};
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
static const uint8_t Bank1_Reg14[]= {
|
||||
0x41,0x20,0x08,0x04,0x81,0x20,0xcf,0xF7,0xfe,0xff,0xff
|
||||
};
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// Bank0 register initialization value
|
||||
static const uint8_t Bank0_Reg[][2]= {
|
||||
|
||||
#if 0
|
||||
{BK_CONFIG, BK_CONFIG_PWR_UP | BK_CONFIG_PRIM_RX }, // (0) 0x0F=Rx, PowerUp, no crc, all interrupts enabled
|
||||
{BK_EN_AA, 0x00}, // (1) 0x00=No auto acknowledge packets on all 6 data pipes (0..5)
|
||||
{BK_EN_RXADDR, 0x02}, // (2) 0x01=1 or 2 out of 6 data pipes enabled (pairing heartbeat and my tx)
|
||||
{BK_SETUP_AW, 0x03}, // (3) 0x01=3 byte address width
|
||||
#else
|
||||
{BK_CONFIG, BK_CONFIG_EN_CRC | BK_CONFIG_CRCO | BK_CONFIG_PWR_UP | BK_CONFIG_PRIM_RX }, // (0) 0x0F=Rx, PowerUp, crc16, all interrupts enabled
|
||||
{BK_EN_AA, 0x00}, // (1) 0x00=No auto acknowledge packets on all 6 data pipes (0..5)
|
||||
{BK_EN_RXADDR, 0x03}, // (2) 0x01=1 or 2 out of 6 data pipes enabled (pairing heartbeat and my tx)
|
||||
{BK_SETUP_AW, 0x03}, // (3) 0x03=5 byte address width
|
||||
#endif
|
||||
{BK_SETUP_RETR, 0x00}, // (4) 0x00=No retransmissions
|
||||
{BK_RF_CH, 0x17}, // (5) 0x17=2423Mhz default frequency
|
||||
|
||||
// Comment in Beken code says that 0x0F or 0x2F=2Mbps; 0x07=1Mbps; 0x27=250Kbps
|
||||
#if (TX_SPEED == 2000)
|
||||
{BK_RF_SETUP, 0x2F}, // (6) 0x2F=2Mbps, high gain, high txpower
|
||||
#elif (TX_SPEED == 1000)
|
||||
{BK_RF_SETUP, 0x07}, // (6) 0x07=1Mbps, high gain, high txpower
|
||||
#elif (TX_SPEED == 250)
|
||||
{BK_RF_SETUP, 0x27}, // (6) 0x27=250kbps
|
||||
//{BK_RF_SETUP, 0x21}, // (6) 0x27=250kbps, lowest txpower
|
||||
#endif
|
||||
|
||||
{BK_STATUS, 0x07}, // (7) 7=no effect
|
||||
{BK_OBSERVE_TX, 0x00}, // (8) (no effect)
|
||||
{BK_CD, 0x00}, // (9) Carrier detect (no effect)
|
||||
// (10) = 5 byte register
|
||||
// (11) = 5 byte register
|
||||
{BK_RX_ADDR_P2, 0xc3}, // (12) rx address for data pipe 2
|
||||
{BK_RX_ADDR_P3, 0xc4}, // (13) rx address for data pipe 3
|
||||
{BK_RX_ADDR_P4, 0xc5}, // (14) rx address for data pipe 4
|
||||
{BK_RX_ADDR_P5, 0xc6}, // (15) rx address for data pipe 5
|
||||
// (16) = 5 byte register
|
||||
{BK_RX_PW_P0, PACKET_LENGTH_RX_CTRL}, // (17) size of rx data pipe 0
|
||||
{BK_RX_PW_P1, PACKET_LENGTH_RX_BIND}, // (18) size of rx data pipe 1
|
||||
{BK_RX_PW_P2, 0x20}, // (19) size of rx data pipe 2
|
||||
{BK_RX_PW_P3, 0x20}, // (20) size of rx data pipe 3
|
||||
{BK_RX_PW_P4, 0x20}, // (21) size of rx data pipe 4
|
||||
{BK_RX_PW_P5, 0x20}, // (22) size of rx data pipe 5
|
||||
{BK_FIFO_STATUS,0x00}, // (23) fifo status
|
||||
// (24,25,26,27)
|
||||
{BK_DYNPD, 0x3F}, // (28) 0x3f=enable dynamic payload length for all 6 data pipes
|
||||
{BK_FEATURE, BK_FEATURE_EN_DPL | BK_FEATURE_EN_ACK_PAY | BK_FEATURE_EN_DYN_ACK } // (29) 7=enable ack, no ack, dynamic payload length
|
||||
};
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
const uint8_t RegPower[8][2] = {
|
||||
{ OUTPUT_POWER_REG4_0, OUTPUT_POWER_REG6_0 },
|
||||
{ OUTPUT_POWER_REG4_1, OUTPUT_POWER_REG6_1 },
|
||||
{ OUTPUT_POWER_REG4_2, OUTPUT_POWER_REG6_2 },
|
||||
{ OUTPUT_POWER_REG4_3, OUTPUT_POWER_REG6_3 },
|
||||
{ OUTPUT_POWER_REG4_4, OUTPUT_POWER_REG6_4 },
|
||||
{ OUTPUT_POWER_REG4_5, OUTPUT_POWER_REG6_5 },
|
||||
{ OUTPUT_POWER_REG4_6, OUTPUT_POWER_REG6_6 },
|
||||
{ OUTPUT_POWER_REG4_7, OUTPUT_POWER_REG6_7 },
|
||||
};
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// Generic functions
|
||||
// --------------------------------------------------------------------
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// constructor
|
||||
Radio_Beken::Radio_Beken(AP_HAL::OwnPtr<AP_HAL::SPIDevice> _dev) :
|
||||
dev(std::move(_dev))
|
||||
{
|
||||
ResetAddress();
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// Use the default addresses
|
||||
void Radio_Beken::ResetAddress(void)
|
||||
{
|
||||
// Set the default address
|
||||
TX_Address[0] = 0x33;
|
||||
TX_Address[1] = RX0_Address[1] = 0x00;
|
||||
TX_Address[2] = RX0_Address[2] = 0x59;
|
||||
TX_Address[3] = RX0_Address[3] = 0x00;
|
||||
TX_Address[4] = RX0_Address[4] = 0x00;
|
||||
RX0_Address[0] = 0x31;
|
||||
RX1_Address[0] = 0x32;
|
||||
RX1_Address[1] = 0x99;
|
||||
RX1_Address[2] = 0x59;
|
||||
RX1_Address[3] = 0xC6;
|
||||
RX1_Address[4] = 0x2D;
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// Raw SPI access functions
|
||||
// --------------------------------------------------------------------
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
void Radio_Beken::ReadRegisterMulti(uint8_t address, uint8_t *data, uint8_t len)
|
||||
{
|
||||
uint8_t tx[len+1];
|
||||
uint8_t rx[len+1];
|
||||
memset(tx, 0, len+1);
|
||||
memset(rx, 0, len+1);
|
||||
tx[0] = address;
|
||||
DEBUG2_HIGH();
|
||||
(void)dev->transfer_fullduplex(tx, rx, len+1);
|
||||
DEBUG2_LOW();
|
||||
memcpy(data, &rx[1], len);
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
void Radio_Beken::WriteRegisterMulti(uint8_t address, const uint8_t *data, uint8_t len)
|
||||
{
|
||||
uint8_t tx[len+1];
|
||||
uint8_t rx[len+1];
|
||||
memset(rx, 0, len+1);
|
||||
tx[0] = address;
|
||||
memcpy(&tx[1], data, len);
|
||||
DEBUG2_HIGH();
|
||||
(void)dev->transfer_fullduplex(tx, rx, len+1);
|
||||
DEBUG2_LOW();
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// Low-level Beken functions
|
||||
// --------------------------------------------------------------------
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
uint8_t Radio_Beken::ReadStatus(void)
|
||||
{
|
||||
uint8_t tx = BK_NOP;
|
||||
uint8_t rx = 0;
|
||||
DEBUG2_HIGH();
|
||||
(void)dev->transfer_fullduplex(&tx, &rx, 1);
|
||||
DEBUG2_LOW();
|
||||
return rx; // Status
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
uint8_t Radio_Beken::ReadReg(uint8_t reg)
|
||||
{
|
||||
uint8_t tx[2];
|
||||
uint8_t rx[2];
|
||||
memset(tx, 0, 2);
|
||||
memset(rx, 0, 2);
|
||||
tx[0] = reg | BK_READ_REG;
|
||||
DEBUG2_HIGH();
|
||||
(void)dev->transfer_fullduplex(tx, rx, 2);
|
||||
DEBUG2_LOW();
|
||||
return rx[1];
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
uint8_t Radio_Beken::Strobe(uint8_t address)
|
||||
{
|
||||
uint8_t tx = address;
|
||||
uint8_t rx = 0;
|
||||
DEBUG2_HIGH();
|
||||
(void)dev->transfer_fullduplex(&tx, &rx, 1);
|
||||
DEBUG2_LOW();
|
||||
return rx; // Status
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// Set which register bank we are accessing
|
||||
void Radio_Beken::SetRBank(uint8_t bank) // 1:Bank1 0:Bank0
|
||||
{
|
||||
uint8_t lastbank = ReadStatus() & BK_STATUS_RBANK;
|
||||
if (!lastbank != !bank) {
|
||||
uint8_t tx[2];
|
||||
uint8_t rx[2];
|
||||
tx[0] = BK_ACTIVATE_CMD;
|
||||
tx[1] = 0x53;
|
||||
DEBUG2_HIGH();
|
||||
(void)dev->transfer_fullduplex(&tx[0], &rx[0], 2);
|
||||
DEBUG2_LOW();
|
||||
}
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
void Radio_Beken::WriteReg(uint8_t address, uint8_t data)
|
||||
{
|
||||
uint8_t tx[2];
|
||||
uint8_t rx[2];
|
||||
memset(rx, 0, 2);
|
||||
tx[0] = address; // done by caller | BK_WRITE_REG;
|
||||
tx[1] = data;
|
||||
DEBUG2_HIGH();
|
||||
(void)dev->transfer_fullduplex(tx, rx, 2);
|
||||
DEBUG2_LOW();
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
void Radio_Beken::WriteRegisterMultiBank1(uint8_t address, const uint8_t *data, uint8_t length)
|
||||
{
|
||||
SetRBank(1);
|
||||
WriteRegisterMulti(address, data, length);
|
||||
SetRBank(0);
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// High-level Beken functions
|
||||
// --------------------------------------------------------------------
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// Set the radio transmission power of the beken
|
||||
// Prerequisite: We should be in idle mode before calling this function
|
||||
void Radio_Beken::SetPower(uint8_t power)
|
||||
{
|
||||
if (power > 7) {
|
||||
power = 7;
|
||||
}
|
||||
uint8_t oldready = bkReady;
|
||||
bkReady = 0;
|
||||
hal.scheduler->delay(100); // delay more than 50ms.
|
||||
SetRBank(1);
|
||||
{
|
||||
const uint8_t* p = &Bank1_RegTable[fcc.CW_mode ? ITX_CARRIER : gTxSpeed][IREG1_4][0];
|
||||
uint8_t idx = *p++;
|
||||
uint8_t buf[4];
|
||||
buf[0] = *p++;
|
||||
buf[1] = *p++;
|
||||
buf[2] = *p++;
|
||||
buf[3] = *p++;
|
||||
buf[0] &= ~0x38;
|
||||
buf[0] |= (RegPower[power][0] << 3); // Bits 27..29
|
||||
WriteRegisterMulti((BK_WRITE_REG|idx), buf, 4);
|
||||
}
|
||||
hal.scheduler->delay(100); // delay more than 50ms.
|
||||
SetRBank(0);
|
||||
hal.scheduler->delay(100);
|
||||
|
||||
uint8_t setup = ReadReg(BK_RF_SETUP);
|
||||
setup &= ~(3 << 1);
|
||||
setup |= (RegPower[power][1] << 1); // Bits 1..2
|
||||
if (fcc.CW_mode) {
|
||||
setup |= 0x10;
|
||||
}
|
||||
WriteReg(BK_WRITE_REG|BK_RF_SETUP, setup);
|
||||
bkReady = oldready;
|
||||
fcc.power = power;
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// Set the physical radio transmission frequency of the beken
|
||||
void Radio_Beken::SetChannel(uint8_t freq)
|
||||
{
|
||||
lastTxChannel = freq;
|
||||
WriteReg(BK_WRITE_REG|BK_RF_CH, freq);
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// Set the radio transmission mode of the beken
|
||||
// Enable/disable the carrier sending mode
|
||||
// Prerequisite: We should be in idle mode before calling this function
|
||||
void Radio_Beken::SetCwMode(uint8_t cw)
|
||||
{
|
||||
uint8_t oldready = bkReady;
|
||||
bkReady = 0;
|
||||
hal.scheduler->delay(100); // delay more than 50ms.
|
||||
SetRBank(1);
|
||||
{
|
||||
const uint8_t* p = &Bank1_RegTable[cw ? ITX_CARRIER : gTxSpeed][IREG1_4][0];
|
||||
uint8_t idx = *p++;
|
||||
uint8_t buf[4];
|
||||
buf[0] = *p++;
|
||||
buf[1] = *p++;
|
||||
buf[2] = *p++;
|
||||
buf[3] = *p++;
|
||||
buf[0] &= ~0x38;
|
||||
buf[0] |= (RegPower[fcc.power & 7][0] << 3); // Bits 27..29
|
||||
WriteRegisterMulti((BK_WRITE_REG|idx), buf, 4);
|
||||
}
|
||||
hal.scheduler->delay(100); // delay more than 50ms.
|
||||
SetRBank(0);
|
||||
hal.scheduler->delay(100); // delay more than 50ms.
|
||||
|
||||
uint8_t setup = ReadReg(BK_RF_SETUP);
|
||||
setup &= ~(3 << 1);
|
||||
setup |= (RegPower[fcc.power & 7][1] << 1); // Bits 1..2
|
||||
if (cw) {
|
||||
setup |= 0x10;
|
||||
}
|
||||
WriteReg((BK_WRITE_REG|BK_RF_SETUP), setup);
|
||||
fcc.CW_mode = cw != 0;
|
||||
bkReady = oldready;
|
||||
}
|
||||
|
||||
// --------------------------------------------------------------------
|
||||
// Enable/disable the CRC receive mode
|
||||
// Prerequisite: We should be in idle mode before calling this function
|
||||
void Radio_Beken::SetCrcMode(uint8_t disable_crc)
|
||||
{
|
||||
uint8_t oldready = bkReady;
|
||||
bkReady = 0;
|
||||
|
||||
uint8_t config = ReadReg(BK_CONFIG);
|
||||
if (disable_crc) {
|
||||
config &= ~(BK_CONFIG_EN_CRC | BK_CONFIG_CRCO); // Disable CRC
|
||||
} else {
|
||||
config |= (BK_CONFIG_EN_CRC | BK_CONFIG_CRCO); // Enable CRC
|
||||
}
|
||||
WriteReg((BK_WRITE_REG|BK_CONFIG), config);
|
||||
fcc.disable_crc = (disable_crc != 0);
|
||||
bkReady = oldready;
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
// Enable the carrier detect feature: Bank1 Reg5 Bit 18
|
||||
void Radio_Beken::EnableCarrierDetect(bool bEnable)
|
||||
{
|
||||
if (bEnable == fcc.enable_cd) {
|
||||
return;
|
||||
}
|
||||
uint8_t oldready = bkReady;
|
||||
bkReady = 0;
|
||||
SetRBank(1);
|
||||
{
|
||||
const uint8_t* p = &Bank1_RegTable[gTxSpeed][IREG1_5][0];
|
||||
uint8_t idx = *p++;
|
||||
uint8_t buf[4];
|
||||
buf[0] = *p++;
|
||||
buf[1] = *p++;
|
||||
buf[2] = *p++;
|
||||
buf[3] = *p++;
|
||||
if (bEnable) {
|
||||
buf[1] &= ~0x04;
|
||||
}
|
||||
WriteRegisterMulti((BK_WRITE_REG|idx), buf, 4);
|
||||
}
|
||||
SetRBank(0);
|
||||
bkReady = oldready;
|
||||
fcc.enable_cd = bEnable;
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
// Returns true if a carrier is detected
|
||||
bool Radio_Beken::CarrierDetect(void)
|
||||
{
|
||||
if (fcc.enable_cd) {
|
||||
if (ReadReg(BK_CD) & 0x01) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
void Radio_Beken::SetFactoryMode(uint8_t factory)
|
||||
{
|
||||
uint8_t oldready = bkReady;
|
||||
bkReady = 0;
|
||||
|
||||
// Set receive/transmit addresses
|
||||
if (factory) {
|
||||
// For factory modes, use fixed addresses
|
||||
TX_Address[0] = 0x35;
|
||||
TX_Address[1] = RX1_Address[1] = RX0_Address[1] = 0x99;
|
||||
TX_Address[2] = RX1_Address[2] = RX0_Address[2] = 0x59;
|
||||
TX_Address[3] = RX1_Address[3] = RX0_Address[3] = 0xC6;
|
||||
TX_Address[4] = RX1_Address[4] = RX0_Address[4] = factory;
|
||||
RX0_Address[0] = 0x34;
|
||||
RX1_Address[0] = 0x43;
|
||||
} else {
|
||||
// For normal modes, use the default addresses
|
||||
ResetAddress();
|
||||
}
|
||||
|
||||
// Write the addresses to the registers
|
||||
WriteRegisterMulti((BK_WRITE_REG|BK_RX_ADDR_P0), RX0_Address, 5);
|
||||
WriteRegisterMulti((BK_WRITE_REG|BK_RX_ADDR_P1), RX1_Address, 5);
|
||||
WriteRegisterMulti((BK_WRITE_REG|BK_TX_ADDR), TX_Address, 5);
|
||||
WriteReg(BK_WRITE_REG|BK_EN_RXADDR, 0x03);
|
||||
|
||||
// Frequency is set by the caller
|
||||
fcc.factory_mode = factory;
|
||||
bkReady = oldready;
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
bool Radio_Beken::Reset(void)
|
||||
{
|
||||
//...
|
||||
hal.scheduler->delay_microseconds(1000);
|
||||
return 0;
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
// Delay after changing chip-enable
|
||||
// This can be called from within the interrupt response thread
|
||||
void Radio_Beken::DelayCE(void)
|
||||
{
|
||||
DEBUG1_LOW();
|
||||
hal.scheduler->delay_microseconds(50);
|
||||
DEBUG1_HIGH();
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
bool Radio_Beken::WasTxMode(void)
|
||||
{
|
||||
// Were we transmitting something?
|
||||
return bkMode == BKRADIO_TX;
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
bool Radio_Beken::WasRxMode(void)
|
||||
{
|
||||
// Were we receiving something?
|
||||
return bkMode == BKRADIO_RX;
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
// Switch to Rx mode
|
||||
void Radio_Beken::SwitchToRxMode(void)
|
||||
{
|
||||
uint8_t value;
|
||||
|
||||
Strobe(BK_FLUSH_RX); // flush Rx
|
||||
value = ReadStatus(); // read register STATUS's value
|
||||
WriteReg(BK_WRITE_REG|BK_STATUS, value); // clear RX_DR or TX_DS or MAX_RT interrupt flag
|
||||
|
||||
BEKEN_CE_LOW();
|
||||
DelayCE();
|
||||
value = ReadReg(BK_CONFIG); // read register CONFIG's value
|
||||
value |= BK_CONFIG_PRIM_RX; // set bit 0
|
||||
value |= BK_CONFIG_PWR_UP;
|
||||
WriteReg(BK_WRITE_REG | BK_CONFIG, value); // Set PWR_UP bit, enable CRC(2 length) & Prim:RX. RX_DR enabled..
|
||||
|
||||
BEKEN_CE_HIGH();
|
||||
//BEKEN_PA_LOW(); // we dont have a PA on the RX side
|
||||
bkMode = BKRADIO_RX;
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
// switch to Tx mode
|
||||
void Radio_Beken::SwitchToTxMode(void)
|
||||
{
|
||||
uint8_t value;
|
||||
Strobe(BK_FLUSH_TX); // flush half-sent Tx
|
||||
Strobe(BK_FLUSH_RX); // flush half-received rx
|
||||
|
||||
// BEKEN_PA_HIGH();
|
||||
BEKEN_CE_LOW();
|
||||
DelayCE();
|
||||
value = ReadReg(BK_CONFIG); // read register CONFIG's value
|
||||
value &= ~BK_CONFIG_PRIM_RX; // Clear bit 0 (PTX)
|
||||
value |= BK_CONFIG_PWR_UP;
|
||||
WriteReg(BK_WRITE_REG | BK_CONFIG, value); // Set PWR_UP bit, enable CRC(2 length) & Prim:RX. RX_DR enabled.
|
||||
// BEKEN_CE_HIGH();
|
||||
bkMode = BKRADIO_TX;
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
// switch to Idle mode
|
||||
void Radio_Beken::SwitchToIdleMode(void)
|
||||
{
|
||||
Strobe(BK_FLUSH_TX); // flush Tx
|
||||
Strobe(BK_FLUSH_RX); // flush Rx
|
||||
|
||||
BEKEN_PA_LOW();
|
||||
BEKEN_CE_LOW();
|
||||
DelayCE();
|
||||
bkMode = BKRADIO_IDLE;
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
// Switch to Sleep mode
|
||||
void Radio_Beken::SwitchToSleepMode(void)
|
||||
{
|
||||
uint8_t value;
|
||||
Strobe(BK_FLUSH_RX); // flush Rx
|
||||
Strobe(BK_FLUSH_TX); // flush Tx
|
||||
value = ReadStatus(); // read register STATUS's value
|
||||
WriteReg(BK_WRITE_REG|BK_STATUS, value); // clear RX_DR or TX_DS or MAX_RT interrupt flag
|
||||
|
||||
BEKEN_PA_LOW();
|
||||
BEKEN_CE_LOW();
|
||||
DelayCE();
|
||||
value = ReadReg(BK_CONFIG); // read register CONFIG's value
|
||||
value |= BK_CONFIG_PRIM_RX; // Receive mode
|
||||
value &= ~BK_CONFIG_PWR_UP; // Power down
|
||||
WriteReg(BK_WRITE_REG | BK_CONFIG, value); // Clear PWR_UP bit, enable CRC(2 length) & Prim:RX. RX_DR enabled..
|
||||
// Stay low
|
||||
BEKEN_CE_LOW();
|
||||
bkMode = BKRADIO_SLEEP;
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
void Radio_Beken::InitBank0Registers(ITX_SPEED spd)
|
||||
{
|
||||
int8_t i;
|
||||
|
||||
//********************Write Bank0 register******************
|
||||
for (i=20; i >= 0; i--) { // From BK_FIFO_STATUS back to beginning of table
|
||||
uint8_t idx = Bank0_Reg[i][0];
|
||||
uint8_t value = Bank0_Reg[i][1];
|
||||
if (idx == BK_RF_SETUP) { // Adjust for speed
|
||||
value = Bank0_Reg6[spd][1];
|
||||
}
|
||||
WriteReg((BK_WRITE_REG|idx), value);
|
||||
}
|
||||
|
||||
// Enable features
|
||||
i = ReadReg(BK_FEATURE);
|
||||
if (i == 0) { // i!=0 showed that chip has been actived. So do not active again (as that would toggle these features off again).
|
||||
WriteReg(BK_ACTIVATE_CMD,0x73); // Activate the BK_FEATURE register. (This command must NOT have BK_WRITE_REG set)
|
||||
}
|
||||
for (i = 22; i >= 21; i--) {
|
||||
WriteReg((BK_WRITE_REG|Bank0_Reg[i][0]),Bank0_Reg[i][1]);
|
||||
}
|
||||
|
||||
// Set the various 5 byte addresses
|
||||
WriteRegisterMulti((BK_WRITE_REG|BK_RX_ADDR_P0),RX0_Address,5); // reg 10 - Rx0 addr
|
||||
WriteRegisterMulti((BK_WRITE_REG|BK_RX_ADDR_P1),RX1_Address,5); // REG 11 - Rx1 addr
|
||||
WriteRegisterMulti((BK_WRITE_REG|BK_TX_ADDR),TX_Address,5); // REG 16 - TX addr
|
||||
WriteReg(BK_WRITE_REG|BK_EN_RXADDR, 0x03);
|
||||
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
void Radio_Beken::InitBank1Registers(ITX_SPEED spd)
|
||||
{
|
||||
int16_t i;
|
||||
|
||||
for (i = IREG1_4; i <= IREG1_13; i++) {
|
||||
const uint8_t* p = &Bank1_RegTable[spd][i][0];
|
||||
uint8_t idx = *p++;
|
||||
WriteRegisterMulti((BK_WRITE_REG|idx), p, 4);
|
||||
}
|
||||
WriteRegisterMulti((BK_WRITE_REG|BK2425_R1_14),&(Bank1_Reg14[0]),11);
|
||||
|
||||
//toggle REG4<25,26>
|
||||
{
|
||||
const uint8_t* p = &Bank1_RegTable[spd][IREG1_4A][0];
|
||||
uint8_t idx = *p++;
|
||||
WriteRegisterMulti((BK_WRITE_REG|idx), p, 4);
|
||||
}
|
||||
{
|
||||
const uint8_t* p = &Bank1_RegTable[spd][IREG1_4][0];
|
||||
uint8_t idx = *p++;
|
||||
WriteRegisterMulti((BK_WRITE_REG|idx), p, 4);
|
||||
}
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
// Set the rx and tx addresses
|
||||
void Radio_Beken::SetAddresses(const uint8_t* txaddr)
|
||||
{
|
||||
TX_Address[1] = RX0_Address[1] = txaddr[1];
|
||||
TX_Address[3] = RX0_Address[3] = txaddr[3];
|
||||
TX_Address[4] = RX0_Address[4] = txaddr[4];
|
||||
WriteRegisterMulti((BK_WRITE_REG|BK_RX_ADDR_P0), RX0_Address, 5);
|
||||
WriteRegisterMulti((BK_WRITE_REG|BK_TX_ADDR), TX_Address, 5);
|
||||
WriteReg(BK_WRITE_REG|BK_EN_RXADDR, 0x03);
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
bool Radio_Beken::ClearAckOverflow(void)
|
||||
{
|
||||
uint8_t status = ReadStatus();
|
||||
if ((BK_STATUS_MAX_RT & status) == 0) {
|
||||
return false;
|
||||
} else {
|
||||
WriteReg((BK_WRITE_REG|BK_STATUS), BK_STATUS_MAX_RT);
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
// Write a data packet
|
||||
bool Radio_Beken::SendPacket(uint8_t type, ///< WR_TX_PLOAD or W_TX_PAYLOAD_NOACK_CMD
|
||||
const uint8_t* pbuf, ///< a buffer pointer
|
||||
uint8_t len) ///< packet length in bytes
|
||||
{
|
||||
uint8_t fifo_sta = ReadReg(BK_FIFO_STATUS); // read register FIFO_STATUS's value
|
||||
bool returnValue = ClearAckOverflow();
|
||||
|
||||
if (!(fifo_sta & BK_FIFO_STATUS_TX_FULL)) { // if not full, send data
|
||||
numTxPackets++;
|
||||
WriteRegisterMulti(type, pbuf, len); // Writes data to buffer A0,B0,A8
|
||||
BEKEN_CE_HIGH(); // Wait until FIFO has the data before sending it.
|
||||
}
|
||||
return returnValue;
|
||||
}
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
// For debugging - tell us the current beken register values (from bank 0)
|
||||
// This just prints it to the UART rather than to the console over WiFi
|
||||
void Radio_Beken::DumpRegisters(void)
|
||||
{
|
||||
uint8_t i;
|
||||
for (i = 0; i <= BK_FEATURE; ++i) {
|
||||
uint8_t len = 1;
|
||||
switch (i) {
|
||||
case 10: case 11: case 16: len = 5; break;
|
||||
case 24: case 25: case 26: case 27: len = 0; break;
|
||||
default: len = 1; break;
|
||||
};
|
||||
if (len == 1) {
|
||||
//printf("Bank0reg%d : %x\r\n", i, ReadReg(i));
|
||||
} else if (len == 5) {
|
||||
uint8_t data[5];
|
||||
ReadRegisterMulti(i, &data[0], len);
|
||||
//printf("Bank0reg%d : %x %x %x %x %x\r\n", i, data[0], data[1], data[2], data[3], data[4]);
|
||||
}
|
||||
}
|
||||
SetRBank(1);
|
||||
for (i = IREG1_4; i <= IREG1_13; ++i) {
|
||||
uint8_t len = 4;
|
||||
uint8_t data[4];
|
||||
ReadRegisterMulti(i, &data[0], len);
|
||||
//uint8_t idx = Bank1_RegTable[0][i][0];
|
||||
//printf("Bank1reg%d : %x %x %x %x\r\n", idx, data[0], data[1], data[2], data[3]);
|
||||
}
|
||||
SetRBank(0);
|
||||
}
|
||||
|
||||
#endif // HAL_RCINPUT_WITH_AP_RADIO
|
439
libraries/AP_Radio/driver_bk2425.h
Normal file
439
libraries/AP_Radio/driver_bk2425.h
Normal file
@ -0,0 +1,439 @@
|
||||
//----------------------------------------------------------------------------------
|
||||
// low level driver for the Beken BK2425 radio on SPI
|
||||
// Note: Under ChiBios the knowledge of which pins are which is not in the driver.
|
||||
// But ultimately comes from hwdef.dat
|
||||
//----------------------------------------------------------------------------------
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <AP_HAL/AP_HAL.h>
|
||||
|
||||
#if defined(HAL_RCINPUT_WITH_AP_RADIO) && CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_CHIBIOS_SKYVIPER_F412
|
||||
|
||||
#define SUPPORT_BK_DEBUG_PINS 0 // 0=UART6 is for GPS, 1=UART6 is debug gpio
|
||||
#define TX_SPEED 250u // Default transmit speed in kilobits per second.
|
||||
|
||||
/** Channel hopping parameters. Values are in MHz from 2400Mhz. */
|
||||
enum CHANNEL_MHZ_e {
|
||||
CHANNEL_MIN_PHYSICAL = 0, ///< Minimum physical channel that is possible
|
||||
CHANNEL_MAX_PHYSICAL = 83, ///< Maximum physical channel that is possible
|
||||
CHANNEL_FCC_LOW = 10, ///< Minimum physical channel that will pass the FCC tests
|
||||
CHANNEL_FCC_HIGH = 72, ///< Maximum physical channel that will pass the FCC tests
|
||||
CHANNEL_FCC_MID = 41, ///< A representative physical channel
|
||||
};
|
||||
|
||||
enum {
|
||||
CHANNEL_COUNT_LOGICAL = 16, ///< The maximum number of entries in each frequency table
|
||||
CHANNEL_BASE_TABLE = 0, ///< The table used for non wifi boards
|
||||
CHANNEL_SAFE_TABLE = 3, ///< A table that will receive packets even if wrong
|
||||
CHANNEL_NUM_TABLES = 6, ///< The number of tables
|
||||
CHANNEL_COUNT_TEST = 16, ///< The number of test mode tables
|
||||
};
|
||||
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
// Packet format definition
|
||||
// ----------------------------------------------------------------------------
|
||||
|
||||
/** The type of packets being sent between controller and drone */
|
||||
enum BK_PKT_TYPE_E {
|
||||
BK_PKT_TYPE_INVALID = 0, ///< Invalid packet from empty packets or bad CRC
|
||||
BK_PKT_TYPE_CTRL_FOUND = 0x10, ///< (Tx->Drone) User control - known receiver
|
||||
BK_PKT_TYPE_CTRL_LOST = 0x11, ///< (Tx->Drone) User control - unknown receiver
|
||||
BK_PKT_TYPE_BIND_AUTO = 0x12, ///< (Tx->Drone) Tell drones this tx is broadcasting
|
||||
BK_PKT_TYPE_TELEMETRY = 0x13, ///< (Drone->Tx) Send telemetry to tx
|
||||
BK_PKT_TYPE_DFU = 0x14, ///< (Drone->Tx) Send new firmware to tx
|
||||
BK_PKT_TYPE_BIND_MANUAL = 0x15, ///< (Tx->Drone) Tell drones this tx is broadcasting
|
||||
BK_PKT_TYPE_TUNE = 0x16, ///< (Drone->Tx) Send musical tune to tx
|
||||
};
|
||||
typedef uint8_t BK_PKT_TYPE;
|
||||
|
||||
/** A bitset of the buttons on this controller. */
|
||||
enum button_bits {
|
||||
BUTTON_NONE = 0x00, ///< No buttons are held
|
||||
BUTTON_RIGHT = 0x01, ///< SW1 = The right button (mode)
|
||||
BUTTON_LEFT = 0x02, ///< SW2 = The left button (launch/land)
|
||||
BUTTON_MIDDLE = 0x04, ///< SW3 = The middle button (GPS)
|
||||
BUTTON_LEFT_SHOULDER = 0x08, ///< SW4 = The left shoulder button (stunt)
|
||||
BUTTON_RIGHT_SHOULDER = 0x10, ///< SW5 = The right shoulder button (video)
|
||||
BUTTON_POWER = 0x20, ///< SW6 = The top button (POWER)
|
||||
};
|
||||
|
||||
/** The type of info being sent in control packets */
|
||||
enum BK_INFO_TYPE_E {
|
||||
BK_INFO_MIN = 1,
|
||||
BK_INFO_FW_VER = 1,
|
||||
BK_INFO_DFU_RX = 2,
|
||||
BK_INFO_FW_CRC_LO = 3,
|
||||
BK_INFO_FW_CRC_HI = 4,
|
||||
BK_INFO_FW_YM = 5,
|
||||
BK_INFO_FW_DAY = 6,
|
||||
BK_INFO_MODEL = 7,
|
||||
BK_INFO_PPS = 8,
|
||||
BK_INFO_BATTERY = 9,
|
||||
BK_INFO_COUNTDOWN = 10,
|
||||
BK_INFO_HOPPING0 = 11,
|
||||
BK_INFO_HOPPING1 = 12,
|
||||
BK_INFO_DRONEID0 = 13,
|
||||
BK_INFO_DRONEID1 = 14,
|
||||
BK_INFO_MAX
|
||||
};
|
||||
typedef uint8_t BK_INFO_TYPE;
|
||||
|
||||
/** Data for packets that are not droneid packets
|
||||
Onair order = little-endian */
|
||||
typedef struct packetDataDeviceCtrl_s {
|
||||
uint8_t roll; ///< 2: Low 8 bits of the roll joystick
|
||||
uint8_t pitch; ///< 3: Low 8 bits of the pitch joystick
|
||||
uint8_t throttle; ///< 4: Low 8 bits of the throttle joystick
|
||||
uint8_t yaw; ///< 5: Low 8 bits of the yaw joystick
|
||||
uint8_t msb; ///< 6: High 2 bits of roll (7..6), pitch (5..4), throttle (3..2), yaw (1..0)
|
||||
uint8_t buttons_held; ///< 7: The buttons
|
||||
uint8_t buttons_toggled; ///< 8: The buttons
|
||||
uint8_t data_type; ///< 9: Type of extra data being sent
|
||||
uint8_t data_value_lo; ///< 10: Value of extra data being sent
|
||||
uint8_t data_value_hi; ///< 11: Value of extra data being sent
|
||||
} packetDataDeviceCtrl;
|
||||
|
||||
enum { SZ_ADDRESS = 5 }; ///< Size of address for transmission packets (40 bits)
|
||||
enum { SZ_CRC_GUID = 4 }; ///< Size of UUID for drone (32 bits)
|
||||
enum { SZ_DFU = 16 }; ///< Size of DFU packets
|
||||
|
||||
/** Data for packets that are binding packets
|
||||
Onair order = little-endian */
|
||||
typedef struct packetDataDeviceBind_s {
|
||||
uint8_t bind_address[SZ_ADDRESS]; ///< The address being used by control packets
|
||||
uint8_t hopping; ///< The hopping table in use for this connection
|
||||
uint8_t droneid[SZ_CRC_GUID]; ///<
|
||||
} packetDataDeviceBind;
|
||||
|
||||
/** Data structure for data packet transmitted from device (controller) to host (drone) */
|
||||
typedef struct packetDataDevice_s {
|
||||
BK_PKT_TYPE packetType; ///< 0: The packet type
|
||||
uint8_t channel; ///< 1: Next channel I will broadcast on
|
||||
union packetDataDevice_u { ///< The variant part of the packets
|
||||
packetDataDeviceCtrl ctrl; ///< Control packets
|
||||
packetDataDeviceBind bind; ///< Binding packets
|
||||
} u;
|
||||
} packetFormatRx;
|
||||
|
||||
/** Data structure for data packet transmitted from host (drone) to device (controller) */
|
||||
typedef struct packetDataDrone_s {
|
||||
BK_PKT_TYPE packetType; ///< 0: The packet type
|
||||
uint8_t channel; ///< 1: Next channel I will broadcast on
|
||||
uint8_t pps; ///< 2: Packets per second the drone received
|
||||
uint8_t flags; ///< 3: Flags
|
||||
uint8_t droneid[SZ_CRC_GUID]; ///< 4...7: CRC of the drone
|
||||
uint8_t flight_mode; ///< 8:
|
||||
uint8_t wifi; ///< 9: Wifi channel + 24 * tx power.
|
||||
uint8_t note_adjust; ///< 10: note adjust for the tx buzzer (should this be sent so often?)
|
||||
uint8_t hopping; ///< 11: The adapative hopping byte we want to use
|
||||
} packetFormatTx;
|
||||
|
||||
typedef struct packetDataDfu_s {
|
||||
BK_PKT_TYPE packetType; ///< 0: The packet type
|
||||
uint8_t channel; ///< 1: Next channel I will broadcast on
|
||||
uint8_t address_lo; ///< 2:
|
||||
uint8_t address_hi; ///< 3:
|
||||
uint8_t data[SZ_DFU]; ///< 4...19:
|
||||
} packetFormatDfu;
|
||||
|
||||
|
||||
// ----------------------------------------------------------------------------
|
||||
// BK2425 chip definition
|
||||
// ----------------------------------------------------------------------------
|
||||
|
||||
|
||||
//----------------------------------------------------------------------------------
|
||||
/** SPI register commands for the BK2425 and nrf24L01+ chips */
|
||||
typedef enum {
|
||||
// General commands
|
||||
BK_REG_MASK = 0x1F, // The range of registers that can be read and written
|
||||
BK_READ_REG = 0x00, // Define read command to register (0..1F)
|
||||
BK_WRITE_REG = 0x20, // Define write command to register (0..1F)
|
||||
BK_ACTIVATE_CMD = 0x50,
|
||||
BK_R_RX_PL_WID_CMD = 0x60,
|
||||
BK_RD_RX_PLOAD = 0x61, // Define RX payload register address
|
||||
BK_WR_TX_PLOAD = 0xA0, // Define TX payload register address
|
||||
BK_W_ACK_PAYLOAD_CMD = 0xA8, // (nrf: +pipe 0..7)
|
||||
BK_W_TX_PAYLOAD_NOACK_CMD = 0xB0,
|
||||
BK_FLUSH_TX = 0xE1, // Define flush TX register command
|
||||
BK_FLUSH_RX = 0xE2, // Define flush RX register command
|
||||
BK_REUSE_TX_PL = 0xE3, // Define reuse TX payload register command
|
||||
BK_NOP = 0xFF, // Define No Operation, might be used to read status register
|
||||
|
||||
// BK2425 bank 0 register addresses
|
||||
BK_CONFIG = 0x00, // 'Config' register address
|
||||
BK_EN_AA = 0x01, // 'Enable Auto Acknowledgment' register address
|
||||
BK_EN_RXADDR = 0x02, // 'Enabled RX addresses' register address
|
||||
BK_SETUP_AW = 0x03, // 'Setup address width' register address
|
||||
BK_SETUP_RETR = 0x04, // 'Setup Auto. Retrans' register address
|
||||
BK_RF_CH = 0x05, // 'RF channel' register address
|
||||
BK_RF_SETUP = 0x06, // 'RF setup' register address
|
||||
BK_STATUS = 0x07, // 'Status' register address
|
||||
BK_OBSERVE_TX = 0x08, // 'Observe TX' register address (lost packets, retransmitted packets on this frequency)
|
||||
BK_CD = 0x09, // 'Carrier Detect' register address
|
||||
BK_RX_ADDR_P0 = 0x0A, // 'RX address pipe0' register address (5 bytes)
|
||||
BK_RX_ADDR_P1 = 0x0B, // 'RX address pipe1' register address (5 bytes)
|
||||
BK_RX_ADDR_P2 = 0x0C, // 'RX address pipe2' register address (1 byte)
|
||||
BK_RX_ADDR_P3 = 0x0D, // 'RX address pipe3' register address (1 byte)
|
||||
BK_RX_ADDR_P4 = 0x0E, // 'RX address pipe4' register address (1 byte)
|
||||
BK_RX_ADDR_P5 = 0x0F, // 'RX address pipe5' register address (1 byte)
|
||||
BK_TX_ADDR = 0x10, // 'TX address' register address (5 bytes)
|
||||
BK_RX_PW_P0 = 0x11, // 'RX payload width, pipe0' register address
|
||||
BK_RX_PW_P1 = 0x12, // 'RX payload width, pipe1' register address
|
||||
BK_RX_PW_P2 = 0x13, // 'RX payload width, pipe2' register address
|
||||
BK_RX_PW_P3 = 0x14, // 'RX payload width, pipe3' register address
|
||||
BK_RX_PW_P4 = 0x15, // 'RX payload width, pipe4' register address
|
||||
BK_RX_PW_P5 = 0x16, // 'RX payload width, pipe5' register address
|
||||
BK_FIFO_STATUS = 0x17, // 'FIFO Status Register' register address
|
||||
BK_DYNPD = 0x1c, // 'Enable dynamic payload length' register address
|
||||
BK_FEATURE = 0x1d, // 'Feature' register address
|
||||
BK_PAYLOAD_WIDTH = 0x1f, // 'payload length of 256 bytes modes register address
|
||||
|
||||
// BK2425 bank 1 register addresses
|
||||
BK2425_R1_4 = 0x04,
|
||||
BK2425_R1_5 = 0x05,
|
||||
BK2425_R1_WHOAMI = 0x08, // Register to read that contains the chip id
|
||||
BK2425_R1_12 = 0x0C, // PLL speed 120 or 130us
|
||||
BK2425_R1_13 = 0x0D,
|
||||
BK2425_R1_14 = 0x0E,
|
||||
} BK_SPI_CMD;
|
||||
|
||||
//----------------------------------------------------------------------------------
|
||||
// Chip Status Byte
|
||||
//----------------------------------------------------------------------------------
|
||||
|
||||
enum {
|
||||
BK_CHIP_ID_BK2425 = 0x63, // The expected value of reading BK2425_R1_WHOAMI
|
||||
};
|
||||
|
||||
// Meanings of the BK_STATUS register
|
||||
enum {
|
||||
BK_STATUS_RBANK = 0x80, // Register bank 1 is in use
|
||||
BK_STATUS_RX_DR = 0x40, // Data ready
|
||||
BK_STATUS_TX_DS = 0x20, // Data sent
|
||||
BK_STATUS_MAX_RT = 0x10, // Max retries failed
|
||||
BK_STATUS_RX_MASK = 0x0E, // Mask for the receptions bit
|
||||
BK_STATUS_RX_EMPTY = 0x0E,
|
||||
BK_STATUS_RX_P_5 = 0x0A, // Data pipe 5 has some data ready
|
||||
BK_STATUS_RX_P_4 = 0x08, // Data pipe 4 has some data ready
|
||||
BK_STATUS_RX_P_3 = 0x06, // Data pipe 3 has some data ready
|
||||
BK_STATUS_RX_P_2 = 0x04, // Data pipe 2 has some data ready
|
||||
BK_STATUS_RX_P_1 = 0x02, // Data pipe 1 has some data ready
|
||||
BK_STATUS_RX_P_0 = 0x00, // Data pipe 0 has some data ready
|
||||
BK_STATUS_TX_FULL = 0x01 // Tx buffer full
|
||||
};
|
||||
|
||||
// Meanings of the FIFO_STATUS register
|
||||
enum {
|
||||
BK_FIFO_STATUS_TX_REUSE = 0x40,
|
||||
BK_FIFO_STATUS_TX_FULL = 0x20,
|
||||
BK_FIFO_STATUS_TX_EMPTY = 0x10,
|
||||
BK_FIFO_STATUS_RX_FULL = 0x02,
|
||||
BK_FIFO_STATUS_RX_EMPTY = 0x01
|
||||
};
|
||||
|
||||
// Meanings of the BK_CONFIG register
|
||||
enum {
|
||||
BK_CONFIG_MASK_RX_DR = 0x40, // Mask interrupt caused by RX_DR
|
||||
BK_CONFIG_MASK_TX_DS = 0x20, // Mask interrupt caused by TX_DS
|
||||
BK_CONFIG_MASK_MAX_RT = 0x10, // Mask interrupt caused by MAX_RT
|
||||
BK_CONFIG_EN_CRC = 0x08, // Enable CRC. Forced high if one of the bits in the EN_AA is high
|
||||
BK_CONFIG_CRCO = 0x04, // CRC encoding scheme (0=8 bits, 1=16 bits)
|
||||
BK_CONFIG_PWR_UP = 0x02, // POWER UP
|
||||
BK_CONFIG_PRIM_RX = 0x01, // Receive/transmit
|
||||
};
|
||||
|
||||
enum {
|
||||
BK_FEATURE_EN_DPL = 0x04, //
|
||||
BK_FEATURE_EN_ACK_PAY = 0x02, //
|
||||
BK_FEATURE_EN_DYN_ACK = 0x01, //
|
||||
};
|
||||
|
||||
// (Lets make it one radio interface for both projects)
|
||||
|
||||
/** The baud rate of the GFSK modulation */
|
||||
typedef enum ITX_SPEED_e {
|
||||
ITX_250, ///< 250kbps (slowest but furthest range)
|
||||
ITX_1000, ///< 1000kbps (balanced)
|
||||
ITX_2000, ///< 2000kbps (fastest hence least congested)
|
||||
ITX_CARRIER, ///< 0kbps (constant carrier wave)
|
||||
ITX_MAX
|
||||
} ITX_SPEED;
|
||||
|
||||
enum {
|
||||
PACKET_LENGTH_RX_CTRL = 12,
|
||||
PACKET_LENGTH_RX_BIND = 12,
|
||||
PACKET_LENGTH_RX_MAX = 12,
|
||||
PACKET_LENGTH_TX_TELEMETRY = 12,
|
||||
PACKET_LENGTH_TX_DFU = 20,
|
||||
PACKET_LENGTH_TX_MAX = 20,
|
||||
};
|
||||
|
||||
// Note that bank 1 registers 0...8 are MSB first; others are LSB first
|
||||
|
||||
#define PLL_SPEED { BK2425_R1_12, 0x00,0x12,0x73,0x05 } // 0x00127305ul, // PLL locking time 130us compatible with nRF24L01;
|
||||
|
||||
// In the array Bank1_Reg0_13[],all the register values are the byte reversed!
|
||||
enum {
|
||||
IREG1_4,
|
||||
IREG1_5,
|
||||
IREG1_12,
|
||||
IREG1_13,
|
||||
IREG1_4A,
|
||||
IREG_MAX
|
||||
};
|
||||
|
||||
#define BK_MAX_PACKET_LEN 32 // max value is 32 bytes
|
||||
#define BK_RCV_TIMEOUT 30
|
||||
|
||||
//----------------------------------------------------------------------------------
|
||||
// Translate output power into a number
|
||||
// must match up with the table RegPower[]
|
||||
#define OUTPUT_POWER_REG6_0 0 // -25dB
|
||||
#define OUTPUT_POWER_REG6_1 0 // -18dB
|
||||
#define OUTPUT_POWER_REG6_2 1 // -18dB
|
||||
#define OUTPUT_POWER_REG6_3 1 // -12dB
|
||||
#define OUTPUT_POWER_REG6_4 1 // -12dB
|
||||
#define OUTPUT_POWER_REG6_5 2 // -7dB
|
||||
#define OUTPUT_POWER_REG6_6 3 // -1dB
|
||||
#define OUTPUT_POWER_REG6_7 3 // +4dB
|
||||
|
||||
// Register 4 in bank 1 only applies to Beken chip
|
||||
#define OUTPUT_POWER_REG4_0 0 // -25dB
|
||||
#define OUTPUT_POWER_REG4_1 3 // -18dB
|
||||
#define OUTPUT_POWER_REG4_2 0 // -18dB
|
||||
#define OUTPUT_POWER_REG4_3 3 // -12dB
|
||||
#define OUTPUT_POWER_REG4_4 2 // -12dB
|
||||
#define OUTPUT_POWER_REG4_5 0 // -7dB
|
||||
#define OUTPUT_POWER_REG4_6 0 // -1dB
|
||||
#define OUTPUT_POWER_REG4_7 7 // +4dB
|
||||
|
||||
// Generic support
|
||||
#define TOKENPASTE(x, y) x ## y
|
||||
#define TOKENPASTE2(x, y) TOKENPASTE(x, y)
|
||||
// The default register values that are for the default power setting
|
||||
#define DEFAULT_OUTPUT_REG6 TOKENPASTE2(OUTPUT_POWER_REG6_,DEFAULT_OUTPUT_POWER)
|
||||
#define DEFAULT_OUTPUT_REG4 TOKENPASTE2(OUTPUT_POWER_REG4_,DEFAULT_OUTPUT_POWER)
|
||||
|
||||
// This assumes we are using ChiBios instead of the pixhawk o/s for accessing GPIO
|
||||
#if CONFIG_HAL_BOARD_SUBTYPE == HAL_BOARD_SUBTYPE_CHIBIOS_SKYVIPER_F412
|
||||
#define BEKEN_SELECT() (dev->set_chip_select(true))
|
||||
#define BEKEN_DESELECT() (dev->set_chip_select(false))
|
||||
#define BEKEN_CE_HIGH() (palSetLine(HAL_GPIO_PIN_RADIO_CE)) // (hal.gpio->write(HAL_CHIBIOS_GPIO_RADIO_CE, 1))
|
||||
#define BEKEN_CE_LOW() (palClearLine(HAL_GPIO_PIN_RADIO_CE)) // (hal.gpio->write(HAL_CHIBIOS_GPIO_RADIO_CE, 0))
|
||||
#define BEKEN_PA_HIGH() (palSetLine(HAL_GPIO_PIN_RADIO_PA_CTL)) // (hal.gpio->write(HAL_CHIBIOS_GPIO_RADIO_PA_CTL, 1))
|
||||
#define BEKEN_PA_LOW() (palClearLine(HAL_GPIO_PIN_RADIO_PA_CTL)) // (hal.gpio->write(HAL_CHIBIOS_GPIO_RADIO_PA_CTL, 0))
|
||||
#if SUPPORT_BK_DEBUG_PINS
|
||||
#define DEBUG1_HIGH() (palSetLine(HAL_GPIO_PIN_DEBUG1))
|
||||
#define DEBUG1_LOW() (palClearLine(HAL_GPIO_PIN_DEBUG1))
|
||||
#define DEBUG2_HIGH() (palSetLine(HAL_GPIO_PIN_DEBUG2))
|
||||
#define DEBUG2_LOW() (palClearLine(HAL_GPIO_PIN_DEBUG2))
|
||||
#else
|
||||
#define DEBUG1_HIGH() do {} while (0)
|
||||
#define DEBUG1_LOW() do {} while (0)
|
||||
#define DEBUG2_HIGH() do {} while (0)
|
||||
#define DEBUG2_LOW() do {} while (0)
|
||||
#endif
|
||||
#else
|
||||
#error This configuration is not supported.
|
||||
#endif
|
||||
|
||||
/** Parameters used by the fcc pretests */
|
||||
typedef struct FccParams_s {
|
||||
uint8_t fcc_mode; ///< The value (0..6) last set by the user that we are using. Non-zero iff we are sending test signals
|
||||
bool scan_mode; ///< true for scanning, false for fixed frequencies
|
||||
bool CW_mode; ///< true for carrier wave, false for packets
|
||||
bool disable_crc_mode; ///< false for CRCs enabled, true for CRCs ignored on reception
|
||||
uint8_t scan_count; ///< In scan mode, packet count before incrementing scan
|
||||
uint8_t channel; ///< Current frequency 8..70
|
||||
uint8_t power; ///< Current power 1..8
|
||||
bool disable_crc; ///< true=crc is physically disabled
|
||||
uint8_t factory_mode; ///< factory test mode 0..8
|
||||
bool enable_cd; ///< enable carrier detect
|
||||
bool last_cd; ///< last carrier detect on a packet received
|
||||
} FccParams;
|
||||
|
||||
typedef enum BkRadioMode_e {
|
||||
BKRADIO_SLEEP,
|
||||
BKRADIO_IDLE,
|
||||
BKRADIO_TX,
|
||||
BKRADIO_RX,
|
||||
BKRADIO_STANDBY1, // Not visible to the code yet
|
||||
BKRADIO_STANDBY2, // Not visible to the code yet
|
||||
} BkRadioMode;
|
||||
|
||||
//----------------------------------------------------------------------------------
|
||||
// BEKEN driver class
|
||||
class Radio_Beken
|
||||
{
|
||||
public:
|
||||
// Generic functions
|
||||
Radio_Beken(AP_HAL::OwnPtr<AP_HAL::SPIDevice> _dev);
|
||||
bool lock_bus(void)
|
||||
{
|
||||
return dev->get_semaphore()->take(HAL_SEMAPHORE_BLOCK_FOREVER);
|
||||
}
|
||||
void unlock_bus(void)
|
||||
{
|
||||
dev->get_semaphore()->give();
|
||||
}
|
||||
|
||||
// Raw SPI access functions
|
||||
void ReadRegisterMulti(uint8_t address, uint8_t *data, uint8_t len);
|
||||
void WriteRegisterMulti(uint8_t address, const uint8_t *data, uint8_t len);
|
||||
|
||||
// Low-level Beken functions
|
||||
uint8_t ReadStatus(void);
|
||||
uint8_t ReadReg(uint8_t reg);
|
||||
uint8_t Strobe(uint8_t address);
|
||||
void SetRBank(uint8_t bank);
|
||||
void WriteReg(uint8_t address, uint8_t data);
|
||||
void WriteRegisterMultiBank1(uint8_t address, const uint8_t *data, uint8_t len);
|
||||
|
||||
// High-level Beken functions
|
||||
void SetPower(uint8_t power);
|
||||
void SetChannel(uint8_t channel); // Sets the physical channel
|
||||
void SetCwMode(uint8_t cw);
|
||||
void SetCrcMode(uint8_t disable_crc); // non-zero means crc is ignored
|
||||
void SetFactoryMode(uint8_t factory);
|
||||
bool Reset(void);
|
||||
void SwitchToRxMode(void);
|
||||
void SwitchToTxMode(void);
|
||||
void SwitchToIdleMode(void);
|
||||
void SwitchToSleepMode(void);
|
||||
void InitBank0Registers(ITX_SPEED spd);
|
||||
void InitBank1Registers(ITX_SPEED spd);
|
||||
void SetAddresses(const uint8_t* txaddr); // Set the rx and tx addresses
|
||||
bool ClearAckOverflow(void);
|
||||
bool SendPacket(uint8_t type, const uint8_t* pbuf, uint8_t len);
|
||||
void DelayCE(void);
|
||||
void DumpRegisters(void);
|
||||
bool WasTxMode(void);
|
||||
bool WasRxMode(void);
|
||||
void ResetAddress(void);
|
||||
void EnableCarrierDetect(bool bEnable);
|
||||
bool CarrierDetect(void);
|
||||
|
||||
// Visible public variables (naughty)
|
||||
volatile uint8_t bkReady; // initialised in AP_Radio_bk2425.h radio_init() at the very end. Similar to a semaphore.
|
||||
static ITX_SPEED gTxSpeed;
|
||||
FccParams fcc;
|
||||
packetFormatTx pktDataTx; // Packet data to send (telemetry)
|
||||
packetFormatDfu pktDataDfu; // Packet data to send (DFU)
|
||||
uint8_t TX_Address[5]; // For sending telemetry and DFU
|
||||
|
||||
private:
|
||||
AP_HAL::OwnPtr<AP_HAL::SPIDevice> dev;
|
||||
uint8_t bFreshData; // Have we received a packet since we last processed one
|
||||
uint32_t numTxPackets;
|
||||
packetFormatRx pktDataRx; // Last valid packet that has been received
|
||||
packetFormatRx pktDataRecv; // Packet data in process of being received
|
||||
uint8_t lastTxChannel; // 0..CHANNEL_COUNT_LOGICAL
|
||||
uint8_t RX0_Address[5]; // The data address
|
||||
uint8_t RX1_Address[5]; // The fixed binding address
|
||||
BkRadioMode bkMode;
|
||||
};
|
||||
|
||||
#endif
|
@ -7,7 +7,7 @@
|
||||
#include "driver_cc2500.h"
|
||||
#include <utility>
|
||||
|
||||
// #pragma GCC optimize("O0")
|
||||
#pragma GCC optimize("O0")
|
||||
|
||||
extern const AP_HAL::HAL& hal;
|
||||
|
||||
@ -61,7 +61,7 @@ void Radio_CC2500::WriteReg(uint8_t address, uint8_t data)
|
||||
void Radio_CC2500::SetPower(uint8_t power)
|
||||
{
|
||||
const uint8_t patable[8] = {
|
||||
0xC5, // -12dbm
|
||||
0xC6, // -12dbm
|
||||
0x97, // -10dbm
|
||||
0x6E, // -8dbm
|
||||
0x7F, // -6dbm
|
||||
@ -73,7 +73,10 @@ void Radio_CC2500::SetPower(uint8_t power)
|
||||
if (power > 7) {
|
||||
power = 7;
|
||||
}
|
||||
if (power != last_power) {
|
||||
WriteReg(CC2500_3E_PATABLE, patable[power]);
|
||||
last_power = power;
|
||||
}
|
||||
}
|
||||
|
||||
bool Radio_CC2500::Reset(void)
|
||||
@ -95,6 +98,8 @@ void Radio_CC2500::WriteRegCheck(uint8_t address, uint8_t data)
|
||||
while (--tries) {
|
||||
dev->write_register(address, data);
|
||||
uint8_t v = ReadReg(address);
|
||||
if (v == data) break;
|
||||
if (v == data) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@ -87,21 +87,21 @@ enum {
|
||||
#define CC2500_SRES 0x30 // Reset chip.
|
||||
#define CC2500_SFSTXON \
|
||||
0x31 // Enable and calibrate frequency synthesizer (if MCSM0.FS_AUTOCAL=1).
|
||||
// If in RX/TX: Go to a wait state where only the synthesizer is
|
||||
// running (for quick RX / TX turnaround).
|
||||
// If in RX/TX: Go to a wait state where only the synthesizer is
|
||||
// running (for quick RX / TX turnaround).
|
||||
#define CC2500_SXOFF 0x32 // Turn off crystal oscillator.
|
||||
#define CC2500_SCAL 0x33 // Calibrate frequency synthesizer and turn it off
|
||||
// (enables quick start).
|
||||
// (enables quick start).
|
||||
#define CC2500_SRX \
|
||||
0x34 // Enable RX. Perform calibration first if coming from IDLE and
|
||||
// MCSM0.FS_AUTOCAL=1.
|
||||
// MCSM0.FS_AUTOCAL=1.
|
||||
#define CC2500_STX \
|
||||
0x35 // In IDLE state: Enable TX. Perform calibration first if
|
||||
// MCSM0.FS_AUTOCAL=1. If in RX state and CCA is enabled:
|
||||
// Only go to TX if channel is clear.
|
||||
// MCSM0.FS_AUTOCAL=1. If in RX state and CCA is enabled:
|
||||
// Only go to TX if channel is clear.
|
||||
#define CC2500_SIDLE \
|
||||
0x36 // Exit RX / TX, turn off frequency synthesizer and exit
|
||||
// Wake-On-Radio mode if applicable.
|
||||
// Wake-On-Radio mode if applicable.
|
||||
#define CC2500_SAFC 0x37 // Perform AFC adjustment of the frequency synthesizer
|
||||
#define CC2500_SWOR 0x38 // Start automatic RX polling sequence (Wake-on-Radio)
|
||||
#define CC2500_SPWD 0x39 // Enter power down mode when CSn goes high.
|
||||
@ -110,7 +110,7 @@ enum {
|
||||
#define CC2500_SWORRST 0x3C // Reset real time clock.
|
||||
#define CC2500_SNOP \
|
||||
0x3D // No operation. May be used to pad strobe commands to two
|
||||
// bytes for simpler software.
|
||||
// bytes for simpler software.
|
||||
//----------------------------------------------------------------------------------
|
||||
// Chip Status Byte
|
||||
//----------------------------------------------------------------------------------
|
||||
@ -137,7 +137,8 @@ enum {
|
||||
#define CC2500_LQI_EST_BM 0x7F
|
||||
|
||||
// CC2500 driver class
|
||||
class Radio_CC2500 {
|
||||
class Radio_CC2500
|
||||
{
|
||||
public:
|
||||
Radio_CC2500(AP_HAL::OwnPtr<AP_HAL::SPIDevice> _dev);
|
||||
|
||||
@ -154,13 +155,16 @@ public:
|
||||
void SetPower(uint8_t power);
|
||||
bool Reset(void);
|
||||
|
||||
bool lock_bus(void) {
|
||||
bool lock_bus(void)
|
||||
{
|
||||
return dev && dev->get_semaphore()->take(HAL_SEMAPHORE_BLOCK_FOREVER);
|
||||
}
|
||||
void unlock_bus(void) {
|
||||
void unlock_bus(void)
|
||||
{
|
||||
dev->get_semaphore()->give();
|
||||
}
|
||||
|
||||
private:
|
||||
AP_HAL::OwnPtr<AP_HAL::SPIDevice> dev;
|
||||
uint8_t last_power;
|
||||
};
|
||||
|
@ -24,7 +24,7 @@ struct PACKED telem_status {
|
||||
uint8_t pps; // packets per second received
|
||||
uint8_t rssi; // lowpass rssi
|
||||
uint8_t flags; // TELEM_FLAG_*
|
||||
uint8_t flight_mode; // flight mode
|
||||
uint8_t flight_mode; // flight mode (with profile number in high bit)
|
||||
uint8_t wifi_chan; // wifi channel number on Sonix
|
||||
uint8_t tx_max; // max TX power
|
||||
uint8_t note_adjust; // buzzer tone adjustment
|
||||
@ -35,7 +35,7 @@ struct PACKED telem_firmware {
|
||||
uint8_t seq;
|
||||
uint8_t len;
|
||||
uint16_t offset;
|
||||
uint8_t data[8];
|
||||
uint8_t data[4];
|
||||
};
|
||||
|
||||
/*
|
||||
@ -51,6 +51,20 @@ struct PACKED telem_packet_cypress {
|
||||
} payload;
|
||||
};
|
||||
|
||||
/*
|
||||
cc2500 specific protocol structures
|
||||
*/
|
||||
struct PACKED telem_status_cc2500 {
|
||||
uint8_t pps; // packets per second received
|
||||
uint8_t rssi; // lowpass rssi
|
||||
uint8_t flags; // TELEM_FLAG_*
|
||||
uint8_t flight_mode; // flight mode (with profile number in high bit)
|
||||
uint8_t wifi_chan; // wifi channel number on Sonix
|
||||
uint8_t tx_max; // max TX power
|
||||
uint8_t note_adjust; // buzzer tone adjustment
|
||||
uint8_t rxid[2]; // 16 bit ID for cc2500 to prevent double binds
|
||||
};
|
||||
|
||||
/*
|
||||
telemetry packet from RX to TX for cc2500
|
||||
*/
|
||||
@ -59,10 +73,23 @@ struct PACKED telem_packet_cc2500 {
|
||||
uint8_t type;
|
||||
uint8_t txid[2];
|
||||
union {
|
||||
uint8_t pkt[12];
|
||||
struct telem_status status;
|
||||
uint8_t pkt[9];
|
||||
struct telem_status_cc2500 status;
|
||||
struct telem_firmware fw;
|
||||
} payload;
|
||||
};
|
||||
|
||||
/*
|
||||
autobind packet from TX to RX for cc2500
|
||||
*/
|
||||
struct PACKED autobind_packet_cc2500 {
|
||||
uint8_t length;
|
||||
uint8_t magic1; // 0xC5
|
||||
uint8_t magic2; // 0xA2
|
||||
uint8_t txid[2];
|
||||
uint8_t txid_inverse[2];
|
||||
uint8_t wifi_chan;
|
||||
uint8_t pad[3]; // pad to 13 bytes for fixed packet length
|
||||
uint8_t crc[2];
|
||||
};
|
||||
|
||||
@ -79,11 +106,13 @@ enum packet_type {
|
||||
PKTYPE_TELEM_PPS = 5,
|
||||
PKTYPE_BL_VERSION = 6,
|
||||
PKTYPE_FW_ACK = 7,
|
||||
PKTYPE_NUM_TYPES = 8 // used for modulus
|
||||
PKTYPE_RXID1 = 8,
|
||||
PKTYPE_RXID2 = 9,
|
||||
PKTYPE_NUM_TYPES = 10 // used for modulus
|
||||
};
|
||||
|
||||
/*
|
||||
skyrocket specific packet for cc2500
|
||||
skyrocket specific packet from TX to RX for cc2500
|
||||
*/
|
||||
struct PACKED srt_packet {
|
||||
uint8_t length; // required for cc2500 FIFO
|
||||
@ -99,5 +128,4 @@ struct PACKED srt_packet {
|
||||
uint8_t buttons; // see channels.h
|
||||
uint8_t channr;
|
||||
uint8_t chanskip;
|
||||
uint8_t crc[2];
|
||||
};
|
||||
|
Loading…
Reference in New Issue
Block a user