From nobody Sun Feb 8 06:56:20 2026 Received: from mail-ej1-f48.google.com (mail-ej1-f48.google.com [209.85.218.48]) (using TLSv1.2 with cipher ECDHE-RSA-AES128-GCM-SHA256 (128/128 bits)) (No client certificate requested) by smtp.subspace.kernel.org (Postfix) with ESMTPS id F0403311C3D for ; Mon, 27 Oct 2025 16:16:27 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=209.85.218.48 ARC-Seal: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1761581790; cv=none; b=aap/0x3O2X2ubmKkfl0XVRpfPLz27BFelKLaDXSK6VgbqUrWAWOrkY6x4isur5B60Zb4K5i+yJpUb3qh6kCuv9971JDdC2DbnOdqKp6NP2MkLjMc4/xqWUIot8hVEy0MpIwjeu9F6yEcr30kaBIxRIgbN+7kEblqnn1hGjd1xa0= ARC-Message-Signature: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1761581790; c=relaxed/simple; bh=S0TJRT9I6u85I8rir2bVhEGUHAKgwkhgyEcDxxXmpA4=; h=From:To:Cc:Subject:Date:Message-ID:MIME-Version; b=XUFNq8X77HGNGxxjBqwCs/AkYKgESj4wrQ6LyimKt3OtIY6miph/LH6u3z4arumBNdryb1MqS+UKl9jExoieA4baHEM/sf1hMW/VmmW9I8xAMZmllvrwko8/hwTMawUGUwyapOp+y2xhPfqdWJKMFFZaqDN80Gjm2m73qXjsWmM= ARC-Authentication-Results: i=1; smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=amarulasolutions.com; spf=pass smtp.mailfrom=amarulasolutions.com; dkim=pass (1024-bit key) header.d=amarulasolutions.com header.i=@amarulasolutions.com header.b=ksadmCAq; arc=none smtp.client-ip=209.85.218.48 Authentication-Results: smtp.subspace.kernel.org; dmarc=pass (p=none dis=none) header.from=amarulasolutions.com Authentication-Results: smtp.subspace.kernel.org; spf=pass smtp.mailfrom=amarulasolutions.com Authentication-Results: smtp.subspace.kernel.org; dkim=pass (1024-bit key) header.d=amarulasolutions.com header.i=@amarulasolutions.com header.b="ksadmCAq" Received: by mail-ej1-f48.google.com with SMTP id a640c23a62f3a-b5a8184144dso773557066b.1 for ; Mon, 27 Oct 2025 09:16:27 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=amarulasolutions.com; s=google; t=1761581786; x=1762186586; darn=vger.kernel.org; h=content-transfer-encoding:mime-version:message-id:date:subject:cc :to:from:from:to:cc:subject:date:message-id:reply-to; bh=dPYho8Qu09IQiF/d4ybgLxhO4EcHqRatvEGSDjzG4FM=; b=ksadmCAqJi261GxurTRjUDxtbEl8zIO/E/DqB4LkibCwb5zF0iS39qvEXxfgjjspVa kxHKVpx7TZvOtYS8RTYOUTuU9/6lVB0Ehs0EOSu6POSM6IlleG8YM3msBFq2IzyjEeDk rFD5hu45RCJC3zG8gsJW0kCgIDFJ+QHxLWbu8= X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20230601; t=1761581786; x=1762186586; h=content-transfer-encoding:mime-version:message-id:date:subject:cc :to:from:x-gm-message-state:from:to:cc:subject:date:message-id :reply-to; bh=dPYho8Qu09IQiF/d4ybgLxhO4EcHqRatvEGSDjzG4FM=; b=dHGYPe7FQ05D4npeNDtlsjiY5LyNmgOO92MAkhP5yEf00krWLmDYqTh7+eXEjQtP0O qnaneK0YPykIVUbN3tu2s37keydIiatyr0cn0PEGzCv1P37NncxWDd+qX5JHhJdOoeEl GIpEkNJzsS4oHlmklQLC/43ZXHMlFH4gidieKC0YsQkcNhVNQaxcg6CTLuDBxOcYgf58 6NfOiaOjOHOcxb5TjYhO7ynhIHCSxZXLUQdGXOm5M7F/CWCwHf488uJBvslYsgosY8Hc 2IVvjLzVTa3ktFRtfSlCkQ1J6lAQVpMHQjBhwI8JzO7IK1mahX9mVM7yi6BMgk/1r1fi wi8w== X-Gm-Message-State: AOJu0YwBMCq4+pMGtY7pkW8djcKOVXVMfgNFdbq7uM0Bcs5vmIhn+B75 WyVw5QqCuZ3jMJVjEiXYzwU60k0WffQhqGFSbdccK18zkYe8Jr/k/I8GbOS1mqdXCjACjgxBAZh gVW0U X-Gm-Gg: ASbGncvsxmc1qZyPACwfO9/6OqVKgGckS7Hkz6dm6q//eCdMmAuh+IVjwd5VmgUU/Av YeWguPzQR8z/lLiETezGW4xWyQx2SJsRwf4UeNSyduOFqkbaXVjadCu1JSjXaUYs1mPWWxNxNih sVdAWgeAaJKSdCD4ebHs+/AX19Cim8B5OK2Ctn2cFxzv7yET+z7IgmsPz4TnJV9PqQpBaEDB/nf AVIRRsJ4ReFnI100dwA0mQZi7UyAJA+fw1Ka/AJQ0j6BJveCOAah4CkBbYsz0GnMd43XVteSmw4 kVVhc/q9aA3mqeb6mkA8tJZKHjQh66Qi7UlwS6o/4zwbQCAKzJXcved4oBdnBADyjisb6E7ayQ8 aLhGaoGJBFKMEnLA6dpZluia2K7L27Cw7oPOkNW96ZSRSc9N82goa0Lf9i3FbunWt+heqfV0uag VGXVO3v1FVYK3FEauzvtF9ka7mw3Z8ZvYKkEOpbmt4Q9brAfegK3CpcDLOw3Z3uMR1zA3FfRL1m e32QjVbGwR+EcSh7kYpJh1S5heCcPVCDVwStI7trxOC X-Google-Smtp-Source: AGHT+IGC3DtKxJ+lK6DUYWoBdT0g+Qnm0l1Sp8U84Jz677mjx6t11jEpUUoTlzXS0zEOB/FeUQ6H7Q== X-Received: by 2002:a17:907:94c7:b0:b6d:573d:bbc5 with SMTP id a640c23a62f3a-b6dba525d87mr43276466b.37.1761581785729; Mon, 27 Oct 2025 09:16:25 -0700 (PDT) Received: from dario-ThinkPad-P14s-Gen-5.homenet.telecomitalia.it (host-82-50-34-170.retail.telecomitalia.it. [82.50.34.170]) by smtp.gmail.com with ESMTPSA id a640c23a62f3a-b6d854430dbsm798403666b.63.2025.10.27.09.16.24 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Mon, 27 Oct 2025 09:16:24 -0700 (PDT) From: Dario Binacchi To: linux-kernel@vger.kernel.org Cc: linux-amarula@amarulasolutions.com, Dario Binacchi , Dmitry Torokhov , Jens Reidel , Wolfram Sang , linux-input@vger.kernel.org Subject: [RESEND PATCH v2] Input: edt-ft5x06 - fix report rate handling by sysfs Date: Mon, 27 Oct 2025 17:16:05 +0100 Message-ID: <20251027161622.2021180-1-dario.binacchi@amarulasolutions.com> X-Mailer: git-send-email 2.43.0 Precedence: bulk X-Mailing-List: linux-kernel@vger.kernel.org List-Id: List-Subscribe: List-Unsubscribe: MIME-Version: 1.0 Content-Transfer-Encoding: quoted-printable Content-Type: text/plain; charset="utf-8" In the driver probe, the report-rate-hz value from device tree is written directly to the M12 controller register, while for the M06 it is divided by 10 since the controller expects the value in units of 10 Hz. That logic was missing in the sysfs handling, leading to inconsistent behavior depending on whether the value came from device tree or sysfs. This patch makes the report-rate handling consistent by applying the same logic in both cases. Two dedicated functions, report_rate_hz_{show,store}, were added for the following reasons: - Avoid modifying the more generic edt_ft5x06_setting_{show,store} and thus prevent regressions. - Properly enforce lower and upper limits for the M06 case. The previous version accepted invalid values for M06, since it relied on the M12 limits. - Return an error when the property is not supported (e.g. M09), to avoid misleading users into thinking the property is handled by the controller. Signed-off-by: Dario Binacchi --- Changes in v2: - Drop the patch: 1/2 Input: edt-ft5x06 - rename sysfs attribute report_rate to report_rate= _hz because not accepted. drivers/input/touchscreen/edt-ft5x06.c | 158 +++++++++++++++++++++---- 1 file changed, 135 insertions(+), 23 deletions(-) diff --git a/drivers/input/touchscreen/edt-ft5x06.c b/drivers/input/touchsc= reen/edt-ft5x06.c index bf498bd4dea9..d7a269a0528f 100644 --- a/drivers/input/touchscreen/edt-ft5x06.c +++ b/drivers/input/touchscreen/edt-ft5x06.c @@ -516,9 +516,136 @@ static EDT_ATTR(offset_y, S_IWUSR | S_IRUGO, NO_REGIS= TER, NO_REGISTER, /* m06: range 20 to 80, m09: range 0 to 30, m12: range 1 to 255... */ static EDT_ATTR(threshold, S_IWUSR | S_IRUGO, WORK_REGISTER_THRESHOLD, M09_REGISTER_THRESHOLD, EV_REGISTER_THRESHOLD, 0, 255); -/* m06: range 3 to 14, m12: range 1 to 255 */ -static EDT_ATTR(report_rate, S_IWUSR | S_IRUGO, WORK_REGISTER_REPORT_RATE, - M12_REGISTER_REPORT_RATE, NO_REGISTER, 0, 255); + +static int edt_ft5x06_report_rate_get(struct edt_ft5x06_ts_data *tsdata) +{ + unsigned int val; + int error; + + if (tsdata->reg_addr.reg_report_rate =3D=3D NO_REGISTER) + return -EOPNOTSUPP; + + error =3D regmap_read(tsdata->regmap, tsdata->reg_addr.reg_report_rate, + &val); + if (error) + return error; + + if (tsdata->version =3D=3D EDT_M06) + val *=3D 10; + + if (val !=3D tsdata->report_rate) { + dev_warn(&tsdata->client->dev, + "report-rate: read (%d) and stored value (%d) differ\n", + val, tsdata->report_rate); + tsdata->report_rate =3D val; + } + + return 0; +} + +static ssize_t report_rate_show(struct device *dev, + struct device_attribute *dattr, char *buf) +{ + struct i2c_client *client =3D to_i2c_client(dev); + struct edt_ft5x06_ts_data *tsdata =3D i2c_get_clientdata(client); + size_t count; + int error; + + mutex_lock(&tsdata->mutex); + + if (tsdata->factory_mode) { + error =3D -EIO; + goto out; + } + + error =3D edt_ft5x06_report_rate_get(tsdata); + if (error) { + dev_err(&tsdata->client->dev, + "Failed to fetch attribute %s, error %d\n", + dattr->attr.name, error); + goto out; + } + + count =3D sysfs_emit(buf, "%d\n", tsdata->report_rate); +out: + mutex_unlock(&tsdata->mutex); + return error ?: count; +} + +static int edt_ft5x06_report_rate_set(struct edt_ft5x06_ts_data *tsdata, + unsigned int val) +{ + if (tsdata->reg_addr.reg_report_rate =3D=3D NO_REGISTER) + return -EOPNOTSUPP; + + if (tsdata->version =3D=3D EDT_M06) + tsdata->report_rate =3D clamp_val(val, 30, 140); + else + tsdata->report_rate =3D clamp_val(val, 1, 255); + + if (val !=3D tsdata->report_rate) { + dev_warn(&tsdata->client->dev, + "report-rate %dHz is unsupported, use %dHz\n", + val, tsdata->report_rate); + val =3D tsdata->report_rate; + } + + if (tsdata->version =3D=3D EDT_M06) + val /=3D 10; + + return regmap_write(tsdata->regmap, tsdata->reg_addr.reg_report_rate, + val); +} + +static ssize_t report_rate_store(struct device *dev, + struct device_attribute *dattr, + const char *buf, size_t count) +{ + struct i2c_client *client =3D to_i2c_client(dev); + struct edt_ft5x06_ts_data *tsdata =3D i2c_get_clientdata(client); + unsigned int val; + u8 limit_low; + u8 limit_high; + int error; + + mutex_lock(&tsdata->mutex); + + if (tsdata->factory_mode) { + error =3D -EIO; + goto out; + } + + error =3D kstrtouint(buf, 0, &val); + if (error) + goto out; + + if (tsdata->version =3D=3D EDT_M06) { + limit_low =3D 30; + limit_high =3D 140; + } else { + limit_low =3D 1; + limit_high =3D 255; + } + + if (val < limit_low || val > limit_high) { + error =3D -ERANGE; + goto out; + } + + error =3D edt_ft5x06_report_rate_set(tsdata, val); + if (error) { + dev_err(&tsdata->client->dev, + "Failed to update attribute %s, error: %d\n", + dattr->attr.name, error); + goto out; + } + +out: + mutex_unlock(&tsdata->mutex); + return error ?: count; +} + +static DEVICE_ATTR_RW(report_rate); =20 static ssize_t model_show(struct device *dev, struct device_attribute *att= r, char *buf) @@ -572,7 +699,7 @@ static struct attribute *edt_ft5x06_attrs[] =3D { &edt_ft5x06_attr_offset_x.dattr.attr, &edt_ft5x06_attr_offset_y.dattr.attr, &edt_ft5x06_attr_threshold.dattr.attr, - &edt_ft5x06_attr_report_rate.dattr.attr, + &dev_attr_report_rate.attr, &dev_attr_model.attr, &dev_attr_fw_version.attr, &dev_attr_header_errors.attr, @@ -595,8 +722,7 @@ static void edt_ft5x06_restore_reg_parameters(struct ed= t_ft5x06_ts_data *tsdata) if (reg_addr->reg_offset_y !=3D NO_REGISTER) regmap_write(regmap, reg_addr->reg_offset_y, tsdata->offset_y); if (reg_addr->reg_report_rate !=3D NO_REGISTER) - regmap_write(regmap, reg_addr->reg_report_rate, - tsdata->report_rate); + edt_ft5x06_report_rate_set(tsdata, tsdata->report_rate); } =20 #ifdef CONFIG_DEBUG_FS @@ -1029,8 +1155,8 @@ static void edt_ft5x06_ts_get_parameters(struct edt_f= t5x06_ts_data *tsdata) if (reg_addr->reg_offset_y !=3D NO_REGISTER) regmap_read(regmap, reg_addr->reg_offset_y, &tsdata->offset_y); if (reg_addr->reg_report_rate !=3D NO_REGISTER) - regmap_read(regmap, reg_addr->reg_report_rate, - &tsdata->report_rate); + edt_ft5x06_report_rate_get(tsdata); + tsdata->num_x =3D EDT_DEFAULT_NUM_X; if (reg_addr->reg_num_x !=3D NO_REGISTER) { if (!regmap_read(regmap, reg_addr->reg_num_x, &val)) @@ -1289,21 +1415,7 @@ static int edt_ft5x06_ts_probe(struct i2c_client *cl= ient) if (tsdata->reg_addr.reg_report_rate !=3D NO_REGISTER && !device_property_read_u32(&client->dev, "report-rate-hz", &report_rate)) { - if (tsdata->version =3D=3D EDT_M06) - tsdata->report_rate =3D clamp_val(report_rate, 30, 140); - else - tsdata->report_rate =3D clamp_val(report_rate, 1, 255); - - if (report_rate !=3D tsdata->report_rate) - dev_warn(&client->dev, - "report-rate %dHz is unsupported, use %dHz\n", - report_rate, tsdata->report_rate); - - if (tsdata->version =3D=3D EDT_M06) - tsdata->report_rate /=3D 10; - - regmap_write(tsdata->regmap, tsdata->reg_addr.reg_report_rate, - tsdata->report_rate); + edt_ft5x06_report_rate_set(tsdata, report_rate); } =20 dev_dbg(&client->dev, --=20 2.43.0 base-commit: dcb6fa37fd7bc9c3d2b066329b0d27dedf8becaa branch: edt-ft5x06-fix-report-rate