| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181 |
- /*
- * Copyright (C) 2003 by egnite Software GmbH. All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- *
- * 1. Redistributions of source code must retain the above copyright
- * notice, this list of conditions and the following disclaimer.
- * 2. Redistributions in binary form must reproduce the above copyright
- * notice, this list of conditions and the following disclaimer in the
- * documentation and/or other materials provided with the distribution.
- * 3. Neither the name of the copyright holders nor the names of
- * contributors may be used to endorse or promote products derived
- * from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
- * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
- * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
- * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
- * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
- * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
- * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
- * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
- * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
- * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
- * SUCH DAMAGE.
- *
- * For additional information see http://www.ethernut.de/
- *
- */
- /*!
- * \file arch/avr/dev/irsony.c
- * \brief AVR support for Sony IR protocol.
- *
- * \verbatim
- * $Id: irsony.c 4706 2012-10-06 17:42:01Z haraldkipp $
- * \endverbatim
- */
- #include <cfg/medianut.h>
- #include <sys/atom.h>
- #include <sys/event.h>
- #ifdef __IMAGECRAFT__
- #pragma interrupt_handler INT4_vect:iv_INT4
- #pragma interrupt_handler TIMER2_OVF_vect:iv_TIMER2_OVF
- #endif
- /*!
- * \addtogroup xgIrSony
- */
- /*@{*/
- #include <dev/irsony.h>
- #define IRTIMER_START 0xF0
- #define IRTIMER_SCALE 0x01
- /*!
- * \brief Last valid infrared code received.
- */
- long nut_ircode;
- /*!
- * \brief Queue of threads waiting for infrared input.
- */
- HANDLE nut_irqueue;
- /*
- * Tick counter, incremented on each timer overflow and
- * cleared on each edge detection on the infrared signal.
- */
- static uint16_t irticks;
- /*
- * Number of bits received from infrared decoder.
- */
- static uint8_t irbitnum;
- /*! \fn TIMER2_OVF_vect(void)
- * \brief Timer 2 overflow handler.
- */
- SIGNAL(TIMER2_OVF_vect)
- {
- /* Set the timer value. */
- outb(TCNT2, IRTIMER_START);
- /*
- * Increment our tick counter. If it overflows, then wait
- * for the next start bit and disable the timer.
- */
- if (++irticks == 0) {
- irbitnum = 0;
- cbi(TIMSK, 6);
- }
- }
- /*! \fn INT4_vect(void)
- * \brief Infrared decoder signal edge handler.
- */
- SIGNAL(INT4_vect)
- {
- static uint16_t minset; /* Min. length of bit value 1, calculated from start bit. */
- static uint16_t ccode; /* Current code. */
- static uint16_t lcode; /* Last code. */
- static uint8_t ncode; /* Number of equal codes. */
- uint16_t bitlen = irticks; /* Length of the current bit. */
- irticks = 0;
- /* Rising egde marks end of bit. */
- if (inb(IR_SIGNAL_PIN) & _BV(IR_SIGNAL_BIT)) {
- /* Start bit. */
- if (irbitnum++ == 0) {
- ccode = 0;
- minset = bitlen / 3;
- }
- /* Data bits. */
- else {
- /* If its length is greater than a third of the start bit,
- then we assume this bit set. */
- if (bitlen >= minset)
- ccode |= _BV(12);
- ccode >>= 1;
- /* All data bits collected? */
- if (irbitnum > 12) {
- cbi(TIMSK, 6);
- irbitnum = 0;
- if (ncode++) {
- if (lcode != ccode)
- ncode = 0;
- /* If we have two equal codes, pass it to the
- application. */
- else if (ncode > 1) {
- ncode = 0;
- nut_ircode = ccode;
- NutEventPostFromIrq(&nut_irqueue);
- }
- } else
- lcode = ccode;
- }
- }
- }
- /* Falling edges mark the start of a bit. Enable timer
- overflow interrupts. */
- else if (irbitnum == 0) {
- sbi(TIMSK, 6);
- }
- }
- /*!
- * \brief Enable Sony infrared remote control.
- */
- int NutIrInitSony(void)
- {
- NutEnterCritical();
- /*
- * Initialize timer 2 and enable overflow interrupts.
- */
- outb(TCNT2, IRTIMER_START);
- outb(TCCR2, IRTIMER_SCALE);
- sbi(TIMSK, 6);
- /*
- * Enable infrared decoder interrupts on both edges.
- */
- cbi(IR_SIGNAL_DDR, IR_SIGNAL_BIT);
- sbi(EICR, 0);
- cbi(EICR, 1);
- sbi(EIMSK, IR_SIGNAL_BIT);
- NutExitCritical();
- return 0;
- }
- /*@}*/
|