2010-07-27 03:31:45 +04:00
/*
2016-11-22 00:55:53 +03:00
* serial_ir . c
2010-07-27 03:31:45 +04:00
*
2016-11-22 00:55:53 +03:00
* serial_ir - Device driver that records pulse - and pause - lengths
2010-07-27 03:31:45 +04:00
* ( space - lengths ) between DDCD event on a serial port .
*
* Copyright ( C ) 1996 , 97 Ralph Metzler < rjkm @ thp . uni - koeln . de >
* Copyright ( C ) 1998 Trent Piepho < xyzzy @ u . washington . edu >
* Copyright ( C ) 1998 Ben Pfaff < blp @ gnu . org >
* Copyright ( C ) 1999 Christoph Bartelmus < lirc @ bartelmus . de >
* Copyright ( C ) 2007 Andrei Tanas < andrei @ tanas . ca > ( suspend / resume support )
2016-11-22 00:55:51 +03:00
* Copyright ( C ) 2016 Sean Young < sean @ mess . org > ( port to rc - core )
2010-07-27 03:31:45 +04:00
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation ; either version 2 of the License , or
* ( at your option ) any later version .
*
* 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 .
*/
2012-11-08 22:55:09 +04:00
# define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
2010-07-27 03:31:45 +04:00
# include <linux/module.h>
# include <linux/errno.h>
# include <linux/interrupt.h>
# include <linux/kernel.h>
# include <linux/serial_reg.h>
# include <linux/types.h>
# include <linux/delay.h>
# include <linux/platform_device.h>
# include <linux/spinlock.h>
2016-11-22 00:55:51 +03:00
# include <media/rc-core.h>
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:51 +03:00
struct serial_ir_hw {
2010-07-27 03:31:45 +04:00
int signal_pin ;
int signal_pin_change ;
u8 on ;
u8 off ;
2016-11-22 00:55:51 +03:00
unsigned set_send_carrier : 1 ;
unsigned set_duty_cycle : 1 ;
2016-11-22 00:55:52 +03:00
void ( * send_pulse ) ( unsigned int length , ktime_t edge ) ;
void ( * send_space ) ( void ) ;
2010-07-27 03:31:45 +04:00
spinlock_t lock ;
} ;
2016-11-22 00:55:51 +03:00
# define IR_HOMEBREW 0
# define IR_IRDEO 1
# define IR_IRDEO_REMOTE 2
# define IR_ANIMAX 3
# define IR_IGOR 4
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:51 +03:00
/* module parameters */
2010-07-27 03:31:45 +04:00
static int type ;
static int io ;
static int irq ;
2012-01-13 03:02:20 +04:00
static bool iommap ;
2010-07-27 03:31:45 +04:00
static int ioshift ;
2015-01-26 16:21:09 +03:00
static bool softcarrier = true ;
2012-01-13 03:02:20 +04:00
static bool share_irq ;
2010-07-27 03:31:45 +04:00
static int sense = - 1 ; /* -1 = auto, 0 = active high, 1 = active low */
2012-01-13 03:02:20 +04:00
static bool txsense ; /* 0 = active high, 1 = active low */
2010-07-27 03:31:45 +04:00
/* forward declarations */
2016-11-22 00:55:52 +03:00
static void send_pulse_irdeo ( unsigned int length , ktime_t edge ) ;
static void send_space_irdeo ( void ) ;
2016-11-22 00:55:51 +03:00
# ifdef CONFIG_IR_SERIAL_TRANSMITTER
2016-11-22 00:55:52 +03:00
static void send_pulse_homebrew ( unsigned int length , ktime_t edge ) ;
static void send_space_homebrew ( void ) ;
2016-11-22 00:55:51 +03:00
# endif
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:51 +03:00
static struct serial_ir_hw hardware [ ] = {
[ IR_HOMEBREW ] = {
. lock = __SPIN_LOCK_UNLOCKED ( hardware [ IR_HOMEBREW ] . lock ) ,
. signal_pin = UART_MSR_DCD ,
2010-07-27 03:31:45 +04:00
. signal_pin_change = UART_MSR_DDCD ,
. on = ( UART_MCR_RTS | UART_MCR_OUT2 | UART_MCR_DTR ) ,
. off = ( UART_MCR_RTS | UART_MCR_OUT2 ) ,
2016-11-22 00:55:51 +03:00
# ifdef CONFIG_IR_SERIAL_TRANSMITTER
2010-07-27 03:31:45 +04:00
. send_pulse = send_pulse_homebrew ,
. send_space = send_space_homebrew ,
2016-11-22 00:55:51 +03:00
. set_send_carrier = true ,
. set_duty_cycle = true ,
2010-07-27 03:31:45 +04:00
# endif
} ,
2016-11-22 00:55:51 +03:00
[ IR_IRDEO ] = {
. lock = __SPIN_LOCK_UNLOCKED ( hardware [ IR_IRDEO ] . lock ) ,
. signal_pin = UART_MSR_DSR ,
2010-07-27 03:31:45 +04:00
. signal_pin_change = UART_MSR_DDSR ,
. on = UART_MCR_OUT2 ,
. off = ( UART_MCR_RTS | UART_MCR_DTR | UART_MCR_OUT2 ) ,
2016-11-22 00:55:51 +03:00
. send_pulse = send_pulse_irdeo ,
. send_space = send_space_irdeo ,
. set_duty_cycle = true ,
2010-07-27 03:31:45 +04:00
} ,
2016-11-22 00:55:51 +03:00
[ IR_IRDEO_REMOTE ] = {
. lock = __SPIN_LOCK_UNLOCKED ( hardware [ IR_IRDEO_REMOTE ] . lock ) ,
. signal_pin = UART_MSR_DSR ,
2010-07-27 03:31:45 +04:00
. signal_pin_change = UART_MSR_DDSR ,
. on = ( UART_MCR_RTS | UART_MCR_DTR | UART_MCR_OUT2 ) ,
. off = ( UART_MCR_RTS | UART_MCR_DTR | UART_MCR_OUT2 ) ,
2016-11-22 00:55:51 +03:00
. send_pulse = send_pulse_irdeo ,
. send_space = send_space_irdeo ,
. set_duty_cycle = true ,
2010-07-27 03:31:45 +04:00
} ,
2016-11-22 00:55:51 +03:00
[ IR_ANIMAX ] = {
. lock = __SPIN_LOCK_UNLOCKED ( hardware [ IR_ANIMAX ] . lock ) ,
. signal_pin = UART_MSR_DCD ,
2010-07-27 03:31:45 +04:00
. signal_pin_change = UART_MSR_DDCD ,
. on = 0 ,
. off = ( UART_MCR_RTS | UART_MCR_DTR | UART_MCR_OUT2 ) ,
} ,
2016-11-22 00:55:51 +03:00
[ IR_IGOR ] = {
. lock = __SPIN_LOCK_UNLOCKED ( hardware [ IR_IGOR ] . lock ) ,
. signal_pin = UART_MSR_DSR ,
2010-07-27 03:31:45 +04:00
. signal_pin_change = UART_MSR_DDSR ,
. on = ( UART_MCR_RTS | UART_MCR_OUT2 | UART_MCR_DTR ) ,
. off = ( UART_MCR_RTS | UART_MCR_OUT2 ) ,
2016-11-22 00:55:51 +03:00
# ifdef CONFIG_IR_SERIAL_TRANSMITTER
2010-07-27 03:31:45 +04:00
. send_pulse = send_pulse_homebrew ,
. send_space = send_space_homebrew ,
2016-11-22 00:55:51 +03:00
. set_send_carrier = true ,
. set_duty_cycle = true ,
2010-07-27 03:31:45 +04:00
# endif
} ,
} ;
# define RS_ISR_PASS_LIMIT 256
2016-11-22 00:55:51 +03:00
struct serial_ir {
ktime_t lastkt ;
struct rc_dev * rcdev ;
struct platform_device * pdev ;
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:51 +03:00
unsigned int freq ;
unsigned int duty_cycle ;
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:52 +03:00
unsigned int pulse_width , space_width ;
2016-11-22 00:55:51 +03:00
} ;
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:51 +03:00
static struct serial_ir serial_ir ;
2010-07-27 03:31:45 +04:00
/* fetch serial input packet (1 byte) from register offset */
static u8 sinp ( int offset )
{
2015-01-26 16:21:09 +03:00
if ( iommap )
2010-07-27 03:31:45 +04:00
/* the register is memory-mapped */
offset < < = ioshift ;
return inb ( io + offset ) ;
}
/* write serial output packet (1 byte) of value to register offset */
static void soutp ( int offset , u8 value )
{
2015-01-26 16:21:09 +03:00
if ( iommap )
2010-07-27 03:31:45 +04:00
/* the register is memory-mapped */
offset < < = ioshift ;
outb ( value , io + offset ) ;
}
static void on ( void )
{
if ( txsense )
soutp ( UART_MCR , hardware [ type ] . off ) ;
else
soutp ( UART_MCR , hardware [ type ] . on ) ;
}
static void off ( void )
{
if ( txsense )
soutp ( UART_MCR , hardware [ type ] . on ) ;
else
soutp ( UART_MCR , hardware [ type ] . off ) ;
}
2016-11-22 00:55:52 +03:00
static void init_timing_params ( unsigned int new_duty_cycle ,
unsigned int new_freq )
2010-07-27 03:31:45 +04:00
{
2016-11-22 00:55:51 +03:00
serial_ir . duty_cycle = new_duty_cycle ;
serial_ir . freq = new_freq ;
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:52 +03:00
serial_ir . pulse_width = DIV_ROUND_CLOSEST (
new_duty_cycle * NSEC_PER_SEC , new_freq * 100l ) ;
serial_ir . space_width = DIV_ROUND_CLOSEST (
( 100l - new_duty_cycle ) * NSEC_PER_SEC , new_freq * 100l ) ;
2010-07-27 03:31:45 +04:00
}
2016-11-22 00:55:52 +03:00
static void send_pulse_irdeo ( unsigned int length , ktime_t target )
2010-07-27 03:31:45 +04:00
{
2016-11-22 00:55:52 +03:00
long rawbits ;
2010-07-27 03:31:45 +04:00
int i ;
unsigned char output ;
unsigned char chunk , shifted ;
/* how many bits have to be sent ? */
rawbits = length * 1152 / 10000 ;
2016-11-22 00:55:51 +03:00
if ( serial_ir . duty_cycle > 50 )
2010-07-27 03:31:45 +04:00
chunk = 3 ;
else
chunk = 1 ;
for ( i = 0 , output = 0x7f ; rawbits > 0 ; rawbits - = 3 ) {
shifted = chunk < < ( i * 3 ) ;
shifted > > = 1 ;
output & = ( ~ shifted ) ;
i + + ;
if ( i = = 3 ) {
soutp ( UART_TX , output ) ;
while ( ! ( sinp ( UART_LSR ) & UART_LSR_THRE ) )
;
output = 0x7f ;
i = 0 ;
}
}
if ( i ! = 0 ) {
soutp ( UART_TX , output ) ;
while ( ! ( sinp ( UART_LSR ) & UART_LSR_TEMT ) )
;
}
}
2016-11-22 00:55:52 +03:00
static void send_space_irdeo ( void )
2016-11-22 00:55:51 +03:00
{
}
# ifdef CONFIG_IR_SERIAL_TRANSMITTER
2016-11-22 00:55:52 +03:00
static void send_pulse_homebrew_softcarrier ( unsigned int length , ktime_t edge )
2010-07-27 03:31:45 +04:00
{
2016-11-22 00:55:52 +03:00
ktime_t now , target = ktime_add_us ( edge , length ) ;
/*
* delta should never exceed 4 seconds and on m68k
* ndelay ( s64 ) does not compile ; so use s32 rather than s64 .
*/
s32 delta ;
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:52 +03:00
for ( ; ; ) {
now = ktime_get ( ) ;
if ( ktime_compare ( now , target ) > = 0 )
break ;
on ( ) ;
edge = ktime_add_ns ( edge , serial_ir . pulse_width ) ;
delta = ktime_to_ns ( ktime_sub ( edge , now ) ) ;
if ( delta > 0 )
ndelay ( delta ) ;
now = ktime_get ( ) ;
off ( ) ;
if ( ktime_compare ( now , target ) > = 0 )
break ;
edge = ktime_add_ns ( edge , serial_ir . space_width ) ;
delta = ktime_to_ns ( ktime_sub ( edge , now ) ) ;
if ( delta > 0 )
ndelay ( delta ) ;
2010-07-27 03:31:45 +04:00
}
}
2016-11-22 00:55:52 +03:00
static void send_pulse_homebrew ( unsigned int length , ktime_t edge )
2010-07-27 03:31:45 +04:00
{
if ( softcarrier )
2016-11-22 00:55:52 +03:00
send_pulse_homebrew_softcarrier ( length , edge ) ;
else
on ( ) ;
2010-07-27 03:31:45 +04:00
}
2016-11-22 00:55:52 +03:00
static void send_space_homebrew ( void )
2010-07-27 03:31:45 +04:00
{
off ( ) ;
}
2016-11-22 00:55:51 +03:00
# endif
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:51 +03:00
static void frbwrite ( unsigned int l , bool is_pulse )
2010-07-27 03:31:45 +04:00
{
/* simple noise filter */
2016-11-22 00:55:51 +03:00
static unsigned int ptr , pulse , space ;
DEFINE_IR_RAW_EVENT ( ev ) ;
if ( ptr > 0 & & is_pulse ) {
pulse + = l ;
if ( pulse > 250000 ) {
ev . duration = space ;
ev . pulse = false ;
ir_raw_event_store_with_filter ( serial_ir . rcdev , & ev ) ;
ev . duration = pulse ;
ev . pulse = true ;
ir_raw_event_store_with_filter ( serial_ir . rcdev , & ev ) ;
2010-07-27 03:31:45 +04:00
ptr = 0 ;
pulse = 0 ;
}
return ;
}
2016-11-22 00:55:51 +03:00
if ( ! is_pulse ) {
2010-07-27 03:31:45 +04:00
if ( ptr = = 0 ) {
2016-11-22 00:55:51 +03:00
if ( l > 20000000 ) {
2010-07-27 03:31:45 +04:00
space = l ;
ptr + + ;
return ;
}
} else {
2016-11-22 00:55:51 +03:00
if ( l > 20000000 ) {
2010-07-27 03:31:45 +04:00
space + = pulse ;
2016-11-22 00:55:51 +03:00
if ( space > IR_MAX_DURATION )
space = IR_MAX_DURATION ;
2010-07-27 03:31:45 +04:00
space + = l ;
2016-11-22 00:55:51 +03:00
if ( space > IR_MAX_DURATION )
space = IR_MAX_DURATION ;
2010-07-27 03:31:45 +04:00
pulse = 0 ;
return ;
}
2016-11-22 00:55:51 +03:00
ev . duration = space ;
ev . pulse = false ;
ir_raw_event_store_with_filter ( serial_ir . rcdev , & ev ) ;
ev . duration = pulse ;
ev . pulse = true ;
ir_raw_event_store_with_filter ( serial_ir . rcdev , & ev ) ;
2010-07-27 03:31:45 +04:00
ptr = 0 ;
pulse = 0 ;
}
}
2016-11-22 00:55:51 +03:00
ev . duration = l ;
ev . pulse = is_pulse ;
ir_raw_event_store_with_filter ( serial_ir . rcdev , & ev ) ;
2010-07-27 03:31:45 +04:00
}
2016-11-22 00:55:51 +03:00
static irqreturn_t serial_ir_irq_handler ( int i , void * blah )
2010-07-27 03:31:45 +04:00
{
2015-11-25 18:11:55 +03:00
ktime_t kt ;
2010-07-27 03:31:45 +04:00
int counter , dcd ;
u8 status ;
2015-11-25 18:11:55 +03:00
ktime_t delkt ;
2016-11-22 00:55:51 +03:00
unsigned int data ;
2010-07-27 03:31:45 +04:00
static int last_dcd = - 1 ;
if ( ( sinp ( UART_IIR ) & UART_IIR_NO_INT ) ) {
/* not our interrupt */
return IRQ_NONE ;
}
counter = 0 ;
do {
counter + + ;
status = sinp ( UART_MSR ) ;
if ( counter > RS_ISR_PASS_LIMIT ) {
2016-11-22 00:55:51 +03:00
dev_err ( & serial_ir . pdev - > dev , " Trapped in interrupt " ) ;
2010-07-27 03:31:45 +04:00
break ;
}
2016-11-22 10:46:14 +03:00
if ( ( status & hardware [ type ] . signal_pin_change ) & &
sense ! = - 1 ) {
2010-07-27 03:31:45 +04:00
/* get current time */
2015-11-25 18:11:55 +03:00
kt = ktime_get ( ) ;
2010-07-27 03:31:45 +04:00
/*
2016-11-22 00:55:51 +03:00
* The driver needs to know if your receiver is
* active high or active low , or the space / pulse
* sense could be inverted .
2010-07-27 03:31:45 +04:00
*/
2016-11-22 00:55:51 +03:00
/* calc time since last interrupt in nanoseconds */
2010-07-27 03:31:45 +04:00
dcd = ( status & hardware [ type ] . signal_pin ) ? 1 : 0 ;
if ( dcd = = last_dcd ) {
2016-11-22 00:55:51 +03:00
dev_err ( & serial_ir . pdev - > dev ,
" ignoring spike: %d %d %lldns %lldns \n " ,
dcd , sense , ktime_to_ns ( kt ) ,
ktime_to_ns ( serial_ir . lastkt ) ) ;
2010-07-27 03:31:45 +04:00
continue ;
}
2016-11-22 00:55:51 +03:00
delkt = ktime_sub ( kt , serial_ir . lastkt ) ;
2015-11-25 18:11:55 +03:00
if ( ktime_compare ( delkt , ktime_set ( 15 , 0 ) ) > 0 ) {
2016-11-22 00:55:51 +03:00
data = IR_MAX_DURATION ; /* really long time */
2016-11-22 10:46:14 +03:00
if ( ! ( dcd ^ sense ) ) {
2010-07-27 03:31:45 +04:00
/* sanity check */
2016-11-22 00:55:51 +03:00
dev_err ( & serial_ir . pdev - > dev ,
" dcd unexpected: %d %d %lldns %lldns \n " ,
dcd , sense , ktime_to_ns ( kt ) ,
ktime_to_ns ( serial_ir . lastkt ) ) ;
2010-07-27 03:31:45 +04:00
/*
* detecting pulse while this
* MUST be a space !
*/
sense = sense ? 0 : 1 ;
}
2016-11-22 10:46:14 +03:00
} else {
2016-11-22 00:55:51 +03:00
data = ktime_to_ns ( delkt ) ;
2016-11-22 10:46:14 +03:00
}
2016-11-22 00:55:51 +03:00
frbwrite ( data , ! ( dcd ^ sense ) ) ;
serial_ir . lastkt = kt ;
2010-07-27 03:31:45 +04:00
last_dcd = dcd ;
2016-11-22 00:55:51 +03:00
ir_raw_event_handle ( serial_ir . rcdev ) ;
2010-07-27 03:31:45 +04:00
}
} while ( ! ( sinp ( UART_IIR ) & UART_IIR_NO_INT ) ) ; /* still pending ? */
return IRQ_HANDLED ;
}
static int hardware_init_port ( void )
{
u8 scratch , scratch2 , scratch3 ;
/*
* This is a simple port existence test , borrowed from the autoconfig
2016-11-22 11:17:44 +03:00
* function in drivers / tty / serial / 8250 / 8250 _port . c
2010-07-27 03:31:45 +04:00
*/
scratch = sinp ( UART_IER ) ;
soutp ( UART_IER , 0 ) ;
# ifdef __i386__
outb ( 0xff , 0x080 ) ;
# endif
scratch2 = sinp ( UART_IER ) & 0x0f ;
soutp ( UART_IER , 0x0f ) ;
# ifdef __i386__
outb ( 0x00 , 0x080 ) ;
# endif
scratch3 = sinp ( UART_IER ) & 0x0f ;
soutp ( UART_IER , scratch ) ;
if ( scratch2 ! = 0 | | scratch3 ! = 0x0f ) {
/* we fail, there's nothing here */
2012-11-08 22:55:09 +04:00
pr_err ( " port existence test failed, cannot continue \n " ) ;
2011-11-16 08:53:35 +04:00
return - ENODEV ;
2010-07-27 03:31:45 +04:00
}
/* Set DLAB 0. */
soutp ( UART_LCR , sinp ( UART_LCR ) & ( ~ UART_LCR_DLAB ) ) ;
/* First of all, disable all interrupts */
soutp ( UART_IER , sinp ( UART_IER ) &
2016-11-22 10:46:14 +03:00
( ~ ( UART_IER_MSI | UART_IER_RLSI | UART_IER_THRI | UART_IER_RDI ) ) ) ;
2010-07-27 03:31:45 +04:00
/* Clear registers. */
sinp ( UART_LSR ) ;
sinp ( UART_RX ) ;
sinp ( UART_IIR ) ;
sinp ( UART_MSR ) ;
/* Set line for power source */
off ( ) ;
/* Clear registers again to be sure. */
sinp ( UART_LSR ) ;
sinp ( UART_RX ) ;
sinp ( UART_IIR ) ;
sinp ( UART_MSR ) ;
switch ( type ) {
2016-11-22 00:55:51 +03:00
case IR_IRDEO :
case IR_IRDEO_REMOTE :
2010-07-27 03:31:45 +04:00
/* setup port to 7N1 @ 115200 Baud */
/* 7N1+start = 9 bits at 115200 ~ 3 bits at 38kHz */
/* Set DLAB 1. */
soutp ( UART_LCR , sinp ( UART_LCR ) | UART_LCR_DLAB ) ;
/* Set divisor to 1 => 115200 Baud */
soutp ( UART_DLM , 0 ) ;
soutp ( UART_DLL , 1 ) ;
/* Set DLAB 0 + 7N1 */
soutp ( UART_LCR , UART_LCR_WLEN7 ) ;
/* THR interrupt already disabled at this point */
break ;
default :
break ;
}
return 0 ;
}
2016-11-22 00:55:51 +03:00
static int serial_ir_probe ( struct platform_device * dev )
2010-07-27 03:31:45 +04:00
{
2011-06-13 22:32:26 +04:00
int i , nlow , nhigh , result ;
2016-11-22 00:55:51 +03:00
result = devm_request_irq ( & dev - > dev , irq , serial_ir_irq_handler ,
share_irq ? IRQF_SHARED : 0 ,
KBUILD_MODNAME , & hardware ) ;
2011-11-16 08:54:04 +04:00
if ( result < 0 ) {
if ( result = = - EBUSY )
2012-11-08 22:55:09 +04:00
dev_err ( & dev - > dev , " IRQ %d busy \n " , irq ) ;
2011-11-16 08:54:04 +04:00
else if ( result = = - EINVAL )
2012-11-08 22:55:09 +04:00
dev_err ( & dev - > dev , " Bad irq number or handler \n " ) ;
2011-11-16 08:54:04 +04:00
return result ;
}
2010-07-27 03:31:45 +04:00
/* Reserve io region. */
2016-11-22 10:46:14 +03:00
if ( ( iommap & &
( devm_request_mem_region ( & dev - > dev , iommap , 8 < < ioshift ,
KBUILD_MODNAME ) = = NULL ) ) | |
( ! iommap & & ( devm_request_region ( & dev - > dev , io , 8 ,
KBUILD_MODNAME ) = = NULL ) ) ) {
2012-11-08 22:55:09 +04:00
dev_err ( & dev - > dev , " port %04x already in use \n " , io ) ;
dev_warn ( & dev - > dev , " use 'setserial /dev/ttySX uart none' \n " ) ;
dev_warn ( & dev - > dev ,
" or compile the serial port driver as module and \n " ) ;
dev_warn ( & dev - > dev , " make sure this module is loaded first \n " ) ;
2014-07-03 23:38:40 +04:00
return - EBUSY ;
2010-07-27 03:31:45 +04:00
}
2011-11-16 08:53:35 +04:00
result = hardware_init_port ( ) ;
if ( result < 0 )
2014-07-03 23:38:40 +04:00
return result ;
2010-07-27 03:31:45 +04:00
/* Initialize pulse/space widths */
2016-11-22 00:55:51 +03:00
init_timing_params ( 50 , 38000 ) ;
2010-07-27 03:31:45 +04:00
/* If pin is high, then this must be an active low receiver. */
if ( sense = = - 1 ) {
/* wait 1/2 sec for the power supply */
msleep ( 500 ) ;
/*
* probe 9 times every 0.04 s , collect " votes " for
* active high / low
*/
nlow = 0 ;
nhigh = 0 ;
for ( i = 0 ; i < 9 ; i + + ) {
if ( sinp ( UART_MSR ) & hardware [ type ] . signal_pin )
nlow + + ;
else
nhigh + + ;
msleep ( 40 ) ;
}
2015-03-13 20:46:57 +03:00
sense = nlow > = nhigh ? 1 : 0 ;
2012-11-08 22:55:09 +04:00
dev_info ( & dev - > dev , " auto-detected active %s receiver \n " ,
sense ? " low " : " high " ) ;
2010-07-27 03:31:45 +04:00
} else
2012-11-08 22:55:09 +04:00
dev_info ( & dev - > dev , " Manually using active %s receiver \n " ,
sense ? " low " : " high " ) ;
2010-07-27 03:31:45 +04:00
2015-09-04 23:04:05 +03:00
dev_dbg ( & dev - > dev , " Interrupt %d, port %04x obtained \n " , irq , io ) ;
2010-07-27 03:31:45 +04:00
return 0 ;
}
2016-11-22 00:55:51 +03:00
static int serial_ir_open ( struct rc_dev * rcdev )
2010-07-27 03:31:45 +04:00
{
unsigned long flags ;
/* initialize timestamp */
2016-11-22 00:55:51 +03:00
serial_ir . lastkt = ktime_get ( ) ;
2010-07-27 03:31:45 +04:00
spin_lock_irqsave ( & hardware [ type ] . lock , flags ) ;
/* Set DLAB 0. */
soutp ( UART_LCR , sinp ( UART_LCR ) & ( ~ UART_LCR_DLAB ) ) ;
2016-11-22 10:46:14 +03:00
soutp ( UART_IER , sinp ( UART_IER ) | UART_IER_MSI ) ;
2010-07-27 03:31:45 +04:00
spin_unlock_irqrestore ( & hardware [ type ] . lock , flags ) ;
return 0 ;
}
2016-11-22 00:55:51 +03:00
static void serial_ir_close ( struct rc_dev * rcdev )
{
unsigned long flags ;
2010-07-27 03:31:45 +04:00
spin_lock_irqsave ( & hardware [ type ] . lock , flags ) ;
/* Set DLAB 0. */
soutp ( UART_LCR , sinp ( UART_LCR ) & ( ~ UART_LCR_DLAB ) ) ;
/* First of all, disable all interrupts */
soutp ( UART_IER , sinp ( UART_IER ) &
2016-11-22 10:46:14 +03:00
( ~ ( UART_IER_MSI | UART_IER_RLSI | UART_IER_THRI | UART_IER_RDI ) ) ) ;
2010-07-27 03:31:45 +04:00
spin_unlock_irqrestore ( & hardware [ type ] . lock , flags ) ;
}
2016-11-22 00:55:51 +03:00
static int serial_ir_tx ( struct rc_dev * dev , unsigned int * txbuf ,
unsigned int count )
2010-07-27 03:31:45 +04:00
{
unsigned long flags ;
2016-11-22 00:55:52 +03:00
ktime_t edge ;
s64 delta ;
2016-11-22 00:55:51 +03:00
int i ;
2010-07-27 03:31:45 +04:00
spin_lock_irqsave ( & hardware [ type ] . lock , flags ) ;
2016-11-22 00:55:51 +03:00
if ( type = = IR_IRDEO ) {
2010-07-27 03:31:45 +04:00
/* DTR, RTS down */
on ( ) ;
}
2016-11-22 00:55:52 +03:00
edge = ktime_get ( ) ;
2010-07-27 03:31:45 +04:00
for ( i = 0 ; i < count ; i + + ) {
2016-11-22 10:46:14 +03:00
if ( i % 2 )
2016-11-22 00:55:52 +03:00
hardware [ type ] . send_space ( ) ;
2010-07-27 03:31:45 +04:00
else
2016-11-22 00:55:52 +03:00
hardware [ type ] . send_pulse ( txbuf [ i ] , edge ) ;
edge = ktime_add_us ( edge , txbuf [ i ] ) ;
delta = ktime_us_delta ( edge , ktime_get ( ) ) ;
if ( delta > 25 ) {
spin_unlock_irqrestore ( & hardware [ type ] . lock , flags ) ;
usleep_range ( delta - 25 , delta + 25 ) ;
spin_lock_irqsave ( & hardware [ type ] . lock , flags ) ;
2016-11-22 10:46:14 +03:00
} else if ( delta > 0 ) {
2016-11-22 00:55:52 +03:00
udelay ( delta ) ;
2016-11-22 10:46:14 +03:00
}
2010-07-27 03:31:45 +04:00
}
off ( ) ;
spin_unlock_irqrestore ( & hardware [ type ] . lock , flags ) ;
2016-11-22 00:55:51 +03:00
return count ;
2010-07-27 03:31:45 +04:00
}
2016-11-22 00:55:51 +03:00
static int serial_ir_tx_duty_cycle ( struct rc_dev * dev , u32 cycle )
2010-07-27 03:31:45 +04:00
{
2016-11-22 00:55:52 +03:00
init_timing_params ( cycle , serial_ir . freq ) ;
return 0 ;
2010-07-27 03:31:45 +04:00
}
2016-11-22 00:55:51 +03:00
static int serial_ir_tx_carrier ( struct rc_dev * dev , u32 carrier )
{
if ( carrier > 500000 | | carrier < 20000 )
return - EINVAL ;
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:52 +03:00
init_timing_params ( serial_ir . duty_cycle , carrier ) ;
return 0 ;
2016-11-22 00:55:51 +03:00
}
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:51 +03:00
static int serial_ir_suspend ( struct platform_device * dev ,
pm_message_t state )
2010-07-27 03:31:45 +04:00
{
/* Set DLAB 0. */
soutp ( UART_LCR , sinp ( UART_LCR ) & ( ~ UART_LCR_DLAB ) ) ;
/* Disable all interrupts */
soutp ( UART_IER , sinp ( UART_IER ) &
2016-11-22 10:46:14 +03:00
( ~ ( UART_IER_MSI | UART_IER_RLSI | UART_IER_THRI | UART_IER_RDI ) ) ) ;
2010-07-27 03:31:45 +04:00
/* Clear registers. */
sinp ( UART_LSR ) ;
sinp ( UART_RX ) ;
sinp ( UART_IIR ) ;
sinp ( UART_MSR ) ;
return 0 ;
}
2016-11-22 00:55:51 +03:00
static int serial_ir_resume ( struct platform_device * dev )
2010-07-27 03:31:45 +04:00
{
unsigned long flags ;
2011-11-16 08:53:35 +04:00
int result ;
2010-07-27 03:31:45 +04:00
2011-11-16 08:53:35 +04:00
result = hardware_init_port ( ) ;
if ( result < 0 )
return result ;
2010-07-27 03:31:45 +04:00
spin_lock_irqsave ( & hardware [ type ] . lock , flags ) ;
/* Enable Interrupt */
2016-11-22 00:55:51 +03:00
serial_ir . lastkt = ktime_get ( ) ;
2016-11-22 10:46:14 +03:00
soutp ( UART_IER , sinp ( UART_IER ) | UART_IER_MSI ) ;
2010-07-27 03:31:45 +04:00
off ( ) ;
spin_unlock_irqrestore ( & hardware [ type ] . lock , flags ) ;
return 0 ;
}
2016-11-22 00:55:51 +03:00
static struct platform_driver serial_ir_driver = {
. probe = serial_ir_probe ,
. suspend = serial_ir_suspend ,
. resume = serial_ir_resume ,
2010-07-27 03:31:45 +04:00
. driver = {
2016-11-22 00:55:51 +03:00
. name = " serial_ir " ,
2010-07-27 03:31:45 +04:00
} ,
} ;
2016-11-22 00:55:51 +03:00
static int __init serial_ir_init ( void )
2010-07-27 03:31:45 +04:00
{
int result ;
2016-11-22 00:55:51 +03:00
result = platform_driver_register ( & serial_ir_driver ) ;
if ( result )
2011-11-16 08:53:35 +04:00
return result ;
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:51 +03:00
serial_ir . pdev = platform_device_alloc ( " serial_ir " , 0 ) ;
if ( ! serial_ir . pdev ) {
2010-07-27 03:31:45 +04:00
result = - ENOMEM ;
goto exit_driver_unregister ;
}
2016-11-22 00:55:51 +03:00
result = platform_device_add ( serial_ir . pdev ) ;
2010-07-27 03:31:45 +04:00
if ( result )
goto exit_device_put ;
return 0 ;
exit_device_put :
2016-11-22 00:55:51 +03:00
platform_device_put ( serial_ir . pdev ) ;
2010-07-27 03:31:45 +04:00
exit_driver_unregister :
2016-11-22 00:55:51 +03:00
platform_driver_unregister ( & serial_ir_driver ) ;
2010-07-27 03:31:45 +04:00
return result ;
}
2016-11-22 00:55:51 +03:00
static void serial_ir_exit ( void )
2010-07-27 03:31:45 +04:00
{
2016-11-22 00:55:51 +03:00
platform_device_unregister ( serial_ir . pdev ) ;
platform_driver_unregister ( & serial_ir_driver ) ;
2010-07-27 03:31:45 +04:00
}
2016-11-22 00:55:51 +03:00
static int __init serial_ir_init_module ( void )
2010-07-27 03:31:45 +04:00
{
2016-11-22 00:55:51 +03:00
struct rc_dev * rcdev ;
2010-07-27 03:31:45 +04:00
int result ;
switch ( type ) {
2016-11-22 00:55:51 +03:00
case IR_HOMEBREW :
case IR_IRDEO :
case IR_IRDEO_REMOTE :
case IR_ANIMAX :
case IR_IGOR :
2010-07-27 03:31:45 +04:00
/* if nothing specified, use ttyS0/com1 and irq 4 */
io = io ? io : 0x3f8 ;
irq = irq ? irq : 4 ;
break ;
default :
2011-11-16 08:49:41 +04:00
return - EINVAL ;
2010-07-27 03:31:45 +04:00
}
if ( ! softcarrier ) {
switch ( type ) {
2016-11-22 00:55:51 +03:00
case IR_HOMEBREW :
case IR_IGOR :
hardware [ type ] . set_send_carrier = false ;
hardware [ type ] . set_duty_cycle = false ;
2010-07-27 03:31:45 +04:00
break ;
}
}
2012-10-14 22:50:11 +04:00
/* make sure sense is either -1, 0, or 1 */
if ( sense ! = - 1 )
sense = ! ! sense ;
2016-11-22 00:55:51 +03:00
result = serial_ir_init ( ) ;
2011-11-16 08:49:41 +04:00
if ( result )
return result ;
2016-11-22 00:55:51 +03:00
rcdev = devm_rc_allocate_device ( & serial_ir . pdev - > dev ) ;
if ( ! rcdev ) {
result = - ENOMEM ;
goto serial_cleanup ;
2010-07-27 03:31:45 +04:00
}
2016-11-22 00:55:51 +03:00
if ( hardware [ type ] . send_pulse & & hardware [ type ] . send_space )
rcdev - > tx_ir = serial_ir_tx ;
if ( hardware [ type ] . set_send_carrier )
rcdev - > s_tx_carrier = serial_ir_tx_carrier ;
if ( hardware [ type ] . set_duty_cycle )
rcdev - > s_tx_duty_cycle = serial_ir_tx_duty_cycle ;
switch ( type ) {
case IR_HOMEBREW :
rcdev - > input_name = " Serial IR type home-brew " ;
break ;
case IR_IRDEO :
rcdev - > input_name = " Serial IR type IRdeo " ;
break ;
case IR_IRDEO_REMOTE :
rcdev - > input_name = " Serial IR type IRdeo remote " ;
break ;
case IR_ANIMAX :
rcdev - > input_name = " Serial IR type AnimaX " ;
break ;
case IR_IGOR :
rcdev - > input_name = " Serial IR type IgorPlug " ;
break ;
}
rcdev - > input_phys = KBUILD_MODNAME " /input0 " ;
rcdev - > input_id . bustype = BUS_HOST ;
rcdev - > input_id . vendor = 0x0001 ;
rcdev - > input_id . product = 0x0001 ;
rcdev - > input_id . version = 0x0100 ;
rcdev - > open = serial_ir_open ;
rcdev - > close = serial_ir_close ;
rcdev - > dev . parent = & serial_ir . pdev - > dev ;
rcdev - > driver_type = RC_DRIVER_IR_RAW ;
rcdev - > allowed_protocols = RC_BIT_ALL ;
rcdev - > driver_name = KBUILD_MODNAME ;
rcdev - > map_name = RC_MAP_RC6_MCE ;
rcdev - > timeout = IR_DEFAULT_TIMEOUT ;
rcdev - > rx_resolution = 250000 ;
serial_ir . rcdev = rcdev ;
result = rc_register_device ( rcdev ) ;
if ( ! result )
return 0 ;
serial_cleanup :
serial_ir_exit ( ) ;
return result ;
2010-07-27 03:31:45 +04:00
}
2016-11-22 00:55:51 +03:00
static void __exit serial_ir_exit_module ( void )
2010-07-27 03:31:45 +04:00
{
2016-11-22 00:55:51 +03:00
rc_unregister_device ( serial_ir . rcdev ) ;
serial_ir_exit ( ) ;
2010-07-27 03:31:45 +04:00
}
2016-11-22 00:55:51 +03:00
module_init ( serial_ir_init_module ) ;
module_exit ( serial_ir_exit_module ) ;
2010-07-27 03:31:45 +04:00
MODULE_DESCRIPTION ( " Infra-red receiver driver for serial ports. " ) ;
2016-11-22 00:55:51 +03:00
MODULE_AUTHOR ( " Ralph Metzler, Trent Piepho, Ben Pfaff, Christoph Bartelmus, Andrei Tanas " ) ;
2010-07-27 03:31:45 +04:00
MODULE_LICENSE ( " GPL " ) ;
2016-11-22 00:55:51 +03:00
module_param ( type , int , 0444 ) ;
MODULE_PARM_DESC ( type , " Hardware type (0 = home-brew, 1 = IRdeo, 2 = IRdeo Remote, 3 = AnimaX, 4 = IgorPlug " ) ;
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:51 +03:00
module_param ( io , int , 0444 ) ;
2010-07-27 03:31:45 +04:00
MODULE_PARM_DESC ( io , " I/O address base (0x3f8 or 0x2f8) " ) ;
/* some architectures (e.g. intel xscale) have memory mapped registers */
2016-11-22 00:55:51 +03:00
module_param ( iommap , bool , 0444 ) ;
MODULE_PARM_DESC ( iommap , " physical base for memory mapped I/O (0 = no memory mapped io) " ) ;
2010-07-27 03:31:45 +04:00
/*
* some architectures ( e . g . intel xscale ) align the 8 bit serial registers
* on 32 bit word boundaries .
2012-03-05 18:49:26 +04:00
* See linux - kernel / drivers / tty / serial / 8250 / 8250. c serial_in ( ) / out ( )
2010-07-27 03:31:45 +04:00
*/
2016-11-22 00:55:51 +03:00
module_param ( ioshift , int , 0444 ) ;
2010-07-27 03:31:45 +04:00
MODULE_PARM_DESC ( ioshift , " shift I/O register offset (0 = no shift) " ) ;
2016-11-22 00:55:51 +03:00
module_param ( irq , int , 0444 ) ;
2010-07-27 03:31:45 +04:00
MODULE_PARM_DESC ( irq , " Interrupt (4 or 3) " ) ;
2016-11-22 00:55:51 +03:00
module_param ( share_irq , bool , 0444 ) ;
2010-07-27 03:31:45 +04:00
MODULE_PARM_DESC ( share_irq , " Share interrupts (0 = off, 1 = on) " ) ;
2016-11-22 00:55:51 +03:00
module_param ( sense , int , 0444 ) ;
MODULE_PARM_DESC ( sense , " Override autodetection of IR receiver circuit (0 = active high, 1 = active low ) " ) ;
2010-07-27 03:31:45 +04:00
2016-11-22 00:55:51 +03:00
# ifdef CONFIG_IR_SERIAL_TRANSMITTER
module_param ( txsense , bool , 0444 ) ;
MODULE_PARM_DESC ( txsense , " Sense of transmitter circuit (0 = active high, 1 = active low ) " ) ;
2010-07-27 03:31:45 +04:00
# endif
2016-11-22 00:55:51 +03:00
module_param ( softcarrier , bool , 0444 ) ;
2010-07-27 03:31:45 +04:00
MODULE_PARM_DESC ( softcarrier , " Software carrier (0 = off, 1 = on, default on) " ) ;