From patchwork Thu Dec 16 23:35:57 2021 Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: 7bit X-Patchwork-Submitter: Tim Harvey X-Patchwork-Id: 12683041 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from vger.kernel.org (vger.kernel.org [23.128.96.18]) by smtp.lore.kernel.org (Postfix) with ESMTP id 4BE5EC4332F for ; Thu, 16 Dec 2021 23:36:02 +0000 (UTC) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S229673AbhLPXgB (ORCPT ); Thu, 16 Dec 2021 18:36:01 -0500 Received: from finn.gateworks.com ([108.161.129.64]:36194 "EHLO finn.localdomain" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S229577AbhLPXgB (ORCPT ); Thu, 16 Dec 2021 18:36:01 -0500 Received: from 068-189-091-139.biz.spectrum.com ([68.189.91.139] helo=tharvey.pdc.gateworks.com) by finn.localdomain with esmtp (Exim 4.93) (envelope-from ) id 1my0I4-0093bq-Im; Thu, 16 Dec 2021 23:36:00 +0000 From: Tim Harvey To: linux-input@vger.kernel.org, Rob Herring , Dmitry Torokhov Cc: Tim Harvey Subject: [RFC PATCH 3/3] input: touchscreen: add dfr0550 support Date: Thu, 16 Dec 2021 15:35:57 -0800 Message-Id: <20211216233557.9393-4-tharvey@gateworks.com> X-Mailer: git-send-email 2.17.1 In-Reply-To: <20211216233557.9393-1-tharvey@gateworks.com> References: <20211216233557.9393-1-tharvey@gateworks.com> Precedence: bulk List-ID: X-Mailing-List: linux-input@vger.kernel.org Add support for the touch controller found on a DFROBOT DFR0550/DFR0776 touchscreen display. These touchscreen displays are intended to be compatible with the official Raspberry Pi 7in display which has an FTx06 touch controller directly attached to the 15pin connector to the host processor. However these displays have an FTx06 touch controller that connected to an I2C master on a STM32F103 micro controller which polls the FTx06 and emulates a virtual I2C device connected to the 15pin connector to the host processor. The emulated FTx06 implements a subset of the FTx06 register set but must be read with individual transactions between reading the number of points and the point data itself. Additionally there is no IRQ made available so this is a polling driver. Signed-off-by: Tim Harvey --- MAINTAINERS | 5 + drivers/input/touchscreen/Kconfig | 12 ++ drivers/input/touchscreen/Makefile | 1 + drivers/input/touchscreen/dfr0550.c | 208 ++++++++++++++++++++++++++++ 4 files changed, 226 insertions(+) create mode 100644 drivers/input/touchscreen/dfr0550.c diff --git a/MAINTAINERS b/MAINTAINERS index 3b79fd441dde..07d581447f4b 100644 --- a/MAINTAINERS +++ b/MAINTAINERS @@ -5458,6 +5458,11 @@ F: include/net/devlink.h F: include/uapi/linux/devlink.h F: net/core/devlink.c +DFR0550 TOUCHSCREEN DRIVER +M: Tim Harvey +S: Supported +F: drivers/input/touchscreen/dfr0550.c + DH ELECTRONICS IMX6 DHCOM BOARD SUPPORT M: Christoph Niedermaier L: kernel@dh-electronics.com diff --git a/drivers/input/touchscreen/Kconfig b/drivers/input/touchscreen/Kconfig index d4e74738c5a8..438742314157 100644 --- a/drivers/input/touchscreen/Kconfig +++ b/drivers/input/touchscreen/Kconfig @@ -309,6 +309,18 @@ config TOUCHSCREEN_DA9052 To compile this driver as a module, choose M here: the module will be called da9052_tsi. +config TOUCHSCREEN_DFR0550 + tristate "Touchscreen support for DFROBOT DFR0550" + depends on I2C + help + Say Y here to enable the support for the touchscreen found + on DFROBOT DFR0550 and DFR0678 displays. + + If unsure, say N. + + To compile this driver as a module, choose M here: the + module will be called dfr0550. + config TOUCHSCREEN_DYNAPRO tristate "Dynapro serial touchscreen" select SERIO diff --git a/drivers/input/touchscreen/Makefile b/drivers/input/touchscreen/Makefile index 7d34100f7f22..8212314975cb 100644 --- a/drivers/input/touchscreen/Makefile +++ b/drivers/input/touchscreen/Makefile @@ -31,6 +31,7 @@ obj-$(CONFIG_TOUCHSCREEN_CYTTSP4_I2C) += cyttsp4_i2c.o cyttsp_i2c_common.o obj-$(CONFIG_TOUCHSCREEN_CYTTSP4_SPI) += cyttsp4_spi.o obj-$(CONFIG_TOUCHSCREEN_DA9034) += da9034-ts.o obj-$(CONFIG_TOUCHSCREEN_DA9052) += da9052_tsi.o +obj-$(CONFIG_TOUCHSCREEN_DFR0550) += dfr0550.o obj-$(CONFIG_TOUCHSCREEN_DYNAPRO) += dynapro.o obj-$(CONFIG_TOUCHSCREEN_EDT_FT5X06) += edt-ft5x06.o obj-$(CONFIG_TOUCHSCREEN_HAMPSHIRE) += hampshire.o diff --git a/drivers/input/touchscreen/dfr0550.c b/drivers/input/touchscreen/dfr0550.c new file mode 100644 index 000000000000..a337efd36bb6 --- /dev/null +++ b/drivers/input/touchscreen/dfr0550.c @@ -0,0 +1,208 @@ +// SPDX-License-Identifier: GPL-2.0 +/* + * Driver for the touch controller found on a DFROBOT DFR0550/DFR0776 + * touchscreen display. + * + * These touchscreen displays are intended to be compatible with the official + * Raspberry Pi 7in display which has an FTx06 touch controller directly + * attached to the 15pin connector to the host processor. However these + * displays have an FTx06 touch controller that connected to an I2C master + * on a STM32F103 micro controller which polls the FTx06 and emulates a + * virtual I2C device connected to the 15pin connector to the host processor. + * The emulated FTx06 implements a subset of the FTx06 register set but + * must be read with individual transactions between reading the number + * of points and the point data itself. + * + * Additionally there is no IRQ made available so this is a polling driver. + */ + +#include +#include +#include +#include +#include +#include + +#define TOUCH_EVENT_DOWN 0x00 +#define TOUCH_EVENT_UP 0x01 +#define TOUCH_EVENT_ON 0x02 +#define TOUCH_EVENT_RESERVED 0x03 + +struct dfr0550_ts_data { + struct i2c_client *client; + struct input_dev *input; + struct touchscreen_properties prop; + int max_support_points; + unsigned int known_ids; + u16 num_x; + u16 num_y; +}; + +static int dfr0550_i2c_read(struct i2c_client *client, u8 reg, u8 *buf, int len) +{ + struct i2c_msg msgs[2]; + int ret; + + msgs[0].flags = 0; + msgs[0].addr = client->addr; + msgs[0].len = 1; + msgs[0].buf = ® + + msgs[1].flags = I2C_M_RD; + msgs[1].addr = client->addr; + msgs[1].len = len; + msgs[1].buf = buf; + + ret = i2c_transfer(client->adapter, msgs, 2); + return reg < 0 ? ret : (ret != ARRAY_SIZE(msgs) ? -EIO : 0); +} + +static void dfr0550_ts_poll(struct input_dev *input) +{ + struct dfr0550_ts_data *ts = input_get_drvdata(input); + unsigned int active_ids = 0; + int i, type, x, y, id, points; + long released_ids; + u8 buf[4]; + int error; + + error = dfr0550_i2c_read(ts->client, 0x02, buf, 1); + if (error) + goto error; + /* official rpi 7in display with ft5x06 shows 0xff until touched */ + if (buf[0] == 0xff) + return; + points = min(buf[0] & 0xf, ts->max_support_points); + + for (i = 0; i < points; i++) { + error = dfr0550_i2c_read(ts->client, 3+6*i, buf, 4); + if (error) + goto error; + + type = (buf[0] >> 6) & 0x3; + id = (buf[2] >> 4) & 0xf; + x = ((buf[0] & 0xf) << 8) + buf[1]; + y = ((buf[2] & 0xf) << 8) + buf[3]; + + active_ids |= BIT(id); + if (type == TOUCH_EVENT_UP || + type == TOUCH_EVENT_ON) { + input_mt_slot(input, id); + input_mt_report_slot_state(input, MT_TOOL_FINGER, 1); + touchscreen_report_pos(input, &ts->prop, x, y, true); + } + } + + released_ids = ts->known_ids & ~active_ids; + for_each_set_bit(i, &released_ids, ts->max_support_points) { + input_mt_slot(input, i); + input_mt_report_slot_inactive(input); + } + ts->known_ids = active_ids; + + input_mt_sync_frame(input); + input_mt_report_pointer_emulation(input, true); + input_sync(input); + + return; + +error: + dev_err_ratelimited(&ts->client->dev, + "Unable to fetch data, error: %d\n", error); +} + +static int dfr0550_ts_probe(struct i2c_client *client, + const struct i2c_device_id *id) +{ + struct dfr0550_ts_data *tsdata; + struct input_dev *input; + u32 poll_interval = 0; + int error; + + tsdata = devm_kzalloc(&client->dev, sizeof(*tsdata), GFP_KERNEL); + if (!tsdata) + return -ENOMEM; + + input = devm_input_allocate_device(&client->dev); + if (!input) { + dev_err(&client->dev, "failed to allocate input device.\n"); + return -ENOMEM; + } + + tsdata->client = client; + tsdata->input = input; + + input->name = "dfr0550-ts"; + input->id.bustype = BUS_I2C; + input->dev.parent = &client->dev; + + input_set_abs_params(input, ABS_MT_POSITION_X, + 0, tsdata->num_x * 64 - 1, 0, 0); + input_set_abs_params(input, ABS_MT_POSITION_Y, + 0, tsdata->num_y * 64 - 1, 0, 0); + + touchscreen_parse_properties(input, true, &tsdata->prop); + + tsdata->max_support_points = 5; + error = input_mt_init_slots(input, tsdata->max_support_points, + INPUT_MT_DIRECT); + if (error) { + dev_err(&client->dev, "Unable to init MT slots.\n"); + return error; + } + + i2c_set_clientdata(client, tsdata); + + device_property_read_u32(&client->dev, "poll-interval", + &poll_interval); + error = input_setup_polling(input, dfr0550_ts_poll); + if (error) { + dev_err(&client->dev, + "Unable to set up polling mode: %d\n", error); + return error; + } + input_set_drvdata(input, tsdata); + input_set_poll_interval(input, poll_interval); + + error = input_register_device(input); + if (error) + return error; + + dev_info(&client->dev, "Polling device at %dms\n", poll_interval); + + return 0; +} + +static int dfr0550_ts_remove(struct i2c_client *client) +{ + return 0; +} + +static const struct i2c_device_id dfr0550_ts_id[] = { + { .name = "dfr,dfr0550" }, + { /* sentinel */ } +}; +MODULE_DEVICE_TABLE(i2c, dfr0550_ts_id); + +static const struct of_device_id dfr0550_of_match[] = { + { .compatible = "dfr,dfr0550", }, + { /* sentinel */ } +}; +MODULE_DEVICE_TABLE(of, dfr0550_of_match); + +static struct i2c_driver dfr0550_ts_driver = { + .driver = { + .name = "dfr0550", + .of_match_table = dfr0550_of_match, + .probe_type = PROBE_PREFER_ASYNCHRONOUS, + }, + .id_table = dfr0550_ts_id, + .probe = dfr0550_ts_probe, + .remove = dfr0550_ts_remove, +}; + +module_i2c_driver(dfr0550_ts_driver); + +MODULE_AUTHOR("Tim Harvey "); +MODULE_DESCRIPTION("DFROBOT DFR0550 I2C Touchscreen Driver"); +MODULE_LICENSE("GPL v2");