| /* | 
 |  * Driver for Feature Integration Technology Inc. (aka Fintek) LPC CIR | 
 |  * | 
 |  * Copyright (C) 2011 Jarod Wilson <jarod@redhat.com> | 
 |  * | 
 |  * Special thanks to Fintek for providing hardware and spec sheets. | 
 |  * This driver is based upon the nuvoton, ite and ene drivers for | 
 |  * similar hardware. | 
 |  * | 
 |  * 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. | 
 |  */ | 
 |  | 
 | #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt | 
 |  | 
 | #include <linux/kernel.h> | 
 | #include <linux/module.h> | 
 | #include <linux/pnp.h> | 
 | #include <linux/io.h> | 
 | #include <linux/interrupt.h> | 
 | #include <linux/sched.h> | 
 | #include <linux/slab.h> | 
 | #include <media/rc-core.h> | 
 |  | 
 | #include "fintek-cir.h" | 
 |  | 
 | /* write val to config reg */ | 
 | static inline void fintek_cr_write(struct fintek_dev *fintek, u8 val, u8 reg) | 
 | { | 
 | 	fit_dbg("%s: reg 0x%02x, val 0x%02x  (ip/dp: %02x/%02x)", | 
 | 		__func__, reg, val, fintek->cr_ip, fintek->cr_dp); | 
 | 	outb(reg, fintek->cr_ip); | 
 | 	outb(val, fintek->cr_dp); | 
 | } | 
 |  | 
 | /* read val from config reg */ | 
 | static inline u8 fintek_cr_read(struct fintek_dev *fintek, u8 reg) | 
 | { | 
 | 	u8 val; | 
 |  | 
 | 	outb(reg, fintek->cr_ip); | 
 | 	val = inb(fintek->cr_dp); | 
 |  | 
 | 	fit_dbg("%s: reg 0x%02x, val 0x%02x  (ip/dp: %02x/%02x)", | 
 | 		__func__, reg, val, fintek->cr_ip, fintek->cr_dp); | 
 | 	return val; | 
 | } | 
 |  | 
 | /* update config register bit without changing other bits */ | 
 | static inline void fintek_set_reg_bit(struct fintek_dev *fintek, u8 val, u8 reg) | 
 | { | 
 | 	u8 tmp = fintek_cr_read(fintek, reg) | val; | 
 | 	fintek_cr_write(fintek, tmp, reg); | 
 | } | 
 |  | 
 | /* clear config register bit without changing other bits */ | 
 | static inline void fintek_clear_reg_bit(struct fintek_dev *fintek, u8 val, u8 reg) | 
 | { | 
 | 	u8 tmp = fintek_cr_read(fintek, reg) & ~val; | 
 | 	fintek_cr_write(fintek, tmp, reg); | 
 | } | 
 |  | 
 | /* enter config mode */ | 
 | static inline void fintek_config_mode_enable(struct fintek_dev *fintek) | 
 | { | 
 | 	/* Enabling Config Mode explicitly requires writing 2x */ | 
 | 	outb(CONFIG_REG_ENABLE, fintek->cr_ip); | 
 | 	outb(CONFIG_REG_ENABLE, fintek->cr_ip); | 
 | } | 
 |  | 
 | /* exit config mode */ | 
 | static inline void fintek_config_mode_disable(struct fintek_dev *fintek) | 
 | { | 
 | 	outb(CONFIG_REG_DISABLE, fintek->cr_ip); | 
 | } | 
 |  | 
 | /* | 
 |  * When you want to address a specific logical device, write its logical | 
 |  * device number to GCR_LOGICAL_DEV_NO | 
 |  */ | 
 | static inline void fintek_select_logical_dev(struct fintek_dev *fintek, u8 ldev) | 
 | { | 
 | 	fintek_cr_write(fintek, ldev, GCR_LOGICAL_DEV_NO); | 
 | } | 
 |  | 
 | /* write val to cir config register */ | 
 | static inline void fintek_cir_reg_write(struct fintek_dev *fintek, u8 val, u8 offset) | 
 | { | 
 | 	outb(val, fintek->cir_addr + offset); | 
 | } | 
 |  | 
 | /* read val from cir config register */ | 
 | static u8 fintek_cir_reg_read(struct fintek_dev *fintek, u8 offset) | 
 | { | 
 | 	return inb(fintek->cir_addr + offset); | 
 | } | 
 |  | 
 | /* dump current cir register contents */ | 
 | static void cir_dump_regs(struct fintek_dev *fintek) | 
 | { | 
 | 	fintek_config_mode_enable(fintek); | 
 | 	fintek_select_logical_dev(fintek, fintek->logical_dev_cir); | 
 |  | 
 | 	pr_info("%s: Dump CIR logical device registers:\n", FINTEK_DRIVER_NAME); | 
 | 	pr_info(" * CR CIR BASE ADDR: 0x%x\n", | 
 | 		(fintek_cr_read(fintek, CIR_CR_BASE_ADDR_HI) << 8) | | 
 | 		fintek_cr_read(fintek, CIR_CR_BASE_ADDR_LO)); | 
 | 	pr_info(" * CR CIR IRQ NUM:   0x%x\n", | 
 | 		fintek_cr_read(fintek, CIR_CR_IRQ_SEL)); | 
 |  | 
 | 	fintek_config_mode_disable(fintek); | 
 |  | 
 | 	pr_info("%s: Dump CIR registers:\n", FINTEK_DRIVER_NAME); | 
 | 	pr_info(" * STATUS:     0x%x\n", | 
 | 		fintek_cir_reg_read(fintek, CIR_STATUS)); | 
 | 	pr_info(" * CONTROL:    0x%x\n", | 
 | 		fintek_cir_reg_read(fintek, CIR_CONTROL)); | 
 | 	pr_info(" * RX_DATA:    0x%x\n", | 
 | 		fintek_cir_reg_read(fintek, CIR_RX_DATA)); | 
 | 	pr_info(" * TX_CONTROL: 0x%x\n", | 
 | 		fintek_cir_reg_read(fintek, CIR_TX_CONTROL)); | 
 | 	pr_info(" * TX_DATA:    0x%x\n", | 
 | 		fintek_cir_reg_read(fintek, CIR_TX_DATA)); | 
 | } | 
 |  | 
 | /* detect hardware features */ | 
 | static int fintek_hw_detect(struct fintek_dev *fintek) | 
 | { | 
 | 	unsigned long flags; | 
 | 	u8 chip_major, chip_minor; | 
 | 	u8 vendor_major, vendor_minor; | 
 | 	u8 portsel, ir_class; | 
 | 	u16 vendor, chip; | 
 |  | 
 | 	fintek_config_mode_enable(fintek); | 
 |  | 
 | 	/* Check if we're using config port 0x4e or 0x2e */ | 
 | 	portsel = fintek_cr_read(fintek, GCR_CONFIG_PORT_SEL); | 
 | 	if (portsel == 0xff) { | 
 | 		fit_pr(KERN_INFO, "first portsel read was bunk, trying alt"); | 
 | 		fintek_config_mode_disable(fintek); | 
 | 		fintek->cr_ip = CR_INDEX_PORT2; | 
 | 		fintek->cr_dp = CR_DATA_PORT2; | 
 | 		fintek_config_mode_enable(fintek); | 
 | 		portsel = fintek_cr_read(fintek, GCR_CONFIG_PORT_SEL); | 
 | 	} | 
 | 	fit_dbg("portsel reg: 0x%02x", portsel); | 
 |  | 
 | 	ir_class = fintek_cir_reg_read(fintek, CIR_CR_CLASS); | 
 | 	fit_dbg("ir_class reg: 0x%02x", ir_class); | 
 |  | 
 | 	switch (ir_class) { | 
 | 	case CLASS_RX_2TX: | 
 | 	case CLASS_RX_1TX: | 
 | 		fintek->hw_tx_capable = true; | 
 | 		break; | 
 | 	case CLASS_RX_ONLY: | 
 | 	default: | 
 | 		fintek->hw_tx_capable = false; | 
 | 		break; | 
 | 	} | 
 |  | 
 | 	chip_major = fintek_cr_read(fintek, GCR_CHIP_ID_HI); | 
 | 	chip_minor = fintek_cr_read(fintek, GCR_CHIP_ID_LO); | 
 | 	chip  = chip_major << 8 | chip_minor; | 
 |  | 
 | 	vendor_major = fintek_cr_read(fintek, GCR_VENDOR_ID_HI); | 
 | 	vendor_minor = fintek_cr_read(fintek, GCR_VENDOR_ID_LO); | 
 | 	vendor = vendor_major << 8 | vendor_minor; | 
 |  | 
 | 	if (vendor != VENDOR_ID_FINTEK) | 
 | 		fit_pr(KERN_WARNING, "Unknown vendor ID: 0x%04x", vendor); | 
 | 	else | 
 | 		fit_dbg("Read Fintek vendor ID from chip"); | 
 |  | 
 | 	fintek_config_mode_disable(fintek); | 
 |  | 
 | 	spin_lock_irqsave(&fintek->fintek_lock, flags); | 
 | 	fintek->chip_major  = chip_major; | 
 | 	fintek->chip_minor  = chip_minor; | 
 | 	fintek->chip_vendor = vendor; | 
 |  | 
 | 	/* | 
 | 	 * Newer reviews of this chipset uses port 8 instead of 5 | 
 | 	 */ | 
 | 	if ((chip != 0x0408) && (chip != 0x0804)) | 
 | 		fintek->logical_dev_cir = LOGICAL_DEV_CIR_REV2; | 
 | 	else | 
 | 		fintek->logical_dev_cir = LOGICAL_DEV_CIR_REV1; | 
 |  | 
 | 	spin_unlock_irqrestore(&fintek->fintek_lock, flags); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void fintek_cir_ldev_init(struct fintek_dev *fintek) | 
 | { | 
 | 	/* Select CIR logical device and enable */ | 
 | 	fintek_select_logical_dev(fintek, fintek->logical_dev_cir); | 
 | 	fintek_cr_write(fintek, LOGICAL_DEV_ENABLE, CIR_CR_DEV_EN); | 
 |  | 
 | 	/* Write allocated CIR address and IRQ information to hardware */ | 
 | 	fintek_cr_write(fintek, fintek->cir_addr >> 8, CIR_CR_BASE_ADDR_HI); | 
 | 	fintek_cr_write(fintek, fintek->cir_addr & 0xff, CIR_CR_BASE_ADDR_LO); | 
 |  | 
 | 	fintek_cr_write(fintek, fintek->cir_irq, CIR_CR_IRQ_SEL); | 
 |  | 
 | 	fit_dbg("CIR initialized, base io address: 0x%lx, irq: %d (len: %d)", | 
 | 		fintek->cir_addr, fintek->cir_irq, fintek->cir_port_len); | 
 | } | 
 |  | 
 | /* enable CIR interrupts */ | 
 | static void fintek_enable_cir_irq(struct fintek_dev *fintek) | 
 | { | 
 | 	fintek_cir_reg_write(fintek, CIR_STATUS_IRQ_EN, CIR_STATUS); | 
 | } | 
 |  | 
 | static void fintek_cir_regs_init(struct fintek_dev *fintek) | 
 | { | 
 | 	/* clear any and all stray interrupts */ | 
 | 	fintek_cir_reg_write(fintek, CIR_STATUS_IRQ_MASK, CIR_STATUS); | 
 |  | 
 | 	/* and finally, enable interrupts */ | 
 | 	fintek_enable_cir_irq(fintek); | 
 | } | 
 |  | 
 | static void fintek_enable_wake(struct fintek_dev *fintek) | 
 | { | 
 | 	fintek_config_mode_enable(fintek); | 
 | 	fintek_select_logical_dev(fintek, LOGICAL_DEV_ACPI); | 
 |  | 
 | 	/* Allow CIR PME's to wake system */ | 
 | 	fintek_set_reg_bit(fintek, ACPI_WAKE_EN_CIR_BIT, LDEV_ACPI_WAKE_EN_REG); | 
 | 	/* Enable CIR PME's */ | 
 | 	fintek_set_reg_bit(fintek, ACPI_PME_CIR_BIT, LDEV_ACPI_PME_EN_REG); | 
 | 	/* Clear CIR PME status register */ | 
 | 	fintek_set_reg_bit(fintek, ACPI_PME_CIR_BIT, LDEV_ACPI_PME_CLR_REG); | 
 | 	/* Save state */ | 
 | 	fintek_set_reg_bit(fintek, ACPI_STATE_CIR_BIT, LDEV_ACPI_STATE_REG); | 
 |  | 
 | 	fintek_config_mode_disable(fintek); | 
 | } | 
 |  | 
 | static int fintek_cmdsize(u8 cmd, u8 subcmd) | 
 | { | 
 | 	int datasize = 0; | 
 |  | 
 | 	switch (cmd) { | 
 | 	case BUF_COMMAND_NULL: | 
 | 		if (subcmd == BUF_HW_CMD_HEADER) | 
 | 			datasize = 1; | 
 | 		break; | 
 | 	case BUF_HW_CMD_HEADER: | 
 | 		if (subcmd == BUF_CMD_G_REVISION) | 
 | 			datasize = 2; | 
 | 		break; | 
 | 	case BUF_COMMAND_HEADER: | 
 | 		switch (subcmd) { | 
 | 		case BUF_CMD_S_CARRIER: | 
 | 		case BUF_CMD_S_TIMEOUT: | 
 | 		case BUF_RSP_PULSE_COUNT: | 
 | 			datasize = 2; | 
 | 			break; | 
 | 		case BUF_CMD_SIG_END: | 
 | 		case BUF_CMD_S_TXMASK: | 
 | 		case BUF_CMD_S_RXSENSOR: | 
 | 			datasize = 1; | 
 | 			break; | 
 | 		} | 
 | 	} | 
 |  | 
 | 	return datasize; | 
 | } | 
 |  | 
 | /* process ir data stored in driver buffer */ | 
 | static void fintek_process_rx_ir_data(struct fintek_dev *fintek) | 
 | { | 
 | 	DEFINE_IR_RAW_EVENT(rawir); | 
 | 	u8 sample; | 
 | 	bool event = false; | 
 | 	int i; | 
 |  | 
 | 	for (i = 0; i < fintek->pkts; i++) { | 
 | 		sample = fintek->buf[i]; | 
 | 		switch (fintek->parser_state) { | 
 | 		case CMD_HEADER: | 
 | 			fintek->cmd = sample; | 
 | 			if ((fintek->cmd == BUF_COMMAND_HEADER) || | 
 | 			    ((fintek->cmd & BUF_COMMAND_MASK) != | 
 | 			     BUF_PULSE_BIT)) { | 
 | 				fintek->parser_state = SUBCMD; | 
 | 				continue; | 
 | 			} | 
 | 			fintek->rem = (fintek->cmd & BUF_LEN_MASK); | 
 | 			fit_dbg("%s: rem: 0x%02x", __func__, fintek->rem); | 
 | 			if (fintek->rem) | 
 | 				fintek->parser_state = PARSE_IRDATA; | 
 | 			else | 
 | 				ir_raw_event_reset(fintek->rdev); | 
 | 			break; | 
 | 		case SUBCMD: | 
 | 			fintek->rem = fintek_cmdsize(fintek->cmd, sample); | 
 | 			fintek->parser_state = CMD_DATA; | 
 | 			break; | 
 | 		case CMD_DATA: | 
 | 			fintek->rem--; | 
 | 			break; | 
 | 		case PARSE_IRDATA: | 
 | 			fintek->rem--; | 
 | 			init_ir_raw_event(&rawir); | 
 | 			rawir.pulse = ((sample & BUF_PULSE_BIT) != 0); | 
 | 			rawir.duration = US_TO_NS((sample & BUF_SAMPLE_MASK) | 
 | 					  * CIR_SAMPLE_PERIOD); | 
 |  | 
 | 			fit_dbg("Storing %s with duration %d", | 
 | 				rawir.pulse ? "pulse" : "space", | 
 | 				rawir.duration); | 
 | 			if (ir_raw_event_store_with_filter(fintek->rdev, | 
 | 									&rawir)) | 
 | 				event = true; | 
 | 			break; | 
 | 		} | 
 |  | 
 | 		if ((fintek->parser_state != CMD_HEADER) && !fintek->rem) | 
 | 			fintek->parser_state = CMD_HEADER; | 
 | 	} | 
 |  | 
 | 	fintek->pkts = 0; | 
 |  | 
 | 	if (event) { | 
 | 		fit_dbg("Calling ir_raw_event_handle"); | 
 | 		ir_raw_event_handle(fintek->rdev); | 
 | 	} | 
 | } | 
 |  | 
 | /* copy data from hardware rx register into driver buffer */ | 
 | static void fintek_get_rx_ir_data(struct fintek_dev *fintek, u8 rx_irqs) | 
 | { | 
 | 	unsigned long flags; | 
 | 	u8 sample, status; | 
 |  | 
 | 	spin_lock_irqsave(&fintek->fintek_lock, flags); | 
 |  | 
 | 	/* | 
 | 	 * We must read data from CIR_RX_DATA until the hardware IR buffer | 
 | 	 * is empty and clears the RX_TIMEOUT and/or RX_RECEIVE flags in | 
 | 	 * the CIR_STATUS register | 
 | 	 */ | 
 | 	do { | 
 | 		sample = fintek_cir_reg_read(fintek, CIR_RX_DATA); | 
 | 		fit_dbg("%s: sample: 0x%02x", __func__, sample); | 
 |  | 
 | 		fintek->buf[fintek->pkts] = sample; | 
 | 		fintek->pkts++; | 
 |  | 
 | 		status = fintek_cir_reg_read(fintek, CIR_STATUS); | 
 | 		if (!(status & CIR_STATUS_IRQ_EN)) | 
 | 			break; | 
 | 	} while (status & rx_irqs); | 
 |  | 
 | 	fintek_process_rx_ir_data(fintek); | 
 |  | 
 | 	spin_unlock_irqrestore(&fintek->fintek_lock, flags); | 
 | } | 
 |  | 
 | static void fintek_cir_log_irqs(u8 status) | 
 | { | 
 | 	fit_pr(KERN_INFO, "IRQ 0x%02x:%s%s%s%s%s", status, | 
 | 		status & CIR_STATUS_IRQ_EN	? " IRQEN"	: "", | 
 | 		status & CIR_STATUS_TX_FINISH	? " TXF"	: "", | 
 | 		status & CIR_STATUS_TX_UNDERRUN	? " TXU"	: "", | 
 | 		status & CIR_STATUS_RX_TIMEOUT	? " RXTO"	: "", | 
 | 		status & CIR_STATUS_RX_RECEIVE	? " RXOK"	: ""); | 
 | } | 
 |  | 
 | /* interrupt service routine for incoming and outgoing CIR data */ | 
 | static irqreturn_t fintek_cir_isr(int irq, void *data) | 
 | { | 
 | 	struct fintek_dev *fintek = data; | 
 | 	u8 status, rx_irqs; | 
 |  | 
 | 	fit_dbg_verbose("%s firing", __func__); | 
 |  | 
 | 	fintek_config_mode_enable(fintek); | 
 | 	fintek_select_logical_dev(fintek, fintek->logical_dev_cir); | 
 | 	fintek_config_mode_disable(fintek); | 
 |  | 
 | 	/* | 
 | 	 * Get IR Status register contents. Write 1 to ack/clear | 
 | 	 * | 
 | 	 * bit: reg name    - description | 
 | 	 *   3: TX_FINISH   - TX is finished | 
 | 	 *   2: TX_UNDERRUN - TX underrun | 
 | 	 *   1: RX_TIMEOUT  - RX data timeout | 
 | 	 *   0: RX_RECEIVE  - RX data received | 
 | 	 */ | 
 | 	status = fintek_cir_reg_read(fintek, CIR_STATUS); | 
 | 	if (!(status & CIR_STATUS_IRQ_MASK) || status == 0xff) { | 
 | 		fit_dbg_verbose("%s exiting, IRSTS 0x%02x", __func__, status); | 
 | 		fintek_cir_reg_write(fintek, CIR_STATUS_IRQ_MASK, CIR_STATUS); | 
 | 		return IRQ_RETVAL(IRQ_NONE); | 
 | 	} | 
 |  | 
 | 	if (debug) | 
 | 		fintek_cir_log_irqs(status); | 
 |  | 
 | 	rx_irqs = status & (CIR_STATUS_RX_RECEIVE | CIR_STATUS_RX_TIMEOUT); | 
 | 	if (rx_irqs) | 
 | 		fintek_get_rx_ir_data(fintek, rx_irqs); | 
 |  | 
 | 	/* ack/clear all irq flags we've got */ | 
 | 	fintek_cir_reg_write(fintek, status, CIR_STATUS); | 
 |  | 
 | 	fit_dbg_verbose("%s done", __func__); | 
 | 	return IRQ_RETVAL(IRQ_HANDLED); | 
 | } | 
 |  | 
 | static void fintek_enable_cir(struct fintek_dev *fintek) | 
 | { | 
 | 	/* set IRQ enabled */ | 
 | 	fintek_cir_reg_write(fintek, CIR_STATUS_IRQ_EN, CIR_STATUS); | 
 |  | 
 | 	fintek_config_mode_enable(fintek); | 
 |  | 
 | 	/* enable the CIR logical device */ | 
 | 	fintek_select_logical_dev(fintek, fintek->logical_dev_cir); | 
 | 	fintek_cr_write(fintek, LOGICAL_DEV_ENABLE, CIR_CR_DEV_EN); | 
 |  | 
 | 	fintek_config_mode_disable(fintek); | 
 |  | 
 | 	/* clear all pending interrupts */ | 
 | 	fintek_cir_reg_write(fintek, CIR_STATUS_IRQ_MASK, CIR_STATUS); | 
 |  | 
 | 	/* enable interrupts */ | 
 | 	fintek_enable_cir_irq(fintek); | 
 | } | 
 |  | 
 | static void fintek_disable_cir(struct fintek_dev *fintek) | 
 | { | 
 | 	fintek_config_mode_enable(fintek); | 
 |  | 
 | 	/* disable the CIR logical device */ | 
 | 	fintek_select_logical_dev(fintek, fintek->logical_dev_cir); | 
 | 	fintek_cr_write(fintek, LOGICAL_DEV_DISABLE, CIR_CR_DEV_EN); | 
 |  | 
 | 	fintek_config_mode_disable(fintek); | 
 | } | 
 |  | 
 | static int fintek_open(struct rc_dev *dev) | 
 | { | 
 | 	struct fintek_dev *fintek = dev->priv; | 
 | 	unsigned long flags; | 
 |  | 
 | 	spin_lock_irqsave(&fintek->fintek_lock, flags); | 
 | 	fintek_enable_cir(fintek); | 
 | 	spin_unlock_irqrestore(&fintek->fintek_lock, flags); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void fintek_close(struct rc_dev *dev) | 
 | { | 
 | 	struct fintek_dev *fintek = dev->priv; | 
 | 	unsigned long flags; | 
 |  | 
 | 	spin_lock_irqsave(&fintek->fintek_lock, flags); | 
 | 	fintek_disable_cir(fintek); | 
 | 	spin_unlock_irqrestore(&fintek->fintek_lock, flags); | 
 | } | 
 |  | 
 | /* Allocate memory, probe hardware, and initialize everything */ | 
 | static int fintek_probe(struct pnp_dev *pdev, const struct pnp_device_id *dev_id) | 
 | { | 
 | 	struct fintek_dev *fintek; | 
 | 	struct rc_dev *rdev; | 
 | 	int ret = -ENOMEM; | 
 |  | 
 | 	fintek = kzalloc(sizeof(struct fintek_dev), GFP_KERNEL); | 
 | 	if (!fintek) | 
 | 		return ret; | 
 |  | 
 | 	/* input device for IR remote (and tx) */ | 
 | 	rdev = rc_allocate_device(RC_DRIVER_IR_RAW); | 
 | 	if (!rdev) | 
 | 		goto exit_free_dev_rdev; | 
 |  | 
 | 	ret = -ENODEV; | 
 | 	/* validate pnp resources */ | 
 | 	if (!pnp_port_valid(pdev, 0)) { | 
 | 		dev_err(&pdev->dev, "IR PNP Port not valid!\n"); | 
 | 		goto exit_free_dev_rdev; | 
 | 	} | 
 |  | 
 | 	if (!pnp_irq_valid(pdev, 0)) { | 
 | 		dev_err(&pdev->dev, "IR PNP IRQ not valid!\n"); | 
 | 		goto exit_free_dev_rdev; | 
 | 	} | 
 |  | 
 | 	fintek->cir_addr = pnp_port_start(pdev, 0); | 
 | 	fintek->cir_irq  = pnp_irq(pdev, 0); | 
 | 	fintek->cir_port_len = pnp_port_len(pdev, 0); | 
 |  | 
 | 	fintek->cr_ip = CR_INDEX_PORT; | 
 | 	fintek->cr_dp = CR_DATA_PORT; | 
 |  | 
 | 	spin_lock_init(&fintek->fintek_lock); | 
 |  | 
 | 	pnp_set_drvdata(pdev, fintek); | 
 | 	fintek->pdev = pdev; | 
 |  | 
 | 	ret = fintek_hw_detect(fintek); | 
 | 	if (ret) | 
 | 		goto exit_free_dev_rdev; | 
 |  | 
 | 	/* Initialize CIR & CIR Wake Logical Devices */ | 
 | 	fintek_config_mode_enable(fintek); | 
 | 	fintek_cir_ldev_init(fintek); | 
 | 	fintek_config_mode_disable(fintek); | 
 |  | 
 | 	/* Initialize CIR & CIR Wake Config Registers */ | 
 | 	fintek_cir_regs_init(fintek); | 
 |  | 
 | 	/* Set up the rc device */ | 
 | 	rdev->priv = fintek; | 
 | 	rdev->allowed_protocols = RC_PROTO_BIT_ALL_IR_DECODER; | 
 | 	rdev->open = fintek_open; | 
 | 	rdev->close = fintek_close; | 
 | 	rdev->device_name = FINTEK_DESCRIPTION; | 
 | 	rdev->input_phys = "fintek/cir0"; | 
 | 	rdev->input_id.bustype = BUS_HOST; | 
 | 	rdev->input_id.vendor = VENDOR_ID_FINTEK; | 
 | 	rdev->input_id.product = fintek->chip_major; | 
 | 	rdev->input_id.version = fintek->chip_minor; | 
 | 	rdev->dev.parent = &pdev->dev; | 
 | 	rdev->driver_name = FINTEK_DRIVER_NAME; | 
 | 	rdev->map_name = RC_MAP_RC6_MCE; | 
 | 	rdev->timeout = US_TO_NS(1000); | 
 | 	/* rx resolution is hardwired to 50us atm, 1, 25, 100 also possible */ | 
 | 	rdev->rx_resolution = US_TO_NS(CIR_SAMPLE_PERIOD); | 
 |  | 
 | 	fintek->rdev = rdev; | 
 |  | 
 | 	ret = -EBUSY; | 
 | 	/* now claim resources */ | 
 | 	if (!request_region(fintek->cir_addr, | 
 | 			    fintek->cir_port_len, FINTEK_DRIVER_NAME)) | 
 | 		goto exit_free_dev_rdev; | 
 |  | 
 | 	if (request_irq(fintek->cir_irq, fintek_cir_isr, IRQF_SHARED, | 
 | 			FINTEK_DRIVER_NAME, (void *)fintek)) | 
 | 		goto exit_free_cir_addr; | 
 |  | 
 | 	ret = rc_register_device(rdev); | 
 | 	if (ret) | 
 | 		goto exit_free_irq; | 
 |  | 
 | 	device_init_wakeup(&pdev->dev, true); | 
 |  | 
 | 	fit_pr(KERN_NOTICE, "driver has been successfully loaded\n"); | 
 | 	if (debug) | 
 | 		cir_dump_regs(fintek); | 
 |  | 
 | 	return 0; | 
 |  | 
 | exit_free_irq: | 
 | 	free_irq(fintek->cir_irq, fintek); | 
 | exit_free_cir_addr: | 
 | 	release_region(fintek->cir_addr, fintek->cir_port_len); | 
 | exit_free_dev_rdev: | 
 | 	rc_free_device(rdev); | 
 | 	kfree(fintek); | 
 |  | 
 | 	return ret; | 
 | } | 
 |  | 
 | static void fintek_remove(struct pnp_dev *pdev) | 
 | { | 
 | 	struct fintek_dev *fintek = pnp_get_drvdata(pdev); | 
 | 	unsigned long flags; | 
 |  | 
 | 	spin_lock_irqsave(&fintek->fintek_lock, flags); | 
 | 	/* disable CIR */ | 
 | 	fintek_disable_cir(fintek); | 
 | 	fintek_cir_reg_write(fintek, CIR_STATUS_IRQ_MASK, CIR_STATUS); | 
 | 	/* enable CIR Wake (for IR power-on) */ | 
 | 	fintek_enable_wake(fintek); | 
 | 	spin_unlock_irqrestore(&fintek->fintek_lock, flags); | 
 |  | 
 | 	/* free resources */ | 
 | 	free_irq(fintek->cir_irq, fintek); | 
 | 	release_region(fintek->cir_addr, fintek->cir_port_len); | 
 |  | 
 | 	rc_unregister_device(fintek->rdev); | 
 |  | 
 | 	kfree(fintek); | 
 | } | 
 |  | 
 | static int fintek_suspend(struct pnp_dev *pdev, pm_message_t state) | 
 | { | 
 | 	struct fintek_dev *fintek = pnp_get_drvdata(pdev); | 
 | 	unsigned long flags; | 
 |  | 
 | 	fit_dbg("%s called", __func__); | 
 |  | 
 | 	spin_lock_irqsave(&fintek->fintek_lock, flags); | 
 |  | 
 | 	/* disable all CIR interrupts */ | 
 | 	fintek_cir_reg_write(fintek, CIR_STATUS_IRQ_MASK, CIR_STATUS); | 
 |  | 
 | 	spin_unlock_irqrestore(&fintek->fintek_lock, flags); | 
 |  | 
 | 	fintek_config_mode_enable(fintek); | 
 |  | 
 | 	/* disable cir logical dev */ | 
 | 	fintek_select_logical_dev(fintek, fintek->logical_dev_cir); | 
 | 	fintek_cr_write(fintek, LOGICAL_DEV_DISABLE, CIR_CR_DEV_EN); | 
 |  | 
 | 	fintek_config_mode_disable(fintek); | 
 |  | 
 | 	/* make sure wake is enabled */ | 
 | 	fintek_enable_wake(fintek); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int fintek_resume(struct pnp_dev *pdev) | 
 | { | 
 | 	struct fintek_dev *fintek = pnp_get_drvdata(pdev); | 
 |  | 
 | 	fit_dbg("%s called", __func__); | 
 |  | 
 | 	/* open interrupt */ | 
 | 	fintek_enable_cir_irq(fintek); | 
 |  | 
 | 	/* Enable CIR logical device */ | 
 | 	fintek_config_mode_enable(fintek); | 
 | 	fintek_select_logical_dev(fintek, fintek->logical_dev_cir); | 
 | 	fintek_cr_write(fintek, LOGICAL_DEV_ENABLE, CIR_CR_DEV_EN); | 
 |  | 
 | 	fintek_config_mode_disable(fintek); | 
 |  | 
 | 	fintek_cir_regs_init(fintek); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void fintek_shutdown(struct pnp_dev *pdev) | 
 | { | 
 | 	struct fintek_dev *fintek = pnp_get_drvdata(pdev); | 
 | 	fintek_enable_wake(fintek); | 
 | } | 
 |  | 
 | static const struct pnp_device_id fintek_ids[] = { | 
 | 	{ "FIT0002", 0 },   /* CIR */ | 
 | 	{ "", 0 }, | 
 | }; | 
 |  | 
 | static struct pnp_driver fintek_driver = { | 
 | 	.name		= FINTEK_DRIVER_NAME, | 
 | 	.id_table	= fintek_ids, | 
 | 	.flags		= PNP_DRIVER_RES_DO_NOT_CHANGE, | 
 | 	.probe		= fintek_probe, | 
 | 	.remove		= fintek_remove, | 
 | 	.suspend	= fintek_suspend, | 
 | 	.resume		= fintek_resume, | 
 | 	.shutdown	= fintek_shutdown, | 
 | }; | 
 |  | 
 | module_param(debug, int, S_IRUGO | S_IWUSR); | 
 | MODULE_PARM_DESC(debug, "Enable debugging output"); | 
 |  | 
 | MODULE_DEVICE_TABLE(pnp, fintek_ids); | 
 | MODULE_DESCRIPTION(FINTEK_DESCRIPTION " driver"); | 
 |  | 
 | MODULE_AUTHOR("Jarod Wilson <jarod@redhat.com>"); | 
 | MODULE_LICENSE("GPL"); | 
 |  | 
 | module_pnp_driver(fintek_driver); |