mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
synced 2025-08-31 14:13:39 +00:00

RMI4 F1A implements capacitive keys. Add support for touch keys found in some Synaptics touch controller configurations. Signed-off-by: André Apitzsch <git@apitzsch.eu> Link: https://lore.kernel.org/r/20250707-rmi4_f1a-v1-2-838d83c72e7f@apitzsch.eu Signed-off-by: Dmitry Torokhov <dmitry.torokhov@gmail.com>
144 lines
3.1 KiB
C
144 lines
3.1 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* Copyright (c) 2025 André Apitzsch <git@apitzsch.eu>
|
|
*/
|
|
|
|
#include <linux/input.h>
|
|
#include <linux/property.h>
|
|
#include "rmi_driver.h"
|
|
|
|
struct f1a_data {
|
|
struct input_dev *input;
|
|
|
|
u32 *keymap;
|
|
unsigned int num_keys;
|
|
};
|
|
|
|
static int rmi_f1a_parse_device_properties(struct rmi_function *fn, struct f1a_data *f1a)
|
|
{
|
|
static const char buttons_property[] = "linux,keycodes";
|
|
struct device *dev = &fn->dev;
|
|
u32 *buttonmap;
|
|
int n_keys;
|
|
int error;
|
|
|
|
if (!device_property_present(dev, buttons_property))
|
|
return 0;
|
|
|
|
n_keys = device_property_count_u32(dev, buttons_property);
|
|
if (n_keys <= 0) {
|
|
error = n_keys < 0 ? n_keys : -EINVAL;
|
|
dev_err(dev, "Invalid/malformed '%s' property: %d\n",
|
|
buttons_property, error);
|
|
return error;
|
|
}
|
|
|
|
buttonmap = devm_kmalloc_array(dev, n_keys, sizeof(*buttonmap),
|
|
GFP_KERNEL);
|
|
if (!buttonmap)
|
|
return -ENOMEM;
|
|
|
|
error = device_property_read_u32_array(dev, buttons_property,
|
|
buttonmap, n_keys);
|
|
if (error) {
|
|
dev_err(dev, "Failed to parse '%s' property: %d\n",
|
|
buttons_property, error);
|
|
return error;
|
|
}
|
|
|
|
f1a->keymap = buttonmap;
|
|
f1a->num_keys = n_keys;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static irqreturn_t rmi_f1a_attention(int irq, void *ctx)
|
|
{
|
|
struct rmi_function *fn = ctx;
|
|
struct f1a_data *f1a = dev_get_drvdata(&fn->dev);
|
|
char button_bitmask;
|
|
int key;
|
|
int error;
|
|
|
|
error = rmi_read_block(fn->rmi_dev, fn->fd.data_base_addr,
|
|
&button_bitmask, sizeof(button_bitmask));
|
|
if (error) {
|
|
dev_err(&fn->dev, "Failed to read object data. Code: %d.\n",
|
|
error);
|
|
return IRQ_RETVAL(error);
|
|
}
|
|
|
|
for (key = 0; key < f1a->num_keys; key++)
|
|
input_report_key(f1a->input, f1a->keymap[key],
|
|
button_bitmask & BIT(key));
|
|
|
|
return IRQ_HANDLED;
|
|
}
|
|
|
|
static int rmi_f1a_config(struct rmi_function *fn)
|
|
{
|
|
struct f1a_data *f1a = dev_get_drvdata(&fn->dev);
|
|
struct rmi_driver *drv = fn->rmi_dev->driver;
|
|
|
|
if (f1a->num_keys)
|
|
drv->set_irq_bits(fn->rmi_dev, fn->irq_mask);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int rmi_f1a_initialize(struct rmi_function *fn, struct f1a_data *f1a)
|
|
{
|
|
int error;
|
|
int i;
|
|
|
|
error = rmi_f1a_parse_device_properties(fn, f1a);
|
|
if (error)
|
|
return error;
|
|
|
|
for (i = 0; i < f1a->num_keys; i++)
|
|
input_set_capability(f1a->input, EV_KEY, f1a->keymap[i]);
|
|
|
|
f1a->input->keycode = f1a->keymap;
|
|
f1a->input->keycodemax = f1a->num_keys;
|
|
f1a->input->keycodesize = sizeof(f1a->keymap[0]);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int rmi_f1a_probe(struct rmi_function *fn)
|
|
{
|
|
struct rmi_device *rmi_dev = fn->rmi_dev;
|
|
struct rmi_driver_data *drv_data = dev_get_drvdata(&rmi_dev->dev);
|
|
struct f1a_data *f1a;
|
|
int error;
|
|
|
|
if (!drv_data->input) {
|
|
dev_info(&fn->dev, "F1A: no input device found, ignoring\n");
|
|
return -ENXIO;
|
|
}
|
|
|
|
f1a = devm_kzalloc(&fn->dev, sizeof(*f1a), GFP_KERNEL);
|
|
if (!f1a)
|
|
return -ENOMEM;
|
|
|
|
f1a->input = drv_data->input;
|
|
|
|
error = rmi_f1a_initialize(fn, f1a);
|
|
if (error)
|
|
return error;
|
|
|
|
dev_set_drvdata(&fn->dev, f1a);
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct rmi_function_handler rmi_f1a_handler = {
|
|
.driver = {
|
|
.name = "rmi4_f1a",
|
|
},
|
|
.func = 0x1a,
|
|
.probe = rmi_f1a_probe,
|
|
.config = rmi_f1a_config,
|
|
.attention = rmi_f1a_attention,
|
|
};
|