/* APM_RC.cpp - Radio Control Library for Ardupilot Mega. Arduino Code by Jordi Muņoz and Jose Julio. DIYDrones.com This library is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. RC Input : PPM signal on IC4 pin RC Output : 11 Servo outputs (standard 20ms frame) Methods: Init() : Initialization of interrupts an Timers OutpuCh(ch,pwm) : Output value to servos (range : 900-2100us) ch=0..10 InputCh(ch) : Read a channel input value. ch=0..7 GetState() : Returns the state of the input. 1 => New radio frame to process Automatically resets when we call InputCh to read channels */ #include "APM_RC.h" #include #include "WProgram.h" #if !defined(__AVR_ATmega1280__) && !defined(__AVR_ATmega2560__) # error Please check the Tools/Board menu to ensure you have selected Arduino Mega as your target. #else // Variable definition for Input Capture interrupt volatile unsigned int ICR4_old; volatile unsigned char PPM_Counter=0; volatile uint16_t PWM_RAW[8] = {2400,2400,2400,2400,2400,2400,2400,2400}; volatile unsigned char radio_status=0; /**************************************************** Input Capture Interrupt ICP4 => PPM signal read ****************************************************/ ISR(TIMER4_CAPT_vect) { unsigned int Pulse; unsigned int Pulse_Width; Pulse=ICR4; if (Pulse8000) // SYNC pulse? PPM_Counter=0; else { if (PPM_Counter < (sizeof(PWM_RAW) / sizeof(PWM_RAW[0]))) { PWM_RAW[PPM_Counter++]=Pulse_Width; //Saving pulse. if (PPM_Counter >= NUM_CHANNELS) radio_status = 1; } } ICR4_old = Pulse; } // Constructors //////////////////////////////////////////////////////////////// APM_RC_Class::APM_RC_Class() { } // Public Methods ////////////////////////////////////////////////////////////// void APM_RC_Class::Init(void) { // Init PPM input pinMode(49, INPUT); // ICP4 pin (PL0) (PPM input) // Init PWM Timer 1 pinMode(11,OUTPUT); // (PB5/OC1A) pinMode(12,OUTPUT); // OUT3 (PB6/OC1B) pinMode(13,OUTPUT); // OUT4 (PB7/OC1C) // Init PWM Timer 3 pinMode(2,OUTPUT); // OUT8 (PE4/OC3B) pinMode(3,OUTPUT); // OUT7 (PE5/OC3C) pinMode(4,OUTPUT); // (PE3/OC3A) // Init PWM Timer 4 // not avail // (PH3/OC4A) pinMode(7,OUTPUT); // OUT6 (PH4/OC4B) pinMode(8,OUTPUT); // OUT5 (PH5/OC4C) // Init PWM Timer 5 pinMode(44,OUTPUT); // OUT2 (PL5/OC5C) pinMode(45,OUTPUT); // OUT1 (PL4/OC5B) pinMode(46,OUTPUT); // (PL3/OC5A) TCCR1A =((1<> 1; // Because timer runs at 0.5us we need to do value / 2 result2 = PWM_RAW[ch] >> 1; if (result != result2) result = PWM_RAW[ch] >> 1; // if the results are different we make a third reading (this should be fine) // Limit values to a valid range result = constrain(result, MIN_PULSEWIDTH, MAX_PULSEWIDTH); radio_status = 0; // Radio channel read return(result); } unsigned char APM_RC_Class::GetState(void) { return(radio_status); } // InstantPWM implementation // This function forces the PWM output (reset PWM) on Out0 and Out1 (Timer5). For quadcopters use void APM_RC_Class::Force_Out0_Out1(void) { if (TCNT5>5000) // We take care that there are not a pulse in the output TCNT5=39990; // This forces the PWM output to reset in 5us (10 counts of 0.5us). The counter resets at 40000 } // This function forces the PWM output (reset PWM) on Out2 and Out3 (Timer1). For quadcopters use void APM_RC_Class::Force_Out2_Out3(void) { if (TCNT1>5000) TCNT1=39990; } // This function forces the PWM output (reset PWM) on Out6 and Out7 (Timer3). For quadcopters use void APM_RC_Class::Force_Out6_Out7(void) { if (TCNT3>5000) TCNT3=39990; } // allow HIL override of RC values // A value of -1 means no change // A value of 0 means no override, use the real RC values void APM_RC_Class::setHIL(int16_t v[NUM_CHANNELS]) { for (unsigned char i=0; i