Commit 034b8c2e authored by Hans de Goede's avatar Hans de Goede
Browse files

platform/x86: intel-vbtn: Create 2 separate input-devs for buttons and switches



Create 2 separate input-devs for buttons and switches, this is a
preparation for dynamically registering the switches-input device
for devices which are not on the switches allow-list, but do make
Notify() calls with an event value from the switches sparse-keymap.

This also brings the intel-vbtn driver inline with the intel-hid
driver which is doing the same thing.

Cc: Elia Devito <eliadevito@gmail.com>
Signed-off-by: default avatarHans de Goede <hdegoede@redhat.com>
Link: https://lore.kernel.org/r/20210115161850.117614-2-hdegoede@redhat.com
parent 5862b4df
Loading
Loading
Loading
Loading
+62 −36
Original line number Original line Diff line number Diff line
@@ -44,6 +44,7 @@ static const struct key_entry intel_vbtn_keymap[] = {
	{ KE_IGNORE, 0xC7, { KEY_VOLUMEDOWN } },	/* volume-down key release */
	{ KE_IGNORE, 0xC7, { KEY_VOLUMEDOWN } },	/* volume-down key release */
	{ KE_KEY,    0xC8, { KEY_ROTATE_LOCK_TOGGLE } },	/* rotate-lock key press */
	{ KE_KEY,    0xC8, { KEY_ROTATE_LOCK_TOGGLE } },	/* rotate-lock key press */
	{ KE_KEY,    0xC9, { KEY_ROTATE_LOCK_TOGGLE } },	/* rotate-lock key release */
	{ KE_KEY,    0xC9, { KEY_ROTATE_LOCK_TOGGLE } },	/* rotate-lock key release */
	{ KE_END }
};
};


