/* * Authors (alphabetical order) * - Bertrand Songis * - Bryan J. Rentoul (Gruvin) * - Cameron Weeks * - Erez Raviv * - Jean-Pierre Parisy * - Karl Szmutny * - Michael Blandford * - Michal Hlavinka * - Pat Mackenzie * - Philip Moss * - Rob Thomson * - Romolo Manfredini * - Thomas Husterer * * open9x is based on code named * gruvin9x by Bryan J. Rentoul: http://code.google.com/p/gruvin9x/, * er9x by Erez Raviv: http://code.google.com/p/er9x/, * and the original (and ongoing) project by * Thomas Husterer, th9x: http://code.google.com/p/th9x/ * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. * * This program 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. * */ #include "open9x.h" #if defined(DSM2) // DSM2 control bits #define DSM2_CHANS 6 #define BIND_BIT 0x80 #define RANGECHECK_BIT 0x20 #define FRANCE_BIT 0x10 #define DSMX_BIT 0x08 #define BAD_DATA 0x47 #endif uint8_t s_current_protocol = 255; #ifdef DSM2_SERIAL inline void DSM2_EnableTXD(void) { UCSR0B |= (1 << TXEN0); // enable TX // UCSR0B |= (1 << UDRIE0); // don't enable UDRE0 interrupt, it will be enabled during next setupPulses } #endif void set_timer3_capture( void ) ; void set_timer3_ppm( void ) ; void setupPulsesPPM16( void ) ; void startPulses() { #ifndef SIMU setupPulses(); #ifdef DSM2_SERIAL if (g_model.protocol != PROTO_DSM2) #endif { #if defined(PCBV4) OCR1B = 0xffff; /* Prevent any PPM_PUT pin toggle before the TCNT1 interrupt fires for the first time and sets up the pulse period. */ // TCCR1A |= (1< g_tmr1Latency_max) g_tmr1Latency_max = dt; if (dt < g_tmr1Latency_min) g_tmr1Latency_min = dt; heartbeat |= HEART_TIMER_PULSES; } #endif #define PPM_CENTER 1500 FORCEINLINE void setupPulsesPPM() { int16_t PPM_range = g_model.extendedLimits ? 640*2 : 512*2; //range of 0.7..1.7msec //Total frame length = 22.5msec //each pulse is 0.7..1.7ms long with a 0.3ms stop tail //The pulse ISR is 2mhz that's why everything is multiplied by 2 // G: Found the following reference at th9x. The below code does not seem // to produce quite exactly this, to my eye. *shrug* // http://www.aerodesign.de/peter/2000/PCM/frame_ppm.gif uint16_t *ptr = (uint16_t *)pulses2MHz; uint8_t p = 8+(g_model.ppmNCH*2); // channels count uint16_t q = (g_model.ppmDelay*50+300)*2; //Stoplen *2 uint16_t rest = 22500u*2-q; //Minimum Framelen=22.5 ms rest += (int16_t(g_model.ppmFrameLength))*1000; for (uint8_t i=0; i>8)^pgm_read_word(&CRCTable[(PcmCrc^data) & 0xFF]); } void putPcmPart( uint8_t value ) { PcmByte >>= 2 ; PcmByte |= value ; if ( ++PcmBitCount >= 4 ) { *pulses2MHzWPtr++ = PcmByte ; PcmBitCount = PcmByte = 0 ; } } void putPcmFlush() { while ( PcmBitCount != 0 ) { putPcmPart( 0 ) ; // Empty } *pulses2MHzWPtr = 0 ; // Mark end } void putPcmBit( uint8_t bit ) { if ( bit ) { PcmOnesCount += 1 ; putPcmPart( 0x80 ) ; } else { PcmOnesCount = 0 ; putPcmPart( 0xC0 ) ; } if ( PcmOnesCount >= 5 ) { putPcmBit( 0 ) ; // Stuff a 0 bit in } } void putPcmByte( uint8_t byte ) { uint8_t i ; crc( byte ) ; for (i=0; i<8; i++) { putPcmBit( byte & 0x80 ) ; byte <<= 1 ; } } void putPcmHead() { // send 7E, do not CRC // 01111110 putPcmPart( 0xC0 ) ; putPcmPart( 0x80 ) ; putPcmPart( 0x80 ) ; putPcmPart( 0x80 ) ; putPcmPart( 0x80 ) ; putPcmPart( 0x80 ) ; putPcmPart( 0x80 ) ; putPcmPart( 0xC0 ) ; } void setupPulsesPXX() { uint8_t i ; uint16_t chan ; uint16_t chan_1 ; pulses2MHzWPtr = pulses2MHz; pulses2MHzRPtr = pulses2MHz; PcmCrc = 0 ; PcmBitCount = PcmByte = 0 ; PcmOnesCount = 0 ; putPcmHead() ; putPcmByte( g_model.modelId ) ; // putPcmByte( g_model.rxnum ) ; // putPcmByte( pxxFlag ) ; // First byte of flags putPcmByte( 0 ) ; // Second byte of flags pxxFlag = 0; // reset flag after send for ( i = 0 ; i < 8 ; i += 2 ) // First 8 channels only { chan = g_chans512[i] * 3 / 4 + 2250 ; chan_1 = g_chans512[i+1] * 3 / 4 + 2250 ; putPcmByte( chan ) ; // Low byte of channel putPcmByte( ( ( chan >> 8 ) & 0x0F ) | ( chan_1 << 4) ) ; // 4 bits each from 2 channels putPcmByte( chan_1 >> 4 ) ; // High byte of channel } chan = PcmCrc ; // get the crc putPcmByte( chan ) ; // Checksum lo putPcmByte( chan >> 8 ) ; // Checksum hi putPcmHead( ) ; putPcmFlush() ; OCR1C += 40000 ; // 20mS on PORTB |= (1<>5)+512,1023); *ptr++ = (i<<2) | ((pulse>>8)&0x03); *ptr++ = pulse & 0xff; } pulses2MHzWPtr = (uint8_t *)ptr; pulses2MHzRPtr = pulses2MHz; } void DSM2_Done() { UCSR0B &= ~((1 << TXEN0) | (1 << UDRIE0)); // disable UART TX and interrupt } void DSM2_Init(void) { #ifndef SIMU DDRE &= ~(1 << DDE0); // set RXD0 pin as input PORTE |= (1 << PORTE0); // enable pullup on RXD0 pin #undef BAUD #define BAUD 125000 #include UBRR0H = UBRRH_VALUE; UBRR0L = UBRRL_VALUE; UCSR0A &= ~(1 << U2X0); // disable double speed operation. // set 8N1 (leave TX and RX disabled for now) UCSR0B = 0 | (0 << RXCIE0) | (0 << TXCIE0) | (0 << UDRIE0) | (0 << RXEN0) | (0 << TXEN0) | (0 << UCSZ02); UCSR0C = 0 | (1 << UCSZ01) | (1 << UCSZ00); while (UCSR0A & (1 << RXC0)) UDR0; // flush receive buffer // These should be running right from power up on a FrSky enabled '9X. DSM2_EnableTXD(); // enable DSM2 UART transmitter #endif // SIMU } #endif #if defined(DSM2_PPM) inline void _send_1(uint8_t v) { *pulses2MHzWPtr++ = v; } #define BITLEN_DSM2 (8*2) //125000 Baud void sendByteDsm2(uint8_t b) //max 10changes 0 10 10 10 10 1 { bool lev = 0; uint8_t len = BITLEN_DSM2; //max val: 9*16 < 256 for( uint8_t i=0; i<=8; i++){ //8Bits + Stop=1 bool nlev = b & 1; //lsb first if(lev == nlev){ len += BITLEN_DSM2; }else{ _send_1(len -1); len = BITLEN_DSM2; lev = nlev; } b = (b>>1) | 0x80; //shift in stop bit } _send_1(len+BITLEN_DSM2-1); // 2 stop bits } void setupPulsesDsm2() { static uint8_t dsmDat[2+6*2] = {0xFF,0x00, 0x00,0xAA, 0x05,0xFF, 0x09,0xFF, 0x0D,0xFF, 0x13,0x54, 0x14,0xAA}; uint8_t counter; pulses2MHzWPtr = pulses2MHz; // If more channels needed make sure the pulses union/array is large enough if (dsmDat[0] & BAD_DATA) // first time through, setup header { switch(g_model.ppmNCH) { case LPXDSM2: dsmDat[0] = BIND_BIT; break; case DSM2only: dsmDat[0] = 0x90; break; default: dsmDat[0] = 0x98; // DSMX bind mode break; } } if ((dsmDat[0] & BIND_BIT) && (!keyState(SW_Trainer))) dsmDat[0] &= ~BIND_BIT; // clear bind bit if trainer not pulled // TODO find a way to do that, FUNC SWITCH: if ((!(dsmDat[0] & BIND_BIT)) && getSwitch(MAX_DRSWITCH-1, 0, 0)) dsmDat[0] |= RANGECHECK_BIT; // range check function // else dsmDat[0] &= ~RANGECHECK_BIT; dsmDat[1] = g_model.modelId; // DSM2 Header second byte for model match for (uint8_t i=0; i>5)+512,1023); dsmDat[2+2*i] = (i<<2) | ((pulse>>8)&0x03); dsmDat[3+2*i] = pulse & 0xff; } for (counter=0; counter<14; counter++) sendByteDsm2(dsmDat[counter]); pulses2MHzWPtr -= 1; //remove last stopbits and _send_1(255); //prolong them _send_1(0); //end of pulse stream pulses2MHzRPtr = pulses2MHz; } #endif void setupPulses() { if (s_current_protocol != g_model.protocol) { s_current_protocol = g_model.protocol; switch (g_model.protocol) { #if defined(DSM2_PPM) case PROTO_DSM2: set_timer3_capture() ; TCCR1B = 0; // Stop counter OCR1C = 200; // 100 uS TCNT1 = 300; // Past the OCR1C value ICR1 = 44000; // Next frame starts in 22 mS #if defined(PCBV4) TIMSK1 &= ~0x3C; // All interrupts off TIFR1 = 0x2F; TIMSK1 |= 0x28; // Enable CAPT and COMPB #else TIMSK &= ~0x3C; // All interrupts off TIFR = 0x3C; ETIFR = 0x3F ; TIMSK |= 0x20; // Enable CAPT ETIMSK |= (1< 200) PORTB |= (1<>= 1) == 0 ) { if ( *(++pulses2MHzRPtr) == 0 ) { OCR1B = OCR1C + 2000 ; // 1mS on from OCR1B } } else { *pulses2MHzRPtr = x; } heartbeat |= HEART_TIMER_PULSES; } #endif ISR(TIMER1_COMPC_vect) // DSM2 or PXX end of frame { #if defined(DSM2_PPM) && defined(PXX) if ( g_model.protocol == PROTO_DSM2 ) { #endif #if defined(DSM2_PPM) ICR1 = 41536 ; // next frame starts in 22ms 41536 = 2*(22000 - 14*11*8) if (OCR1C < 255) { OCR1C = 39000; // delay setup pulses by 19.5ms to reduce system latency } else { OCR1C = 200; // sei will be called inside setupPulses() setupPulses(); } #endif #if defined(DSM2_PPM) && defined(PXX) } else { #endif #if defined(PXX) // must be PXX setupPulses() ; #endif #if defined(DSM2_PPM) && defined(PXX) } #endif } #endif #endif void set_timer3_capture() { #ifndef SIMU #if defined (PCBV4) TIMSK3 &= ~( (1<