mirror of
https://github.com/pascallanger/DIY-Multiprotocol-TX-Module.git
synced 2025-02-04 18:38:13 +00:00
4a626eaf14
Loads of protocols have been touched by this change. Some testing has been done but please test on all your models. The XN297 emulation selects in this order: - the CC2500 if it is available and bitrate=250K. Configure the option field automatically for RF tune. - the NRF for all bitrates if it is available - if NRF is not available and bitrate=1M then an invalid protocol is sent automatically to the radio. CC2500 @250K can now receive normal and enhanced payloads. OMP protocol supports telemetry on CC2500 and is also for NRF only modules including telemetry. Separation of E016H (new protocol) from E01X due to different structure. MJXQ, MT99XX, Q303 and XK: some sub protocols available on CC2500 only.
225 lines
6.4 KiB
C++
225 lines
6.4 KiB
C++
/*
|
|
This project 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.
|
|
|
|
Multiprotocol 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 Multiprotocol. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
// Compatible with X450 and X420/X520 plane.
|
|
|
|
#if defined(XK_CCNRF_INO)
|
|
|
|
#include "iface_nrf250k.h"
|
|
|
|
//#define FORCE_XK_ORIGINAL_ID
|
|
|
|
#define XK_INITIAL_WAIT 500
|
|
#define XK_PACKET_PERIOD 4000
|
|
#define XK_RF_BIND_NUM_CHANNELS 8
|
|
#define XK_RF_NUM_CHANNELS 4
|
|
#define XK_PAYLOAD_SIZE 16
|
|
#define XK_BIND_COUNT 750 //3sec
|
|
|
|
static uint16_t __attribute__((unused)) XK_convert_channel(uint8_t num)
|
|
{
|
|
uint16_t val;
|
|
if(num==RUDDER)
|
|
{// introduce deadband on rudder to prevent twitching
|
|
//debug("RUD:%d",val);
|
|
val=convert_channel_8b_limit_deadband(RUDDER,0x00,0x80, 0xFF, 40)<<2;
|
|
//debugln(",%d",val);
|
|
}
|
|
else
|
|
val=convert_channel_10b(num, false);
|
|
|
|
// 1FF..01=left, 00=center, 200..3FF=right
|
|
if(val==0x200)
|
|
val=0; // 0
|
|
else
|
|
if(val>0x200)
|
|
val--; // 200..3FE
|
|
else
|
|
{
|
|
val=0x200-val; // 200..01
|
|
if(val==0x200)
|
|
val--; // 1FF..01
|
|
}
|
|
return val;
|
|
}
|
|
|
|
static void __attribute__((unused)) XK_send_packet()
|
|
{
|
|
// RF channel
|
|
XN297_Hopping((IS_BIND_IN_PROGRESS?0:XK_RF_BIND_NUM_CHANNELS)+(hopping_frequency_no>>1));
|
|
hopping_frequency_no++;
|
|
if(hopping_frequency_no >= (IS_BIND_IN_PROGRESS?XK_RF_BIND_NUM_CHANNELS*2:XK_RF_NUM_CHANNELS*2))
|
|
hopping_frequency_no=0;
|
|
|
|
// Build packet
|
|
memset(packet,0x00,7);
|
|
memset(&packet[10],0x00,5);
|
|
|
|
packet[12]=0x40;
|
|
packet[13]=0x40;
|
|
if(IS_BIND_IN_PROGRESS)
|
|
packet[14] = 0xC0;
|
|
else
|
|
{
|
|
uint16_t val=convert_channel_10b(THROTTLE, false);
|
|
packet[0] = val>>2; // 0..255
|
|
packet[12] |= val & 2;
|
|
val=XK_convert_channel(RUDDER);
|
|
packet[1] = val>>2;
|
|
packet[12] |= (val & 2)<<2;
|
|
val=XK_convert_channel(ELEVATOR);
|
|
packet[2] = val>>2;
|
|
packet[13] |= val & 2;
|
|
val=XK_convert_channel(AILERON);
|
|
packet[3] = val>>2;
|
|
packet[13] |= (val & 2)<<2;
|
|
|
|
memset(&packet[4],0x40,3); // Trims
|
|
|
|
if(Channel_data[CH5] > CHANNEL_MAX_COMMAND)
|
|
packet[10] = 0x10; // V-Mode
|
|
else
|
|
if(Channel_data[CH5] > CHANNEL_MIN_COMMAND)
|
|
packet[10] = 0x04; // 6G-Mode
|
|
//0x00 default M-Mode
|
|
|
|
packet[10] |= GET_FLAG(CH7_SW,0x80); // Emergency stop momentary switch
|
|
|
|
packet[11] = GET_FLAG(CH8_SW,0x03) // 3D/6G momentary switch
|
|
|GET_FLAG(CH6_SW,0x40); // Take off momentary switch
|
|
packet[14] = GET_FLAG(CH9_SW,0x01) // Photo momentary switch
|
|
|GET_FLAG(CH10_SW,0x2); // Video momentary switch
|
|
//debugln("P1:%02X,P12:%02X",packet[1],packet[12]);
|
|
}
|
|
|
|
crc=packet[0];
|
|
for(uint8_t i=1; i<XK_PAYLOAD_SIZE-1;i++)
|
|
crc+=packet[i];
|
|
packet[15]=crc;
|
|
|
|
// debug("C: %02X, P:",hopping_frequency[rf_ch_num]);
|
|
// for(uint8_t i=0; i<XK_PAYLOAD_SIZE; i++)
|
|
// debug(" %02X",packet[i]);
|
|
// debugln("");
|
|
|
|
// Send
|
|
XN297_SetPower(); // Set tx_power
|
|
XN297_SetTxRxMode(TX_EN);
|
|
XN297_SetFreqOffset(); // Set frequency offset
|
|
XN297_WritePayload(packet, XK_PAYLOAD_SIZE);
|
|
}
|
|
|
|
const uint8_t PROGMEM XK_bind_hop[XK_RF_BIND_NUM_CHANNELS]= { 0x07, 0x24, 0x3E, 0x2B, 0x47, 0x0E, 0x39, 0x1C }; // Bind
|
|
|
|
const uint8_t PROGMEM XK_tx_addr[]= { 0xB3, 0x67, 0xE9, 0x98, 0x3A, 0xEC, 0xA6, 0x59, 0xB2, 0x94, 0x2B, 0xA5, 0x37, 0xC5, 0x4A, 0xD3,
|
|
0x49, 0xA6, 0x83, 0xEB, 0x4B, 0xC9, 0x59, 0xD2, 0x65, 0x34, 0x6A, 0xD3, 0x2C, 0x96, 0x2A, 0xA9,
|
|
0x32, 0xB2, 0xB4, 0x49, 0xD3, 0x37, 0xE9 };
|
|
|
|
const uint8_t PROGMEM XK_hop[]= { 0x47, 0x3A, 0x4C, 0x39, 0x4D, 0x34, 0x4A, 0x3F, 0x45, 0x3E, 0x4B, 0x3D, 0x3B, 0x48, 0x40, 0x49,
|
|
0x46, 0x3C, 0x43, 0x38, 0x35, 0x42, 0x33, 0x44, 0x4E, 0x37, 0x44, 0x35, 0x37, 0x4E, 0x36, 0x41 };
|
|
|
|
static void __attribute__((unused)) XK_initialize_txid()
|
|
{
|
|
//bind hop
|
|
for(uint8_t i=0; i<XK_RF_BIND_NUM_CHANNELS; i++)
|
|
hopping_frequency[i]=pgm_read_byte_near( &XK_bind_hop[i] );
|
|
|
|
//GID
|
|
packet[7]=rx_tx_addr[1];
|
|
packet[8]=rx_tx_addr[2];
|
|
packet[9]=rx_tx_addr[3];
|
|
uint8_t sum=packet[7]+packet[8]+packet[9];
|
|
// debugln("GID=%02X %02X %02X, sum=%d", packet[7],packet[8],packet[9],sum);
|
|
|
|
//Normal hop
|
|
uint8_t start=(sum&0x07)<<2;
|
|
// debug("start=%d, hop=",start);
|
|
for(uint8_t i=0; i<XK_RF_NUM_CHANNELS; i++)
|
|
{
|
|
hopping_frequency[ i + XK_RF_BIND_NUM_CHANNELS ]=pgm_read_byte_near( &XK_hop[ start + i ] );
|
|
// debug("%02X ", hopping_frequency[ i + XK_RF_BIND_NUM_CHANNELS ]);
|
|
}
|
|
// debugln("");
|
|
//Normal packet address
|
|
start=(sum&0x1F)+((sum>>5)&0x03);
|
|
// debug("start=%d, addr=",start);
|
|
for(uint8_t i=0; i<5; i++)
|
|
{
|
|
rx_tx_addr[i]=pgm_read_byte_near( &XK_tx_addr[ start + i ] );
|
|
// debug("%02X ", rx_tx_addr[ i ]);
|
|
}
|
|
// debugln("");
|
|
|
|
#ifdef FORCE_XK_ORIGINAL_ID
|
|
switch(RX_num%2)
|
|
{
|
|
default:
|
|
//TX1 X8 X450
|
|
//GID
|
|
packet[7]=0x04;
|
|
packet[8]=0x15;
|
|
packet[9]=0x22;
|
|
//Normal hop
|
|
memcpy(&hopping_frequency[XK_RF_BIND_NUM_CHANNELS],(uint8_t*)"\x3B\x48\x40\x49", XK_RF_NUM_CHANNELS); // freq and order verified
|
|
//Normal packet address
|
|
memcpy(rx_tx_addr,(uint8_t*)"\x2C\x96\x2A\xA9\x32",5);
|
|
break;
|
|
case 1:
|
|
//TX2 X4 X420
|
|
//GID
|
|
packet[7]=0x13;
|
|
packet[8]=0x24;
|
|
packet[9]=0x18;
|
|
//Normal hop
|
|
memcpy(&hopping_frequency[XK_RF_BIND_NUM_CHANNELS],(uint8_t*)"\x36\x41\x37\x4E", XK_RF_NUM_CHANNELS); // freq ok and order from xn297dump auto
|
|
//Normal packet address
|
|
memcpy(rx_tx_addr,(uint8_t*)"\xA6\x83\xEB\x4B\xC9",5);
|
|
break;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
static void __attribute__((unused)) XK_RF_init()
|
|
{
|
|
XN297_Configure(XN297_CRCEN, XN297_SCRAMBLED, sub_protocol==X420 ? XN297_1M : XN297_250K);
|
|
XN297_SetTXAddr((uint8_t*)"\x68\x94\xA6\xD5\xC3", 5); // Bind address
|
|
XN297_HoppingCalib(XK_RF_BIND_NUM_CHANNELS+XK_RF_NUM_CHANNELS); // Calibrate all channels
|
|
}
|
|
|
|
uint16_t XK_callback()
|
|
{
|
|
#ifdef MULTI_SYNC
|
|
telemetry_set_input_sync(XK_PACKET_PERIOD);
|
|
#endif
|
|
if(bind_counter)
|
|
if(--bind_counter==0)
|
|
{
|
|
BIND_DONE;
|
|
XN297_SetTXAddr(rx_tx_addr, 5); // Normal packets address
|
|
}
|
|
XK_send_packet();
|
|
return XK_PACKET_PERIOD;
|
|
}
|
|
|
|
void XK_init()
|
|
{
|
|
BIND_IN_PROGRESS; // Autobind protocol
|
|
XK_initialize_txid();
|
|
XK_RF_init();
|
|
hopping_frequency_no = 0;
|
|
bind_counter=XK_BIND_COUNT;
|
|
}
|
|
|
|
#endif
|