// SPDX-License-Identifier: GPL-2.0 /* * adv_swbutton.c - Software Button Interface Driver. * * (C) Copyright 2020 Advantech Corporation, Inc * */ #include #include #include #include #include #define ACPI_BUTTON_HID_SWBTN "AHC0310" #define ACPI_BUTTON_NOTIFY_SWBTN_RELEASE 0x86 #define ACPI_BUTTON_NOTIFY_SWBTN_PRESSED 0x85 struct adv_swbutton { struct input_dev *input; char phys[32]; }; /*------------------------------------------------------------------------- * Driver Interface *-------------------------------------------------------------------------- */ static void adv_swbutton_notify(acpi_handle handle, u32 event, void *context) { struct platform_device *device = context; struct adv_swbutton *button = dev_get_drvdata(&device->dev); switch (event) { case ACPI_BUTTON_NOTIFY_SWBTN_RELEASE: input_report_key(button->input, KEY_PROG1, 0); input_sync(button->input); break; case ACPI_BUTTON_NOTIFY_SWBTN_PRESSED: input_report_key(button->input, KEY_PROG1, 1); input_sync(button->input); break; default: dev_dbg(&device->dev, "Unsupported event [0x%x]\n", event); } } static int adv_swbutton_probe(struct platform_device *device) { struct adv_swbutton *button; struct input_dev *input; acpi_handle handle = ACPI_HANDLE(&device->dev); acpi_status status; int error; button = devm_kzalloc(&device->dev, sizeof(*button), GFP_KERNEL); if (!button) return -ENOMEM; dev_set_drvdata(&device->dev, button); input = devm_input_allocate_device(&device->dev); if (!input) return -ENOMEM; button->input = input; snprintf(button->phys, sizeof(button->phys), "%s/button/input0", ACPI_BUTTON_HID_SWBTN); input->name = "Advantech Software Button"; input->phys = button->phys; input->id.bustype = BUS_HOST; input->dev.parent = &device->dev; set_bit(EV_REP, input->evbit); input_set_capability(input, EV_KEY, KEY_PROG1); error = input_register_device(input); if (error) return error; device_init_wakeup(&device->dev, true); status = acpi_install_notify_handler(handle, ACPI_DEVICE_NOTIFY, adv_swbutton_notify, device); if (ACPI_FAILURE(status)) { dev_err(&device->dev, "Error installing notify handler\n"); return -EIO; } return 0; } static void adv_swbutton_remove(struct platform_device *device) { acpi_handle handle = ACPI_HANDLE(&device->dev); acpi_remove_notify_handler(handle, ACPI_DEVICE_NOTIFY, adv_swbutton_notify); } static const struct acpi_device_id button_device_ids[] = { {ACPI_BUTTON_HID_SWBTN, 0}, {"", 0}, }; MODULE_DEVICE_TABLE(acpi, button_device_ids); static struct platform_driver adv_swbutton_driver = { .driver = { .name = "adv_swbutton", .acpi_match_table = button_device_ids, }, .probe = adv_swbutton_probe, .remove_new = adv_swbutton_remove, }; module_platform_driver(adv_swbutton_driver); MODULE_AUTHOR("Andrea Ho"); MODULE_DESCRIPTION("Advantech ACPI SW Button Driver"); MODULE_LICENSE("GPL v2");