static const struct key_entry intel_vbtn_switchmap[] = {
static const struct key_entry intel_vbtn_switchmap[] = {
@@ -51,14 +52,15 @@ static const struct key_entry intel_vbtn_switchmap[] = {
	{ KE_SW,     0xCB, { .sw = { SW_DOCK, 0 } } },		/* Undocked */
	{ KE_SW,     0xCB, { .sw = { SW_DOCK, 0 } } },		/* Undocked */
	{ KE_SW,     0xCC, { .sw = { SW_TABLET_MODE, 1 } } },	/* Tablet */
	{ KE_SW,     0xCC, { .sw = { SW_TABLET_MODE, 1 } } },	/* Tablet */
	{ KE_SW,     0xCD, { .sw = { SW_TABLET_MODE, 0 } } },	/* Laptop */
	{ KE_SW,     0xCD, { .sw = { SW_TABLET_MODE, 0 } } },	/* Laptop */
	{ KE_END }
};
};


#define KEYMAP_LEN \
#define KEYMAP_LEN \
	(ARRAY_SIZE(intel_vbtn_keymap) + ARRAY_SIZE(intel_vbtn_switchmap) + 1)
	(ARRAY_SIZE(intel_vbtn_keymap) + ARRAY_SIZE(intel_vbtn_switchmap) + 1)


struct intel_vbtn_priv {
struct intel_vbtn_priv {
	struct key_entry keymap[KEYMAP_LEN];
	struct input_dev *buttons_dev;
	struct input_dev *input_dev;
	struct input_dev *switches_dev;
	bool has_buttons;
	bool has_buttons;
	bool has_switches;
	bool has_switches;
	bool wakeup_mode;
	bool wakeup_mode;
@@ -77,48 +79,62 @@ static void detect_tablet_mode(struct platform_device *device)
		return;
		return;


	m = !(vgbs & VGBS_TABLET_MODE_FLAGS);
	m = !(vgbs & VGBS_TABLET_MODE_FLAGS);
	input_report_switch(priv->input_dev, SW_TABLET_MODE, m);
	input_report_switch(priv->switches_dev, SW_TABLET_MODE, m);
	m = (vgbs & VGBS_DOCK_MODE_FLAG) ? 1 : 0;
	m = (vgbs & VGBS_DOCK_MODE_FLAG) ? 1 : 0;
	input_report_switch(priv->input_dev, SW_DOCK, m);
	input_report_switch(priv->switches_dev, SW_DOCK, m);
}
}


/*
 * Note this unconditionally creates the 2 input_dev-s and sets up
 * the sparse-keymaps. Only the registration is conditional on
 * have_buttons / have_switches. This is done so that the notify
 * handler can always call sparse_keymap_entry_from_scancode()
 * on the input_dev-s do determine the event type.
 */
static int intel_vbtn_input_setup(struct platform_device *device)
static int intel_vbtn_input_setup(struct platform_device *device)
{
{
	struct intel_vbtn_priv *priv = dev_get_drvdata(&device->dev);
	struct intel_vbtn_priv *priv = dev_get_drvdata(&device->dev);
	int ret, keymap_len = 0;
	int ret;


	if (priv->has_buttons) {
	priv->buttons_dev = devm_input_allocate_device(&device->dev);
		memcpy(&priv->keymap[keymap_len], intel_vbtn_keymap,
	if (!priv->buttons_dev)
		       ARRAY_SIZE(intel_vbtn_keymap) *
		return -ENOMEM;
		       sizeof(struct key_entry));
		keymap_len += ARRAY_SIZE(intel_vbtn_keymap);
	}


	if (priv->has_switches) {
	ret = sparse_keymap_setup(priv->buttons_dev, intel_vbtn_keymap, NULL);
		memcpy(&priv->keymap[keymap_len], intel_vbtn_switchmap,
	if (ret)
		       ARRAY_SIZE(intel_vbtn_switchmap) *
		return ret;
		       sizeof(struct key_entry));

		keymap_len += ARRAY_SIZE(intel_vbtn_switchmap);
	priv->buttons_dev->dev.parent = &device->dev;
	}
	priv->buttons_dev->name = "Intel Virtual Buttons";
	priv->buttons_dev->id.bustype = BUS_HOST;


	priv->keymap[keymap_len].type = KE_END;
	if (priv->has_buttons) {
		ret = input_register_device(priv->buttons_dev);
		if (ret)
			return ret;
	}


	priv->input_dev = devm_input_allocate_device(&device->dev);
	priv->switches_dev = devm_input_allocate_device(&device->dev);
	if (!priv->input_dev)
	if (!priv->switches_dev)
		return -ENOMEM;
		return -ENOMEM;


	ret = sparse_keymap_setup(priv->input_dev, priv->keymap, NULL);
	ret = sparse_keymap_setup(priv->switches_dev, intel_vbtn_switchmap, NULL);
	if (ret)
	if (ret)
		return ret;
		return ret;


	priv->input_dev->dev.parent = &device->dev;
	priv->switches_dev->dev.parent = &device->dev;
	priv->input_dev->name = "Intel Virtual Button driver";
	priv->switches_dev->name = "Intel Virtual Switches";
	priv->input_dev->id.bustype = BUS_HOST;
	priv->switches_dev->id.bustype = BUS_HOST;


	if (priv->has_switches)
	if (priv->has_switches) {
		detect_tablet_mode(device);
		detect_tablet_mode(device);


	return input_register_device(priv->input_dev);
		ret = input_register_device(priv->switches_dev);
		if (ret)
			return ret;
	}

	return 0;
}
}


static void notify_handler(acpi_handle handle, u32 event, void *context)
static void notify_handler(acpi_handle handle, u32 event, void *context)
@@ -127,13 +143,27 @@ static void notify_handler(acpi_handle handle, u32 event, void *context)
	struct intel_vbtn_priv *priv = dev_get_drvdata(&device->dev);
	struct intel_vbtn_priv *priv = dev_get_drvdata(&device->dev);
	unsigned int val = !(event & 1); /* Even=press, Odd=release */
	unsigned int val = !(event & 1); /* Even=press, Odd=release */
	const struct key_entry *ke, *ke_rel;
	const struct key_entry *ke, *ke_rel;
	struct input_dev *input_dev;
	bool autorelease;
	bool autorelease;


	if (priv->wakeup_mode) {
	if ((ke = sparse_keymap_entry_from_scancode(priv->buttons_dev, event))) {
		ke = sparse_keymap_entry_from_scancode(priv->input_dev, event);
		if (!priv->has_buttons) {
		if (!ke)
			dev_warn(&device->dev, "Warning: received a button event on a device without buttons, please report this.\n");
			goto out_unknown;
			return;
		}
		input_dev = priv->buttons_dev;
	} else if ((ke = sparse_keymap_entry_from_scancode(priv->switches_dev, event))) {
		if (!priv->has_switches) {
			dev_warn(&device->dev, "Warning: received a switches event on a device without switchess, please report this.\n");
			return;
		}
		input_dev = priv->switches_dev;
	} else {
		dev_dbg(&device->dev, "unknown event index 0x%x\n", event);
		return;
	}


	if (priv->wakeup_mode) {
		pm_wakeup_hard_event(&device->dev);
		pm_wakeup_hard_event(&device->dev);


		/*
		/*
@@ -148,14 +178,10 @@ static void notify_handler(acpi_handle handle, u32 event, void *context)
	 * Even press events are autorelease if there is no corresponding odd
	 * Even press events are autorelease if there is no corresponding odd
	 * release event, or if the odd event is KE_IGNORE.
	 * release event, or if the odd event is KE_IGNORE.
	 */
	 */
	ke_rel = sparse_keymap_entry_from_scancode(priv->input_dev, event | 1);
	ke_rel = sparse_keymap_entry_from_scancode(input_dev, event | 1);
	autorelease = val && (!ke_rel || ke_rel->type == KE_IGNORE);
	autorelease = val && (!ke_rel || ke_rel->type == KE_IGNORE);


	if (sparse_keymap_report_event(priv->input_dev, event, val, autorelease))
	sparse_keymap_report_event(input_dev, event, val, autorelease);
		return;

out_unknown:
	dev_dbg(&device->dev, "unknown event index 0x%x\n", event);
}
}


static bool intel_vbtn_has_buttons(acpi_handle handle)
static bool intel_vbtn_has_buttons(acpi_handle handle)