Reading RC Receiver PPM Signal Using Arduino




About: I love making and flying RC planes.

RC transmitter is very well suitable for the projects requires a wireless link as it has encrypted link with good range.

All RC receiver is made to drive servo motors. so there are 3 pins for each channel: ground, Vcc, and signal. here special kind of signal is used. It sends pulses at some interval. When servo receives 1.5ms pulse it sets to 90 degrees and by varying this value from 1-2ms servo goes to minimum and maximum value.
(In above photo a 6 channel receiver is shown with PPM pin)

So easy method is to measuring pulse width on each pin and map that data as per requirement. but here problem is that:

  • For each channel of the receiver, we need to connect a wire to Arduino pin.
  • this not only requires lots of connection but also consume lots of pins on an Arduino.
  • Most of Arduino just has 2 interrupt pin so if we uses more than 2 channel read it add some delay to our code which may be problematic for some application.

To solve this problem many receivers comes with an extra pin called PPM. this PPM pin transmits data of all channel in a single signal.

Step 1: How PPM Pin Signal Composed:

This signal consists of data packets separated by blank space.
Here space in between peaks represents the value of the channel. in this case, I have used 6 channel receiver so there are 7 pulses.

So in our code first, we need to detect separation space and then start recording data from the pulses for each channel.

As can be seen in the second image,

all data is separated with approx 11500 microseconds of time. than 6 values are for each channel.

Step 2: Making and Using Code:

Here read_me() specified as function:

a=micros(); c=a-b; b=a; x[i]=c; i=i+1;

if(i==15){for(int j=0;j<15;j++) {ch1[j]=x[j];} i=0; } }

this part runs on interrupt pin and take 15 time values and store it in array.

another function read_rc()

this part look for any space which is higher than 10000microsecond, in our case it detects separation space and as it detects that space code moves to next section.

after this space next 6 values are for each channel position it is stored in array named ch[ channel no],
here ch[1], ch[2], ch[3], ch[4], ch[5], ch[6] represents value of each chanel.

Step 3: Finally...

As we are using PPM signal with interrupt

  • This code do not add delay to our main functions
  • It consumes just 1 pin from Arduino



  • Tape Contest

    Tape Contest
  • Arduino Contest 2019

    Arduino Contest 2019
  • Trash to Treasure

    Trash to Treasure

15 Discussions


6 months ago on Step 3

Thanks a lot! Good idea! I optimised it a litte. Maybe this is interesting for someone else:

#define RECEIVE_PIN 2
#define DETECTION_SPACE 2500

int ch[CHANNEL_AMOUNT + 1];

void setup()
attachInterrupt(digitalPinToInterrupt(RECEIVE_PIN), ppm_interrupt, METHOD);

void loop()

void ppm_write()
static unsigned long int t;
if (millis() - t < 100)
return 0;
for (byte i = 0; i < CHANNEL_AMOUNT + 1; i++)
t = millis();

void ppm_interrupt()
static byte i;
static unsigned long int t_old;
unsigned long int t = micros(); //store time value a when pin value falling/rising
unsigned long int dt = t - t_old; //calculating time inbetween two peaks
t_old = t;

i = 0;
ch[i++] = dt;


Question 1 year ago

why nothing happens in my serial monitor if i use teensyduino instead arduino?


1 year ago on Step 2

Great explanatory video thanks a lot :) It's great to see in action this modulation!

alopez dominguez

1 year ago

Very usefull code!! i had to modify it a little bit because my receiver is usually on high output and the spikes go down instad of up, got it working fast, just one little question , how do the variables a , b and c work? i didnt quite understand em

1 reply

a,b and c is used to measure the spacing between two peaks in term of time.

when a new data comes we store it in "b" (last data already there as old data named: "a")
than take difference of "a" and "b" which is spacing in between two peaks.
now as we have used latest data it is not new data (which we store as "b"), so we update old data. and make value of "a" equal to "b".
now we again take new reading store it as "b".


so this process keep happens hundred of times per second.


1 year ago

Thanks for the guide it's great, although I couldn't get it to work.

I'm using Flysky TGY-iA6C RX - (PPM mode)

I tested first the pulseIn(3,HIGH) * had to set it to HIGH in order to get values, and it works fine, my separator is >6000 && < 9000.

When trying to use your full code example I only get -1000 values

Looking back at pulseIn values I noticed the number of values between separators change to somewhere between 8,9 even 12, so I don't know what's going on.

Is there any way to solve or address that ? thanks :)

1 reply

Reply 1 year ago


Can please confirm that signal from ppm pin goes to any interrupt enable pin ..

if that is ok. go to read_me function and check the output for variable "c". as mentioned in the code it is spacing between two peaks and should be similar spacing and channel value as data you capture using pulseIn function. if data in c is right than there may be some problem in code.


Reply 1 year ago

My receiver (of 6 channels) sends PPM for 8 channel (out of 2 are meaningless). I have chosen 15 just to make coding easy. so that, in any case, there is one set of 8 values which represents a value of each channel.
for an example
(here each number represents a channel number)
2 3 4 5 6 7 8 1 2 3 4 5 6 7 8
7 8 1 2 3 4 5 6 7 8 1 2 3 4 5
5 6 7 8 1 2 3 4 5 6 7 8 1 2 3
8 1 2 3 4 5 6 7 8 1 2 3 4 5 6
here we can just pick 8 value after channel 1

if you use only 8 values it will become
6 7 8 1 2 3 4 5
1 2 3 4 5 6 7 8
8 1 2 3 4 5 6 7
so, in this case, we need to make rearrangement which is also not very difficult but I preferred easier approach.

hope now you are clear about it. If you have any query let me know.


2 years ago

my receiver only has pwm outputs. is it okay if i add a ppm encoder between the receiver and the arduino?

1 reply

Reply 2 years ago

Yes, It should work.
if it not works, just check signal and do modification in code as per separation (step 1).

If you don't have any problem with a delay in the reading channel (around 15milliseconds/channel) or do not wont to purchase PPM encoder you can directly use the pulseIn function for each channel.