mirror of
https://github.com/pascallanger/DIY-Multiprotocol-TX-Module.git
synced 2025-02-04 17:48:11 +00:00
bc42dbf88a
Lot of changes in this new master ChangeLog: - Core: LED flashing when an invalid protocol has been selected - Core: Channels 5 to 12 available as switches for all protocols: code and size optimization - Documentation (readme.md): fully updated, all protocols/sub protocols/channels described, models example, many improvements - All protocols have been updated in some way, here are some highlights: * Bayang: added picture, video and inverted channels * CG023->H8_3D: added light and calibration channels * CX10: added sub protocols Q282, JC3015_1, JC3015_2, MK33041 * ESky: added new protocol - untested * Hubsan: added compatibility with the new Hubsan Plus protocol * KN: fully rewritten protocol: added sub protocols WLTOYS and FEILUN, 11 channels support New version successfully tested on all my models: Flysky RX/F939/V911 protocol Flysky, Frsky RX protocol Frsky, Hubsan X4 protocol Hubsan, Hisky HCP100/HCP80 protocol Hisky, HK-3000/HK3100 RX protocol Hisky/HK310, XINXUN X39 protocol YD717/XINXUN, Symax X5C-1 protocol SymaX/SYMAX, Cheerson CX-10A protocol CX10/BLUE, Eachine 3D-X4 protocol CG023. To access new protocols from er9x/ersky9x, you need to build a version from this github repository https://github.com/pascallanger/mbtx based on the latest er9x r820 and ersky9x r218.
251 lines
7.4 KiB
C++
251 lines
7.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/>.
|
|
*/
|
|
// Last sync with hexfet new_protocols/yd717_nrf24l01.c dated 2015-09-28
|
|
|
|
#if defined(YD717_NRF24L01_INO)
|
|
|
|
#include "iface_nrf24l01.h"
|
|
|
|
#define YD717_BIND_COUNT 60
|
|
#define YD717_PACKET_PERIOD 8000 // Timeout for callback in uSec, 8ms=8000us for YD717
|
|
#define YD717_INITIAL_WAIT 50000 // Initial wait before starting callbacks
|
|
#define YD717_PACKET_CHKTIME 500 // Time to wait if packet not yet acknowledged or timed out
|
|
|
|
// Stock tx fixed frequency is 0x3C. Receiver only binds on this freq.
|
|
#define YD717_RF_CHANNEL 0x3C
|
|
|
|
#define YD717_FLAG_FLIP 0x0F
|
|
#define YD717_FLAG_LIGHT 0x80
|
|
#define YD717_FLAG_PICTURE 0x40
|
|
#define YD717_FLAG_VIDEO 0x20
|
|
#define YD717_FLAG_HEADLESS 0x10
|
|
|
|
#define YD717_PAYLOADSIZE 8 // receive data pipes set to this size, but unused
|
|
|
|
enum {
|
|
YD717_INIT1 = 0,
|
|
YD717_BIND2,
|
|
YD717_BIND3,
|
|
YD717_DATA
|
|
};
|
|
|
|
static void yd717_send_packet(uint8_t bind)
|
|
{
|
|
uint8_t rudder_trim, elevator_trim, aileron_trim;
|
|
if (bind)
|
|
{
|
|
packet[0]= rx_tx_addr[0]; // send data phase address in first 4 bytes
|
|
packet[1]= rx_tx_addr[1];
|
|
packet[2]= rx_tx_addr[2];
|
|
packet[3]= rx_tx_addr[3];
|
|
packet[4] = 0x56;
|
|
packet[5] = 0xAA;
|
|
packet[6] = (sub_protocol == NIHUI) ? 0x00 : 0x32;
|
|
packet[7] = 0x00;
|
|
}
|
|
else
|
|
{
|
|
// Throttle
|
|
packet[0] = convert_channel_8b(THROTTLE);
|
|
// Rudder
|
|
if( sub_protocol==XINXUN )
|
|
{
|
|
rudder = convert_channel_8b(RUDDER);
|
|
rudder_trim = (0xff - rudder) >> 1;
|
|
}
|
|
else
|
|
{
|
|
rudder = 0xff - convert_channel_8b(RUDDER);
|
|
rudder_trim = rudder >> 1;
|
|
}
|
|
packet[1] = rudder;
|
|
// Elevator
|
|
elevator = convert_channel_8b(ELEVATOR);
|
|
elevator_trim = elevator >> 1;
|
|
packet[3] = elevator;
|
|
// Aileron
|
|
aileron = 0xff - convert_channel_8b(AILERON);
|
|
aileron_trim = aileron >> 1;
|
|
packet[4] = aileron;
|
|
// Trims
|
|
if( sub_protocol == YD717 )
|
|
{
|
|
packet[2] = elevator_trim;
|
|
packet[5] = aileron_trim;
|
|
packet[6] = rudder_trim;
|
|
}
|
|
else
|
|
{
|
|
packet[2] = rudder_trim;
|
|
packet[5] = elevator_trim;
|
|
packet[6] = aileron_trim;
|
|
}
|
|
// Flags
|
|
flags=0;
|
|
// Channel 5
|
|
if (Servo_AUX1) flags = YD717_FLAG_FLIP;
|
|
// Channel 6
|
|
if (Servo_AUX2) flags |= YD717_FLAG_LIGHT;
|
|
// Channel 7
|
|
if (Servo_AUX3) flags |= YD717_FLAG_PICTURE;
|
|
// Channel 8
|
|
if (Servo_AUX4) flags |= YD717_FLAG_VIDEO;
|
|
// Channel 9
|
|
if (Servo_AUX5) flags |= YD717_FLAG_HEADLESS;
|
|
packet[7] = flags;
|
|
}
|
|
|
|
// clear packet status bits and TX FIFO
|
|
NRF24L01_WriteReg(NRF24L01_07_STATUS, (BV(NRF24L01_07_TX_DS) | BV(NRF24L01_07_MAX_RT)));
|
|
NRF24L01_FlushTx();
|
|
|
|
if( sub_protocol == YD717 )
|
|
NRF24L01_WritePayload(packet, 8);
|
|
else
|
|
{
|
|
packet[8] = packet[0]; // checksum
|
|
for(uint8_t i=1; i < 8; i++)
|
|
packet[8] += packet[i];
|
|
packet[8] = ~packet[8];
|
|
NRF24L01_WritePayload(packet, 9);
|
|
}
|
|
|
|
NRF24L01_SetPower(); // Set tx_power
|
|
}
|
|
|
|
static void yd717_init()
|
|
{
|
|
NRF24L01_Initialize();
|
|
|
|
// CRC, radio on
|
|
NRF24L01_SetTxRxMode(TX_EN);
|
|
NRF24L01_WriteReg(NRF24L01_00_CONFIG, BV(NRF24L01_00_EN_CRC) | BV(NRF24L01_00_PWR_UP));
|
|
NRF24L01_WriteReg(NRF24L01_01_EN_AA, 0x3F); // Auto Acknoledgement on all data pipes
|
|
NRF24L01_WriteReg(NRF24L01_02_EN_RXADDR, 0x3F); // Enable all data pipes
|
|
NRF24L01_WriteReg(NRF24L01_03_SETUP_AW, 0x03); // 5-byte RX/TX address
|
|
NRF24L01_WriteReg(NRF24L01_04_SETUP_RETR, 0x1A); // 500uS retransmit t/o, 10 tries
|
|
NRF24L01_WriteReg(NRF24L01_05_RF_CH, YD717_RF_CHANNEL); // Channel 3C
|
|
NRF24L01_SetBitrate(NRF24L01_BR_1M); // 1Mbps
|
|
NRF24L01_SetPower();
|
|
NRF24L01_WriteReg(NRF24L01_07_STATUS, 0x70); // Clear data ready, data sent, and retransmit
|
|
NRF24L01_WriteReg(NRF24L01_0C_RX_ADDR_P2, 0xC3); // LSB byte of pipe 2 receive address
|
|
NRF24L01_WriteReg(NRF24L01_0D_RX_ADDR_P3, 0xC4);
|
|
NRF24L01_WriteReg(NRF24L01_0E_RX_ADDR_P4, 0xC5);
|
|
NRF24L01_WriteReg(NRF24L01_0F_RX_ADDR_P5, 0xC6);
|
|
NRF24L01_WriteReg(NRF24L01_11_RX_PW_P0, YD717_PAYLOADSIZE); // bytes of data payload for pipe 1
|
|
NRF24L01_WriteReg(NRF24L01_12_RX_PW_P1, YD717_PAYLOADSIZE);
|
|
NRF24L01_WriteReg(NRF24L01_13_RX_PW_P2, YD717_PAYLOADSIZE);
|
|
NRF24L01_WriteReg(NRF24L01_14_RX_PW_P3, YD717_PAYLOADSIZE);
|
|
NRF24L01_WriteReg(NRF24L01_15_RX_PW_P4, YD717_PAYLOADSIZE);
|
|
NRF24L01_WriteReg(NRF24L01_16_RX_PW_P5, YD717_PAYLOADSIZE);
|
|
NRF24L01_WriteReg(NRF24L01_17_FIFO_STATUS, 0x00); // Just in case, no real bits to write here
|
|
|
|
NRF24L01_Activate(0x73); // Activate feature register
|
|
NRF24L01_WriteReg(NRF24L01_1C_DYNPD, 0x3F); // Enable dynamic payload length on all pipes
|
|
NRF24L01_WriteReg(NRF24L01_1D_FEATURE, 0x07); // Set feature bits on
|
|
NRF24L01_Activate(0x73);
|
|
|
|
// set tx id
|
|
NRF24L01_WriteRegisterMulti(NRF24L01_0A_RX_ADDR_P0, rx_tx_addr, 5);
|
|
NRF24L01_WriteRegisterMulti(NRF24L01_10_TX_ADDR, rx_tx_addr, 5);
|
|
}
|
|
|
|
static void YD717_init1()
|
|
{
|
|
// for bind packets set address to prearranged value known to receiver
|
|
uint8_t bind_rx_tx_addr[] = {0x65, 0x65, 0x65, 0x65, 0x65};
|
|
uint8_t i;
|
|
if( sub_protocol==SYMAX4 )
|
|
for(i=0; i < 5; i++)
|
|
bind_rx_tx_addr[i] = 0x60;
|
|
else
|
|
if( sub_protocol==NIHUI )
|
|
for(i=0; i < 5; i++)
|
|
bind_rx_tx_addr[i] = 0x64;
|
|
|
|
NRF24L01_WriteRegisterMulti(NRF24L01_0A_RX_ADDR_P0, bind_rx_tx_addr, 5);
|
|
NRF24L01_WriteRegisterMulti(NRF24L01_10_TX_ADDR, bind_rx_tx_addr, 5);
|
|
}
|
|
|
|
static void YD717_init2()
|
|
{
|
|
// set rx/tx address for data phase
|
|
NRF24L01_WriteRegisterMulti(NRF24L01_0A_RX_ADDR_P0, rx_tx_addr, 5);
|
|
NRF24L01_WriteRegisterMulti(NRF24L01_10_TX_ADDR, rx_tx_addr, 5);
|
|
}
|
|
|
|
uint16_t yd717_callback()
|
|
{
|
|
switch (phase)
|
|
{
|
|
case YD717_INIT1:
|
|
yd717_send_packet(0); // receiver doesn't re-enter bind mode if connection lost...check if already bound
|
|
phase = YD717_BIND3;
|
|
break;
|
|
case YD717_BIND2:
|
|
if (counter == 0)
|
|
{
|
|
if (NRF24L01_packet_ack() == PKT_PENDING)
|
|
return YD717_PACKET_CHKTIME; // packet send not yet complete
|
|
YD717_init2(); // change to data phase rx/tx address
|
|
yd717_send_packet(0);
|
|
phase = YD717_BIND3;
|
|
}
|
|
else
|
|
{
|
|
if (NRF24L01_packet_ack() == PKT_PENDING)
|
|
return YD717_PACKET_CHKTIME; // packet send not yet complete;
|
|
yd717_send_packet(1);
|
|
counter--;
|
|
}
|
|
break;
|
|
case YD717_BIND3:
|
|
switch (NRF24L01_packet_ack())
|
|
{
|
|
case PKT_PENDING:
|
|
return YD717_PACKET_CHKTIME; // packet send not yet complete
|
|
case PKT_ACKED:
|
|
phase = YD717_DATA;
|
|
BIND_DONE; // bind complete
|
|
break;
|
|
case PKT_TIMEOUT:
|
|
YD717_init1(); // change to bind rx/tx address
|
|
counter = YD717_BIND_COUNT;
|
|
phase = YD717_BIND2;
|
|
yd717_send_packet(1);
|
|
}
|
|
break;
|
|
case YD717_DATA:
|
|
if (NRF24L01_packet_ack() == PKT_PENDING)
|
|
return YD717_PACKET_CHKTIME; // packet send not yet complete
|
|
yd717_send_packet(0);
|
|
break;
|
|
}
|
|
return YD717_PACKET_PERIOD; // Packet every 8ms
|
|
}
|
|
|
|
uint16_t initYD717()
|
|
{
|
|
rx_tx_addr[4] = 0xC1; // always uses first data port
|
|
yd717_init();
|
|
phase = YD717_INIT1;
|
|
BIND_IN_PROGRESS; // autobind protocol
|
|
|
|
// Call callback in 50ms
|
|
return YD717_INITIAL_WAIT;
|
|
}
|
|
|
|
#endif
|