|  | // SPDX-License-Identifier: GPL-2.0-or-later | 
|  | /* | 
|  | * wm97xx-core.c  --  Touch screen driver core for Wolfson WM9705, WM9712 | 
|  | *                    and WM9713 AC97 Codecs. | 
|  | * | 
|  | * Copyright 2003, 2004, 2005, 2006, 2007, 2008 Wolfson Microelectronics PLC. | 
|  | * Author: Liam Girdwood <lrg@slimlogic.co.uk> | 
|  | * Parts Copyright : Ian Molton <spyro@f2s.com> | 
|  | *                   Andrew Zabolotny <zap@homelink.ru> | 
|  | *                   Russell King <rmk@arm.linux.org.uk> | 
|  | * | 
|  | * Notes: | 
|  | * | 
|  | *  Features: | 
|  | *       - supports WM9705, WM9712, WM9713 | 
|  | *       - polling mode | 
|  | *       - continuous mode (arch-dependent) | 
|  | *       - adjustable rpu/dpp settings | 
|  | *       - adjustable pressure current | 
|  | *       - adjustable sample settle delay | 
|  | *       - 4 and 5 wire touchscreens (5 wire is WM9712 only) | 
|  | *       - pen down detection | 
|  | *       - battery monitor | 
|  | *       - sample AUX adcs | 
|  | *       - power management | 
|  | *       - codec GPIO | 
|  | *       - codec event notification | 
|  | * Todo | 
|  | *       - Support for async sampling control for noisy LCDs. | 
|  | */ | 
|  |  | 
|  | #include <linux/module.h> | 
|  | #include <linux/moduleparam.h> | 
|  | #include <linux/kernel.h> | 
|  | #include <linux/init.h> | 
|  | #include <linux/delay.h> | 
|  | #include <linux/string.h> | 
|  | #include <linux/proc_fs.h> | 
|  | #include <linux/pm.h> | 
|  | #include <linux/interrupt.h> | 
|  | #include <linux/bitops.h> | 
|  | #include <linux/mfd/wm97xx.h> | 
|  | #include <linux/workqueue.h> | 
|  | #include <linux/wm97xx.h> | 
|  | #include <linux/uaccess.h> | 
|  | #include <linux/io.h> | 
|  | #include <linux/slab.h> | 
|  |  | 
|  | #define TS_NAME			"wm97xx" | 
|  | #define WM_CORE_VERSION		"1.00" | 
|  | #define DEFAULT_PRESSURE	0xb0c0 | 
|  |  | 
|  |  | 
|  | /* | 
|  | * Touchscreen absolute values | 
|  | * | 
|  | * These parameters are used to help the input layer discard out of | 
|  | * range readings and reduce jitter etc. | 
|  | * | 
|  | *   o min, max:- indicate the min and max values your touch screen returns | 
|  | *   o fuzz:- use a higher number to reduce jitter | 
|  | * | 
|  | * The default values correspond to Mainstone II in QVGA mode | 
|  | * | 
|  | * Please read | 
|  | * Documentation/input/input-programming.rst for more details. | 
|  | */ | 
|  |  | 
|  | static int abs_x[3] = {150, 4000, 5}; | 
|  | module_param_array(abs_x, int, NULL, 0); | 
|  | MODULE_PARM_DESC(abs_x, "Touchscreen absolute X min, max, fuzz"); | 
|  |  | 
|  | static int abs_y[3] = {200, 4000, 40}; | 
|  | module_param_array(abs_y, int, NULL, 0); | 
|  | MODULE_PARM_DESC(abs_y, "Touchscreen absolute Y min, max, fuzz"); | 
|  |  | 
|  | static int abs_p[3] = {0, 150, 4}; | 
|  | module_param_array(abs_p, int, NULL, 0); | 
|  | MODULE_PARM_DESC(abs_p, "Touchscreen absolute Pressure min, max, fuzz"); | 
|  |  | 
|  | /* | 
|  | * wm97xx IO access, all IO locking done by AC97 layer | 
|  | */ | 
|  | int wm97xx_reg_read(struct wm97xx *wm, u16 reg) | 
|  | { | 
|  | if (wm->ac97) | 
|  | return wm->ac97->bus->ops->read(wm->ac97, reg); | 
|  | else | 
|  | return -1; | 
|  | } | 
|  | EXPORT_SYMBOL_GPL(wm97xx_reg_read); | 
|  |  | 
|  | void wm97xx_reg_write(struct wm97xx *wm, u16 reg, u16 val) | 
|  | { | 
|  | /* cache digitiser registers */ | 
|  | if (reg >= AC97_WM9713_DIG1 && reg <= AC97_WM9713_DIG3) | 
|  | wm->dig[(reg - AC97_WM9713_DIG1) >> 1] = val; | 
|  |  | 
|  | /* cache gpio regs */ | 
|  | if (reg >= AC97_GPIO_CFG && reg <= AC97_MISC_AFE) | 
|  | wm->gpio[(reg - AC97_GPIO_CFG) >> 1] = val; | 
|  |  | 
|  | /* wm9713 irq reg */ | 
|  | if (reg == 0x5a) | 
|  | wm->misc = val; | 
|  |  | 
|  | if (wm->ac97) | 
|  | wm->ac97->bus->ops->write(wm->ac97, reg, val); | 
|  | } | 
|  | EXPORT_SYMBOL_GPL(wm97xx_reg_write); | 
|  |  | 
|  | /** | 
|  | * wm97xx_read_aux_adc - Read the aux adc. | 
|  | * @wm: wm97xx device. | 
|  | * @adcsel: codec ADC to be read | 
|  | * | 
|  | * Reads the selected AUX ADC. | 
|  | */ | 
|  |  | 
|  | int wm97xx_read_aux_adc(struct wm97xx *wm, u16 adcsel) | 
|  | { | 
|  | int power_adc = 0, auxval; | 
|  | u16 power = 0; | 
|  | int rc = 0; | 
|  | int timeout = 0; | 
|  |  | 
|  | /* get codec */ | 
|  | mutex_lock(&wm->codec_mutex); | 
|  |  | 
|  | /* When the touchscreen is not in use, we may have to power up | 
|  | * the AUX ADC before we can use sample the AUX inputs-> | 
|  | */ | 
|  | if (wm->id == WM9713_ID2 && | 
|  | (power = wm97xx_reg_read(wm, AC97_EXTENDED_MID)) & 0x8000) { | 
|  | power_adc = 1; | 
|  | wm97xx_reg_write(wm, AC97_EXTENDED_MID, power & 0x7fff); | 
|  | } | 
|  |  | 
|  | /* Prepare the codec for AUX reading */ | 
|  | wm->codec->aux_prepare(wm); | 
|  |  | 
|  | /* Turn polling mode on to read AUX ADC */ | 
|  | wm->pen_probably_down = 1; | 
|  |  | 
|  | while (rc != RC_VALID && timeout++ < 5) | 
|  | rc = wm->codec->poll_sample(wm, adcsel, &auxval); | 
|  |  | 
|  | if (power_adc) | 
|  | wm97xx_reg_write(wm, AC97_EXTENDED_MID, power | 0x8000); | 
|  |  | 
|  | wm->codec->dig_restore(wm); | 
|  |  | 
|  | wm->pen_probably_down = 0; | 
|  |  | 
|  | if (timeout >= 5) { | 
|  | dev_err(wm->dev, | 
|  | "timeout reading auxadc %d, disabling digitiser\n", | 
|  | adcsel); | 
|  | wm->codec->dig_enable(wm, false); | 
|  | } | 
|  |  | 
|  | mutex_unlock(&wm->codec_mutex); | 
|  | return (rc == RC_VALID ? auxval & 0xfff : -EBUSY); | 
|  | } | 
|  | EXPORT_SYMBOL_GPL(wm97xx_read_aux_adc); | 
|  |  | 
|  | /** | 
|  | * wm97xx_get_gpio - Get the status of a codec GPIO. | 
|  | * @wm: wm97xx device. | 
|  | * @gpio: gpio | 
|  | * | 
|  | * Get the status of a codec GPIO pin | 
|  | */ | 
|  |  | 
|  | enum wm97xx_gpio_status wm97xx_get_gpio(struct wm97xx *wm, u32 gpio) | 
|  | { | 
|  | u16 status; | 
|  | enum wm97xx_gpio_status ret; | 
|  |  | 
|  | mutex_lock(&wm->codec_mutex); | 
|  | status = wm97xx_reg_read(wm, AC97_GPIO_STATUS); | 
|  |  | 
|  | if (status & gpio) | 
|  | ret = WM97XX_GPIO_HIGH; | 
|  | else | 
|  | ret = WM97XX_GPIO_LOW; | 
|  |  | 
|  | mutex_unlock(&wm->codec_mutex); | 
|  | return ret; | 
|  | } | 
|  | EXPORT_SYMBOL_GPL(wm97xx_get_gpio); | 
|  |  | 
|  | /** | 
|  | * wm97xx_set_gpio - Set the status of a codec GPIO. | 
|  | * @wm: wm97xx device. | 
|  | * @gpio: gpio | 
|  | * | 
|  | * | 
|  | * Set the status of a codec GPIO pin | 
|  | */ | 
|  |  | 
|  | void wm97xx_set_gpio(struct wm97xx *wm, u32 gpio, | 
|  | enum wm97xx_gpio_status status) | 
|  | { | 
|  | u16 reg; | 
|  |  | 
|  | mutex_lock(&wm->codec_mutex); | 
|  | reg = wm97xx_reg_read(wm, AC97_GPIO_STATUS); | 
|  |  | 
|  | if (status == WM97XX_GPIO_HIGH) | 
|  | reg |= gpio; | 
|  | else | 
|  | reg &= ~gpio; | 
|  |  | 
|  | if (wm->id == WM9712_ID2 && wm->variant != WM97xx_WM1613) | 
|  | wm97xx_reg_write(wm, AC97_GPIO_STATUS, reg << 1); | 
|  | else | 
|  | wm97xx_reg_write(wm, AC97_GPIO_STATUS, reg); | 
|  | mutex_unlock(&wm->codec_mutex); | 
|  | } | 
|  | EXPORT_SYMBOL_GPL(wm97xx_set_gpio); | 
|  |  | 
|  | /* | 
|  | * Codec GPIO pin configuration, this sets pin direction, polarity, | 
|  | * stickyness and wake up. | 
|  | */ | 
|  | void wm97xx_config_gpio(struct wm97xx *wm, u32 gpio, enum wm97xx_gpio_dir dir, | 
|  | enum wm97xx_gpio_pol pol, enum wm97xx_gpio_sticky sticky, | 
|  | enum wm97xx_gpio_wake wake) | 
|  | { | 
|  | u16 reg; | 
|  |  | 
|  | mutex_lock(&wm->codec_mutex); | 
|  | reg = wm97xx_reg_read(wm, AC97_GPIO_POLARITY); | 
|  |  | 
|  | if (pol == WM97XX_GPIO_POL_HIGH) | 
|  | reg |= gpio; | 
|  | else | 
|  | reg &= ~gpio; | 
|  |  | 
|  | wm97xx_reg_write(wm, AC97_GPIO_POLARITY, reg); | 
|  | reg = wm97xx_reg_read(wm, AC97_GPIO_STICKY); | 
|  |  | 
|  | if (sticky == WM97XX_GPIO_STICKY) | 
|  | reg |= gpio; | 
|  | else | 
|  | reg &= ~gpio; | 
|  |  | 
|  | wm97xx_reg_write(wm, AC97_GPIO_STICKY, reg); | 
|  | reg = wm97xx_reg_read(wm, AC97_GPIO_WAKEUP); | 
|  |  | 
|  | if (wake == WM97XX_GPIO_WAKE) | 
|  | reg |= gpio; | 
|  | else | 
|  | reg &= ~gpio; | 
|  |  | 
|  | wm97xx_reg_write(wm, AC97_GPIO_WAKEUP, reg); | 
|  | reg = wm97xx_reg_read(wm, AC97_GPIO_CFG); | 
|  |  | 
|  | if (dir == WM97XX_GPIO_IN) | 
|  | reg |= gpio; | 
|  | else | 
|  | reg &= ~gpio; | 
|  |  | 
|  | wm97xx_reg_write(wm, AC97_GPIO_CFG, reg); | 
|  | mutex_unlock(&wm->codec_mutex); | 
|  | } | 
|  | EXPORT_SYMBOL_GPL(wm97xx_config_gpio); | 
|  |  | 
|  | /* | 
|  | * Configure the WM97XX_PRP value to use while system is suspended. | 
|  | * If a value other than 0 is set then WM97xx pen detection will be | 
|  | * left enabled in the configured mode while the system is in suspend, | 
|  | * the device has users and suspend has not been disabled via the | 
|  | * wakeup sysfs entries. | 
|  | * | 
|  | * @wm:   WM97xx device to configure | 
|  | * @mode: WM97XX_PRP value to configure while suspended | 
|  | */ | 
|  | void wm97xx_set_suspend_mode(struct wm97xx *wm, u16 mode) | 
|  | { | 
|  | wm->suspend_mode = mode; | 
|  | device_init_wakeup(&wm->input_dev->dev, mode != 0); | 
|  | } | 
|  | EXPORT_SYMBOL_GPL(wm97xx_set_suspend_mode); | 
|  |  | 
|  | /* | 
|  | * Handle a pen down interrupt. | 
|  | */ | 
|  | static void wm97xx_pen_irq_worker(struct work_struct *work) | 
|  | { | 
|  | struct wm97xx *wm = container_of(work, struct wm97xx, pen_event_work); | 
|  | int pen_was_down = wm->pen_is_down; | 
|  |  | 
|  | /* do we need to enable the touch panel reader */ | 
|  | if (wm->id == WM9705_ID2) { | 
|  | if (wm97xx_reg_read(wm, AC97_WM97XX_DIGITISER_RD) & | 
|  | WM97XX_PEN_DOWN) | 
|  | wm->pen_is_down = 1; | 
|  | else | 
|  | wm->pen_is_down = 0; | 
|  | } else { | 
|  | u16 status, pol; | 
|  | mutex_lock(&wm->codec_mutex); | 
|  | status = wm97xx_reg_read(wm, AC97_GPIO_STATUS); | 
|  | pol = wm97xx_reg_read(wm, AC97_GPIO_POLARITY); | 
|  |  | 
|  | if (WM97XX_GPIO_13 & pol & status) { | 
|  | wm->pen_is_down = 1; | 
|  | wm97xx_reg_write(wm, AC97_GPIO_POLARITY, pol & | 
|  | ~WM97XX_GPIO_13); | 
|  | } else { | 
|  | wm->pen_is_down = 0; | 
|  | wm97xx_reg_write(wm, AC97_GPIO_POLARITY, pol | | 
|  | WM97XX_GPIO_13); | 
|  | } | 
|  |  | 
|  | if (wm->id == WM9712_ID2 && wm->variant != WM97xx_WM1613) | 
|  | wm97xx_reg_write(wm, AC97_GPIO_STATUS, (status & | 
|  | ~WM97XX_GPIO_13) << 1); | 
|  | else | 
|  | wm97xx_reg_write(wm, AC97_GPIO_STATUS, status & | 
|  | ~WM97XX_GPIO_13); | 
|  | mutex_unlock(&wm->codec_mutex); | 
|  | } | 
|  |  | 
|  | /* If the system is not using continuous mode or it provides a | 
|  | * pen down operation then we need to schedule polls while the | 
|  | * pen is down.  Otherwise the machine driver is responsible | 
|  | * for scheduling reads. | 
|  | */ | 
|  | if (!wm->mach_ops->acc_enabled || wm->mach_ops->acc_pen_down) { | 
|  | if (wm->pen_is_down && !pen_was_down) { | 
|  | /* Data is not available immediately on pen down */ | 
|  | queue_delayed_work(wm->ts_workq, &wm->ts_reader, 1); | 
|  | } | 
|  |  | 
|  | /* Let ts_reader report the pen up for debounce. */ | 
|  | if (!wm->pen_is_down && pen_was_down) | 
|  | wm->pen_is_down = 1; | 
|  | } | 
|  |  | 
|  | if (!wm->pen_is_down && wm->mach_ops->acc_enabled) | 
|  | wm->mach_ops->acc_pen_up(wm); | 
|  |  | 
|  | wm->mach_ops->irq_enable(wm, 1); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Codec PENDOWN irq handler | 
|  | * | 
|  | * We have to disable the codec interrupt in the handler because it | 
|  | * can take up to 1ms to clear the interrupt source. We schedule a task | 
|  | * in a work queue to do the actual interaction with the chip.  The | 
|  | * interrupt is then enabled again in the slow handler when the source | 
|  | * has been cleared. | 
|  | */ | 
|  | static irqreturn_t wm97xx_pen_interrupt(int irq, void *dev_id) | 
|  | { | 
|  | struct wm97xx *wm = dev_id; | 
|  |  | 
|  | if (!work_pending(&wm->pen_event_work)) { | 
|  | wm->mach_ops->irq_enable(wm, 0); | 
|  | queue_work(wm->ts_workq, &wm->pen_event_work); | 
|  | } | 
|  |  | 
|  | return IRQ_HANDLED; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * initialise pen IRQ handler and workqueue | 
|  | */ | 
|  | static int wm97xx_init_pen_irq(struct wm97xx *wm) | 
|  | { | 
|  | u16 reg; | 
|  |  | 
|  | /* If an interrupt is supplied an IRQ enable operation must also be | 
|  | * provided. */ | 
|  | BUG_ON(!wm->mach_ops->irq_enable); | 
|  |  | 
|  | if (request_irq(wm->pen_irq, wm97xx_pen_interrupt, IRQF_SHARED, | 
|  | "wm97xx-pen", wm)) { | 
|  | dev_err(wm->dev, | 
|  | "Failed to register pen down interrupt, polling"); | 
|  | wm->pen_irq = 0; | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | /* Configure GPIO as interrupt source on WM971x */ | 
|  | if (wm->id != WM9705_ID2) { | 
|  | BUG_ON(!wm->mach_ops->irq_gpio); | 
|  | reg = wm97xx_reg_read(wm, AC97_MISC_AFE); | 
|  | wm97xx_reg_write(wm, AC97_MISC_AFE, | 
|  | reg & ~(wm->mach_ops->irq_gpio)); | 
|  | reg = wm97xx_reg_read(wm, 0x5a); | 
|  | wm97xx_reg_write(wm, 0x5a, reg & ~0x0001); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int wm97xx_read_samples(struct wm97xx *wm) | 
|  | { | 
|  | struct wm97xx_data data; | 
|  | int rc; | 
|  |  | 
|  | mutex_lock(&wm->codec_mutex); | 
|  |  | 
|  | if (wm->mach_ops && wm->mach_ops->acc_enabled) | 
|  | rc = wm->mach_ops->acc_pen_down(wm); | 
|  | else | 
|  | rc = wm->codec->poll_touch(wm, &data); | 
|  |  | 
|  | if (rc & RC_PENUP) { | 
|  | if (wm->pen_is_down) { | 
|  | wm->pen_is_down = 0; | 
|  | dev_dbg(wm->dev, "pen up\n"); | 
|  | input_report_abs(wm->input_dev, ABS_PRESSURE, 0); | 
|  | input_report_key(wm->input_dev, BTN_TOUCH, 0); | 
|  | input_sync(wm->input_dev); | 
|  | } else if (!(rc & RC_AGAIN)) { | 
|  | /* We need high frequency updates only while | 
|  | * pen is down, the user never will be able to | 
|  | * touch screen faster than a few times per | 
|  | * second... On the other hand, when the user | 
|  | * is actively working with the touchscreen we | 
|  | * don't want to lose the quick response. So we | 
|  | * will slowly increase sleep time after the | 
|  | * pen is up and quicky restore it to ~one task | 
|  | * switch when pen is down again. | 
|  | */ | 
|  | if (wm->ts_reader_interval < HZ / 10) | 
|  | wm->ts_reader_interval++; | 
|  | } | 
|  |  | 
|  | } else if (rc & RC_VALID) { | 
|  | dev_dbg(wm->dev, | 
|  | "pen down: x=%x:%d, y=%x:%d, pressure=%x:%d\n", | 
|  | data.x >> 12, data.x & 0xfff, data.y >> 12, | 
|  | data.y & 0xfff, data.p >> 12, data.p & 0xfff); | 
|  |  | 
|  | if (abs_x[0] > (data.x & 0xfff) || | 
|  | abs_x[1] < (data.x & 0xfff) || | 
|  | abs_y[0] > (data.y & 0xfff) || | 
|  | abs_y[1] < (data.y & 0xfff)) { | 
|  | dev_dbg(wm->dev, "Measurement out of range, dropping it\n"); | 
|  | rc = RC_AGAIN; | 
|  | goto out; | 
|  | } | 
|  |  | 
|  | input_report_abs(wm->input_dev, ABS_X, data.x & 0xfff); | 
|  | input_report_abs(wm->input_dev, ABS_Y, data.y & 0xfff); | 
|  | input_report_abs(wm->input_dev, ABS_PRESSURE, data.p & 0xfff); | 
|  | input_report_key(wm->input_dev, BTN_TOUCH, 1); | 
|  | input_sync(wm->input_dev); | 
|  | wm->pen_is_down = 1; | 
|  | wm->ts_reader_interval = wm->ts_reader_min_interval; | 
|  | } else if (rc & RC_PENDOWN) { | 
|  | dev_dbg(wm->dev, "pen down\n"); | 
|  | wm->pen_is_down = 1; | 
|  | wm->ts_reader_interval = wm->ts_reader_min_interval; | 
|  | } | 
|  |  | 
|  | out: | 
|  | mutex_unlock(&wm->codec_mutex); | 
|  | return rc; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * The touchscreen sample reader. | 
|  | */ | 
|  | static void wm97xx_ts_reader(struct work_struct *work) | 
|  | { | 
|  | int rc; | 
|  | struct wm97xx *wm = container_of(work, struct wm97xx, ts_reader.work); | 
|  |  | 
|  | BUG_ON(!wm->codec); | 
|  |  | 
|  | do { | 
|  | rc = wm97xx_read_samples(wm); | 
|  | } while (rc & RC_AGAIN); | 
|  |  | 
|  | if (wm->pen_is_down || !wm->pen_irq) | 
|  | queue_delayed_work(wm->ts_workq, &wm->ts_reader, | 
|  | wm->ts_reader_interval); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * wm97xx_ts_input_open - Open the touch screen input device. | 
|  | * @idev:	Input device to be opened. | 
|  | * | 
|  | * Called by the input sub system to open a wm97xx touchscreen device. | 
|  | * Starts the touchscreen thread and touch digitiser. | 
|  | */ | 
|  | static int wm97xx_ts_input_open(struct input_dev *idev) | 
|  | { | 
|  | struct wm97xx *wm = input_get_drvdata(idev); | 
|  |  | 
|  | wm->ts_workq = alloc_ordered_workqueue("kwm97xx", 0); | 
|  | if (wm->ts_workq == NULL) { | 
|  | dev_err(wm->dev, | 
|  | "Failed to create workqueue\n"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | /* start digitiser */ | 
|  | if (wm->mach_ops && wm->mach_ops->acc_enabled) | 
|  | wm->codec->acc_enable(wm, 1); | 
|  | wm->codec->dig_enable(wm, 1); | 
|  |  | 
|  | INIT_DELAYED_WORK(&wm->ts_reader, wm97xx_ts_reader); | 
|  | INIT_WORK(&wm->pen_event_work, wm97xx_pen_irq_worker); | 
|  |  | 
|  | wm->ts_reader_min_interval = HZ >= 100 ? HZ / 100 : 1; | 
|  | if (wm->ts_reader_min_interval < 1) | 
|  | wm->ts_reader_min_interval = 1; | 
|  | wm->ts_reader_interval = wm->ts_reader_min_interval; | 
|  |  | 
|  | wm->pen_is_down = 0; | 
|  | if (wm->pen_irq) | 
|  | wm97xx_init_pen_irq(wm); | 
|  | else | 
|  | dev_err(wm->dev, "No IRQ specified\n"); | 
|  |  | 
|  | /* If we either don't have an interrupt for pen down events or | 
|  | * failed to acquire it then we need to poll. | 
|  | */ | 
|  | if (wm->pen_irq == 0) | 
|  | queue_delayed_work(wm->ts_workq, &wm->ts_reader, | 
|  | wm->ts_reader_interval); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * wm97xx_ts_input_close - Close the touch screen input device. | 
|  | * @idev:	Input device to be closed. | 
|  | * | 
|  | * Called by the input sub system to close a wm97xx touchscreen | 
|  | * device.  Kills the touchscreen thread and stops the touch | 
|  | * digitiser. | 
|  | */ | 
|  |  | 
|  | static void wm97xx_ts_input_close(struct input_dev *idev) | 
|  | { | 
|  | struct wm97xx *wm = input_get_drvdata(idev); | 
|  | u16 reg; | 
|  |  | 
|  | if (wm->pen_irq) { | 
|  | /* Return the interrupt to GPIO usage (disabling it) */ | 
|  | if (wm->id != WM9705_ID2) { | 
|  | BUG_ON(!wm->mach_ops->irq_gpio); | 
|  | reg = wm97xx_reg_read(wm, AC97_MISC_AFE); | 
|  | wm97xx_reg_write(wm, AC97_MISC_AFE, | 
|  | reg | wm->mach_ops->irq_gpio); | 
|  | } | 
|  |  | 
|  | free_irq(wm->pen_irq, wm); | 
|  | } | 
|  |  | 
|  | wm->pen_is_down = 0; | 
|  |  | 
|  | /* Balance out interrupt disables/enables */ | 
|  | if (cancel_work_sync(&wm->pen_event_work)) | 
|  | wm->mach_ops->irq_enable(wm, 1); | 
|  |  | 
|  | /* ts_reader rearms itself so we need to explicitly stop it | 
|  | * before we destroy the workqueue. | 
|  | */ | 
|  | cancel_delayed_work_sync(&wm->ts_reader); | 
|  |  | 
|  | destroy_workqueue(wm->ts_workq); | 
|  |  | 
|  | /* stop digitiser */ | 
|  | wm->codec->dig_enable(wm, 0); | 
|  | if (wm->mach_ops && wm->mach_ops->acc_enabled) | 
|  | wm->codec->acc_enable(wm, 0); | 
|  | } | 
|  |  | 
|  | static int wm97xx_register_touch(struct wm97xx *wm) | 
|  | { | 
|  | struct wm97xx_pdata *pdata = dev_get_platdata(wm->dev); | 
|  | int ret; | 
|  |  | 
|  | wm->input_dev = devm_input_allocate_device(wm->dev); | 
|  | if (wm->input_dev == NULL) | 
|  | return -ENOMEM; | 
|  |  | 
|  | /* set up touch configuration */ | 
|  | wm->input_dev->name = "wm97xx touchscreen"; | 
|  | wm->input_dev->phys = "wm97xx"; | 
|  | wm->input_dev->open = wm97xx_ts_input_open; | 
|  | wm->input_dev->close = wm97xx_ts_input_close; | 
|  |  | 
|  | __set_bit(EV_ABS, wm->input_dev->evbit); | 
|  | __set_bit(EV_KEY, wm->input_dev->evbit); | 
|  | __set_bit(BTN_TOUCH, wm->input_dev->keybit); | 
|  |  | 
|  | input_set_abs_params(wm->input_dev, ABS_X, abs_x[0], abs_x[1], | 
|  | abs_x[2], 0); | 
|  | input_set_abs_params(wm->input_dev, ABS_Y, abs_y[0], abs_y[1], | 
|  | abs_y[2], 0); | 
|  | input_set_abs_params(wm->input_dev, ABS_PRESSURE, abs_p[0], abs_p[1], | 
|  | abs_p[2], 0); | 
|  |  | 
|  | input_set_drvdata(wm->input_dev, wm); | 
|  | wm->input_dev->dev.parent = wm->dev; | 
|  |  | 
|  | ret = input_register_device(wm->input_dev); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | /* | 
|  | * register our extended touch device (for machine specific | 
|  | * extensions) | 
|  | */ | 
|  | wm->touch_dev = platform_device_alloc("wm97xx-touch", -1); | 
|  | if (!wm->touch_dev) { | 
|  | ret = -ENOMEM; | 
|  | goto touch_err; | 
|  | } | 
|  | platform_set_drvdata(wm->touch_dev, wm); | 
|  | wm->touch_dev->dev.parent = wm->dev; | 
|  | wm->touch_dev->dev.platform_data = pdata; | 
|  | ret = platform_device_add(wm->touch_dev); | 
|  | if (ret < 0) | 
|  | goto touch_reg_err; | 
|  |  | 
|  | return 0; | 
|  | touch_reg_err: | 
|  | platform_device_put(wm->touch_dev); | 
|  | touch_err: | 
|  | input_unregister_device(wm->input_dev); | 
|  | wm->input_dev = NULL; | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static void wm97xx_unregister_touch(struct wm97xx *wm) | 
|  | { | 
|  | platform_device_unregister(wm->touch_dev); | 
|  | input_unregister_device(wm->input_dev); | 
|  | wm->input_dev = NULL; | 
|  | } | 
|  |  | 
|  | static int _wm97xx_probe(struct wm97xx *wm) | 
|  | { | 
|  | int id = 0; | 
|  |  | 
|  | mutex_init(&wm->codec_mutex); | 
|  | dev_set_drvdata(wm->dev, wm); | 
|  |  | 
|  | /* check that we have a supported codec */ | 
|  | id = wm97xx_reg_read(wm, AC97_VENDOR_ID1); | 
|  | if (id != WM97XX_ID1) { | 
|  | dev_err(wm->dev, | 
|  | "Device with vendor %04x is not a wm97xx\n", id); | 
|  | return -ENODEV; | 
|  | } | 
|  |  | 
|  | wm->id = wm97xx_reg_read(wm, AC97_VENDOR_ID2); | 
|  |  | 
|  | wm->variant = WM97xx_GENERIC; | 
|  |  | 
|  | dev_info(wm->dev, "detected a wm97%02x codec\n", wm->id & 0xff); | 
|  |  | 
|  | switch (wm->id & 0xff) { | 
|  | #ifdef CONFIG_TOUCHSCREEN_WM9705 | 
|  | case 0x05: | 
|  | wm->codec = &wm9705_codec; | 
|  | break; | 
|  | #endif | 
|  | #ifdef CONFIG_TOUCHSCREEN_WM9712 | 
|  | case 0x12: | 
|  | wm->codec = &wm9712_codec; | 
|  | break; | 
|  | #endif | 
|  | #ifdef CONFIG_TOUCHSCREEN_WM9713 | 
|  | case 0x13: | 
|  | wm->codec = &wm9713_codec; | 
|  | break; | 
|  | #endif | 
|  | default: | 
|  | dev_err(wm->dev, "Support for wm97%02x not compiled in.\n", | 
|  | wm->id & 0xff); | 
|  | return -ENODEV; | 
|  | } | 
|  |  | 
|  | /* set up physical characteristics */ | 
|  | wm->codec->phy_init(wm); | 
|  |  | 
|  | /* load gpio cache */ | 
|  | wm->gpio[0] = wm97xx_reg_read(wm, AC97_GPIO_CFG); | 
|  | wm->gpio[1] = wm97xx_reg_read(wm, AC97_GPIO_POLARITY); | 
|  | wm->gpio[2] = wm97xx_reg_read(wm, AC97_GPIO_STICKY); | 
|  | wm->gpio[3] = wm97xx_reg_read(wm, AC97_GPIO_WAKEUP); | 
|  | wm->gpio[4] = wm97xx_reg_read(wm, AC97_GPIO_STATUS); | 
|  | wm->gpio[5] = wm97xx_reg_read(wm, AC97_MISC_AFE); | 
|  |  | 
|  | return wm97xx_register_touch(wm); | 
|  | } | 
|  |  | 
|  | static void wm97xx_remove_battery(struct wm97xx *wm) | 
|  | { | 
|  | platform_device_unregister(wm->battery_dev); | 
|  | } | 
|  |  | 
|  | static int wm97xx_add_battery(struct wm97xx *wm, | 
|  | struct wm97xx_batt_pdata *pdata) | 
|  | { | 
|  | int ret; | 
|  |  | 
|  | wm->battery_dev = platform_device_alloc("wm97xx-battery", -1); | 
|  | if (!wm->battery_dev) | 
|  | return -ENOMEM; | 
|  |  | 
|  | platform_set_drvdata(wm->battery_dev, wm); | 
|  | wm->battery_dev->dev.parent = wm->dev; | 
|  | wm->battery_dev->dev.platform_data = pdata; | 
|  | ret = platform_device_add(wm->battery_dev); | 
|  | if (ret) | 
|  | platform_device_put(wm->battery_dev); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int wm97xx_probe(struct device *dev) | 
|  | { | 
|  | struct wm97xx *wm; | 
|  | int ret; | 
|  | struct wm97xx_pdata *pdata = dev_get_platdata(dev); | 
|  |  | 
|  | wm = devm_kzalloc(dev, sizeof(struct wm97xx), GFP_KERNEL); | 
|  | if (!wm) | 
|  | return -ENOMEM; | 
|  |  | 
|  | wm->dev = dev; | 
|  | wm->ac97 = to_ac97_t(dev); | 
|  |  | 
|  | ret =  _wm97xx_probe(wm); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | ret = wm97xx_add_battery(wm, pdata ? pdata->batt_pdata : NULL); | 
|  | if (ret < 0) | 
|  | goto batt_err; | 
|  |  | 
|  | return ret; | 
|  |  | 
|  | batt_err: | 
|  | wm97xx_unregister_touch(wm); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int wm97xx_remove(struct device *dev) | 
|  | { | 
|  | struct wm97xx *wm = dev_get_drvdata(dev); | 
|  |  | 
|  | wm97xx_remove_battery(wm); | 
|  | wm97xx_unregister_touch(wm); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int wm97xx_mfd_probe(struct platform_device *pdev) | 
|  | { | 
|  | struct wm97xx *wm; | 
|  | struct wm97xx_platform_data *mfd_pdata = dev_get_platdata(&pdev->dev); | 
|  | int ret; | 
|  |  | 
|  | wm = devm_kzalloc(&pdev->dev, sizeof(struct wm97xx), GFP_KERNEL); | 
|  | if (!wm) | 
|  | return -ENOMEM; | 
|  |  | 
|  | wm->dev = &pdev->dev; | 
|  | wm->ac97 = mfd_pdata->ac97; | 
|  |  | 
|  | ret =  _wm97xx_probe(wm); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | ret = wm97xx_add_battery(wm, mfd_pdata->batt_pdata); | 
|  | if (ret < 0) | 
|  | goto batt_err; | 
|  |  | 
|  | return ret; | 
|  |  | 
|  | batt_err: | 
|  | wm97xx_unregister_touch(wm); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static int wm97xx_mfd_remove(struct platform_device *pdev) | 
|  | { | 
|  | return wm97xx_remove(&pdev->dev); | 
|  | } | 
|  |  | 
|  | static int __maybe_unused wm97xx_suspend(struct device *dev) | 
|  | { | 
|  | struct wm97xx *wm = dev_get_drvdata(dev); | 
|  | u16 reg; | 
|  | int suspend_mode; | 
|  |  | 
|  | if (device_may_wakeup(&wm->input_dev->dev)) | 
|  | suspend_mode = wm->suspend_mode; | 
|  | else | 
|  | suspend_mode = 0; | 
|  |  | 
|  | if (wm->input_dev->users) | 
|  | cancel_delayed_work_sync(&wm->ts_reader); | 
|  |  | 
|  | /* Power down the digitiser (bypassing the cache for resume) */ | 
|  | reg = wm97xx_reg_read(wm, AC97_WM97XX_DIGITISER2); | 
|  | reg &= ~WM97XX_PRP_DET_DIG; | 
|  | if (wm->input_dev->users) | 
|  | reg |= suspend_mode; | 
|  | wm->ac97->bus->ops->write(wm->ac97, AC97_WM97XX_DIGITISER2, reg); | 
|  |  | 
|  | /* WM9713 has an additional power bit - turn it off if there | 
|  | * are no users or if suspend mode is zero. */ | 
|  | if (wm->id == WM9713_ID2 && | 
|  | (!wm->input_dev->users || !suspend_mode)) { | 
|  | reg = wm97xx_reg_read(wm, AC97_EXTENDED_MID) | 0x8000; | 
|  | wm97xx_reg_write(wm, AC97_EXTENDED_MID, reg); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int __maybe_unused wm97xx_resume(struct device *dev) | 
|  | { | 
|  | struct wm97xx *wm = dev_get_drvdata(dev); | 
|  |  | 
|  | /* restore digitiser and gpios */ | 
|  | if (wm->id == WM9713_ID2) { | 
|  | wm97xx_reg_write(wm, AC97_WM9713_DIG1, wm->dig[0]); | 
|  | wm97xx_reg_write(wm, 0x5a, wm->misc); | 
|  | if (wm->input_dev->users) { | 
|  | u16 reg; | 
|  | reg = wm97xx_reg_read(wm, AC97_EXTENDED_MID) & 0x7fff; | 
|  | wm97xx_reg_write(wm, AC97_EXTENDED_MID, reg); | 
|  | } | 
|  | } | 
|  |  | 
|  | wm97xx_reg_write(wm, AC97_WM9713_DIG2, wm->dig[1]); | 
|  | wm97xx_reg_write(wm, AC97_WM9713_DIG3, wm->dig[2]); | 
|  |  | 
|  | wm97xx_reg_write(wm, AC97_GPIO_CFG, wm->gpio[0]); | 
|  | wm97xx_reg_write(wm, AC97_GPIO_POLARITY, wm->gpio[1]); | 
|  | wm97xx_reg_write(wm, AC97_GPIO_STICKY, wm->gpio[2]); | 
|  | wm97xx_reg_write(wm, AC97_GPIO_WAKEUP, wm->gpio[3]); | 
|  | wm97xx_reg_write(wm, AC97_GPIO_STATUS, wm->gpio[4]); | 
|  | wm97xx_reg_write(wm, AC97_MISC_AFE, wm->gpio[5]); | 
|  |  | 
|  | if (wm->input_dev->users && !wm->pen_irq) { | 
|  | wm->ts_reader_interval = wm->ts_reader_min_interval; | 
|  | queue_delayed_work(wm->ts_workq, &wm->ts_reader, | 
|  | wm->ts_reader_interval); | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static SIMPLE_DEV_PM_OPS(wm97xx_pm_ops, wm97xx_suspend, wm97xx_resume); | 
|  |  | 
|  | /* | 
|  | * Machine specific operations | 
|  | */ | 
|  | int wm97xx_register_mach_ops(struct wm97xx *wm, | 
|  | struct wm97xx_mach_ops *mach_ops) | 
|  | { | 
|  | mutex_lock(&wm->codec_mutex); | 
|  | if (wm->mach_ops) { | 
|  | mutex_unlock(&wm->codec_mutex); | 
|  | return -EINVAL; | 
|  | } | 
|  | wm->mach_ops = mach_ops; | 
|  | mutex_unlock(&wm->codec_mutex); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  | EXPORT_SYMBOL_GPL(wm97xx_register_mach_ops); | 
|  |  | 
|  | void wm97xx_unregister_mach_ops(struct wm97xx *wm) | 
|  | { | 
|  | mutex_lock(&wm->codec_mutex); | 
|  | wm->mach_ops = NULL; | 
|  | mutex_unlock(&wm->codec_mutex); | 
|  | } | 
|  | EXPORT_SYMBOL_GPL(wm97xx_unregister_mach_ops); | 
|  |  | 
|  | static struct device_driver wm97xx_driver = { | 
|  | .name =		"wm97xx-ts", | 
|  | #ifdef CONFIG_AC97_BUS | 
|  | .bus =		&ac97_bus_type, | 
|  | #endif | 
|  | .owner =	THIS_MODULE, | 
|  | .probe =	wm97xx_probe, | 
|  | .remove =	wm97xx_remove, | 
|  | .pm =		&wm97xx_pm_ops, | 
|  | }; | 
|  |  | 
|  | static struct platform_driver wm97xx_mfd_driver = { | 
|  | .driver = { | 
|  | .name =		"wm97xx-ts", | 
|  | .pm =		&wm97xx_pm_ops, | 
|  | }, | 
|  | .probe =	wm97xx_mfd_probe, | 
|  | .remove =	wm97xx_mfd_remove, | 
|  | }; | 
|  |  | 
|  | static int __init wm97xx_init(void) | 
|  | { | 
|  | int ret; | 
|  |  | 
|  | ret = platform_driver_register(&wm97xx_mfd_driver); | 
|  | if (ret) | 
|  | return ret; | 
|  |  | 
|  | if (IS_BUILTIN(CONFIG_AC97_BUS)) | 
|  | ret =  driver_register(&wm97xx_driver); | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static void __exit wm97xx_exit(void) | 
|  | { | 
|  | if (IS_BUILTIN(CONFIG_AC97_BUS)) | 
|  | driver_unregister(&wm97xx_driver); | 
|  | platform_driver_unregister(&wm97xx_mfd_driver); | 
|  | } | 
|  |  | 
|  | module_init(wm97xx_init); | 
|  | module_exit(wm97xx_exit); | 
|  |  | 
|  | /* Module information */ | 
|  | MODULE_AUTHOR("Liam Girdwood <lrg@slimlogic.co.uk>"); | 
|  | MODULE_DESCRIPTION("WM97xx Core - Touch Screen / AUX ADC / GPIO Driver"); | 
|  | MODULE_LICENSE("GPL"); |