Received: by 10.213.65.68 with SMTP id h4csp89293imn; Mon, 26 Mar 2018 15:54:53 -0700 (PDT) X-Google-Smtp-Source: AG47ELv+gHZ4eR7abqQ6QPKwWMKPexFoqxXyN1wgrNV5mYZTahNGBSLXWFGvjYt1W+8ELyJUgeqc X-Received: by 2002:a17:902:5acf:: with SMTP id g15-v6mr37208801plm.138.1522104893477; Mon, 26 Mar 2018 15:54:53 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1522104893; cv=none; d=google.com; s=arc-20160816; b=v8PNQOc6+hH+5/xW4Oepxjls0+s0oNNJQzG3UYEU3oE/GiERs3CoxXR3rM3sBhNJt5 sDOUDZff4anGRdbG57fFIQMwNFf2F90nUJEKEUcGsMbb8txtRB7spe9AWmNti7g3Fg29 qYlr7mOqZFVnA1qgJw6yM54wNUPxKKv/xEjMhGqq0ivojBCA6FUWtqWwlp6hQOoU6WXJ J+5B06xcBpamqAexByOxDfoz+OawBZPBk13CZ72s/E9UY4OwDKHv89yoGuYYqroqM6ab fnI2EpsvKBYOe6cAfZWc5Wuc3EruAUYzsKN+xPuyWxvUXAHK0ifiLX0tYhKfCIHDjmUF OoNA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:user-agent:in-reply-to :content-disposition:mime-version:references:message-id:subject:cc :to:from:date:dkim-signature:arc-authentication-results; bh=J2pkPcSnCiXRc/iOTfG8WYwemVfmTorNvZAsfnG6ung=; b=NQv3c0QtXd5R3jE2Xb5DXFJEljekqzg9Na08dk5yRDhl5F/9qfWQicqU+UHiu4e3x2 1jGxBnsOfrJSIfHIeQ8o3L8yO4YT7mBKZeWj6EBcLkAveS4uqhI5hLfxvQLAEwBiblYI wHmpE6gEPFnYlDSqqvUv82rJDs02WHOC8qX2MZoHrsXsViN8EMy+6xtNFuBjXzqg09kD dChFLksXpOKs3quR4GfhA71pI4OmxWGmIRMlgzHYWdRAPFBSBS1+O67RJFFkM3ctnSoI N2DnYfBTyiMooXBb2g7L7K9UiMw+qdwU9WSre0hYJVieD7X93iKzX49Kl+Z7sBOo1UuC 4eIA== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b=DRGLiLC2; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id z62si10768004pgd.819.2018.03.26.15.54.38; Mon, 26 Mar 2018 15:54:53 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@gmail.com header.s=20161025 header.b=DRGLiLC2; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=QUARANTINE dis=NONE) header.from=gmail.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1752168AbeCZWxm (ORCPT + 99 others); Mon, 26 Mar 2018 18:53:42 -0400 Received: from mail-pf0-f194.google.com ([209.85.192.194]:33357 "EHLO mail-pf0-f194.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1751739AbeCZWxj (ORCPT ); Mon, 26 Mar 2018 18:53:39 -0400 Received: by mail-pf0-f194.google.com with SMTP id f15so2371790pfn.0; Mon, 26 Mar 2018 15:53:39 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20161025; h=date:from:to:cc:subject:message-id:references:mime-version :content-disposition:in-reply-to:user-agent; bh=J2pkPcSnCiXRc/iOTfG8WYwemVfmTorNvZAsfnG6ung=; b=DRGLiLC2zk7DzuLyNGMXFWaJ6lrDoyGJjN0jIBx0U2GhCqOLcRRDB6mtXy2n8OdXp2 VKyJ8kRgUN7F3ZTaazGajbiklQ6vUuNRPw8TXMzkjszKnMarVPM40ulw5l/YPui+6ORl zYtJahxneuEVjlBhs28QF2TTYgw86qiF2ehljMGKn/N31MXzYk2iJ8kHGBeLO/F6a6fK yzJHNXtns3sTRJCklePFqYeY7JGdXyDkYcQUuiUZgZWckV0HLnG+ouzvwFy2TypvKntx NMNHdBO07ekkEupYCRpuJ99rjtvlmXxgu2E0U+XEmZ1QMx7ju3Ny7KdxGC4Zqwwh58/r ck0g== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:date:from:to:cc:subject:message-id:references :mime-version:content-disposition:in-reply-to:user-agent; bh=J2pkPcSnCiXRc/iOTfG8WYwemVfmTorNvZAsfnG6ung=; b=uCbou9JRubCXxa7KInbWT1hjvgRfVS+P4GfB69amdtusx3SlO5CyUJ/b2KipJ979UH EBGbtquM+s/mYSKoiXlPKqvUpLjHhkky5ffh7uj9DPaaxhzGBibYSyUCIq2S7ItGhAeL 5zX0b/e5muKX/DoWunNW+/5ll5aQquhGZMahCYlTjRlkawtEXisE8vuALWNFxclULgTU Xosx30tRuVeTNGrVO2MIlkzw7FFUlhywZ0XgS7pR/suJE+XsBCeZn2tI+msdxtlkeTs4 A5I2UEYAJgsCcTtRgpucp7FDEMRKQaOAkgAHRdSnIH4GcKa+qb48eENfLpvO1GjJXnyX d3aw== X-Gm-Message-State: AElRT7EXW6fq2AgybY3jAiU0SpuT7I352fI6PlZf+ojqe4KSHZAnwRoF yPFiYycBI04IdNHtsDAv5k4= X-Received: by 10.99.169.1 with SMTP id u1mr17628829pge.251.1522104818480; Mon, 26 Mar 2018 15:53:38 -0700 (PDT) Received: from dtor-ws ([2620:0:1000:1511:8de6:27a8:ed13:2ef5]) by smtp.gmail.com with ESMTPSA id c201sm25317734pfb.30.2018.03.26.15.53.37 (version=TLS1_2 cipher=ECDHE-RSA-CHACHA20-POLY1305 bits=256/256); Mon, 26 Mar 2018 15:53:37 -0700 (PDT) Date: Mon, 26 Mar 2018 15:53:36 -0700 From: Dmitry Torokhov To: Mark Jonas Cc: Rob Herring , Mark Rutland , linux-input@vger.kernel.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, hs@denx.de, Zhu Yi Subject: Re: [PATCH] Input: add bu21029 touch driver Message-ID: <20180326225336.GD204225@dtor-ws> References: <1521651874-15379-1-git-send-email-mark.jonas@de.bosch.com> MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: <1521651874-15379-1-git-send-email-mark.jonas@de.bosch.com> User-Agent: Mutt/1.9.2 (2017-12-15) Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Hi Mark, On Wed, Mar 21, 2018 at 06:04:34PM +0100, Mark Jonas wrote: > From: Zhu Yi > > Add the ROHM BU21029 resistive touch panel controller > support with i2c interface. > > Signed-off-by: Zhu Yi > Signed-off-by: Mark Jonas > Reviewed-by: Heiko Schocher > --- > .../bindings/input/touchscreen/bu21029.txt | 30 ++ > drivers/input/touchscreen/Kconfig | 12 + > drivers/input/touchscreen/Makefile | 1 + > drivers/input/touchscreen/bu21029_ts.c | 456 +++++++++++++++++++++ > 4 files changed, 499 insertions(+) > create mode 100644 Documentation/devicetree/bindings/input/touchscreen/bu21029.txt > create mode 100644 drivers/input/touchscreen/bu21029_ts.c > > diff --git a/Documentation/devicetree/bindings/input/touchscreen/bu21029.txt b/Documentation/devicetree/bindings/input/touchscreen/bu21029.txt > new file mode 100644 > index 0000000..7b61602 > --- /dev/null > +++ b/Documentation/devicetree/bindings/input/touchscreen/bu21029.txt > @@ -0,0 +1,30 @@ > +* Rohm BU21029 Touch Screen Controller > + > +Required properties: > + - compatible : must be "rohm,bu21029" > + - reg : i2c device address of the chip > + - interrupt-parent : the phandle for the gpio controller > + - interrupts : (gpio) interrupt to which the chip is connected > + - reset-gpios : gpio pin to reset the chip > + - rohm,x-plate-ohms : x-plate resistance in ohms > + > +Optional properties: > + - touchscreen-max-pressure: maximum pressure value > + > +Example: > + > + &i2c1 { > + /* ... */ > + > + bu21029: bu21029@40 { > + compatible = "rohm,bu21029"; > + reg = <0x40>; > + interrupt-parent = <&gpio1>; > + interrupts = <4 IRQ_TYPE_EDGE_FALLING>; > + reset-gpios = <&gpio6 16 GPIO_ACTIVE_LOW>; > + rohm,x-plate-ohms = <600>; > + touchscreen-max-pressure = <4095>; > + }; > + > + /* ... */ > + }; > diff --git a/drivers/input/touchscreen/Kconfig b/drivers/input/touchscreen/Kconfig > index 4f15496..e09fe8f 100644 > --- a/drivers/input/touchscreen/Kconfig > +++ b/drivers/input/touchscreen/Kconfig > @@ -151,6 +151,18 @@ config TOUCHSCREEN_BU21013 > To compile this driver as a module, choose M here: the > module will be called bu21013_ts. > > +config TOUCHSCREEN_BU21029 > + tristate "Rohm BU21029 based touch panel controllers" > + depends on I2C > + help > + Say Y here if you have a Rohm BU21029 touchscreen controller > + connected to your system. > + > + If unsure, say N. > + > + To compile this driver as a module, choose M here: the > + module will be called bu21029_ts. > + > config TOUCHSCREEN_CHIPONE_ICN8318 > tristate "chipone icn8318 touchscreen controller" > depends on GPIOLIB || COMPILE_TEST > diff --git a/drivers/input/touchscreen/Makefile b/drivers/input/touchscreen/Makefile > index dddae79..f50624c 100644 > --- a/drivers/input/touchscreen/Makefile > +++ b/drivers/input/touchscreen/Makefile > @@ -18,6 +18,7 @@ obj-$(CONFIG_TOUCHSCREEN_AR1021_I2C) += ar1021_i2c.o > obj-$(CONFIG_TOUCHSCREEN_ATMEL_MXT) += atmel_mxt_ts.o > obj-$(CONFIG_TOUCHSCREEN_AUO_PIXCIR) += auo-pixcir-ts.o > obj-$(CONFIG_TOUCHSCREEN_BU21013) += bu21013_ts.o > +obj-$(CONFIG_TOUCHSCREEN_BU21029) += bu21029_ts.o > obj-$(CONFIG_TOUCHSCREEN_CHIPONE_ICN8318) += chipone_icn8318.o > obj-$(CONFIG_TOUCHSCREEN_CY8CTMG110) += cy8ctmg110_ts.o > obj-$(CONFIG_TOUCHSCREEN_CYTTSP_CORE) += cyttsp_core.o > diff --git a/drivers/input/touchscreen/bu21029_ts.c b/drivers/input/touchscreen/bu21029_ts.c > new file mode 100644 > index 0000000..d5cbf11 > --- /dev/null > +++ b/drivers/input/touchscreen/bu21029_ts.c > @@ -0,0 +1,456 @@ Please add SPDX tag for the driver. > +/* > + * Rohm BU21029 touchscreen controller driver > + * > + * Copyright (C) 2015 Bosch Sicherheitssysteme GmbH > + * > + * This program is free software; you can redistribute it and/or modify > + * it under the terms of the GNU General Public License version 2 as > + * published by the Free Software Foundation. > + */ > + > +#include > +#include > +#include > +#include > +#include > +#include > +#include > +#include > +#include Please use GPIOD API (and include linux/gpio/consumer.h instead of of_gpio.h). > +#include > + > +/* HW_ID1 Register (PAGE=0, ADDR=0x0E, Reset value=0x02, Read only) Multi-line comments start with empty comment line: /* * Multi * line. */ > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | D7 | D6 | D5 | D4 | D3 | D2 | D1 | D0 | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | HW_IDH | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * HW_ID2 Register (PAGE=0, ADDR=0x0F, Reset value=0x29, Read only) > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | D7 | D6 | D5 | D4 | D3 | D2 | D1 | D0 | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | HW_IDL | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * HW_IDH: high 8bits of IC's ID > + * HW_IDL: low 8bits of IC's ID > + */ > +#define BU21029_HWID_REG (0x0E << 3) > +#define SUPPORTED_HWID 0x0229 > + > +/* CFR0 Register (PAGE=0, ADDR=0x00, Reset value=0x20) > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | D7 | D6 | D5 | D4 | D3 | D2 | D1 | D0 | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | 0 | 0 | CALIB | INTRM | 0 | 0 | 0 | 0 | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * CALIB: 0 = not to use calibration result (*) > + * 1 = use calibration result > + * INTRM: 0 = INT output depend on "pen down" (*) > + * 1 = INT output always "0" > + */ > +#define BU21029_CFR0_REG (0x00 << 3) > +#define CFR0_VALUE 0x00 > + > +/* CFR1 Register (PAGE=0, ADDR=0x01, Reset value=0xA6) > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | D7 | D6 | D5 | D4 | D3 | D2 | D1 | D0 | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | MAV | AVE[2:0] | 0 | SMPL[2:0] | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * MAV: 0 = median average filter off > + * 1 = median average filter on (*) > + * AVE: AVE+1 = number of average samples for MAV, > + * if AVE>SMPL, then AVE=SMPL (=3) > + * SMPL: SMPL+1 = number of conversion samples for MAV (=7) > + */ > +#define BU21029_CFR1_REG (0x01 << 3) > +#define CFR1_VALUE 0xA6 > + > +/* CFR2 Register (PAGE=0, ADDR=0x02, Reset value=0x04) > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | D7 | D6 | D5 | D4 | D3 | D2 | D1 | D0 | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | INTVL_TIME[3:0] | TIME_ST_ADC[3:0] | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * INTVL_TIME: waiting time between completion of conversion > + * and start of next conversion, only usable in > + * autoscan mode (=20.480ms) > + * TIME_ST_ADC: waiting time between application of voltage > + * to panel and start of A/D conversion (=100us) > + */ > +#define BU21029_CFR2_REG (0x02 << 3) > +#define CFR2_VALUE 0xC9 > + > +/* CFR3 Register (PAGE=0, ADDR=0x0B, Reset value=0x72) > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | D7 | D6 | D5 | D4 | D3 | D2 | D1 | D0 | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | RM8 | STRETCH| PU90K | DUAL | PIDAC_OFS[3:0] | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * RM8: 0 = coordinate resolution is 12bit (*) > + * 1 = coordinate resolution is 8bit > + * STRETCH: 0 = SCL_STRETCH function off > + * 1 = SCL_STRETCH function on (*) > + * PU90K: 0 = internal pull-up resistance for touch detection is ~50kohms (*) > + * 1 = internal pull-up resistance for touch detection is ~90kohms > + * DUAL: 0 = dual touch detection off (*) > + * 1 = dual touch detection on > + * PIDAC_OFS: dual touch detection circuit adjustment, it is not necessary > + * to change this from initial value > + */ > +#define BU21029_CFR3_REG (0x0B << 3) > +#define CFR3_VALUE 0x42 > + > +/* LDO Register (PAGE=0, ADDR=0x0C, Reset value=0x00) > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | D7 | D6 | D5 | D4 | D3 | D2 | D1 | D0 | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | 0 | PVDD[2:0] | 0 | AVDD[2:0] | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * PVDD: output voltage of panel output regulator (=2.000V) > + * AVDD: output voltage of analog circuit regulator (=2.000V) > + */ > +#define BU21029_LDO_REG (0x0C << 3) > +#define LDO_VALUE 0x77 > + > +/* Serial Interface Command Byte 1 (CID=1) > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | D7 | D6 | D5 | D4 | D3 | D2 | D1 | D0 | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * | 1 | CF | CMSK | PDM | STP | > + * +--------+--------+--------+--------+--------+--------+--------+--------+ > + * CF: conversion function, see table 3 in datasheet p6 (=0000, automatic scan) > + * CMSK: 0 = executes convert function (*) > + * 1 = reads the convert result > + * PDM: 0 = power down after convert function stops (*) > + * 1 = keep power on after convert function stops > + * STP: 1 = abort current conversion and power down, set to "0" automatically > + */ > +#define BU21029_AUTOSCAN 0x80 > + > +/* The timeout value needs to be larger than INTVL_TIME + tConv4 (sample and > + * conversion time), where tConv4 is calculated by formula: > + * tPON + tDLY1 + (tTIME_ST_ADC + (tADC * tSMPL) * 2 + tDLY2) * 3 > + * see figure 8 in datasheet p15 for details of each field. > + */ > +#define PEN_UP_TIMEOUT msecs_to_jiffies(50) > + > +#define STOP_DELAY_US 50L > +#define START_DELAY_MS 2L > +#define BUF_LEN 8L > +#define SCALE_12BIT (1 << 12) > +#define MAX_12BIT ((1 << 12) - 1) > +#define DRIVER_NAME "bu21029" > + > +struct bu21029_ts_data { > + struct i2c_client *client; > + struct input_dev *in_dev; > + struct timer_list timer; > + u32 reset_gpios; > + u32 reset_gpios_assert; > + u32 x_plate_ohms; > + u32 max_pressure; > +}; > + > +static int bu21029_touch_report(struct bu21029_ts_data *bu21029) > +{ > + struct i2c_client *i2c = bu21029->client; > + u8 buf[BUF_LEN]; > + u16 x, y, z1, z2; > + u32 rz; > + > + /* read touch data and deassert INT (by restarting the autoscan mode) */ > + int error = i2c_smbus_read_i2c_block_data(i2c, > + BU21029_AUTOSCAN, > + BUF_LEN, > + buf); > + if (error < 0) > + return error; > + > + /* compose upper 8 and lower 4 bits into a 12bit value: > + * +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+ > + * | ByteH | ByteL | > + * +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+ > + * |b07|b06|b05|b04|b03|b02|b01|b00|b07|b06|b05|b04|b03|b02|b01|b00| > + * +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+ > + * |v11|v10|v09|v08|v07|v06|v05|v04|v03|v02|v01|v00| 0 | 0 | 0 | 0 | > + * +---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+---+ > + */ > + x = (buf[0] << 4) | (buf[1] >> 4); > + y = (buf[2] << 4) | (buf[3] >> 4); > + z1 = (buf[4] << 4) | (buf[5] >> 4); > + z2 = (buf[6] << 4) | (buf[7] >> 4); > + > + if (z1 == 0 || z2 == 0) > + return 0; > + > + /* calculate Rz (pressure resistance value) by equation: > + * Rz = Rx * (x/Q) * ((z2/z1) - 1), where > + * Rx is x-plate resistance, > + * Q is the touch screen resolution (8bit = 256, 12bit = 4096) > + * x, z1, z2 are the measured positions. > + */ > + rz = z2 - z1; > + rz *= x; > + rz *= bu21029->x_plate_ohms; > + rz /= z1; > + rz = DIV_ROUND_CLOSEST(rz, SCALE_12BIT); > + if (rz <= bu21029->max_pressure) { > + input_report_abs(bu21029->in_dev, ABS_X, x); > + input_report_abs(bu21029->in_dev, ABS_Y, y); > + input_report_abs(bu21029->in_dev, ABS_PRESSURE, rz); What is the values of pressure reported when finger is touching the surface? IOW is 'rz' pressure or resistance? > + input_report_key(bu21029->in_dev, BTN_TOUCH, 1); > + input_sync(bu21029->in_dev); > + } > + > + return 0; > +} > + > +static void bu21029_touch_release(struct timer_list *t) > +{ > + struct bu21029_ts_data *bu21029 = from_timer(bu21029, t, timer); > + > + input_report_abs(bu21029->in_dev, ABS_PRESSURE, 0); > + input_report_key(bu21029->in_dev, BTN_TOUCH, 0); > + input_sync(bu21029->in_dev); > +} > + > +static irqreturn_t bu21029_touch_soft_irq(int irq, void *data) > +{ > + struct bu21029_ts_data *bu21029 = data; > + struct i2c_client *i2c = bu21029->client; > + > + /* report touch and deassert interrupt (will assert again after > + * INTVL_TIME + tConv4 for continuous touch) > + */ > + int error = bu21029_touch_report(bu21029); > + > + if (error) { > + dev_err(&i2c->dev, "failed to report (error: %d)\n", error); > + return IRQ_NONE; > + } > + > + /* reset timer for pen up detection */ > + mod_timer(&bu21029->timer, jiffies + PEN_UP_TIMEOUT); > + > + return IRQ_HANDLED; > +} > + > +static void bu21029_reset_chip(struct bu21029_ts_data *bu21029) > +{ > + gpio_set_value(bu21029->reset_gpios, > + bu21029->reset_gpios_assert); > + udelay(STOP_DELAY_US); > + gpio_set_value(bu21029->reset_gpios, > + !bu21029->reset_gpios_assert); > + mdelay(START_DELAY_MS); > +} > + > +static int bu21029_init_chip(struct bu21029_ts_data *bu21029) > +{ > + struct i2c_client *i2c = bu21029->client; > + struct { > + u8 reg; > + u8 value; > + } init_table[] = { > + {BU21029_CFR0_REG, CFR0_VALUE}, > + {BU21029_CFR1_REG, CFR1_VALUE}, > + {BU21029_CFR2_REG, CFR2_VALUE}, > + {BU21029_CFR3_REG, CFR3_VALUE}, > + {BU21029_LDO_REG, LDO_VALUE} > + }; > + int error, i; > + u16 hwid; > + > + bu21029_reset_chip(bu21029); > + > + error = i2c_smbus_read_i2c_block_data(i2c, > + BU21029_HWID_REG, > + 2, > + (u8 *)&hwid); > + if (error < 0) { > + dev_err(&i2c->dev, "failed to read HW ID\n"); > + return error; > + } > + > + if (cpu_to_be16(hwid) != SUPPORTED_HWID) { > + dev_err(&i2c->dev, "unsupported HW ID 0x%x\n", hwid); > + return -ENODEV; > + } > + > + for (i = 0; i < ARRAY_SIZE(init_table); ++i) { > + error = i2c_smbus_write_byte_data(i2c, > + init_table[i].reg, > + init_table[i].value); > + if (error < 0) { > + dev_err(&i2c->dev, > + "failed to write 0x%x to register 0x%x\n", > + init_table[i].value, > + init_table[i].reg); > + return error; > + } > + } > + > + error = i2c_smbus_write_byte(i2c, BU21029_AUTOSCAN); > + if (error < 0) { > + dev_err(&i2c->dev, "failed to start autoscan\n"); > + return error; > + } > + > + return 0; > +} > + > +static int bu21029_parse_dt(struct bu21029_ts_data *bu21029) > +{ > + struct device *dev = &bu21029->client->dev; > + struct device_node *np = dev->of_node; > + enum of_gpio_flags flags; > + u32 val32; > + int gpio; > + > + if (!np) { > + dev_err(dev, "no device tree data\n"); > + return -EINVAL; > + } > + > + gpio = of_get_named_gpio_flags(np, "reset-gpios", 0, &flags); > + if (!gpio_is_valid(gpio)) { > + dev_err(dev, "invalid 'reset-gpios' supplied\n"); > + return -EINVAL; > + } > + bu21029->reset_gpios = gpio; > + bu21029->reset_gpios_assert = (flags & OF_GPIO_ACTIVE_LOW) ? 0 : 1; > + > + if (of_property_read_u32(np, "rohm,x-plate-ohms", &val32)) { > + dev_err(dev, "invalid 'x-plate-ohms' supplied\n"); > + return -EINVAL; > + } > + bu21029->x_plate_ohms = val32; > + > + if (of_property_read_u32(np, "touchscreen-max-pressure", &val32)) > + bu21029->max_pressure = MAX_12BIT; > + else > + bu21029->max_pressure = val32; Please use infrastructure form include/linux/input/touchscreen.h so that you handle different sizes and orientations. > + > + return 0; > +} > + > +static int bu21029_probe(struct i2c_client *client, > + const struct i2c_device_id *id) > +{ > + struct bu21029_ts_data *bu21029; > + struct input_dev *in_dev; > + int error; > + > + if (!i2c_check_functionality(client->adapter, > + I2C_FUNC_SMBUS_WRITE_BYTE | > + I2C_FUNC_SMBUS_WRITE_BYTE_DATA | > + I2C_FUNC_SMBUS_READ_I2C_BLOCK)) { > + dev_err(&client->dev, > + "i2c functionality support is not sufficient\n"); > + return -EIO; > + } > + > + bu21029 = devm_kzalloc(&client->dev, sizeof(*bu21029), GFP_KERNEL); > + if (!bu21029) > + return -ENOMEM; > + > + in_dev = devm_input_allocate_device(&client->dev); > + if (!in_dev) { > + dev_err(&client->dev, "unable to allocate input device\n"); > + return -ENOMEM; > + } > + > + bu21029->client = client; > + bu21029->in_dev = in_dev; > + timer_setup(&bu21029->timer, bu21029_touch_release, 0); > + > + error = bu21029_parse_dt(bu21029); > + if (error) > + return error; > + > + error = devm_gpio_request_one(&client->dev, > + bu21029->reset_gpios, > + GPIOF_OUT_INIT_HIGH, > + DRIVER_NAME); > + if (error) { > + dev_err(&client->dev, "unable to request reset-gpios\n"); > + return error; > + } > + > + error = bu21029_init_chip(bu21029); > + if (error) { > + dev_err(&client->dev, "unable to config bu21029\n"); > + return error; > + } > + > + in_dev->name = DRIVER_NAME; > + in_dev->id.bustype = BUS_I2C; > + in_dev->dev.parent = &client->dev; Not needed with devm_input_allocate_device(). > + > + __set_bit(EV_SYN, in_dev->evbit); > + __set_bit(EV_KEY, in_dev->evbit); > + __set_bit(EV_ABS, in_dev->evbit); > + __set_bit(ABS_X, in_dev->absbit); > + __set_bit(ABS_Y, in_dev->absbit); > + __set_bit(ABS_PRESSURE, in_dev->absbit); > + __set_bit(BTN_TOUCH, in_dev->keybit); > + > + input_set_abs_params(in_dev, ABS_X, 0, MAX_12BIT, 0, 0); > + input_set_abs_params(in_dev, ABS_Y, 0, MAX_12BIT, 0, 0); > + input_set_abs_params(in_dev, ABS_PRESSURE, > + 0, bu21029->max_pressure, 0, 0); > + input_set_drvdata(in_dev, bu21029); > + > + error = input_register_device(in_dev); > + if (error) { > + dev_err(&client->dev, "unable to register input device\n"); > + return error; > + } > + > + i2c_set_clientdata(client, bu21029); > + > + error = devm_request_threaded_irq(&client->dev, > + client->irq, > + NULL, > + bu21029_touch_soft_irq, > + IRQF_ONESHOT, > + DRIVER_NAME, > + bu21029); > + if (error) { > + dev_err(&client->dev, "unable to request touch irq\n"); > + return error; > + } > + > + return 0; > +} > + > +static int bu21029_remove(struct i2c_client *client) > +{ > + struct bu21029_ts_data *bu21029 = i2c_get_clientdata(client); > + > + del_timer_sync(&bu21029->timer); If interrupt comes here kernel will be unhappy. You need to either work canceling timer into devm unwid stream (devm_add_action_or_reset()) or somehow make sure that you shut off interrupts before canceling the timer. > + > + return 0; > +} > + > +static const struct i2c_device_id bu21029_ids[] = { > + {DRIVER_NAME, 0}, > + {} > +}; > +MODULE_DEVICE_TABLE(i2c, bu21029_ids); > + > +static struct i2c_driver bu21029_driver = { > + .driver = { > + .name = DRIVER_NAME, > + .owner = THIS_MODULE, Not needed. > + }, > + .id_table = bu21029_ids, > + .probe = bu21029_probe, > + .remove = bu21029_remove, > +}; > +module_i2c_driver(bu21029_driver); > + > +MODULE_AUTHOR("Zhu Yi "); > +MODULE_DESCRIPTION("Rohm BU21029 touchscreen controller driver"); > +MODULE_LICENSE("GPL v2"); > -- > 2.7.4 > Thanks. -- Dmitry