/* 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_xn297.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(sub_protocol != XK_CARS) { // Introduce deadband on all channels to prevent twitching //debug("val:%d",val); val=convert_channel_8b_limit_deadband(num,0x00,0x80, 0xFF, 40)<<2; //debugln(",%d",val); } else val=convert_channel_16b_limit(num,0x00,0x3FF); // 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,0x02) // Video momentary switch |GET_FLAG(CH11_SW,0x04) // Flip |GET_FLAG(CH12_SW,0x10); // Light //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==X450 ? XN297_250K : XN297_1M ); 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() { if(sub_protocol != XK_CARS) BIND_IN_PROGRESS; // Autobind protocol XK_initialize_txid(); XK_RF_init(); hopping_frequency_no = 0; bind_counter=XK_BIND_COUNT; } #endif