2020-01-14 12:19:12 +01:00
|
|
|
/*
|
|
|
|
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/>.
|
|
|
|
*/
|
2020-01-14 22:51:47 +01:00
|
|
|
// Compatible with X450 and X420/X520 plane.
|
2020-01-14 12:19:12 +01:00
|
|
|
|
|
|
|
#if defined(XK_NRF24L01_INO)
|
|
|
|
|
2020-02-17 11:44:53 +01:00
|
|
|
#include "iface_nrf250k.h"
|
2020-01-14 12:19:12 +01:00
|
|
|
|
2020-01-16 23:43:29 +01:00
|
|
|
//#define FORCE_XK_ORIGINAL_ID
|
2020-01-14 12:19:12 +01:00
|
|
|
|
|
|
|
#define XK_INITIAL_WAIT 500
|
2020-01-14 21:25:43 +01:00
|
|
|
#define XK_PACKET_PERIOD 4000
|
2020-01-14 12:19:12 +01:00
|
|
|
#define XK_RF_BIND_NUM_CHANNELS 8
|
|
|
|
#define XK_RF_NUM_CHANNELS 4
|
|
|
|
#define XK_PAYLOAD_SIZE 16
|
2020-01-14 22:51:47 +01:00
|
|
|
#define XK_BIND_COUNT 750 //3sec
|
2020-01-14 12:19:12 +01:00
|
|
|
|
|
|
|
static uint16_t __attribute__((unused)) XK_convert_channel(uint8_t num)
|
|
|
|
{
|
2020-11-30 13:36:29 +01:00
|
|
|
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
|
2021-01-09 18:39:31 +01:00
|
|
|
val=convert_channel_10b(num, false);
|
2020-11-30 13:36:29 +01:00
|
|
|
|
2020-01-14 12:19:12 +01:00
|
|
|
// 1FF..01=left, 00=center, 200..3FF=right
|
|
|
|
if(val==0x200)
|
2020-01-14 22:51:47 +01:00
|
|
|
val=0; // 0
|
2020-01-14 12:19:12 +01:00
|
|
|
else
|
|
|
|
if(val>0x200)
|
2020-01-14 22:51:47 +01:00
|
|
|
val--; // 200..3FE
|
2020-01-14 12:19:12 +01:00
|
|
|
else
|
|
|
|
{
|
2020-01-14 22:51:47 +01:00
|
|
|
val=0x200-val; // 200..01
|
2020-01-14 12:19:12 +01:00
|
|
|
if(val==0x200)
|
2020-01-14 22:51:47 +01:00
|
|
|
val--; // 1FF..01
|
2020-01-14 12:19:12 +01:00
|
|
|
}
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void __attribute__((unused)) XK_send_packet()
|
|
|
|
{
|
2020-01-14 22:51:47 +01:00
|
|
|
memset(packet,0x00,7);
|
|
|
|
memset(&packet[10],0x00,5);
|
|
|
|
|
2020-01-14 12:19:12 +01:00
|
|
|
packet[12]=0x40;
|
|
|
|
packet[13]=0x40;
|
|
|
|
if(IS_BIND_IN_PROGRESS)
|
|
|
|
packet[14] = 0xC0;
|
|
|
|
else
|
|
|
|
{
|
2021-01-09 18:39:31 +01:00
|
|
|
uint16_t val=convert_channel_10b(THROTTLE, false);
|
2020-01-14 12:19:12 +01:00
|
|
|
packet[0] = val>>2; // 0..255
|
2020-11-30 13:36:29 +01:00
|
|
|
packet[12] |= val & 2;
|
2020-01-14 12:19:12 +01:00
|
|
|
val=XK_convert_channel(RUDDER);
|
|
|
|
packet[1] = val>>2;
|
2020-11-30 13:36:29 +01:00
|
|
|
packet[12] |= (val & 2)<<2;
|
2020-01-14 12:19:12 +01:00
|
|
|
val=XK_convert_channel(ELEVATOR);
|
|
|
|
packet[2] = val>>2;
|
2020-11-30 13:36:29 +01:00
|
|
|
packet[13] |= val & 2;
|
2020-01-14 12:19:12 +01:00
|
|
|
val=XK_convert_channel(AILERON);
|
|
|
|
packet[3] = val>>2;
|
2020-11-30 13:36:29 +01:00
|
|
|
packet[13] |= (val & 2)<<2;
|
2020-01-14 12:19:12 +01:00
|
|
|
|
|
|
|
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
|
2020-01-14 21:25:43 +01:00
|
|
|
|
2020-01-14 12:19:12 +01:00
|
|
|
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
|
2020-01-14 21:25:43 +01:00
|
|
|
packet[14] = GET_FLAG(CH9_SW,0x01) // Photo momentary switch
|
|
|
|
|GET_FLAG(CH10_SW,0x2); // Video momentary switch
|
2020-11-30 13:36:29 +01:00
|
|
|
//debugln("P1:%02X,P12:%02X",packet[1],packet[12]);
|
2020-01-14 12:19:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
crc=packet[0];
|
|
|
|
for(uint8_t i=1; i<XK_PAYLOAD_SIZE-1;i++)
|
|
|
|
crc+=packet[i];
|
|
|
|
packet[15]=crc;
|
|
|
|
|
2020-01-16 23:43:29 +01:00
|
|
|
// debug("C: %02X, P:",hopping_frequency[(IS_BIND_IN_PROGRESS?0:XK_RF_BIND_NUM_CHANNELS)+(hopping_frequency_no>>1)]);
|
2020-01-14 12:19:12 +01:00
|
|
|
XN297L_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;
|
|
|
|
|
|
|
|
XN297L_WritePayload(packet, XK_PAYLOAD_SIZE);
|
2020-01-16 23:43:29 +01:00
|
|
|
// for(uint8_t i=0; i<XK_PAYLOAD_SIZE; i++)
|
|
|
|
// debug(" %02X",packet[i]);
|
|
|
|
// debugln("");
|
2020-01-14 12:19:12 +01:00
|
|
|
|
|
|
|
XN297L_SetPower(); // Set tx_power
|
|
|
|
XN297L_SetFreqOffset(); // Set frequency offset
|
|
|
|
}
|
|
|
|
|
|
|
|
const uint8_t PROGMEM XK_bind_hop[XK_RF_BIND_NUM_CHANNELS]= { 0x07, 0x24, 0x3E, 0x2B, 0x47, 0x0E, 0x39, 0x1C }; // Bind
|
|
|
|
|
2020-01-16 23:43:29 +01:00
|
|
|
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 };
|
|
|
|
|
2020-01-14 12:19:12 +01:00
|
|
|
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] );
|
|
|
|
|
2020-01-16 23:43:29 +01:00
|
|
|
//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++)
|
2020-01-14 22:51:47 +01:00
|
|
|
{
|
2020-01-16 23:43:29 +01:00
|
|
|
rx_tx_addr[i]=pgm_read_byte_near( &XK_tx_addr[ start + i ] );
|
|
|
|
// debug("%02X ", rx_tx_addr[ i ]);
|
2020-01-14 22:51:47 +01:00
|
|
|
}
|
2020-01-16 23:43:29 +01:00
|
|
|
// 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
|
2020-01-14 12:19:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void __attribute__((unused)) XK_init()
|
|
|
|
{
|
|
|
|
XN297L_Init();
|
2020-01-14 21:25:43 +01:00
|
|
|
XN297L_SetTXAddr((uint8_t*)"\x68\x94\xA6\xD5\xC3", 5); // Bind address
|
|
|
|
XN297L_HoppingCalib(XK_RF_BIND_NUM_CHANNELS+XK_RF_NUM_CHANNELS); // Calibrate all channels
|
2020-01-14 12:19:12 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t XK_callback()
|
|
|
|
{
|
2020-01-14 22:05:30 +01:00
|
|
|
if(sub_protocol==X420)
|
|
|
|
option=0; // Forcing the use of NRF24L01@1Mbps
|
2020-01-14 12:19:12 +01:00
|
|
|
#ifdef MULTI_SYNC
|
|
|
|
telemetry_set_input_sync(XK_PACKET_PERIOD);
|
|
|
|
#endif
|
|
|
|
if(IS_BIND_IN_PROGRESS)
|
|
|
|
if(--bind_counter==0)
|
|
|
|
{
|
|
|
|
BIND_DONE;
|
2020-01-14 22:51:47 +01:00
|
|
|
XN297L_SetTXAddr(rx_tx_addr, 5); // Normal packets address
|
2020-01-14 12:19:12 +01:00
|
|
|
}
|
|
|
|
XK_send_packet();
|
|
|
|
return XK_PACKET_PERIOD;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t initXK()
|
|
|
|
{
|
2020-01-14 22:05:30 +01:00
|
|
|
if(sub_protocol==X420)
|
|
|
|
option=prev_option=0; // Forcing the use of NRF24L01@1Mbps
|
|
|
|
BIND_IN_PROGRESS; // Autobind protocol
|
2020-01-14 12:19:12 +01:00
|
|
|
XK_initialize_txid();
|
|
|
|
XK_init();
|
2020-01-14 22:05:30 +01:00
|
|
|
if(sub_protocol==X420)
|
2020-01-14 22:51:47 +01:00
|
|
|
NRF24L01_SetBitrate(NRF24L01_BR_1M); // X420/X520 runs @1Mbps
|
2020-01-14 12:19:12 +01:00
|
|
|
hopping_frequency_no = 0;
|
|
|
|
bind_counter=XK_BIND_COUNT;
|
|
|
|
return XK_INITIAL_WAIT;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|