2011-09-30 19:12:22 -03:00
// -------------------------------------------------------------
2013-12-16 15:33:45 -04:00
// ArduPPM (PPM Encoder) V2.3.17
2011-09-30 19:12:22 -03:00
// -------------------------------------------------------------
2012-03-16 16:48:55 -03:00
// Improved servo to ppm for ArduPilot MEGA v1.x (ATmega328p),
2012-11-23 16:53:35 -04:00
// PhoneDrone and APM2.x (ATmega32u2)
2011-09-30 19:12:22 -03:00
2012-03-16 16:48:55 -03:00
// By: John Arne Birkeland - 2012
2011-09-30 19:12:22 -03:00
// APM v1.x adaptation and "difficult" receiver testing by Olivier ADLER
// -------------------------------------------------------------
2012-11-23 16:53:35 -04:00
// -------------------------------------------------------------
// ARDUPPM OPERATIONAL DESCRIPTION
// -------------------------------------------------------------
// APM 2.x LED STATUS:
// -------------------
// RX - OFF = No input signal detected
// RX - SLOW TOGGLE = Input signal OK
// RX - FAST TOGGLE = Invalid input signal(s) detected
// RX - ON = Input signal(s) lost during flight and fail-safe activated
// TX - OFF = PPM output disabled
// TX - FAST TOGGLE = PPM output enabled
// TX - SLOW TOGGLE = PPM pass-trough mode
// SERVO INPUT (PWM) MODE:
// -----------------------
// - PPM output will not be enabled unless a input signal has been detected and verified
// - Verified inputs are lost during operaton (lose servo wire or receiver malfunction):
2012-12-16 13:07:55 -04:00
// + The last known value of the lost input channel is keept for ~1 second
// + If the lost input channel is not restored within ~1 second, it will be set to the default fail-safe value
2012-11-23 16:53:35 -04:00
// - Lost channel signal is restored:
2012-12-16 13:07:55 -04:00
// + Normal channel operation is restored using the valid input signal
2012-11-23 16:53:35 -04:00
// PPM PASS-THROUGH MODE (signal pin 2&3 shorted):
// -----------------------------------------------
2013-12-16 15:33:45 -04:00
// - Any input level changes on pin 1 will be passed directly to the PPM output (PPM pass-trough)
// + If no input level changes are detected or if the input signals stop, there will be no PPM output
2012-11-23 16:53:35 -04:00
2011-09-30 19:12:22 -03:00
// Changelog:
// 01-08-2011
// V2.2.3 - Changed back to BLOCKING interrupts.
// Assembly PPM compare interrupt can be switch back to non-blocking, but not recommended.
// V2.2.3 - Implemented 0.5us cut filter to remove servo input capture jitter.
// 04-08-2011
// V2.2.4 - Implemented PPM passtrough funtion.
// Shorting channel 2&3 enabled ppm passtrough on channel 1.
// 04-08-2011
// V2.2.5 - Implemented simple average filter to smooth servo input capture jitter.
// Takes fewer clocks to execute and has better performance then cut filter.
// 05-08-2011
// V2.2.51 - Minor bug fixes.
// 06-08-2011
// V2.2.6 - PPM passtrough failsafe implemented.
// The PPM generator will be activated and output failsafe values while ppm passtrough signal is missing.
// 01-09-2011
// V2.2.61 - Temporary MUX pin always high patch for APM beta board
// 22-09-2011
// V2.2.62 - ATmegaXXU2 USB connection status pin (PC2) for APM UART MUX selection (removed temporary high patch)
// - Removed assembly optimized PPM generator (not usable for production release)
// 23-09-2011
// V2.2.63 - Average filter disabled
// 24-09-2011
// V2.2.64 - Added distincts Power on / Failsafe PPM values
// - Changed CH5 (mode selection) PPM Power on and Failsafe values to 1555 (Flight mode 4)
// - Added brownout detection : Failsafe values are copied after a brownout reset instead of power on values
// 25-09-2011
// V2.2.65 - Implemented PPM output delay until input signal is detected (PWM and PPM pass-trough mode)
// - Changed brownout detection and FailSafe handling to work with XXU2 chips
// - Minor variable and define naming changes to enhance readability
2012-03-16 16:48:55 -03:00
// 15-03-2012
// V2.2.66 - Added APM2 (ATmega32U2) support for using TX and RX status leds to indicate PWM and PPM traffic
2012-11-23 16:53:35 -04:00
// - <RX>: <OFF> = no pwm input detected, <TOGGLE> = speed of toggle indicate how many channel are active, <ON> = input lost (failsafe)
// - <TX>: <OFF> = ppm output not started, <FAST TOGGLE> = normal PWM->PPM output or PPM passtrough failsafe, <SLOW TOGGLE> = PPM passtrough
2012-03-16 16:48:55 -03:00
2012-06-02 21:42:39 -03:00
// 03-06-2012
// V2.2.67 - Implemented detection and failsafe (throttle = 900us) for missing throttle signal.
2012-06-04 17:55:15 -03:00
// 04-06-2012
// V2.2.68 - Fixed possible logic flaw in throttle failsafe reset if _JITTER_FILTER_ is enabled
2012-11-02 12:31:35 -03:00
// 02-11-2012
// V2.2.69 - Added PPM output positive polarity - mainly for standalone PPM encoder board use
2012-11-05 19:45:38 -04:00
// 03-11-2012
// V2.3.0 - Implemented single channel fail-safe detection for all inputs
2012-11-16 07:51:34 -04:00
// 16-11-2012
// V2.3.1 - Improved watchdog timer reset, so that only valid input signals will prevent the watchdog timer from triggering
2012-11-02 12:31:35 -03:00
2012-11-23 16:53:35 -04:00
// 22-11-2012
// V2.3.11 - Very experimental test forcing throttle fail-safe (RTL) on single channel loss. !DO NOT RELEASE TO PUBLIC!
// - Test for active input channels during init
// 23-11-2012 !VERY EXPERIMENTAL!
// V2.3.11 - Nested interrupt PWM output interrupt (PPM generator) for greatly improved jitter performance
// - Active input channel test during init removed
// - Implemented dynamic testing of active input channels
// 23-12-2012
// V2.3.12 - New improved fail-safe detection and handeling for single or multible signal loss and receiver malfuntion
// - Improved LED status for APM 2.x
// - Improved jitter performance (PPM output using nested interrupts)
2012-12-16 13:07:55 -04:00
// 30-11-2012
// V2.3.13pre - Internal dev only pre-release
// - Improved active channel detection requering 100 valid pulses before channel is marked active
// - Removed forced throttle fail-safe after channel loss
// - Changed fail-safe values to 800us, this will be used by APM code to detect channel loss and decide fail safe actions
// 16-12-2012
// V2.3.13 - Official release
// - Fail-safe vales changed back to normal fail-safe values. Use until APM code knows how to handle lost channel signal (800us)
2013-01-11 17:19:36 -04:00
// 10-01-2013
2013-01-10 21:45:26 -04:00
// V2.3.14pre - Internal test release
// - If one or more or all channel(s) are disconnected, throttle is set to fail-safe low (RTL)
// - If the misssing channel(s) are regained, throttle control is regained
2013-01-11 17:19:36 -04:00
// 11-01-2013
// V2.3.14 - temporary release for ArduCopter 2.9
// - fail-safe throttle low can be set with a define
2013-01-11 19:08:13 -04:00
// - recovery from error condition can also be set with a define
2013-01-10 21:45:26 -04:00
2013-01-13 23:53:18 -04:00
// 13-01-2013
// V2.3.15 - very small bug fix: speedup
2013-02-27 08:58:19 -04:00
// 27-02-2013
2013-03-01 19:13:07 -04:00
// V2.3.16 - if channel 1, 2 or 4 is disconnected: set it to center (1500us)
// - if channel 3 (throttle) is disconnected: set it to low (900us)
// - if channel 5-8 are disconnected: set it to last value
// - permanent LED error condition indication is only triggered when throttle is set low (or all channels are disconnected)
2013-02-27 18:58:00 -04:00
2013-12-16 15:33:45 -04:00
// 16-12-2013
// V2.3.17 - Disabled fail-safe output when signal is missing in PPM pass-trough (PPMSUM) mode (feature depreciated by addition of missing signal detection in APM code)
// - ppm_encoder_init() will now make sure PPM output always starts with fail-safe (900us) value on throttle after a brown-out reset
2013-02-27 08:58:19 -04:00
2011-09-30 19:12:22 -03:00
// -------------------------------------------------------------
2016-02-17 21:26:01 -04:00
# pragma once
2011-09-30 19:12:22 -03:00
# include <avr/io.h>
// -------------------------------------------------------------
# include <avr/interrupt.h>
# include <avr/wdt.h>
# include <util/delay.h>
2012-11-02 12:31:35 -03:00
2011-09-30 19:12:22 -03:00
// -------------------------------------------------------------
2012-11-23 16:53:35 -04:00
// SERVO INPUT FILTERS AND PARAMETERS
2011-09-30 19:12:22 -03:00
// -------------------------------------------------------------
// Using both filters is not recommended and may reduce servo input resolution
2012-11-23 16:53:35 -04:00
//#define _AVERAGE_FILTER_ // Average filter to smooth servo input capture jitter
# define _JITTER_FILTER_ 2 // Cut filter to remove servo input capture jitter (1 unit = 0.5us)
# define _INPUT_ERROR_TRIGGER_ 100 // Number of invalid input signals allowed before triggering alarm
2011-09-30 19:12:22 -03:00
// -------------------------------------------------------------
# ifndef F_CPU
2012-11-23 16:53:35 -04:00
# define F_CPU 16000000UL
2011-09-30 19:12:22 -03:00
# endif
# ifndef true
# define true 1
# endif
# ifndef false
# define false 0
# endif
# ifndef bool
# define bool _Bool
# endif
2012-11-05 19:45:38 -04:00
// Version stamp for firmware hex file ( decode hex file using <avr-objdump -s file.hex> and look for "ArduPPM" string )
2013-12-16 15:33:45 -04:00
const char ver [ 15 ] = " ArduPPMv2.3.17 " ;
2012-11-05 19:45:38 -04:00
2011-09-30 19:12:22 -03:00
// -------------------------------------------------------------
2012-11-02 12:31:35 -03:00
// INPUT MODE
2011-09-30 19:12:22 -03:00
// -------------------------------------------------------------
# define JUMPER_SELECT_MODE 0 // Default - PPM passtrough mode selected if channel 2&3 shorted. Normal servo input (pwm) if not shorted.
# define SERVO_PWM_MODE 1 // Normal 8 channel servo (pwm) input
# define PPM_PASSTROUGH_MODE 2 // PPM signal passtrough on channel 1
2012-11-02 12:31:35 -03:00
# define JETI_MODE 3 // Reserved
# define SPEKTRUM_MODE 4 // Reserved for Spektrum satelitte on channel 1
2011-09-30 19:12:22 -03:00
// Servo input mode (jumper (default), pwm, ppm, jeti or spektrum)
volatile uint8_t servo_input_mode = JUMPER_SELECT_MODE ;
2012-12-16 13:07:55 -04:00
// -------------------------------------------------------------
// FAILSAFE MODE
// -------------------------------------------------------------
//#define _APM_FAILSAFE_ // Used to spesify APM 800us channel loss fail safe values, remove to use normal fail safe values (stand alone encoder board)
2013-01-11 19:37:03 -04:00
//#define _THROTTLE_LOW_FAILSAFE_INDICATION //if set, throttle is set to low when a single channel is lost
//#define _THROTTLE_LOW_RECOVERY_POSSIBLE //if set, throttle low recovers from being low when the single channel comes back, only makes sense together with _THROTTLE_LOW_FAILSAFE_INDICATION
2013-01-11 19:08:13 -04:00
2013-12-16 15:33:45 -04:00
# define _PPM_PASSTROUGH_MODE_DISABLE_FAILSAFE_ // Disable fail-safe PPM generator and watchdog timer in ppm pass-trough (PPMSUM) mode
2013-01-11 19:08:13 -04:00
# if defined _THROTTLE_LOW_RECOVERY_POSSIBLE && !defined _THROTTLE_LOW_FAILSAFE_INDICATION
# error failsafe recovery is only possible with throttle_low_failsafe_indication defined as well
# endif
2013-01-11 17:19:36 -04:00
2012-12-16 13:07:55 -04:00
// -------------------------------------------------------------
// SERVO LIMIT VALUES
2011-09-30 19:12:22 -03:00
// -------------------------------------------------------------
// Number of Timer1 ticks in one microsecond
# define ONE_US F_CPU / 8 / 1000 / 1000
// 400us PPM pre pulse
# define PPM_PRE_PULSE ONE_US * 400
// Servo minimum position
# define PPM_SERVO_MIN ONE_US * 900 - PPM_PRE_PULSE
// Servo center position
# define PPM_SERVO_CENTER ONE_US * 1500 - PPM_PRE_PULSE
// Servo maximum position
# define PPM_SERVO_MAX ONE_US * 2100 - PPM_PRE_PULSE
// Throttle default at power on
# define PPM_THROTTLE_DEFAULT ONE_US * 1100 - PPM_PRE_PULSE
// Throttle during failsafe
# define PPM_THROTTLE_FAILSAFE ONE_US * 900 - PPM_PRE_PULSE
2012-12-16 13:07:55 -04:00
// Channel loss failsafe
# define PPM_CHANNEL_LOSS ONE_US * 800 - PPM_PRE_PULSE
2011-09-30 19:12:22 -03:00
// CH5 power on values (mode selection channel)
# define PPM_CH5_MODE_4 ONE_US * 1555 - PPM_PRE_PULSE
// -------------------------------------------------------------
2012-11-05 19:45:38 -04:00
// PPM OUTPUT SETTINGS
// -------------------------------------------------------------
2012-11-23 16:53:35 -04:00
// #define _POSITIVE_PPM_FRAME_ // Switch to positive pulse PPM
2012-11-05 19:45:38 -04:00
// (the actual timing is encoded in the length of the low between two pulses)
2011-09-30 19:12:22 -03:00
// Number of servo input channels
# define SERVO_CHANNELS 8
// PPM period 18.5ms - 26.5ms (54hz - 37Hz)
# define PPM_PERIOD ONE_US * ( 22500 - ( 8 * 1500 ) )
// Size of ppm[..] data array ( servo channels * 2 + 2)
# define PPM_ARRAY_MAX 18
2012-12-16 13:07:55 -04:00
# ifdef _APM_FAILSAFE_
// -------------------------------------------------------------
// APM FAILSAFE VALUES
// -------------------------------------------------------------
volatile uint16_t failsafe_ppm [ PPM_ARRAY_MAX ] =
{
PPM_PRE_PULSE ,
PPM_CHANNEL_LOSS , // Channel 1
PPM_PRE_PULSE ,
PPM_CHANNEL_LOSS , // Channel 2
PPM_PRE_PULSE ,
PPM_CHANNEL_LOSS , // Channel 3 (throttle)
PPM_PRE_PULSE ,
PPM_CHANNEL_LOSS , // Channel 4
PPM_PRE_PULSE ,
PPM_CHANNEL_LOSS , // Channel 5
PPM_PRE_PULSE ,
PPM_CHANNEL_LOSS , // Channel 6
PPM_PRE_PULSE ,
PPM_CHANNEL_LOSS , // Channel 7
PPM_PRE_PULSE ,
PPM_CHANNEL_LOSS , // Channel 8
PPM_PRE_PULSE ,
PPM_PERIOD
} ;
# else
2012-11-05 19:45:38 -04:00
// -------------------------------------------------------------
// SERVO FAILSAFE VALUES
// -------------------------------------------------------------
2012-11-23 16:53:35 -04:00
volatile uint16_t failsafe_ppm [ PPM_ARRAY_MAX ] =
2011-09-30 19:12:22 -03:00
{
PPM_PRE_PULSE ,
2012-11-05 19:45:38 -04:00
PPM_SERVO_CENTER , // Channel 1
2011-09-30 19:12:22 -03:00
PPM_PRE_PULSE ,
PPM_SERVO_CENTER , // Channel 2
PPM_PRE_PULSE ,
2012-11-05 19:45:38 -04:00
PPM_THROTTLE_FAILSAFE , // Channel 3 (throttle)
2011-09-30 19:12:22 -03:00
PPM_PRE_PULSE ,
PPM_SERVO_CENTER , // Channel 4
PPM_PRE_PULSE ,
PPM_CH5_MODE_4 , // Channel 5
PPM_PRE_PULSE ,
PPM_SERVO_CENTER , // Channel 6
PPM_PRE_PULSE ,
PPM_SERVO_CENTER , // Channel 7
PPM_PRE_PULSE ,
PPM_SERVO_CENTER , // Channel 8
PPM_PRE_PULSE ,
PPM_PERIOD
} ;
2012-12-16 13:07:55 -04:00
# endif
2011-09-30 19:12:22 -03:00
2012-11-02 12:31:35 -03:00
// -------------------------------------------------------------
2012-11-05 19:45:38 -04:00
// Data array for storing ppm (8 channels) pulse widths.
2011-09-30 19:12:22 -03:00
// -------------------------------------------------------------
2012-11-05 19:45:38 -04:00
volatile uint16_t ppm [ PPM_ARRAY_MAX ] =
2011-09-30 19:12:22 -03:00
{
PPM_PRE_PULSE ,
2012-11-05 19:45:38 -04:00
PPM_SERVO_CENTER , // Channel 1
2011-09-30 19:12:22 -03:00
PPM_PRE_PULSE ,
PPM_SERVO_CENTER , // Channel 2
PPM_PRE_PULSE ,
2012-11-05 19:45:38 -04:00
PPM_THROTTLE_DEFAULT , // Channel 3 (throttle)
2011-09-30 19:12:22 -03:00
PPM_PRE_PULSE ,
PPM_SERVO_CENTER , // Channel 4
PPM_PRE_PULSE ,
PPM_CH5_MODE_4 , // Channel 5
PPM_PRE_PULSE ,
PPM_SERVO_CENTER , // Channel 6
PPM_PRE_PULSE ,
PPM_SERVO_CENTER , // Channel 7
PPM_PRE_PULSE ,
PPM_SERVO_CENTER , // Channel 8
PPM_PRE_PULSE ,
PPM_PERIOD
} ;
2012-11-05 19:45:38 -04:00
// -------------------------------------------------------------
// Data arraw for storing ppm timeout (missing channel detection)
2011-09-30 19:12:22 -03:00
// -------------------------------------------------------------
2012-11-05 19:45:38 -04:00
# define PPM_TIMEOUT_VALUE 40 // ~1sec before triggering missing channel detection
volatile uint8_t ppm_timeout [ PPM_ARRAY_MAX ] ;
2012-12-16 13:07:55 -04:00
// Servo input channel connected status
# define SERVO_INPUT_CONNECTED_VALUE 100
volatile uint8_t servo_input_connected [ PPM_ARRAY_MAX ] ;
2011-09-30 19:12:22 -03:00
2013-01-11 19:08:13 -04:00
# ifdef _THROTTLE_LOW_RECOVERY_POSSIBLE
2013-01-10 21:45:26 -04:00
// count the channels which have been once connected but then got disconnected
volatile uint8_t disconnected_channels ;
2013-01-11 19:08:13 -04:00
# endif
2013-01-10 21:45:26 -04:00
2012-03-16 16:48:55 -03:00
// AVR parameters for PhoneDrone and APM2 boards using ATmega32u2
2011-09-30 19:12:22 -03:00
# if defined (__AVR_ATmega16U2__) || defined (__AVR_ATmega32U2__)
# define SERVO_DDR DDRB
# define SERVO_PORT PORTB
# define SERVO_INPUT PINB
# define SERVO_INT_VECTOR PCINT0_vect
# define SERVO_INT_MASK PCMSK0
# define SERVO_INT_CLEAR_FLAG PCIF0
# define SERVO_INT_ENABLE PCIE0
# define SERVO_TIMER_CNT TCNT1
# define PPM_DDR DDRC
# define PPM_PORT PORTC
# define PPM_OUTPUT_PIN PC6
# define PPM_INT_VECTOR TIMER1_COMPA_vect
# define PPM_COMPARE OCR1A
# define PPM_COMPARE_FLAG COM1A0
# define PPM_COMPARE_ENABLE OCIE1A
2012-11-02 20:54:57 -03:00
# define PPM_COMPARE_FORCE_MATCH FOC1A
2011-09-30 19:12:22 -03:00
# define USB_DDR DDRC
2012-12-16 13:07:55 -04:00
# define USB_PORT PORTC
2011-09-30 19:12:22 -03:00
# define USB_PIN PC2
2011-11-20 04:06:19 -04:00
// true if we have received a USB device connect event
static bool usb_connected ;
2011-09-30 19:12:22 -03:00
// USB connected event
void EVENT_USB_Device_Connect ( void )
{
// Toggle USB pin high if USB is connected
USB_PORT | = ( 1 < < USB_PIN ) ;
2011-11-20 04:06:19 -04:00
usb_connected = true ;
// this unsets the pin connected to PA1 on the 2560
// when the bit is clear, USB is connected
PORTD & = ~ 1 ;
2011-09-30 19:12:22 -03:00
}
2011-11-20 04:06:19 -04:00
// USB disconnect event
2011-09-30 19:12:22 -03:00
void EVENT_USB_Device_Disconnect ( void )
{
// toggle USB pin low if USB is disconnected
USB_PORT & = ~ ( 1 < < USB_PIN ) ;
2011-11-20 04:06:19 -04:00
usb_connected = false ;
// this sets the pin connected to PA1 on the 2560
// when the bit is clear, USB is connected
PORTD | = 1 ;
2011-09-30 19:12:22 -03:00
}
// AVR parameters for ArduPilot MEGA v1.4 PPM Encoder (ATmega328P)
# elif defined (__AVR_ATmega328P__) || defined (__AVR_ATmega328__)
# define SERVO_DDR DDRD
# define SERVO_PORT PORTD
# define SERVO_INPUT PIND
# define SERVO_INT_VECTOR PCINT2_vect
# define SERVO_INT_MASK PCMSK2
# define SERVO_INT_CLEAR_FLAG PCIF2
# define SERVO_INT_ENABLE PCIE2
# define SERVO_TIMER_CNT TCNT1
# define PPM_DDR DDRB
# define PPM_PORT PORTB
# define PPM_OUTPUT_PIN PB2
# define PPM_INT_VECTOR TIMER1_COMPB_vect
# define PPM_COMPARE OCR1B
# define PPM_COMPARE_FLAG COM1B0
# define PPM_COMPARE_ENABLE OCIE1B
2012-11-02 20:54:57 -03:00
# define PPM_COMPARE_FORCE_MATCH FOC1B
2011-09-30 19:12:22 -03:00
# else
# error NO SUPPORTED DEVICE FOUND! (ATmega16u2 / ATmega32u2 / ATmega328p)
# endif
2012-12-16 13:07:55 -04:00
2011-09-30 19:12:22 -03:00
// Used to indicate invalid SERVO input signals
volatile uint8_t servo_input_errors = 0 ;
// Used to indicate missing SERVO input signals
volatile bool servo_input_missing = true ;
// Used to indicate if PPM generator is active
volatile bool ppm_generator_active = false ;
// Used to indicate a brownout restart
volatile bool brownout_reset = false ;
2013-01-11 19:08:13 -04:00
# ifdef _THROTTLE_LOW_FAILSAFE_INDICATION
2013-01-10 21:45:26 -04:00
// Used to force throttle fail-safe mode (RTL)
volatile bool throttle_failsafe_force = false ;
2013-01-11 19:08:13 -04:00
# endif
2012-11-23 16:53:35 -04:00
2011-09-30 19:12:22 -03:00
// ------------------------------------------------------------------------------
// PPM GENERATOR START - TOGGLE ON COMPARE INTERRUPT ENABLE
// ------------------------------------------------------------------------------
void ppm_start ( void )
{
// Prevent reenabling an already active PPM generator
if ( ppm_generator_active ) return ;
// Store interrupt status and register flags
2012-11-23 16:53:35 -04:00
volatile uint8_t SREG_tmp = SREG ;
2011-09-30 19:12:22 -03:00
// Stop interrupts
cli ( ) ;
2012-11-23 16:53:35 -04:00
// Make sure initial output state is low
2011-09-30 19:12:22 -03:00
PPM_PORT & = ~ ( 1 < < PPM_OUTPUT_PIN ) ;
2012-11-02 12:31:35 -03:00
2011-09-30 19:12:22 -03:00
// Wait for output pin to settle
//_delay_us( 1 );
// Set initial compare toggle to maximum (32ms) to give other parts of the system time to start
SERVO_TIMER_CNT = 0 ;
PPM_COMPARE = 0xFFFF ;
// Set toggle on compare output
TCCR1A = ( 1 < < PPM_COMPARE_FLAG ) ;
// Set TIMER1 8x prescaler
TCCR1B = ( 1 < < CS11 ) ;
2012-11-23 16:53:35 -04:00
# if defined (_POSITIVE_PPM_FRAME_)
2012-11-02 20:54:57 -03:00
// Force output compare to reverse polarity
TCCR1C | = ( 1 < < PPM_COMPARE_FORCE_MATCH ) ;
# endif
2011-09-30 19:12:22 -03:00
// Enable output compare interrupt
TIMSK1 | = ( 1 < < PPM_COMPARE_ENABLE ) ;
// Indicate that PPM generator is active
ppm_generator_active = true ;
2012-11-23 16:53:35 -04:00
# if defined (__AVR_ATmega16U2__) || defined (__AVR_ATmega32U2__)
// Turn on TX led if PPM generator is active
PORTD & = ~ ( 1 < < PD5 ) ;
# endif
2011-09-30 19:12:22 -03:00
// Restore interrupt status and register flags
SREG = SREG_tmp ;
}
// ------------------------------------------------------------------------------
// PPM GENERATOR STOP - TOGGLE ON COMPARE INTERRUPT DISABLE
// ------------------------------------------------------------------------------
void ppm_stop ( void )
{
// Store interrupt status and register flags
2012-11-23 16:53:35 -04:00
volatile uint8_t SREG_tmp = SREG ;
2011-09-30 19:12:22 -03:00
// Stop interrupts
cli ( ) ;
// Disable output compare interrupt
TIMSK1 & = ~ ( 1 < < PPM_COMPARE_ENABLE ) ;
// Reset TIMER1 registers
TCCR1A = 0 ;
TCCR1B = 0 ;
// Indicate that PPM generator is not active
ppm_generator_active = false ;
2012-11-23 16:53:35 -04:00
# if defined (__AVR_ATmega16U2__) || defined (__AVR_ATmega32U2__)
// Turn off TX led if PPM generator is off
PORTD | = ( 1 < < PD5 ) ;
# endif
2011-09-30 19:12:22 -03:00
// Restore interrupt status and register flags
SREG = SREG_tmp ;
}
// ------------------------------------------------------------------------------
// Watchdog Interrupt (interrupt only mode, no reset)
// ------------------------------------------------------------------------------
ISR ( WDT_vect ) // If watchdog is triggered then enable missing signal flag and copy power on or failsafe positions
{
// Use failsafe values if PPM generator is active or if chip has been reset from a brown-out
if ( ppm_generator_active | | brownout_reset )
{
// Copy failsafe values to ppm[..]
2013-02-26 15:20:26 -04:00
//for( uint8_t i = 0; i < PPM_ARRAY_MAX; i++ )
//{
// ppm[ i ] = failsafe_ppm[ i ];
//}
// Set Throttle Low & leave other channels at last value
ppm [ 5 ] = failsafe_ppm [ 5 ] ;
2012-11-23 16:53:35 -04:00
}
2011-09-30 19:12:22 -03:00
// If we are in PPM passtrough mode and a input signal has been detected, or if chip has been reset from a brown_out then start the PPM generator.
if ( ( servo_input_mode = = PPM_PASSTROUGH_MODE & & servo_input_missing = = false ) | | brownout_reset )
{
// Start PPM generator
ppm_start ( ) ;
brownout_reset = false ;
2012-11-23 16:53:35 -04:00
}
# if defined (__AVR_ATmega16U2__) || defined (__AVR_ATmega32U2__)
// Turn on RX led if signal is missing
if ( ! servo_input_missing )
{
PORTD & = ~ ( 1 < < PD4 ) ;
}
# endif
2011-09-30 19:12:22 -03:00
// Set missing receiver signal flag
servo_input_missing = true ;
// Reset servo input error flag
servo_input_errors = 0 ;
2012-11-23 16:53:35 -04:00
2011-09-30 19:12:22 -03:00
}
// ------------------------------------------------------------------------------
// ------------------------------------------------------------------------------
// SERVO/PPM INPUT - PIN CHANGE INTERRUPT
// ------------------------------------------------------------------------------
ISR ( SERVO_INT_VECTOR )
{
// Servo pulse start timing
2012-11-23 16:53:35 -04:00
static uint16_t servo_start [ PPM_ARRAY_MAX ] = { 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 , 0 } ; // We sacrefice some memory but save instructions by working with ppm index count (18) instead of input channel count (8)
2012-03-16 16:48:55 -03:00
2012-11-23 16:53:35 -04:00
# if defined (__AVR_ATmega16U2__) || defined (__AVR_ATmega32U2__)
// Toggle LED delay
static uint8_t led_delay = 0 ;
# endif
2012-06-02 21:42:39 -03:00
2012-11-23 16:53:35 -04:00
// Servo input pin storage
2011-09-30 19:12:22 -03:00
static uint8_t servo_pins_old = 0 ;
// Used to store current servo input pins
uint8_t servo_pins ;
2012-11-23 16:53:35 -04:00
uint8_t servo_change ;
uint8_t servo_pin ;
uint8_t ppm_channel ;
2011-09-30 19:12:22 -03:00
// Read current servo pulse change time
uint16_t servo_time = SERVO_TIMER_CNT ;
// ------------------------------------------------------------------------------
// PPM passtrough mode ( signal passtrough from channel 1 to ppm output pin)
// ------------------------------------------------------------------------------
if ( servo_input_mode = = PPM_PASSTROUGH_MODE )
{
// Has watchdog timer failsafe started PPM generator? If so we need to stop it.
if ( ppm_generator_active )
{
// Stop PPM generator
ppm_stop ( ) ;
}
// PPM (channel 1) input pin is high
if ( SERVO_INPUT & 1 )
{
// Set PPM output pin high
PPM_PORT | = ( 1 < < PPM_OUTPUT_PIN ) ;
}
// PPM (channel 1) input pin is low
else
{
// Set PPM output pin low
PPM_PORT & = ~ ( 1 < < PPM_OUTPUT_PIN ) ;
}
// Reset Watchdog Timer
2013-12-16 15:33:45 -04:00
# ifndef _PPM_PASSTROUGH_MODE_DISABLE_FAILSAFE_
2011-09-30 19:12:22 -03:00
wdt_reset ( ) ;
2013-12-16 15:33:45 -04:00
# endif
2011-09-30 19:12:22 -03:00
// Set servo input missing flag false to indicate that we have received servo input signals
servo_input_missing = false ;
2012-11-23 16:53:35 -04:00
# if defined (__AVR_ATmega16U2__) || defined (__AVR_ATmega32U2__)
// Toggle TX LED at PPM passtrough
if ( + + led_delay > 128 ) // Toggle every 128th pulse
{
// Toggle TX led
PIND | = ( 1 < < PD5 ) ;
led_delay = 0 ;
}
# endif
2011-09-30 19:12:22 -03:00
// Leave interrupt
return ;
}
// ------------------------------------------------------------------------------
// SERVO PWM MODE
// ------------------------------------------------------------------------------
2012-11-23 16:53:35 -04:00
2012-12-16 13:07:55 -04:00
//CHECK_PINS_START: // Start of servo input check
2011-09-30 19:12:22 -03:00
// Store current servo input pins
servo_pins = SERVO_INPUT ;
// Calculate servo input pin change mask
2012-11-23 16:53:35 -04:00
servo_change = servo_pins ^ servo_pins_old ;
// Set initial servo pin and ppm[..] index
servo_pin = 1 ;
ppm_channel = 1 ;
2011-09-30 19:12:22 -03:00
CHECK_PINS_LOOP : // Input servo pin check loop
// Check for pin change on current servo channel
if ( servo_change & servo_pin )
{
2012-11-23 16:53:35 -04:00
// Remove processed pin change from bitmask
servo_change & = ~ servo_pin ;
2011-09-30 19:12:22 -03:00
// High (raising edge)
if ( servo_pins & servo_pin )
{
2012-11-23 16:53:35 -04:00
servo_start [ ppm_channel ] = servo_time ;
2011-09-30 19:12:22 -03:00
}
else
{
// Get servo pulse width
2012-11-23 16:53:35 -04:00
uint16_t servo_width = servo_time - servo_start [ ppm_channel ] - PPM_PRE_PULSE ;
2011-09-30 19:12:22 -03:00
// Check that servo pulse signal is valid before sending to ppm encoder
if ( servo_width > PPM_SERVO_MAX ) goto CHECK_PINS_ERROR ;
if ( servo_width < PPM_SERVO_MIN ) goto CHECK_PINS_ERROR ;
2012-11-23 16:53:35 -04:00
// Reset Watchdog Timer
wdt_reset ( ) ;
2013-12-16 15:33:45 -04:00
2012-11-23 16:53:35 -04:00
// Set servo input missing flag false to indicate that we have received servo input signals
servo_input_missing = false ;
2012-12-16 13:07:55 -04:00
// Count valid signals to mark channel active
if ( servo_input_connected [ ppm_channel ] < SERVO_INPUT_CONNECTED_VALUE )
2012-11-23 16:53:35 -04:00
{
2012-12-16 13:07:55 -04:00
servo_input_connected [ ppm_channel ] + + ;
2012-11-23 16:53:35 -04:00
}
2013-01-10 21:45:26 -04:00
//Reset ppm single channel fail-safe timeout
2012-12-16 13:07:55 -04:00
ppm_timeout [ ppm_channel ] = 0 ;
2013-01-10 21:45:26 -04:00
2013-01-11 17:19:36 -04:00
# ifdef _THROTTLE_LOW_FAILSAFE_INDICATION
2013-01-10 21:45:26 -04:00
// Check for forced throttle fail-safe
if ( throttle_failsafe_force )
{
if ( ppm_channel = = 5 )
{
// Force throttle fail-safe
servo_width = PPM_THROTTLE_FAILSAFE ;
}
}
2013-01-11 17:19:36 -04:00
# endif
2012-12-16 13:07:55 -04:00
2011-09-30 19:12:22 -03:00
# ifdef _AVERAGE_FILTER_
// Average filter to smooth input jitter
2012-11-23 16:53:35 -04:00
servo_width + = ppm [ ppm_channel ] ;
2011-09-30 19:12:22 -03:00
servo_width > > = 1 ;
# endif
# ifdef _JITTER_FILTER_
// 0.5us cut filter to remove input jitter
2012-11-23 16:53:35 -04:00
int16_t ppm_tmp = ppm [ ppm_channel ] - servo_width ;
if ( ppm_tmp < = _JITTER_FILTER_ & & ppm_tmp > = - _JITTER_FILTER_ ) goto CHECK_PINS_NEXT ;
2011-09-30 19:12:22 -03:00
# endif
// Update ppm[..]
2012-11-23 16:53:35 -04:00
ppm [ ppm_channel ] = servo_width ;
2011-09-30 19:12:22 -03:00
}
}
CHECK_PINS_NEXT :
2012-11-23 16:53:35 -04:00
// Are we done processing pins?
if ( servo_change )
{
// Select next ppm[..] index
ppm_channel + = 2 ;
2011-09-30 19:12:22 -03:00
2012-11-23 16:53:35 -04:00
// Select next servo pin
servo_pin < < = 1 ;
// Check next pin
goto CHECK_PINS_LOOP ;
}
2011-09-30 19:12:22 -03:00
2012-11-23 16:53:35 -04:00
// All changed pins have been checked
2011-09-30 19:12:22 -03:00
goto CHECK_PINS_DONE ;
CHECK_PINS_ERROR :
// Used to indicate invalid servo input signals
servo_input_errors + + ;
goto CHECK_PINS_NEXT ;
2012-11-23 16:53:35 -04:00
// Processing done, cleanup and exit
2011-09-30 19:12:22 -03:00
CHECK_PINS_DONE :
2012-11-23 16:53:35 -04:00
// Start PPM generator if not already running
if ( ! ppm_generator_active ) ppm_start ( ) ;
# if defined (__AVR_ATmega16U2__) || defined (__AVR_ATmega32U2__)
// Toggle RX LED when finished receiving servo pulses
if ( + + led_delay = = 0 ) // Toggle led every 128th or 255th interval
{
PIND | = ( 1 < < PD4 ) ;
// Fast toggle on servo input errors
if ( servo_input_errors > _INPUT_ERROR_TRIGGER_ )
{
led_delay = 223 ;
}
}
# endif
2011-09-30 19:12:22 -03:00
// Store current servo input pins for next check
servo_pins_old = servo_pins ;
2012-06-02 21:42:39 -03:00
//Has servo input changed while processing pins, if so we need to re-check pins
2012-11-23 16:53:35 -04:00
//if( servo_pins != SERVO_INPUT ) goto CHECK_PINS_START;
2012-06-02 21:42:39 -03:00
// Clear interrupt event from already processed pin changes
2012-11-23 16:53:35 -04:00
//PCIFR |= (1 << SERVO_INT_CLEAR_FLAG);
2011-09-30 19:12:22 -03:00
}
// ------------------------------------------------------------------------------
// ------------------------------------------------------------------------------
// PPM OUTPUT - TIMER1 COMPARE INTERRUPT
// ------------------------------------------------------------------------------
2012-11-23 16:53:35 -04:00
// Current active ppm channel
volatile uint8_t ppm_out_channel = PPM_ARRAY_MAX - 1 ;
ISR ( PPM_INT_VECTOR , ISR_NOBLOCK )
2011-09-30 19:12:22 -03:00
{
2012-11-23 16:53:35 -04:00
// ------------------------------------------------------------------------------
// !! NESTED INTERRUPT !!
// - ALL VARIABLES SHOULD BE GLOBAL VOLATILE
// - ACCESSING VARIABLES >8BIT MUST BE DONE ATOMIC USING CLI/SEI
// ------------------------------------------------------------------------------
2011-09-30 19:12:22 -03:00
2012-11-23 16:53:35 -04:00
// Update timing for next compare toggle with either current ppm input value, or fail-safe value if there is a channel timeout.
if ( ppm_timeout [ ppm_out_channel ] > PPM_TIMEOUT_VALUE )
2012-11-05 19:45:38 -04:00
{
2013-02-27 08:58:19 -04:00
// Channel 1-4?
2013-02-27 09:15:28 -04:00
if ( ppm_out_channel < 8 )
2013-02-26 22:45:32 -04:00
{
2013-02-27 08:58:19 -04:00
// Channel 1-4 - Use fail-safe value
2013-02-26 22:45:32 -04:00
cli ( ) ;
PPM_COMPARE + = failsafe_ppm [ ppm_out_channel ] ;
2013-02-26 15:20:26 -04:00
sei ( ) ;
2013-02-27 08:58:19 -04:00
// Channel 3
if ( ppm_out_channel = = 5 )
{
// report this to the LED indication
servo_input_missing = true ;
}
2013-02-26 15:20:26 -04:00
}
2013-02-27 08:58:19 -04:00
else
2013-02-26 15:20:26 -04:00
{
2013-02-27 08:58:19 -04:00
// Channel 5-8 - Use last known value
2013-02-26 15:20:26 -04:00
cli ( ) ;
PPM_COMPARE + = ppm [ ppm_out_channel ] ;
sei ( ) ;
}
2013-02-27 08:58:19 -04:00
2013-01-11 19:08:13 -04:00
# if defined _THROTTLE_LOW_RECOVERY_POSSIBLE && defined _THROTTLE_LOW_FAILSAFE_INDICATION
2013-01-10 21:45:26 -04:00
// Count the channel that we have lost
2013-01-14 00:30:35 -04:00
disconnected_channels + + ;
2013-01-11 19:08:13 -04:00
# elif defined _THROTTLE_LOW_FAILSAFE_INDICATION
throttle_failsafe_force = true ;
# endif
2013-01-10 21:45:26 -04:00
2012-12-16 13:07:55 -04:00
# if defined (__AVR_ATmega16U2__) || defined (__AVR_ATmega32U2__)
// Turn on RX LED to indicate a fail-safe condition
PORTD & = ~ ( 1 < < PD4 ) ;
# endif
2012-11-05 19:45:38 -04:00
}
else
{
2012-11-23 16:53:35 -04:00
// Use latest ppm input value
cli ( ) ;
PPM_COMPARE + = ppm [ ppm_out_channel ] ;
2013-02-26 15:20:26 -04:00
sei ( ) ;
2012-11-05 19:45:38 -04:00
2012-12-16 13:07:55 -04:00
// Increment active channel timeout (reset to zero in input interrupt each time a valid signal is detected)
if ( servo_input_connected [ ppm_out_channel ] > = SERVO_INPUT_CONNECTED_VALUE )
{
ppm_timeout [ ppm_out_channel ] + + ;
}
2012-11-05 19:45:38 -04:00
}
2012-12-16 13:07:55 -04:00
2012-11-23 16:53:35 -04:00
if ( + + ppm_out_channel > = PPM_ARRAY_MAX )
{
ppm_out_channel = 0 ;
2013-02-26 15:20:26 -04:00
# ifdef _THROTTLE_LOW_RECOVERY_POSSIBLE
2013-01-11 19:08:13 -04:00
// Did we lose one or more active servo input channel? If so force throttle fail-safe (RTL)
2013-01-10 21:45:26 -04:00
if ( disconnected_channels > 0 )
{
throttle_failsafe_force = true ;
2013-01-13 23:53:18 -04:00
disconnected_channels = 0 ;
2013-01-10 21:45:26 -04:00
}
else
{
throttle_failsafe_force = false ;
}
2013-01-11 19:08:13 -04:00
# endif
2013-01-10 21:45:26 -04:00
2012-11-23 16:53:35 -04:00
# if defined (__AVR_ATmega16U2__) || defined (__AVR_ATmega32U2__)
// Blink TX LED when PPM generator has finished a pulse train
PIND | = ( 1 < < PD5 ) ;
# endif
}
2011-09-30 19:12:22 -03:00
}
// ------------------------------------------------------------------------------
// ------------------------------------------------------------------------------
// PPM READ - INTERRUPT SAFE PPM SERVO CHANNEL READ
// ------------------------------------------------------------------------------
uint16_t ppm_read_channel ( uint8_t channel )
{
// Limit channel to valid value
uint8_t _channel = channel ;
if ( _channel = = 0 ) _channel = 1 ;
if ( _channel > SERVO_CHANNELS ) _channel = SERVO_CHANNELS ;
// Calculate ppm[..] position
uint8_t ppm_index = ( _channel < < 1 ) + 1 ;
// Read ppm[..] in a non blocking interrupt safe manner
uint16_t ppm_tmp = ppm [ ppm_index ] ;
while ( ppm_tmp ! = ppm [ ppm_index ] ) ppm_tmp = ppm [ ppm_index ] ;
// Return as normal servo value
return ppm_tmp + PPM_PRE_PULSE ;
}
// ------------------------------------------------------------------------------
// ------------------------------------------------------------------------------
// PPM ENCODER INIT
// ------------------------------------------------------------------------------
void ppm_encoder_init ( void )
{
// ATmegaXXU2 only init code
// ------------------------------------------------------------------------------
# if defined (__AVR_ATmega16U2__) || defined (__AVR_ATmega32U2__)
// ------------------------------------------------------------------------------
// Reset Source checkings
// ------------------------------------------------------------------------------
if ( MCUSR & 1 ) // Power-on Reset
{
MCUSR = 0 ; // Clear MCU Status register
// custom code here
}
else if ( MCUSR & 2 ) // External Reset
{
MCUSR = 0 ; // Clear MCU Status register
// custom code here
}
else if ( MCUSR & 4 ) // Brown-Out Reset
{
MCUSR = 0 ; // Clear MCU Status register
brownout_reset = true ;
2013-12-16 15:33:45 -04:00
// Make sure PPM generator will always start directly with throttle F/S value (900us) after a brown-out reset
ppm [ 5 ] = failsafe_ppm [ 5 ] ;
2011-09-30 19:12:22 -03:00
}
else // Watchdog Reset
{
MCUSR = 0 ; // Clear MCU Status register
// custom code here
}
// APM USB connection status UART MUX selector pin
// ------------------------------------------------------------------------------
USB_DDR | = ( 1 < < USB_PIN ) ; // Set USB pin to output
2012-11-23 16:53:35 -04:00
DDRD | = ( 1 < < PD4 ) ; // RX LED OUTPUT
DDRD | = ( 1 < < PD5 ) ; // TX LED OUTPUT
PORTD | = ( 1 < < PD4 ) ; // RX LED OFF
PORTD | = ( 1 < < PD5 ) ; // TX LED OFF
2011-09-30 19:12:22 -03:00
# endif
// USE JUMPER TO CHECK FOR PWM OR PPM PASSTROUGH MODE (channel 2&3 shorted)
// ------------------------------------------------------------------------------
if ( servo_input_mode = = JUMPER_SELECT_MODE )
{
// channel 3 status counter
uint8_t channel3_status = 0 ;
// Set channel 3 to input
SERVO_DDR & = ~ ( 1 < < 2 ) ;
// Enable channel 3 pullup
SERVO_PORT | = ( 1 < < 2 ) ;
// Set channel 2 to output
SERVO_DDR | = ( 1 < < 1 ) ;
// Set channel 2 output low
SERVO_PORT & = ~ ( 1 < < 1 ) ;
_delay_us ( 10 ) ;
// Increment channel3_status if channel 3 is set low by channel 2
if ( ( SERVO_INPUT & ( 1 < < 2 ) ) = = 0 ) channel3_status + + ;
// Set channel 2 output high
SERVO_PORT | = ( 1 < < 1 ) ;
_delay_us ( 10 ) ;
// Increment channel3_status if channel 3 is set high by channel 2
if ( ( SERVO_INPUT & ( 1 < < 2 ) ) ! = 0 ) channel3_status + + ;
// Set channel 2 output low
SERVO_PORT & = ~ ( 1 < < 1 ) ;
_delay_us ( 10 ) ;
// Increment channel3_status if channel 3 is set low by channel 2
if ( ( SERVO_INPUT & ( 1 < < 2 ) ) = = 0 ) channel3_status + + ;
// Set servo input mode based on channel3_status
if ( channel3_status = = 3 ) servo_input_mode = PPM_PASSTROUGH_MODE ;
else servo_input_mode = SERVO_PWM_MODE ;
}
// SERVO/PPM INPUT PINS
// ------------------------------------------------------------------------------
// Set all servo input pins to inputs
SERVO_DDR = 0 ;
// Activate pullups on all input pins
SERVO_PORT | = 0xFF ;
2011-11-20 04:06:19 -04:00
# if defined (__AVR_ATmega16U2__) || defined (__AVR_ATmega32U2__)
// on 32U2 set PD0 to be an output, and clear the bit. This tells
// the 2560 that USB is connected. The USB connection event fires
// later to set the right value
DDRD | = 1 ;
if ( usb_connected ) {
2012-11-23 16:53:35 -04:00
PORTD & = ~ 1 ;
2011-11-20 04:06:19 -04:00
} else {
2012-11-23 16:53:35 -04:00
PORTD | = 1 ;
2011-11-20 04:06:19 -04:00
}
# endif
2011-09-30 19:12:22 -03:00
2012-11-23 16:53:35 -04:00
// PPM PASS-THROUGH MODE
2011-09-30 19:12:22 -03:00
if ( servo_input_mode = = PPM_PASSTROUGH_MODE )
{
// Set servo input interrupt pin mask to servo input channel 1
SERVO_INT_MASK = 0x01 ;
}
2012-11-23 16:53:35 -04:00
// SERVO PWM INPUT MODE
// ------------------------------------------------------------------------------
if ( servo_input_mode = = SERVO_PWM_MODE )
{
// Interrupt on all input pins
SERVO_INT_MASK = 0xFF ;
}
2011-09-30 19:12:22 -03:00
// Enable servo input interrupt
PCICR | = ( 1 < < SERVO_INT_ENABLE ) ;
2012-11-23 16:53:35 -04:00
// PPM OUTPUT
2011-09-30 19:12:22 -03:00
// ------------------------------------------------------------------------------
2012-11-23 16:53:35 -04:00
// PPM generator (PWM output timer/counter) is started either by pin change interrupt or by watchdog interrupt
2011-09-30 19:12:22 -03:00
// Set PPM pin to output
PPM_DDR | = ( 1 < < PPM_OUTPUT_PIN ) ;
2012-11-23 16:53:35 -04:00
2011-09-30 19:12:22 -03:00
// ------------------------------------------------------------------------------
// Enable watchdog interrupt mode
// ------------------------------------------------------------------------------
2013-12-16 15:33:45 -04:00
# ifdef _PPM_PASSTROUGH_MODE_DISABLE_FAILSAFE_
if ( servo_input_mode ! = PPM_PASSTROUGH_MODE )
# endif
{
// Disable watchdog
wdt_disable ( ) ;
// Reset watchdog timer
wdt_reset ( ) ;
// Start timed watchdog setup sequence
WDTCSR | = ( 1 < < WDCE ) | ( 1 < < WDE ) ;
// Set 250 ms watchdog timeout and enable interrupt
WDTCSR = ( 1 < < WDIE ) | ( 1 < < WDP2 ) ;
}
2011-09-30 19:12:22 -03:00
}
// ------------------------------------------------------------------------------