From nobody Mon Dec 1 22:07:24 2025 Received: from mail-ej1-f54.google.com (mail-ej1-f54.google.com [209.85.218.54]) (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 7B76C31A553 for ; Thu, 27 Nov 2025 07:46:36 +0000 (UTC) Authentication-Results: smtp.subspace.kernel.org; arc=none smtp.client-ip=209.85.218.54 ARC-Seal: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1764229598; cv=none; b=Ru4aoxOMG6OyDpHm4fbib42Axy0nU7uQQGt9NWmX2qmbPQB3oOS9Xo0fD1WGbq2E2P7/ZwwbY87k8Luhha4mcPwbjVKDvVpEMa5BQoM3sN1yVkAQnBoNZJW0IQ379G24Dy7puWBbm3at+K7uZ59vk7bHDcSFSTA3RaALt7iOiUs= ARC-Message-Signature: i=1; a=rsa-sha256; d=subspace.kernel.org; s=arc-20240116; t=1764229598; c=relaxed/simple; bh=Wm3ZhBOgwnlfY2W5VGIWg1AKkxdXhR9gwKIvXq0Myc0=; h=From:To:Cc:Subject:Date:Message-ID:MIME-Version; b=GTFMvbqLQt4uuAY8T3hPS0S9rUrQD9dn4XluuMbBBTKcOa9xAnAUyAn24C00pS3GJgQsFRdc7V77XWnhFE6bey9Vmhb3g34iQxa6cHom5Yq5BKW2fgSLuPVJFVmpcXvof4KMORGX8Igg2fbh1V29Qqq7T9zDpPGIdtCRWlyCgqY= 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=GV1+3zgH; arc=none smtp.client-ip=209.85.218.54 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="GV1+3zgH" Received: by mail-ej1-f54.google.com with SMTP id a640c23a62f3a-b7380f66a8bso90161766b.2 for ; Wed, 26 Nov 2025 23:46:36 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=amarulasolutions.com; s=google; t=1764229594; x=1764834394; 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=I57iSPGSJdvo4hExZdg2/yXzMQCRNalZqIfRbvKW8Ek=; b=GV1+3zgHKZy8pzwvaIdmyUULRNouPCskMux1saxhSwg0b/pUr1AKg1YB8tGsuN9L+u xzQccoC7e6XO6YW6QA0tHHZklj3oFtv8ku+zRdPeI8SQe0zf4vkr+uOew1UtZmyGvUnq OAH9UHSBbuJZ0/6Qdtcacx/8g+96cvdfaCOzM= X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20230601; t=1764229594; x=1764834394; h=content-transfer-encoding:mime-version:message-id:date:subject:cc :to:from:x-gm-gg:x-gm-message-state:from:to:cc:subject:date :message-id:reply-to; bh=I57iSPGSJdvo4hExZdg2/yXzMQCRNalZqIfRbvKW8Ek=; b=fqlsb1Mqh7cEwiTy/UTGV1JiY4KmbNXm6ZHUvc9gmEzXo0G6yzoCFLWyGMLKhAS6PC EFaFeGDHZByYHifSwK0wKjijSYpiMljRkgyHCtxcJXAZ2Scp16HSTivzu9Io7PbC2vWY KlyKu0NLX7mjLN1eavrgHZQ6E6MuY1G1cy8xMSM6Hlv3YoOZsnJst42lchu9i4xzHoNE T9k8OcNe9fDoE9yOkpB8oKLYeah1TwF0FBf+Y8fCZ02DNZwZzLI6GDU7NTD3wR2ONwmh CjKnNFhPK8SIZcC6Lmcd/Gcdpg5QKjHaQAgEvdiGzyMDQgEyuEnF+vcGm/JEqt7FhCFy uakw== X-Gm-Message-State: AOJu0YzGzDu+Qutr+fJA1k7F1LvdctusnG2qNEgQPfSyX10mMANJ/bPA QxzwL3a1HjNyQ/sdLsCqNTJ40ElxzJmQ+p0OXCTSR8e3nPokB2XEVZ9tWbJaM3CaowS+dxSmo2K dT/QR X-Gm-Gg: ASbGncvPYGTcBF5OXZyfmBC6NAifUS48g9g54uGTwzUc3iwMfXncDAOjuEdsXZAVWyr XLOQIaa0Yudj+9tlqlp9m+KvszdyiNsKKhcvvJ7oKhWqcNtGTGxfy1Pl1OUZhOUpscl0JAIfyxv 2na07fGjmkRJfSqKo5cCZDgI/g8B8FjWb9UKG8E/TIVrFwCT0jE5mLjM+lwpknMD0zU6VWSAXos AZEHHp094ZqJQCSWXKa+QdEpGVk326W0Lc3ARrUCpf3Llk+hT04JdfGkWZlK9JOLZp3GIstg/Yo +PdXL1GOHXSqrYli7RF0LVN2hPPTVGTus93arKIufxpXkmKzMJywc6p+CfowZrxran+P0SgKe5B S5J6EUrPxU/PBHuipjhjFopWRSmGpJ0yUeRU4l39VV2npWrJIxUOSZeiqy3ONoJE1Z/5Tvs4qza RMBGe4sMapl0DOFgnMHssXOhQAAItMJRz7CxnyYxI6EifdnC9ozbKkN0fnAVjug8h4ZzrY2MxBa wodmWuX93EPS4GgWXicnKzmj3qiLCHTskwhQUoHW+yuFH0= X-Google-Smtp-Source: AGHT+IEgx3bglb1EkfZgShJUGyqy+Y6wzWQac9BwM5RQzPqtJgXdrFY9KX/BK/jMNa+Dg7A1opAdfw== X-Received: by 2002:a17:907:26c2:b0:b4e:f7cc:72f1 with SMTP id a640c23a62f3a-b76715aba47mr2280426566b.22.1764229594426; Wed, 26 Nov 2025 23:46:34 -0800 (PST) Received: from dario-ThinkPad-P14s-Gen-5.amarulasolutions.com (host-87-4-36-237.retail.telecomitalia.it. [87.4.36.237]) by smtp.gmail.com with ESMTPSA id a640c23a62f3a-b76f51c55e9sm91435666b.26.2025.11.26.23.46.33 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Wed, 26 Nov 2025 23:46:34 -0800 (PST) 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: Thu, 27 Nov 2025 08:46:21 +0100 Message-ID: <20251127074626.1953741-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: 765e56e41a5af2d456ddda6cbd617b9d3295ab4e branch: edt-ft5x06-fix-report-rate