2019-05-27 14:55:06 +08:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2010-07-27 07:31:45 +08:00
|
|
|
/*
|
2016-11-22 05:55:53 +08:00
|
|
|
* serial_ir.c
|
2010-07-27 07:31:45 +08:00
|
|
|
*
|
2016-11-22 05:55:53 +08:00
|
|
|
* serial_ir - Device driver that records pulse- and pause-lengths
|
2010-07-27 07:31:45 +08: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 05:55:51 +08:00
|
|
|
* Copyright (C) 2016 Sean Young <sean@mess.org> (port to rc-core)
|
2010-07-27 07:31:45 +08:00
|
|
|
*/
|
|
|
|
|
2012-11-09 02:55:09 +08:00
|
|
|
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
|
|
|
|
|
2010-07-27 07:31:45 +08: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 05:55:51 +08:00
|
|
|
#include <media/rc-core.h>
|
2010-07-27 07:31:45 +08:00
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
struct serial_ir_hw {
|
2010-07-27 07:31:45 +08:00
|
|
|
int signal_pin;
|
|
|
|
int signal_pin_change;
|
|
|
|
u8 on;
|
|
|
|
u8 off;
|
2016-11-22 05:55:51 +08:00
|
|
|
unsigned set_send_carrier:1;
|
|
|
|
unsigned set_duty_cycle:1;
|
2016-11-22 05:55:52 +08:00
|
|
|
void (*send_pulse)(unsigned int length, ktime_t edge);
|
|
|
|
void (*send_space)(void);
|
2010-07-27 07:31:45 +08:00
|
|
|
spinlock_t lock;
|
|
|
|
};
|
|
|
|
|
2016-11-22 05:55:51 +08: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 07:31:45 +08:00
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
/* module parameters */
|
2010-07-27 07:31:45 +08:00
|
|
|
static int type;
|
|
|
|
static int io;
|
|
|
|
static int irq;
|
2017-02-14 06:53:23 +08:00
|
|
|
static ulong iommap;
|
2010-07-27 07:31:45 +08:00
|
|
|
static int ioshift;
|
2015-01-26 21:21:09 +08:00
|
|
|
static bool softcarrier = true;
|
2012-01-13 07:02:20 +08:00
|
|
|
static bool share_irq;
|
2010-07-27 07:31:45 +08:00
|
|
|
static int sense = -1; /* -1 = auto, 0 = active high, 1 = active low */
|
2012-01-13 07:02:20 +08:00
|
|
|
static bool txsense; /* 0 = active high, 1 = active low */
|
2010-07-27 07:31:45 +08:00
|
|
|
|
|
|
|
/* forward declarations */
|
2016-11-22 05:55:52 +08:00
|
|
|
static void send_pulse_irdeo(unsigned int length, ktime_t edge);
|
|
|
|
static void send_space_irdeo(void);
|
2016-11-22 05:55:51 +08:00
|
|
|
#ifdef CONFIG_IR_SERIAL_TRANSMITTER
|
2016-11-22 05:55:52 +08:00
|
|
|
static void send_pulse_homebrew(unsigned int length, ktime_t edge);
|
|
|
|
static void send_space_homebrew(void);
|
2016-11-22 05:55:51 +08:00
|
|
|
#endif
|
2010-07-27 07:31:45 +08:00
|
|
|
|
2016-11-22 05:55:51 +08: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 07:31:45 +08: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 05:55:51 +08:00
|
|
|
#ifdef CONFIG_IR_SERIAL_TRANSMITTER
|
2010-07-27 07:31:45 +08:00
|
|
|
.send_pulse = send_pulse_homebrew,
|
|
|
|
.send_space = send_space_homebrew,
|
2016-11-22 05:55:51 +08:00
|
|
|
.set_send_carrier = true,
|
|
|
|
.set_duty_cycle = true,
|
2010-07-27 07:31:45 +08:00
|
|
|
#endif
|
|
|
|
},
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
[IR_IRDEO] = {
|
|
|
|
.lock = __SPIN_LOCK_UNLOCKED(hardware[IR_IRDEO].lock),
|
|
|
|
.signal_pin = UART_MSR_DSR,
|
2010-07-27 07:31:45 +08:00
|
|
|
.signal_pin_change = UART_MSR_DDSR,
|
|
|
|
.on = UART_MCR_OUT2,
|
|
|
|
.off = (UART_MCR_RTS | UART_MCR_DTR | UART_MCR_OUT2),
|
2016-11-22 05:55:51 +08:00
|
|
|
.send_pulse = send_pulse_irdeo,
|
|
|
|
.send_space = send_space_irdeo,
|
|
|
|
.set_duty_cycle = true,
|
2010-07-27 07:31:45 +08:00
|
|
|
},
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
[IR_IRDEO_REMOTE] = {
|
|
|
|
.lock = __SPIN_LOCK_UNLOCKED(hardware[IR_IRDEO_REMOTE].lock),
|
|
|
|
.signal_pin = UART_MSR_DSR,
|
2010-07-27 07:31:45 +08: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 05:55:51 +08:00
|
|
|
.send_pulse = send_pulse_irdeo,
|
|
|
|
.send_space = send_space_irdeo,
|
|
|
|
.set_duty_cycle = true,
|
2010-07-27 07:31:45 +08:00
|
|
|
},
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
[IR_ANIMAX] = {
|
|
|
|
.lock = __SPIN_LOCK_UNLOCKED(hardware[IR_ANIMAX].lock),
|
|
|
|
.signal_pin = UART_MSR_DCD,
|
2010-07-27 07:31:45 +08:00
|
|
|
.signal_pin_change = UART_MSR_DDCD,
|
|
|
|
.on = 0,
|
|
|
|
.off = (UART_MCR_RTS | UART_MCR_DTR | UART_MCR_OUT2),
|
|
|
|
},
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
[IR_IGOR] = {
|
|
|
|
.lock = __SPIN_LOCK_UNLOCKED(hardware[IR_IGOR].lock),
|
|
|
|
.signal_pin = UART_MSR_DSR,
|
2010-07-27 07:31:45 +08: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 05:55:51 +08:00
|
|
|
#ifdef CONFIG_IR_SERIAL_TRANSMITTER
|
2010-07-27 07:31:45 +08:00
|
|
|
.send_pulse = send_pulse_homebrew,
|
|
|
|
.send_space = send_space_homebrew,
|
2016-11-22 05:55:51 +08:00
|
|
|
.set_send_carrier = true,
|
|
|
|
.set_duty_cycle = true,
|
2010-07-27 07:31:45 +08:00
|
|
|
#endif
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
#define RS_ISR_PASS_LIMIT 256
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
struct serial_ir {
|
|
|
|
ktime_t lastkt;
|
|
|
|
struct rc_dev *rcdev;
|
|
|
|
struct platform_device *pdev;
|
2016-12-03 01:16:11 +08:00
|
|
|
struct timer_list timeout_timer;
|
2010-07-27 07:31:45 +08:00
|
|
|
|
2017-08-23 23:06:04 +08:00
|
|
|
unsigned int carrier;
|
2016-11-22 05:55:51 +08:00
|
|
|
unsigned int duty_cycle;
|
|
|
|
};
|
2010-07-27 07:31:45 +08:00
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static struct serial_ir serial_ir;
|
2010-07-27 07:31:45 +08:00
|
|
|
|
|
|
|
/* fetch serial input packet (1 byte) from register offset */
|
|
|
|
static u8 sinp(int offset)
|
|
|
|
{
|
2015-01-26 21:21:09 +08:00
|
|
|
if (iommap)
|
2010-07-27 07:31:45 +08: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 21:21:09 +08:00
|
|
|
if (iommap)
|
2010-07-27 07:31:45 +08: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 05:55:52 +08:00
|
|
|
static void send_pulse_irdeo(unsigned int length, ktime_t target)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
2016-11-22 05:55:52 +08:00
|
|
|
long rawbits;
|
2010-07-27 07:31:45 +08:00
|
|
|
int i;
|
|
|
|
unsigned char output;
|
|
|
|
unsigned char chunk, shifted;
|
|
|
|
|
|
|
|
/* how many bits have to be sent ? */
|
|
|
|
rawbits = length * 1152 / 10000;
|
2016-11-22 05:55:51 +08:00
|
|
|
if (serial_ir.duty_cycle > 50)
|
2010-07-27 07:31:45 +08: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 05:55:52 +08:00
|
|
|
static void send_space_irdeo(void)
|
2016-11-22 05:55:51 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_IR_SERIAL_TRANSMITTER
|
2016-11-22 05:55:52 +08:00
|
|
|
static void send_pulse_homebrew_softcarrier(unsigned int length, ktime_t edge)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
2016-11-22 05:55:52 +08: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;
|
2017-08-23 23:06:04 +08:00
|
|
|
unsigned int pulse, space;
|
|
|
|
|
|
|
|
/* Ensure the dividend fits into 32 bit */
|
|
|
|
pulse = DIV_ROUND_CLOSEST(serial_ir.duty_cycle * (NSEC_PER_SEC / 100),
|
|
|
|
serial_ir.carrier);
|
|
|
|
space = DIV_ROUND_CLOSEST((100 - serial_ir.duty_cycle) *
|
|
|
|
(NSEC_PER_SEC / 100), serial_ir.carrier);
|
2010-07-27 07:31:45 +08:00
|
|
|
|
2016-11-22 05:55:52 +08:00
|
|
|
for (;;) {
|
|
|
|
now = ktime_get();
|
|
|
|
if (ktime_compare(now, target) >= 0)
|
|
|
|
break;
|
|
|
|
on();
|
2017-08-23 23:06:04 +08:00
|
|
|
edge = ktime_add_ns(edge, pulse);
|
2016-11-22 05:55:52 +08:00
|
|
|
delta = ktime_to_ns(ktime_sub(edge, now));
|
|
|
|
if (delta > 0)
|
|
|
|
ndelay(delta);
|
|
|
|
now = ktime_get();
|
|
|
|
off();
|
|
|
|
if (ktime_compare(now, target) >= 0)
|
|
|
|
break;
|
2017-08-23 23:06:04 +08:00
|
|
|
edge = ktime_add_ns(edge, space);
|
2016-11-22 05:55:52 +08:00
|
|
|
delta = ktime_to_ns(ktime_sub(edge, now));
|
|
|
|
if (delta > 0)
|
|
|
|
ndelay(delta);
|
2010-07-27 07:31:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:52 +08:00
|
|
|
static void send_pulse_homebrew(unsigned int length, ktime_t edge)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
|
|
|
if (softcarrier)
|
2016-11-22 05:55:52 +08:00
|
|
|
send_pulse_homebrew_softcarrier(length, edge);
|
|
|
|
else
|
|
|
|
on();
|
2010-07-27 07:31:45 +08:00
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:52 +08:00
|
|
|
static void send_space_homebrew(void)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
|
|
|
off();
|
|
|
|
}
|
2016-11-22 05:55:51 +08:00
|
|
|
#endif
|
2010-07-27 07:31:45 +08:00
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static void frbwrite(unsigned int l, bool is_pulse)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
|
|
|
/* simple noise filter */
|
2016-11-22 05:55:51 +08:00
|
|
|
static unsigned int ptr, pulse, space;
|
2018-08-22 03:57:52 +08:00
|
|
|
struct ir_raw_event ev = {};
|
2016-11-22 05:55:51 +08:00
|
|
|
|
|
|
|
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 07:31:45 +08:00
|
|
|
ptr = 0;
|
|
|
|
pulse = 0;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2016-11-22 05:55:51 +08:00
|
|
|
if (!is_pulse) {
|
2010-07-27 07:31:45 +08:00
|
|
|
if (ptr == 0) {
|
2016-11-22 05:55:51 +08:00
|
|
|
if (l > 20000000) {
|
2010-07-27 07:31:45 +08:00
|
|
|
space = l;
|
|
|
|
ptr++;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
} else {
|
2016-11-22 05:55:51 +08:00
|
|
|
if (l > 20000000) {
|
2010-07-27 07:31:45 +08:00
|
|
|
space += pulse;
|
2016-11-22 05:55:51 +08:00
|
|
|
if (space > IR_MAX_DURATION)
|
|
|
|
space = IR_MAX_DURATION;
|
2010-07-27 07:31:45 +08:00
|
|
|
space += l;
|
2016-11-22 05:55:51 +08:00
|
|
|
if (space > IR_MAX_DURATION)
|
|
|
|
space = IR_MAX_DURATION;
|
2010-07-27 07:31:45 +08:00
|
|
|
pulse = 0;
|
|
|
|
return;
|
|
|
|
}
|
2016-11-22 05:55:51 +08: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 07:31:45 +08:00
|
|
|
ptr = 0;
|
|
|
|
pulse = 0;
|
|
|
|
}
|
|
|
|
}
|
2016-11-22 05:55:51 +08:00
|
|
|
|
|
|
|
ev.duration = l;
|
|
|
|
ev.pulse = is_pulse;
|
|
|
|
ir_raw_event_store_with_filter(serial_ir.rcdev, &ev);
|
2010-07-27 07:31:45 +08:00
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static irqreturn_t serial_ir_irq_handler(int i, void *blah)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
2015-11-25 23:11:55 +08:00
|
|
|
ktime_t kt;
|
2010-07-27 07:31:45 +08:00
|
|
|
int counter, dcd;
|
|
|
|
u8 status;
|
2015-11-25 23:11:55 +08:00
|
|
|
ktime_t delkt;
|
2016-11-22 05:55:51 +08:00
|
|
|
unsigned int data;
|
2010-07-27 07:31:45 +08: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 05:55:51 +08:00
|
|
|
dev_err(&serial_ir.pdev->dev, "Trapped in interrupt");
|
2010-07-27 07:31:45 +08:00
|
|
|
break;
|
|
|
|
}
|
2016-11-22 15:46:14 +08:00
|
|
|
if ((status & hardware[type].signal_pin_change) &&
|
|
|
|
sense != -1) {
|
2010-07-27 07:31:45 +08:00
|
|
|
/* get current time */
|
2015-11-25 23:11:55 +08:00
|
|
|
kt = ktime_get();
|
2010-07-27 07:31:45 +08:00
|
|
|
|
|
|
|
/*
|
2016-11-22 05:55:51 +08: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 07:31:45 +08:00
|
|
|
*/
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
/* calc time since last interrupt in nanoseconds */
|
2010-07-27 07:31:45 +08:00
|
|
|
dcd = (status & hardware[type].signal_pin) ? 1 : 0;
|
|
|
|
|
|
|
|
if (dcd == last_dcd) {
|
2019-12-27 05:28:45 +08:00
|
|
|
dev_dbg(&serial_ir.pdev->dev,
|
2016-11-22 05:55:51 +08:00
|
|
|
"ignoring spike: %d %d %lldns %lldns\n",
|
|
|
|
dcd, sense, ktime_to_ns(kt),
|
|
|
|
ktime_to_ns(serial_ir.lastkt));
|
2010-07-27 07:31:45 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
delkt = ktime_sub(kt, serial_ir.lastkt);
|
2015-11-25 23:11:55 +08:00
|
|
|
if (ktime_compare(delkt, ktime_set(15, 0)) > 0) {
|
2016-11-22 05:55:51 +08:00
|
|
|
data = IR_MAX_DURATION; /* really long time */
|
2016-11-22 15:46:14 +08:00
|
|
|
if (!(dcd ^ sense)) {
|
2010-07-27 07:31:45 +08:00
|
|
|
/* sanity check */
|
2016-11-22 05:55:51 +08: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 07:31:45 +08:00
|
|
|
/*
|
|
|
|
* detecting pulse while this
|
|
|
|
* MUST be a space!
|
|
|
|
*/
|
|
|
|
sense = sense ? 0 : 1;
|
|
|
|
}
|
2016-11-22 15:46:14 +08:00
|
|
|
} else {
|
2016-11-22 05:55:51 +08:00
|
|
|
data = ktime_to_ns(delkt);
|
2016-11-22 15:46:14 +08:00
|
|
|
}
|
2016-11-22 05:55:51 +08:00
|
|
|
frbwrite(data, !(dcd ^ sense));
|
|
|
|
serial_ir.lastkt = kt;
|
2010-07-27 07:31:45 +08:00
|
|
|
last_dcd = dcd;
|
|
|
|
}
|
|
|
|
} while (!(sinp(UART_IIR) & UART_IIR_NO_INT)); /* still pending ? */
|
2016-12-03 01:16:11 +08:00
|
|
|
|
|
|
|
mod_timer(&serial_ir.timeout_timer,
|
|
|
|
jiffies + nsecs_to_jiffies(serial_ir.rcdev->timeout));
|
|
|
|
|
|
|
|
ir_raw_event_handle(serial_ir.rcdev);
|
|
|
|
|
2010-07-27 07:31:45 +08:00
|
|
|
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 16:17:44 +08:00
|
|
|
* function in drivers/tty/serial/8250/8250_port.c
|
2010-07-27 07:31:45 +08: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-09 02:55:09 +08:00
|
|
|
pr_err("port existence test failed, cannot continue\n");
|
2011-11-16 12:53:35 +08:00
|
|
|
return -ENODEV;
|
2010-07-27 07:31:45 +08: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 15:46:14 +08:00
|
|
|
(~(UART_IER_MSI | UART_IER_RLSI | UART_IER_THRI | UART_IER_RDI)));
|
2010-07-27 07:31:45 +08: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 05:55:51 +08:00
|
|
|
case IR_IRDEO:
|
|
|
|
case IR_IRDEO_REMOTE:
|
2010-07-27 07:31:45 +08: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;
|
|
|
|
}
|
|
|
|
|
2017-10-17 07:10:36 +08:00
|
|
|
static void serial_ir_timeout(struct timer_list *unused)
|
2016-12-03 01:16:11 +08:00
|
|
|
{
|
2018-08-22 03:57:52 +08:00
|
|
|
struct ir_raw_event ev = {
|
|
|
|
.timeout = true,
|
|
|
|
.duration = serial_ir.rcdev->timeout
|
|
|
|
};
|
2016-12-03 01:16:11 +08:00
|
|
|
ir_raw_event_store_with_filter(serial_ir.rcdev, &ev);
|
|
|
|
ir_raw_event_handle(serial_ir.rcdev);
|
|
|
|
}
|
|
|
|
|
2017-02-25 19:28:16 +08:00
|
|
|
/* Needed by serial_ir_probe() */
|
|
|
|
static int serial_ir_tx(struct rc_dev *dev, unsigned int *txbuf,
|
|
|
|
unsigned int count);
|
|
|
|
static int serial_ir_tx_duty_cycle(struct rc_dev *dev, u32 cycle);
|
|
|
|
static int serial_ir_tx_carrier(struct rc_dev *dev, u32 carrier);
|
|
|
|
static int serial_ir_open(struct rc_dev *rcdev);
|
|
|
|
static void serial_ir_close(struct rc_dev *rcdev);
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static int serial_ir_probe(struct platform_device *dev)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
2017-02-25 19:28:16 +08:00
|
|
|
struct rc_dev *rcdev;
|
2011-06-14 02:32:26 +08:00
|
|
|
int i, nlow, nhigh, result;
|
|
|
|
|
2017-02-25 19:28:16 +08:00
|
|
|
rcdev = devm_rc_allocate_device(&dev->dev, RC_DRIVER_IR_RAW);
|
|
|
|
if (!rcdev)
|
|
|
|
return -ENOMEM;
|
|
|
|
|
|
|
|
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:
|
2017-07-02 00:13:19 +08:00
|
|
|
rcdev->device_name = "Serial IR type home-brew";
|
2017-02-25 19:28:16 +08:00
|
|
|
break;
|
|
|
|
case IR_IRDEO:
|
2017-07-02 00:13:19 +08:00
|
|
|
rcdev->device_name = "Serial IR type IRdeo";
|
2017-02-25 19:28:16 +08:00
|
|
|
break;
|
|
|
|
case IR_IRDEO_REMOTE:
|
2017-07-02 00:13:19 +08:00
|
|
|
rcdev->device_name = "Serial IR type IRdeo remote";
|
2017-02-25 19:28:16 +08:00
|
|
|
break;
|
|
|
|
case IR_ANIMAX:
|
2017-07-02 00:13:19 +08:00
|
|
|
rcdev->device_name = "Serial IR type AnimaX";
|
2017-02-25 19:28:16 +08:00
|
|
|
break;
|
|
|
|
case IR_IGOR:
|
2017-07-02 00:13:19 +08:00
|
|
|
rcdev->device_name = "Serial IR type IgorPlug";
|
2017-02-25 19:28:16 +08:00
|
|
|
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;
|
2017-08-08 04:20:58 +08:00
|
|
|
rcdev->allowed_protocols = RC_PROTO_BIT_ALL_IR_DECODER;
|
2017-02-25 19:28:16 +08:00
|
|
|
rcdev->driver_name = KBUILD_MODNAME;
|
|
|
|
rcdev->map_name = RC_MAP_RC6_MCE;
|
|
|
|
rcdev->min_timeout = 1;
|
|
|
|
rcdev->timeout = IR_DEFAULT_TIMEOUT;
|
|
|
|
rcdev->max_timeout = 10 * IR_DEFAULT_TIMEOUT;
|
|
|
|
rcdev->rx_resolution = 250000;
|
|
|
|
|
|
|
|
serial_ir.rcdev = rcdev;
|
|
|
|
|
2017-10-17 07:10:36 +08:00
|
|
|
timer_setup(&serial_ir.timeout_timer, serial_ir_timeout, 0);
|
2017-02-25 19:28:16 +08:00
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
result = devm_request_irq(&dev->dev, irq, serial_ir_irq_handler,
|
|
|
|
share_irq ? IRQF_SHARED : 0,
|
|
|
|
KBUILD_MODNAME, &hardware);
|
2011-11-16 12:54:04 +08:00
|
|
|
if (result < 0) {
|
|
|
|
if (result == -EBUSY)
|
2012-11-09 02:55:09 +08:00
|
|
|
dev_err(&dev->dev, "IRQ %d busy\n", irq);
|
2011-11-16 12:54:04 +08:00
|
|
|
else if (result == -EINVAL)
|
2012-11-09 02:55:09 +08:00
|
|
|
dev_err(&dev->dev, "Bad irq number or handler\n");
|
2011-11-16 12:54:04 +08:00
|
|
|
return result;
|
|
|
|
}
|
2010-07-27 07:31:45 +08:00
|
|
|
|
|
|
|
/* Reserve io region. */
|
2016-11-22 15:46:14 +08: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-09 02:55:09 +08: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-04 03:38:40 +08:00
|
|
|
return -EBUSY;
|
2010-07-27 07:31:45 +08:00
|
|
|
}
|
|
|
|
|
2011-11-16 12:53:35 +08:00
|
|
|
result = hardware_init_port();
|
|
|
|
if (result < 0)
|
2014-07-04 03:38:40 +08:00
|
|
|
return result;
|
2010-07-27 07:31:45 +08:00
|
|
|
|
|
|
|
/* Initialize pulse/space widths */
|
2017-08-23 23:06:04 +08:00
|
|
|
serial_ir.duty_cycle = 50;
|
|
|
|
serial_ir.carrier = 38000;
|
2010-07-27 07:31:45 +08: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.04s, 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-14 01:46:57 +08:00
|
|
|
sense = nlow >= nhigh ? 1 : 0;
|
2012-11-09 02:55:09 +08:00
|
|
|
dev_info(&dev->dev, "auto-detected active %s receiver\n",
|
|
|
|
sense ? "low" : "high");
|
2010-07-27 07:31:45 +08:00
|
|
|
} else
|
2012-11-09 02:55:09 +08:00
|
|
|
dev_info(&dev->dev, "Manually using active %s receiver\n",
|
|
|
|
sense ? "low" : "high");
|
2010-07-27 07:31:45 +08:00
|
|
|
|
2015-09-05 04:04:05 +08:00
|
|
|
dev_dbg(&dev->dev, "Interrupt %d, port %04x obtained\n", irq, io);
|
2017-02-25 19:28:16 +08:00
|
|
|
|
|
|
|
return devm_rc_register_device(&dev->dev, rcdev);
|
2010-07-27 07:31:45 +08:00
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static int serial_ir_open(struct rc_dev *rcdev)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
|
|
|
|
/* initialize timestamp */
|
2016-11-22 05:55:51 +08:00
|
|
|
serial_ir.lastkt = ktime_get();
|
2010-07-27 07:31:45 +08:00
|
|
|
|
|
|
|
spin_lock_irqsave(&hardware[type].lock, flags);
|
|
|
|
|
|
|
|
/* Set DLAB 0. */
|
|
|
|
soutp(UART_LCR, sinp(UART_LCR) & (~UART_LCR_DLAB));
|
|
|
|
|
2016-11-22 15:46:14 +08:00
|
|
|
soutp(UART_IER, sinp(UART_IER) | UART_IER_MSI);
|
2010-07-27 07:31:45 +08:00
|
|
|
|
|
|
|
spin_unlock_irqrestore(&hardware[type].lock, flags);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static void serial_ir_close(struct rc_dev *rcdev)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
2010-07-27 07:31:45 +08: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 15:46:14 +08:00
|
|
|
(~(UART_IER_MSI | UART_IER_RLSI | UART_IER_THRI | UART_IER_RDI)));
|
2010-07-27 07:31:45 +08:00
|
|
|
spin_unlock_irqrestore(&hardware[type].lock, flags);
|
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static int serial_ir_tx(struct rc_dev *dev, unsigned int *txbuf,
|
|
|
|
unsigned int count)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
|
|
|
unsigned long flags;
|
2016-11-22 05:55:52 +08:00
|
|
|
ktime_t edge;
|
|
|
|
s64 delta;
|
2016-11-22 05:55:51 +08:00
|
|
|
int i;
|
2010-07-27 07:31:45 +08:00
|
|
|
|
|
|
|
spin_lock_irqsave(&hardware[type].lock, flags);
|
2016-11-22 05:55:51 +08:00
|
|
|
if (type == IR_IRDEO) {
|
2010-07-27 07:31:45 +08:00
|
|
|
/* DTR, RTS down */
|
|
|
|
on();
|
|
|
|
}
|
2016-11-22 05:55:52 +08:00
|
|
|
|
|
|
|
edge = ktime_get();
|
2010-07-27 07:31:45 +08:00
|
|
|
for (i = 0; i < count; i++) {
|
2016-11-22 15:46:14 +08:00
|
|
|
if (i % 2)
|
2016-11-22 05:55:52 +08:00
|
|
|
hardware[type].send_space();
|
2010-07-27 07:31:45 +08:00
|
|
|
else
|
2016-11-22 05:55:52 +08: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 15:46:14 +08:00
|
|
|
} else if (delta > 0) {
|
2016-11-22 05:55:52 +08:00
|
|
|
udelay(delta);
|
2016-11-22 15:46:14 +08:00
|
|
|
}
|
2010-07-27 07:31:45 +08:00
|
|
|
}
|
|
|
|
off();
|
|
|
|
spin_unlock_irqrestore(&hardware[type].lock, flags);
|
2016-11-22 05:55:51 +08:00
|
|
|
return count;
|
2010-07-27 07:31:45 +08:00
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static int serial_ir_tx_duty_cycle(struct rc_dev *dev, u32 cycle)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
2017-08-23 23:06:04 +08:00
|
|
|
serial_ir.duty_cycle = cycle;
|
2016-11-22 05:55:52 +08:00
|
|
|
return 0;
|
2010-07-27 07:31:45 +08:00
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static int serial_ir_tx_carrier(struct rc_dev *dev, u32 carrier)
|
|
|
|
{
|
|
|
|
if (carrier > 500000 || carrier < 20000)
|
|
|
|
return -EINVAL;
|
2010-07-27 07:31:45 +08:00
|
|
|
|
2017-08-23 23:06:04 +08:00
|
|
|
serial_ir.carrier = carrier;
|
2016-11-22 05:55:52 +08:00
|
|
|
return 0;
|
2016-11-22 05:55:51 +08:00
|
|
|
}
|
2010-07-27 07:31:45 +08:00
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static int serial_ir_suspend(struct platform_device *dev,
|
|
|
|
pm_message_t state)
|
2010-07-27 07:31:45 +08: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 15:46:14 +08:00
|
|
|
(~(UART_IER_MSI | UART_IER_RLSI | UART_IER_THRI | UART_IER_RDI)));
|
2010-07-27 07:31:45 +08:00
|
|
|
|
|
|
|
/* Clear registers. */
|
|
|
|
sinp(UART_LSR);
|
|
|
|
sinp(UART_RX);
|
|
|
|
sinp(UART_IIR);
|
|
|
|
sinp(UART_MSR);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static int serial_ir_resume(struct platform_device *dev)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
|
|
|
unsigned long flags;
|
2011-11-16 12:53:35 +08:00
|
|
|
int result;
|
2010-07-27 07:31:45 +08:00
|
|
|
|
2011-11-16 12:53:35 +08:00
|
|
|
result = hardware_init_port();
|
|
|
|
if (result < 0)
|
|
|
|
return result;
|
2010-07-27 07:31:45 +08:00
|
|
|
|
|
|
|
spin_lock_irqsave(&hardware[type].lock, flags);
|
|
|
|
/* Enable Interrupt */
|
2016-11-22 05:55:51 +08:00
|
|
|
serial_ir.lastkt = ktime_get();
|
2016-11-22 15:46:14 +08:00
|
|
|
soutp(UART_IER, sinp(UART_IER) | UART_IER_MSI);
|
2010-07-27 07:31:45 +08:00
|
|
|
off();
|
|
|
|
|
|
|
|
spin_unlock_irqrestore(&hardware[type].lock, flags);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static struct platform_driver serial_ir_driver = {
|
|
|
|
.probe = serial_ir_probe,
|
|
|
|
.suspend = serial_ir_suspend,
|
|
|
|
.resume = serial_ir_resume,
|
2010-07-27 07:31:45 +08:00
|
|
|
.driver = {
|
2016-11-22 05:55:51 +08:00
|
|
|
.name = "serial_ir",
|
2010-07-27 07:31:45 +08:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static int __init serial_ir_init(void)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
|
|
|
int result;
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
result = platform_driver_register(&serial_ir_driver);
|
|
|
|
if (result)
|
2011-11-16 12:53:35 +08:00
|
|
|
return result;
|
2010-07-27 07:31:45 +08:00
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
serial_ir.pdev = platform_device_alloc("serial_ir", 0);
|
|
|
|
if (!serial_ir.pdev) {
|
2010-07-27 07:31:45 +08:00
|
|
|
result = -ENOMEM;
|
|
|
|
goto exit_driver_unregister;
|
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
result = platform_device_add(serial_ir.pdev);
|
2010-07-27 07:31:45 +08:00
|
|
|
if (result)
|
|
|
|
goto exit_device_put;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
exit_device_put:
|
2016-11-22 05:55:51 +08:00
|
|
|
platform_device_put(serial_ir.pdev);
|
2010-07-27 07:31:45 +08:00
|
|
|
exit_driver_unregister:
|
2016-11-22 05:55:51 +08:00
|
|
|
platform_driver_unregister(&serial_ir_driver);
|
2010-07-27 07:31:45 +08:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static void serial_ir_exit(void)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
2016-11-22 05:55:51 +08:00
|
|
|
platform_device_unregister(serial_ir.pdev);
|
|
|
|
platform_driver_unregister(&serial_ir_driver);
|
2010-07-27 07:31:45 +08:00
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static int __init serial_ir_init_module(void)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
|
|
|
switch (type) {
|
2016-11-22 05:55:51 +08:00
|
|
|
case IR_HOMEBREW:
|
|
|
|
case IR_IRDEO:
|
|
|
|
case IR_IRDEO_REMOTE:
|
|
|
|
case IR_ANIMAX:
|
|
|
|
case IR_IGOR:
|
2010-07-27 07:31:45 +08:00
|
|
|
/* if nothing specified, use ttyS0/com1 and irq 4 */
|
|
|
|
io = io ? io : 0x3f8;
|
|
|
|
irq = irq ? irq : 4;
|
|
|
|
break;
|
|
|
|
default:
|
2011-11-16 12:49:41 +08:00
|
|
|
return -EINVAL;
|
2010-07-27 07:31:45 +08:00
|
|
|
}
|
|
|
|
if (!softcarrier) {
|
|
|
|
switch (type) {
|
2016-11-22 05:55:51 +08:00
|
|
|
case IR_HOMEBREW:
|
|
|
|
case IR_IGOR:
|
|
|
|
hardware[type].set_send_carrier = false;
|
|
|
|
hardware[type].set_duty_cycle = false;
|
2010-07-27 07:31:45 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-15 02:50:11 +08:00
|
|
|
/* make sure sense is either -1, 0, or 1 */
|
|
|
|
if (sense != -1)
|
|
|
|
sense = !!sense;
|
|
|
|
|
2019-03-05 13:40:26 +08:00
|
|
|
return serial_ir_init();
|
2010-07-27 07:31:45 +08:00
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
static void __exit serial_ir_exit_module(void)
|
2010-07-27 07:31:45 +08:00
|
|
|
{
|
2016-12-03 01:16:11 +08:00
|
|
|
del_timer_sync(&serial_ir.timeout_timer);
|
2016-11-22 05:55:51 +08:00
|
|
|
serial_ir_exit();
|
2010-07-27 07:31:45 +08:00
|
|
|
}
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
module_init(serial_ir_init_module);
|
|
|
|
module_exit(serial_ir_exit_module);
|
2010-07-27 07:31:45 +08:00
|
|
|
|
|
|
|
MODULE_DESCRIPTION("Infra-red receiver driver for serial ports.");
|
2016-11-22 05:55:51 +08:00
|
|
|
MODULE_AUTHOR("Ralph Metzler, Trent Piepho, Ben Pfaff, Christoph Bartelmus, Andrei Tanas");
|
2010-07-27 07:31:45 +08:00
|
|
|
MODULE_LICENSE("GPL");
|
|
|
|
|
2016-11-22 05:55:51 +08: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 07:31:45 +08:00
|
|
|
|
2017-04-04 23:54:24 +08:00
|
|
|
module_param_hw(io, int, ioport, 0444);
|
2010-07-27 07:31:45 +08:00
|
|
|
MODULE_PARM_DESC(io, "I/O address base (0x3f8 or 0x2f8)");
|
|
|
|
|
|
|
|
/* some architectures (e.g. intel xscale) have memory mapped registers */
|
Annotation of module parameters that specify device settings
-----BEGIN PGP SIGNATURE-----
iQIVAwUAWPiW6vSw1s6N8H32AQLOrw/+NTqGf7bjq+64YKS6NfR0XDgE+wNJltGO
ck7zJW3NHIg76RNu8s0I9xg5aVmwizz3Z5DGROZquaolnezux4tQihZ3AFyxIzLc
+Y3WHYagcML7yFfjl/WznCLRD5EW3yPln4lCvQO0nW/xICRYeRI057JaIbi2Dtek
BhcXt3c4AjXDLdYJkgtHV3p2R2mt8hcdFdWqqx6s7JaIThZNRGNzxAgtbcB9k5IW
HVG9ZEIL73VBYWHrYivzjHYF5rBnNCPt87eOwDQeTOSkhv8te+u9k+bH8vxZw1T0
XUtDrLBndKiuVo2GUfLkkF8LItx3Q9eLCJYy0joaIliyPqTEsPx9KjQ+Af0cxS9s
ZPCZ5SYf96stKmDeL5xaMfrAmeyVHJ4lc4JTOqdzbIT8blsOSfYO/03p0ALShSDv
/RQLaKGlf8Bjoy8PwKFcXb4sIDufcd/U1Av/EMFXxOfgN/u2JUkGKq6EaIM5B68L
fHPje+aR9VNELPmPjwNOWtmN4I79EH3EItQf7zv0KG+UeKhcHLx/EAcSJ3ZRKEkH
Lathg7pPOEJGArPiVO79TZzBG01ADn1aiwv65XObMzNZ+54xI/mN/Y1DNF/kL5jU
XzvNzEjFt8mwMIZGVNdAt4+pDyMfIZGZSyUkSRKFnaQZMIvQrfQIU9RLBYLX5eOx
+/p0VkIwDpg=
=lbS7
-----END PGP SIGNATURE-----
Merge tag 'hwparam-20170420' of git://git.kernel.org/pub/scm/linux/kernel/git/dhowells/linux-fs
Pull hw lockdown support from David Howells:
"Annotation of module parameters that configure hardware resources
including ioports, iomem addresses, irq lines and dma channels.
This allows a future patch to prohibit the use of such module
parameters to prevent that hardware from being abused to gain access
to the running kernel image as part of locking the kernel down under
UEFI secure boot conditions.
Annotations are made by changing:
module_param(n, t, p)
module_param_named(n, v, t, p)
module_param_array(n, t, m, p)
to:
module_param_hw(n, t, hwtype, p)
module_param_hw_named(n, v, t, hwtype, p)
module_param_hw_array(n, t, hwtype, m, p)
where the module parameter refers to a hardware setting
hwtype specifies the type of the resource being configured. This can
be one of:
ioport Module parameter configures an I/O port
iomem Module parameter configures an I/O mem address
ioport_or_iomem Module parameter could be either (runtime set)
irq Module parameter configures an I/O port
dma Module parameter configures a DMA channel
dma_addr Module parameter configures a DMA buffer address
other Module parameter configures some other value
Note that the hwtype is compile checked, but not currently stored (the
lockdown code probably won't require it). It is, however, there for
future use.
A bonus is that the hwtype can also be used for grepping.
The intention is for the kernel to ignore or reject attempts to set
annotated module parameters if lockdown is enabled. This applies to
options passed on the boot command line, passed to insmod/modprobe or
direct twiddling in /sys/module/ parameter files.
The module initialisation then needs to handle the parameter not being
set, by (1) giving an error, (2) probing for a value or (3) using a
reasonable default.
What I can't do is just reject a module out of hand because it may
take a hardware setting in the module parameters. Some important
modules, some ipmi stuff for instance, both probe for hardware and
allow hardware to be manually specified; if the driver is aborts with
any error, you don't get any ipmi hardware.
Further, trying to do this entirely in the module initialisation code
doesn't protect against sysfs twiddling.
[!] Note that in and of itself, this series of patches should have no
effect on the the size of the kernel or code execution - that is
left to a patch in the next series to effect. It does mark
annotated kernel parameters with a KERNEL_PARAM_FL_HWPARAM flag in
an already existing field"
* tag 'hwparam-20170420' of git://git.kernel.org/pub/scm/linux/kernel/git/dhowells/linux-fs: (38 commits)
Annotate hardware config module parameters in sound/pci/
Annotate hardware config module parameters in sound/oss/
Annotate hardware config module parameters in sound/isa/
Annotate hardware config module parameters in sound/drivers/
Annotate hardware config module parameters in fs/pstore/
Annotate hardware config module parameters in drivers/watchdog/
Annotate hardware config module parameters in drivers/video/
Annotate hardware config module parameters in drivers/tty/
Annotate hardware config module parameters in drivers/staging/vme/
Annotate hardware config module parameters in drivers/staging/speakup/
Annotate hardware config module parameters in drivers/staging/media/
Annotate hardware config module parameters in drivers/scsi/
Annotate hardware config module parameters in drivers/pcmcia/
Annotate hardware config module parameters in drivers/pci/hotplug/
Annotate hardware config module parameters in drivers/parport/
Annotate hardware config module parameters in drivers/net/wireless/
Annotate hardware config module parameters in drivers/net/wan/
Annotate hardware config module parameters in drivers/net/irda/
Annotate hardware config module parameters in drivers/net/hamradio/
Annotate hardware config module parameters in drivers/net/ethernet/
...
2017-05-11 10:13:03 +08:00
|
|
|
module_param_hw(iommap, ulong, other, 0444);
|
2016-11-22 05:55:51 +08:00
|
|
|
MODULE_PARM_DESC(iommap, "physical base for memory mapped I/O (0 = no memory mapped io)");
|
2010-07-27 07:31:45 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* some architectures (e.g. intel xscale) align the 8bit serial registers
|
|
|
|
* on 32bit word boundaries.
|
2012-03-05 22:49:26 +08:00
|
|
|
* See linux-kernel/drivers/tty/serial/8250/8250.c serial_in()/out()
|
2010-07-27 07:31:45 +08:00
|
|
|
*/
|
2017-04-04 23:54:24 +08:00
|
|
|
module_param_hw(ioshift, int, other, 0444);
|
2010-07-27 07:31:45 +08:00
|
|
|
MODULE_PARM_DESC(ioshift, "shift I/O register offset (0 = no shift)");
|
|
|
|
|
2017-04-04 23:54:24 +08:00
|
|
|
module_param_hw(irq, int, irq, 0444);
|
2010-07-27 07:31:45 +08:00
|
|
|
MODULE_PARM_DESC(irq, "Interrupt (4 or 3)");
|
|
|
|
|
2017-04-04 23:54:24 +08:00
|
|
|
module_param_hw(share_irq, bool, other, 0444);
|
2010-07-27 07:31:45 +08:00
|
|
|
MODULE_PARM_DESC(share_irq, "Share interrupts (0 = off, 1 = on)");
|
|
|
|
|
2016-11-22 05:55:51 +08: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 07:31:45 +08:00
|
|
|
|
2016-11-22 05:55:51 +08: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 07:31:45 +08:00
|
|
|
#endif
|
|
|
|
|
2016-11-22 05:55:51 +08:00
|
|
|
module_param(softcarrier, bool, 0444);
|
2010-07-27 07:31:45 +08:00
|
|
|
MODULE_PARM_DESC(softcarrier, "Software carrier (0 = off, 1 = on, default on)");
|