Skip to content

Commit 1f19e7c

Browse files
author
robotmaker team
committed
Arduino to Frysky SBUS connection
This allows connection of an Arduino to a Frysky X8R SBUS port to enable additional on-board control, such as LED headlights, sensors and PWM servos, indicator LED's. It is part of a project to connect our 360° infrared proximity sensor for Quadcopters / UAV to enable autonomous flight.
1 parent 932e5a8 commit 1f19e7c

File tree

1 file changed

+253
-0
lines changed

1 file changed

+253
-0
lines changed

SBUS_20.pde

+253
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,253 @@
1+
// Graphical represetentation of Futuba SBUS - also used in FrySky Receivers
2+
// Example by Colin Bacon
3+
// ROBOTmaker
4+
// www.robotmaker.eu
5+
6+
//Thanks goes to Futaba for making this SBUS protocol so challenging for everyone. Talk about "why make things simple if we can make things extermely difficult..."
7+
//To get this to work you'll need to have SBUS conneted receiver that is connected via an Arduiono or FTDI USB thingy to your PC
8+
//As the SBUS is an inverted signal (yes - to make life even more fun), you'll need to invert the signal using a simple transitor+2 restistors or if you have an FTDI there is a program to invert the signals automatically
9+
//Detail of how to do this are on our website
10+
11+
12+
// References
13+
//https://developer.mbed.org/users/Digixx/notebook/futaba-s-bus-controlled-by-mbed/
14+
//http://www.clubamcl.fr/techniques/la-technologie-futaba-s-bus/
15+
//Thanks goes to https://www.ordinoscope.net/index.php/Electronique/Protocoles/SBUS for the arduino code which I converted to processing
16+
17+
18+
import processing.serial.*;
19+
20+
Serial myPort; // The serial port
21+
float x, y;
22+
23+
String myString = null;
24+
int [] buffer = new int [25];
25+
int [] channels = new int [18];
26+
int errors = 0;
27+
boolean failsafe = false;
28+
int idx, inByte, lost, packetErrors;
29+
long last_refresh = 0;
30+
float ymag = 0;
31+
float newYmag = 0;
32+
float xmag = 0;
33+
float newXmag = 0;
34+
35+
void setup()
36+
{
37+
size(1500, 600, FX2D); //give good dashboard resolutions
38+
// size(1500, 600, P3D); //used for drawing cubes
39+
//fullScreen(); //This sets it to full screen, but comment out the size commmand above
40+
pixelDensity(1);
41+
42+
x = width * 0.3;
43+
y = height * 0.5;
44+
45+
// Variable for drawing the cube
46+
47+
48+
// List all the available serial ports
49+
printArray(Serial.list());
50+
// Open the port you are using at the rate you want:
51+
myPort = new Serial(this, Serial.list()[0], 100000,'E',8,2);
52+
myString = myPort.readStringUntil(15); //clean the first buffer
53+
println(myPort); //Debug to show the port opened
54+
}
55+
56+
void draw()
57+
{
58+
59+
//Check for incoming Serial Port Data
60+
if (myPort.available() == 0){
61+
62+
//If no incoming data then count lost packages
63+
packetErrors++;
64+
text ("No Signal", 140,20);
65+
}
66+
67+
//When data is comming in then check for a start byte No.1=B11110000 and stop byte No.25=B00000000
68+
while (myPort.available() > 0) {
69+
inByte = myPort.read(); //Read the Byte that just came in
70+
71+
//if it's a new packet and the start byte is not B00001111 (DEC15) then it's an error.
72+
//Note that the SBUS byte data is MSB, which causes some fun later
73+
if (idx == 0 && inByte != 0x0F) // error - wait for the start byte
74+
{
75+
text("Package Error", 100,100);
76+
77+
}
78+
// if it's a new packet and the start byte is B00001111 (DEC15) then start reading the next 25 bytes.
79+
else {
80+
buffer[idx++] = inByte; // fill the buffer with 25 Bytes
81+
}
82+
83+
// if the buffer of 25 Bytes is reached then start to decode
84+
if (idx == 25)
85+
{
86+
idx = 0; //reset the buffer count for the next cycle
87+
if (buffer[24] != 0x00)
88+
{ //Check that the packet size is 25 bytes long with stop byte b00000000 as the last byte
89+
errors++; //Count the number of errors
90+
println(errors); //Print to the error totals to the dashboard
91+
}
92+
else
93+
{ //Start decoding the bits and bytes
94+
95+
//Buffer[0] contains the start byte value of 15, so this is ignored
96+
//The Channels are 11 bits in length, so the bytes need to be split-up
97+
//To make this conversion more 'interesting' as the bytes arrive as MSB i.e. the highest byte arrives first and the channels are
98+
//assembled as little endian, which means the 1st 3 bits of the second byte need to comes before the 1st byte.
99+
100+
//--------------Channel 1 -------------------
101+
// The 1st channel = 1st byte + 3 bits from the 2nd byte
102+
// This channel packet of 11 bits is then read backwards.
103+
// As the inByte variable is already converted to integers we can move bits along as follows:
104+
// byte 1 = 00000000 00000011 (say the value is 3)
105+
// byte 2 = ‭01100100‬ (say the value of the 2nd byte is 100)
106+
// but we are only interested in the last 3 bits which need to be moved infront of the 1st byte,...
107+
// byte 2 = ‭01100100 00000000 - ...So shift this byte along left by 8 and it looks like this
108+
// byte 2 = | (or) the two bytes together and you get this
109+
// byte 1 = 00000000 00000011
110+
// byte 2 = ‭01100100 00000000
111+
// Channel 1 = 01100100 00000011 <- the intermediate result
112+
// But we were only interested in the 1st 3 bits of byte 2 so we need strip the 1st 5 bits by and-ing with 2047 (11 bits)
113+
// & (and) the two together and you get this....
114+
// Channel1 = 01100100 00000011
115+
// 2047 = 00000111 11111111
116+
// Channel1 = 00000100 00000011 <- the final result
117+
//The bytes are litte Endian which means that to read the full
118+
119+
channels[0] = ((buffer[1]|buffer[2]<<8) & 0x07FF); //The first Channel
120+
121+
//--------------Channel 2 -------------------
122+
// 2nd Channel = last 5 bit of byte2 and 8 bits of byte 3. Here we need to play around some more.
123+
// Remove the 1st 3 bits of byte2 by pushing them off to the right
124+
// byte 2 = 00000000 ‭01100100
125+
// byte 2 = 00000000 ‭00001100 moved 3 to the right
126+
// byte 3 = 00000000 ‭01010100 lets say that byte 3 is this
127+
// byte 3 = 00001010 10000000 moved 5 to the left to get the 5 bytes we need in place (remember this is MSB so bits in byte 3 comes before bits in byte 2)
128+
// | (or) byte 2 and 3 them together to get the 11 bits in place for channels2
129+
// byte 2 = 00000000 ‭00001100
130+
// byte 3 = 00001010 10000000
131+
// Channel2 = 00001010 10001100 < - result of| (or-ing) them together
132+
// again we were only interested in 11 bits so strip the package by and-ing with 2047 (11 bits)
133+
// & (and) the two together and you get this....
134+
// Channel2 = 00001010 10001100 < - interim result of| (or-ing) them together
135+
// 2047 = 00000111 11111111
136+
// Channel2 = 00000010 10001100 < - result of & (and-ing) them together
137+
138+
channels[1] = ((buffer[2]>>3|buffer[3]<<5) & 0x07FF);
139+
140+
//--------------Channel 3 -------------------
141+
// 3rd Channel = last 2 bit of byte3 and 8 bits of byte 4 and some bits from byte 5.
142+
// Here we have even more to play around with. Someone with a sense of humour surely designed this.
143+
// Talk about "why make things simple if we can make it very difficult..."
144+
// Remove the 1st 6 bits of byte3 by pushing them off to the right
145+
// byte 2 = 00000000 ‭00001100
146+
// byte 3 = 00000000 ‭00000001 moved 6 to the right
147+
// byte 4 = 00000000 10101010 lets say that byte 4 is this
148+
// byte 4 = 00000010 10101000 moved 2 to the left to get the bytes we need in place
149+
// | (or) byte 3 and 4 together to get the 11 bits in place for channels3
150+
// byte 3 = 00000000 ‭00000001
151+
// byte 4 = 00000010 10101000
152+
// Channel3 = 00000010 10101001 < - result of| (or-ing) them together
153+
// byte 5 = 00000000 10110101 lets say that byte 5 is this
154+
// byte 5 = 11010100 00000000 moved 10 to the left to get the bytes we need in place
155+
// Channel3 = 00000010 10101001 < - result of prrevious| (or-ing) them together
156+
// Channel3 = 11010110 10101001 <- result of or-ing shift byte 5 and previous or-ing
157+
// | (or) byte 5 together with the current channel 3 to get the 11 bits in place for channels3
158+
// again we were only interested in 11 bits so strip the package by and-ing with 2047 (11 bits)
159+
// & (and) the two together and you get this....
160+
// Channel3 = 11010110 10101001 <- result of or-ing shift byte 5 and previous or-ing
161+
// 2047 = 00000111 11111111
162+
// Channel3 = 00000110 10101001 < - result of & (and-ing) them together
163+
// ...and so on
164+
channels[2] = ((buffer[3]>>6 |buffer[4]<<2 |buffer[5]<<10) & 0x07FF);
165+
channels[3] = ((buffer[5]>>1 |buffer[6]<<7) & 0x07FF);
166+
channels[4] = ((buffer[6]>>4 |buffer[7]<<4) & 0x07FF);
167+
channels[5] = ((buffer[7]>>7 |buffer[8]<<1 |buffer[9]<<9) & 0x07FF);
168+
channels[6] = ((buffer[9]>>2 |buffer[10]<<6) & 0x07FF);
169+
channels[7] = ((buffer[10]>>5|buffer[11]<<3) & 0x07FF);
170+
channels[8] = ((buffer[12] |buffer[13]<<8) & 0x07FF);
171+
channels[9] = ((buffer[13]>>3|buffer[14]<<5) & 0x07FF);
172+
channels[10] = ((buffer[14]>>6|buffer[15]<<2|buffer[16]<<10) & 0x07FF);
173+
channels[11] = ((buffer[16]>>1|buffer[17]<<7) & 0x07FF);
174+
channels[12] = ((buffer[17]>>4|buffer[18]<<4) & 0x07FF);
175+
channels[13] = ((buffer[18]>>7|buffer[19]<<1|buffer[20]<<9) & 0x07FF);
176+
channels[14] = ((buffer[20]>>2|buffer[21]<<6) & 0x07FF);
177+
channels[15] = ((buffer[21]>>5|buffer[22]<<3) & 0x07FF);
178+
channels[16] = ((buffer[23]));
179+
//channels[16] = ((buffer[23]) & 0x0001) ? 2047 : 0;
180+
//channels[17] = ((buffer[23] >> 1) & 0x0001) ? 2047 : 0;
181+
//failsafe = ((buffer[23] >> 3) & 0x0001) ? 1 : 0;
182+
//if ((buffer[23] >> 2) & 0x0001) lost++;
183+
}
184+
} //For loop
185+
186+
//pushMatrix();
187+
int pos1=0,channelText = 0;
188+
//Set the background colour of the bargraphs
189+
background(199);
190+
191+
//Cycle through all channels and draw a bar chart for each channel value
192+
for(int c=0;c<16;c++){
193+
194+
//Set the colour of the bar chart graphics
195+
noStroke(); //remove the graphic boarder lines
196+
fill(17,37,204); //set the colour of bar chart
197+
198+
//Map Channel bar graph
199+
float z= float(channels[c]); //convert to a float
200+
float x1 = map(z,160,2000,0,width*0.95);
201+
int yoffSet = 50;
202+
pos1 = pos1 + 20; //y postion of bar on the canvas
203+
channelText = channelText + 1;
204+
rect(100, pos1+yoffSet , x1, 15);
205+
fill(0);
206+
text("Channel" + channelText + " = ", 5,pos1+10+yoffSet);
207+
text(channels[c],70,pos1+10 + yoffSet);
208+
209+
//Draw the dashboard title and status messages
210+
textSize(26);
211+
text("Open TX SBUS - Status Dashboard", width/2-150, 30); //Place the text in the centre of the dashboard
212+
textSize(10); //Reset text size
213+
214+
//If the bits in byte23 are set, then there is an error. Display a warning
215+
if (buffer[23] >= 12) {
216+
fill(226,72,47);
217+
ellipse(50,40,20,20); //Draw a red warning led
218+
fill(0); //colour the text
219+
textSize(12); //Reset text size
220+
text("Signal Lost",30,20);
221+
textSize(11); //Reset text size
222+
}
223+
else
224+
{
225+
fill(22,205,65);
226+
ellipse(50,40,20,20); //Draw a green warning led
227+
fill(0); //colour the text
228+
textSize(12); //Reset text size
229+
text("Signal OK",30,20);
230+
textSize(11); //Reset text size
231+
}
232+
233+
//Place the error counts message
234+
textSize(12); //Reset text size
235+
text("Errors", 120,20);
236+
text("bytes23= ", 100,40);
237+
text(buffer[23],150,40);
238+
textSize(11); //Reset text size
239+
text("Count= ", 100,55);
240+
text(errors, 150,55);
241+
242+
//Place the lost package message
243+
textSize(12); //Reset text size
244+
text("Lost Packages", 170,20);
245+
textSize(11); //Reset text size
246+
text(packetErrors, 190,40);
247+
248+
249+
} //if
250+
} //While
251+
252+
}
253+

0 commit comments

Comments
 (0)