-
Notifications
You must be signed in to change notification settings - Fork 32
/
SBUS.cpp
139 lines (113 loc) · 5.38 KB
/
SBUS.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
/*
To use this software, you must adhere to the license terms described below, and assume all responsibility for the use
of the software. The user is responsible for all consequences or damage that may result from using this software.
The user is responsible for ensuring that the hardware used to run this software complies with local regulations and that
any radio signal generated from use of this software is legal for that user to generate. The author(s) of this software
assume no liability whatsoever. The author(s) of this software is not responsible for legal or civil consequences of
using this software, including, but not limited to, any damages cause by lost control of a vehicle using this software.
If this software is copied or modified, this disclaimer must accompany all copies.
This PPM algorithm originated from:
https://code.google.com/archive/p/generate-ppm-signal/
under GNU GPL 2. Second version dated Mar 14, 2013.
Modifications and additions released here under GPL.
*/
/*
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.
RC_RX_CABELL_V3_FHSS 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 RC_RX_CABELL_V3_FHSS. If not, see <http://www.gnu.org/licenses/>.
*/
#include "RX.h"
#include "Arduino.h"
#include "SBUS.h"
bool sbusEnabledFlag = false;
volatile uint8_t sbusPacket[25];
//------------------------------------------------------------------------------------------------------------------------
void sbusSetup(){
Serial.begin(100000,SERIAL_8E2);
noInterrupts();
TCCR1A = 0; // set entire TCCR1 register to 0
TCCR1B = 0;
OCR1A = SBUS_PACKET_RATE * 2000; // compare match register. at pre-scalar 8 and 16 Mhz 14000 is 7 ms. This defines the SBUS packet rate
TCCR1B |= (1 << WGM12); // turn on CTC mode
TCCR1B |= (1 << CS11); // 8 pre-scaler: 0,5 microseconds at 16mhz
TIMSK1 |= (1 << OCIE1A); // enable timer compare interrupt
sbusPacket[SBUS_START_BYTE] = 0x0F;
sbusPacket[SBUS_FLAG_BYTE] = 0x00;
sbusPacket[SBUS_END_BYTE] = 0x00;
interrupts();
sbusEnabledFlag = true;
}
bool sbusEnabled() {
return sbusEnabledFlag;
}
//------------------------------------------------------------------------------------------------------------------------
void sbusDisable(){
noInterrupts();
TCCR1A = 0; // set entire TCCR1 register to 0
TCCR1B = 0;
TIMSK1 &= ~(1<<OCIE1A); // Disable Interrupt Counter 1, output compare A (TIMER1_CMPA_vect)
interrupts();
sbusEnabledFlag = false;
Serial.begin(74880);
}
//------------------------------------------------------------------------------------------------------------------------
void setSbusOutputChannelValue(uint8_t channel, uint16_t value) {
uint8_t firstBit = 8 + (constrain(channel,0,15) * 11); // Start byte plus 11 bits per channel. 16 channels
uint8_t byteIndex = firstBit / 8;
uint8_t bitIndex = (firstBit % 8);
int16_t adjustedValue = constrain(value,CHANNEL_MIN_VALUE,CHANNEL_MAX_VALUE);
adjustedValue = map(adjustedValue - CHANNEL_MID_VALUE,(CHANNEL_MIN_VALUE - CHANNEL_MID_VALUE),(CHANNEL_MAX_VALUE - CHANNEL_MID_VALUE),-SBUS_OFFSET_FROM_MID_POINT,SBUS_OFFSET_FROM_MID_POINT);
adjustedValue += SBUS_MID_POINT;
noInterrupts(); //Turn off interrupts so that SBUS_ISR does not run while a value is being updated
for (uint8_t x = 0; x < 11; x++) {
if (adjustedValue & 0x0001) {
sbusPacket[byteIndex] |= _BV(bitIndex);
} else {
sbusPacket[byteIndex] &= ~_BV(bitIndex);
}
adjustedValue >>= 1;
if (bitIndex == 7) {
bitIndex = 0;
byteIndex++;
} else{
bitIndex++;
}
}
interrupts();
}
//------------------------------------------------------------------------------------------------------------------------
void sbusSetFailsafe(bool value) {
noInterrupts(); //Turn off interrupts so that SBUS_ISR does not run while a value is being updated
if (value) {
sbusPacket[SBUS_FLAG_BYTE] |= SBUS_FAILSAFE_MASK;
} else {
sbusPacket[SBUS_FLAG_BYTE] &= ~SBUS_FAILSAFE_MASK;
}
interrupts();
}
//------------------------------------------------------------------------------------------------------------------------
void sbusSetFrameLost(bool value) {
noInterrupts(); //Turn off interrupts so that SBUS_ISR does not run while a value is being updated
if (value) {
sbusPacket[SBUS_FLAG_BYTE] |= SBUS_FRAME_LOST_MASK;
} else {
sbusPacket[SBUS_FLAG_BYTE] &= ~SBUS_FRAME_LOST_MASK;
}
interrupts();
}
//------------------------------------------------------------------------------------------------------------------------
void SBUS_ISR() {
// Copy the sbusPacket to the serial output buffer and send it (8E2 at 100000 baud
uint8_t outBuf[25];
for (uint8_t x = 0; x < sizeof(outBuf); x++) { // code loop instead of using memcpy because sbusPacket is volatile
outBuf[x] = sbusPacket[x];
}
Serial.write(&outBuf[0],sizeof(outBuf));
}