123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817 |
- // SPDX-License-Identifier: GPL-2.0+
- /*
- * lg-laptop.c - LG Gram ACPI features and hotkeys Driver
- *
- * Copyright (C) 2018 Matan Ziv-Av <[email protected]>
- */
- #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
- #include <linux/acpi.h>
- #include <linux/dmi.h>
- #include <linux/input.h>
- #include <linux/input/sparse-keymap.h>
- #include <linux/kernel.h>
- #include <linux/leds.h>
- #include <linux/module.h>
- #include <linux/platform_device.h>
- #include <linux/types.h>
- #include <acpi/battery.h>
- #define LED_DEVICE(_name, max, flag) struct led_classdev _name = { \
- .name = __stringify(_name), \
- .max_brightness = max, \
- .brightness_set = _name##_set, \
- .brightness_get = _name##_get, \
- .flags = flag, \
- }
- MODULE_AUTHOR("Matan Ziv-Av");
- MODULE_DESCRIPTION("LG WMI Hotkey Driver");
- MODULE_LICENSE("GPL");
- #define WMI_EVENT_GUID0 "E4FB94F9-7F2B-4173-AD1A-CD1D95086248"
- #define WMI_EVENT_GUID1 "023B133E-49D1-4E10-B313-698220140DC2"
- #define WMI_EVENT_GUID2 "37BE1AC0-C3F2-4B1F-BFBE-8FDEAF2814D6"
- #define WMI_EVENT_GUID3 "911BAD44-7DF8-4FBB-9319-BABA1C4B293B"
- #define WMI_METHOD_WMAB "C3A72B38-D3EF-42D3-8CBB-D5A57049F66D"
- #define WMI_METHOD_WMBB "2B4F501A-BD3C-4394-8DCF-00A7D2BC8210"
- #define WMI_EVENT_GUID WMI_EVENT_GUID0
- #define WMAB_METHOD "\\XINI.WMAB"
- #define WMBB_METHOD "\\XINI.WMBB"
- #define SB_GGOV_METHOD "\\_SB.GGOV"
- #define GOV_TLED 0x2020008
- #define WM_GET 1
- #define WM_SET 2
- #define WM_KEY_LIGHT 0x400
- #define WM_TLED 0x404
- #define WM_FN_LOCK 0x407
- #define WM_BATT_LIMIT 0x61
- #define WM_READER_MODE 0xBF
- #define WM_FAN_MODE 0x33
- #define WMBB_USB_CHARGE 0x10B
- #define WMBB_BATT_LIMIT 0x10C
- #define PLATFORM_NAME "lg-laptop"
- MODULE_ALIAS("wmi:" WMI_EVENT_GUID0);
- MODULE_ALIAS("wmi:" WMI_EVENT_GUID1);
- MODULE_ALIAS("wmi:" WMI_EVENT_GUID2);
- MODULE_ALIAS("wmi:" WMI_EVENT_GUID3);
- MODULE_ALIAS("wmi:" WMI_METHOD_WMAB);
- MODULE_ALIAS("wmi:" WMI_METHOD_WMBB);
- static struct platform_device *pf_device;
- static struct input_dev *wmi_input_dev;
- static u32 inited;
- #define INIT_INPUT_WMI_0 0x01
- #define INIT_INPUT_WMI_2 0x02
- #define INIT_INPUT_ACPI 0x04
- #define INIT_SPARSE_KEYMAP 0x80
- static int battery_limit_use_wmbb;
- static struct led_classdev kbd_backlight;
- static enum led_brightness get_kbd_backlight_level(void);
- static const struct key_entry wmi_keymap[] = {
- {KE_KEY, 0x70, {KEY_F15} }, /* LG control panel (F1) */
- {KE_KEY, 0x74, {KEY_F21} }, /* Touchpad toggle (F5) */
- {KE_KEY, 0xf020000, {KEY_F14} }, /* Read mode (F9) */
- {KE_KEY, 0x10000000, {KEY_F16} },/* Keyboard backlight (F8) - pressing
- * this key both sends an event and
- * changes backlight level.
- */
- {KE_KEY, 0x80, {KEY_RFKILL} },
- {KE_END, 0}
- };
- static int ggov(u32 arg0)
- {
- union acpi_object args[1];
- union acpi_object *r;
- acpi_status status;
- acpi_handle handle;
- struct acpi_object_list arg;
- struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL };
- int res;
- args[0].type = ACPI_TYPE_INTEGER;
- args[0].integer.value = arg0;
- status = acpi_get_handle(NULL, (acpi_string) SB_GGOV_METHOD, &handle);
- if (ACPI_FAILURE(status)) {
- pr_err("Cannot get handle");
- return -ENODEV;
- }
- arg.count = 1;
- arg.pointer = args;
- status = acpi_evaluate_object(handle, NULL, &arg, &buffer);
- if (ACPI_FAILURE(status)) {
- acpi_handle_err(handle, "GGOV: call failed.\n");
- return -EINVAL;
- }
- r = buffer.pointer;
- if (r->type != ACPI_TYPE_INTEGER) {
- kfree(r);
- return -EINVAL;
- }
- res = r->integer.value;
- kfree(r);
- return res;
- }
- static union acpi_object *lg_wmab(u32 method, u32 arg1, u32 arg2)
- {
- union acpi_object args[3];
- acpi_status status;
- acpi_handle handle;
- struct acpi_object_list arg;
- struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL };
- args[0].type = ACPI_TYPE_INTEGER;
- args[0].integer.value = method;
- args[1].type = ACPI_TYPE_INTEGER;
- args[1].integer.value = arg1;
- args[2].type = ACPI_TYPE_INTEGER;
- args[2].integer.value = arg2;
- status = acpi_get_handle(NULL, (acpi_string) WMAB_METHOD, &handle);
- if (ACPI_FAILURE(status)) {
- pr_err("Cannot get handle");
- return NULL;
- }
- arg.count = 3;
- arg.pointer = args;
- status = acpi_evaluate_object(handle, NULL, &arg, &buffer);
- if (ACPI_FAILURE(status)) {
- acpi_handle_err(handle, "WMAB: call failed.\n");
- return NULL;
- }
- return buffer.pointer;
- }
- static union acpi_object *lg_wmbb(u32 method_id, u32 arg1, u32 arg2)
- {
- union acpi_object args[3];
- acpi_status status;
- acpi_handle handle;
- struct acpi_object_list arg;
- struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL };
- u8 buf[32];
- *(u32 *)buf = method_id;
- *(u32 *)(buf + 4) = arg1;
- *(u32 *)(buf + 16) = arg2;
- args[0].type = ACPI_TYPE_INTEGER;
- args[0].integer.value = 0; /* ignored */
- args[1].type = ACPI_TYPE_INTEGER;
- args[1].integer.value = 1; /* Must be 1 or 2. Does not matter which */
- args[2].type = ACPI_TYPE_BUFFER;
- args[2].buffer.length = 32;
- args[2].buffer.pointer = buf;
- status = acpi_get_handle(NULL, (acpi_string)WMBB_METHOD, &handle);
- if (ACPI_FAILURE(status)) {
- pr_err("Cannot get handle");
- return NULL;
- }
- arg.count = 3;
- arg.pointer = args;
- status = acpi_evaluate_object(handle, NULL, &arg, &buffer);
- if (ACPI_FAILURE(status)) {
- acpi_handle_err(handle, "WMAB: call failed.\n");
- return NULL;
- }
- return (union acpi_object *)buffer.pointer;
- }
- static void wmi_notify(u32 value, void *context)
- {
- struct acpi_buffer response = { ACPI_ALLOCATE_BUFFER, NULL };
- union acpi_object *obj;
- acpi_status status;
- long data = (long)context;
- pr_debug("event guid %li\n", data);
- status = wmi_get_event_data(value, &response);
- if (ACPI_FAILURE(status)) {
- pr_err("Bad event status 0x%x\n", status);
- return;
- }
- obj = (union acpi_object *)response.pointer;
- if (!obj)
- return;
- if (obj->type == ACPI_TYPE_INTEGER) {
- int eventcode = obj->integer.value;
- struct key_entry *key;
- if (eventcode == 0x10000000) {
- led_classdev_notify_brightness_hw_changed(
- &kbd_backlight, get_kbd_backlight_level());
- } else {
- key = sparse_keymap_entry_from_scancode(
- wmi_input_dev, eventcode);
- if (key && key->type == KE_KEY)
- sparse_keymap_report_entry(wmi_input_dev,
- key, 1, true);
- }
- }
- pr_debug("Type: %i Eventcode: 0x%llx\n", obj->type,
- obj->integer.value);
- kfree(response.pointer);
- }
- static void wmi_input_setup(void)
- {
- acpi_status status;
- wmi_input_dev = input_allocate_device();
- if (wmi_input_dev) {
- wmi_input_dev->name = "LG WMI hotkeys";
- wmi_input_dev->phys = "wmi/input0";
- wmi_input_dev->id.bustype = BUS_HOST;
- if (sparse_keymap_setup(wmi_input_dev, wmi_keymap, NULL) ||
- input_register_device(wmi_input_dev)) {
- pr_info("Cannot initialize input device");
- input_free_device(wmi_input_dev);
- return;
- }
- inited |= INIT_SPARSE_KEYMAP;
- status = wmi_install_notify_handler(WMI_EVENT_GUID0, wmi_notify,
- (void *)0);
- if (ACPI_SUCCESS(status))
- inited |= INIT_INPUT_WMI_0;
- status = wmi_install_notify_handler(WMI_EVENT_GUID2, wmi_notify,
- (void *)2);
- if (ACPI_SUCCESS(status))
- inited |= INIT_INPUT_WMI_2;
- } else {
- pr_info("Cannot allocate input device");
- }
- }
- static void acpi_notify(struct acpi_device *device, u32 event)
- {
- struct key_entry *key;
- acpi_handle_debug(device->handle, "notify: %d\n", event);
- if (inited & INIT_SPARSE_KEYMAP) {
- key = sparse_keymap_entry_from_scancode(wmi_input_dev, 0x80);
- if (key && key->type == KE_KEY)
- sparse_keymap_report_entry(wmi_input_dev, key, 1, true);
- }
- }
- static ssize_t fan_mode_store(struct device *dev,
- struct device_attribute *attr,
- const char *buffer, size_t count)
- {
- bool value;
- union acpi_object *r;
- u32 m;
- int ret;
- ret = kstrtobool(buffer, &value);
- if (ret)
- return ret;
- r = lg_wmab(WM_FAN_MODE, WM_GET, 0);
- if (!r)
- return -EIO;
- if (r->type != ACPI_TYPE_INTEGER) {
- kfree(r);
- return -EIO;
- }
- m = r->integer.value;
- kfree(r);
- r = lg_wmab(WM_FAN_MODE, WM_SET, (m & 0xffffff0f) | (value << 4));
- kfree(r);
- r = lg_wmab(WM_FAN_MODE, WM_SET, (m & 0xfffffff0) | value);
- kfree(r);
- return count;
- }
- static ssize_t fan_mode_show(struct device *dev,
- struct device_attribute *attr, char *buffer)
- {
- unsigned int status;
- union acpi_object *r;
- r = lg_wmab(WM_FAN_MODE, WM_GET, 0);
- if (!r)
- return -EIO;
- if (r->type != ACPI_TYPE_INTEGER) {
- kfree(r);
- return -EIO;
- }
- status = r->integer.value & 0x01;
- kfree(r);
- return sysfs_emit(buffer, "%d\n", status);
- }
- static ssize_t usb_charge_store(struct device *dev,
- struct device_attribute *attr,
- const char *buffer, size_t count)
- {
- bool value;
- union acpi_object *r;
- int ret;
- ret = kstrtobool(buffer, &value);
- if (ret)
- return ret;
- r = lg_wmbb(WMBB_USB_CHARGE, WM_SET, value);
- if (!r)
- return -EIO;
- kfree(r);
- return count;
- }
- static ssize_t usb_charge_show(struct device *dev,
- struct device_attribute *attr, char *buffer)
- {
- unsigned int status;
- union acpi_object *r;
- r = lg_wmbb(WMBB_USB_CHARGE, WM_GET, 0);
- if (!r)
- return -EIO;
- if (r->type != ACPI_TYPE_BUFFER) {
- kfree(r);
- return -EIO;
- }
- status = !!r->buffer.pointer[0x10];
- kfree(r);
- return sysfs_emit(buffer, "%d\n", status);
- }
- static ssize_t reader_mode_store(struct device *dev,
- struct device_attribute *attr,
- const char *buffer, size_t count)
- {
- bool value;
- union acpi_object *r;
- int ret;
- ret = kstrtobool(buffer, &value);
- if (ret)
- return ret;
- r = lg_wmab(WM_READER_MODE, WM_SET, value);
- if (!r)
- return -EIO;
- kfree(r);
- return count;
- }
- static ssize_t reader_mode_show(struct device *dev,
- struct device_attribute *attr, char *buffer)
- {
- unsigned int status;
- union acpi_object *r;
- r = lg_wmab(WM_READER_MODE, WM_GET, 0);
- if (!r)
- return -EIO;
- if (r->type != ACPI_TYPE_INTEGER) {
- kfree(r);
- return -EIO;
- }
- status = !!r->integer.value;
- kfree(r);
- return sysfs_emit(buffer, "%d\n", status);
- }
- static ssize_t fn_lock_store(struct device *dev,
- struct device_attribute *attr,
- const char *buffer, size_t count)
- {
- bool value;
- union acpi_object *r;
- int ret;
- ret = kstrtobool(buffer, &value);
- if (ret)
- return ret;
- r = lg_wmab(WM_FN_LOCK, WM_SET, value);
- if (!r)
- return -EIO;
- kfree(r);
- return count;
- }
- static ssize_t fn_lock_show(struct device *dev,
- struct device_attribute *attr, char *buffer)
- {
- unsigned int status;
- union acpi_object *r;
- r = lg_wmab(WM_FN_LOCK, WM_GET, 0);
- if (!r)
- return -EIO;
- if (r->type != ACPI_TYPE_BUFFER) {
- kfree(r);
- return -EIO;
- }
- status = !!r->buffer.pointer[0];
- kfree(r);
- return sysfs_emit(buffer, "%d\n", status);
- }
- static ssize_t charge_control_end_threshold_store(struct device *dev,
- struct device_attribute *attr,
- const char *buf, size_t count)
- {
- unsigned long value;
- int ret;
- ret = kstrtoul(buf, 10, &value);
- if (ret)
- return ret;
- if (value == 100 || value == 80) {
- union acpi_object *r;
- if (battery_limit_use_wmbb)
- r = lg_wmbb(WMBB_BATT_LIMIT, WM_SET, value);
- else
- r = lg_wmab(WM_BATT_LIMIT, WM_SET, value);
- if (!r)
- return -EIO;
- kfree(r);
- return count;
- }
- return -EINVAL;
- }
- static ssize_t charge_control_end_threshold_show(struct device *device,
- struct device_attribute *attr,
- char *buf)
- {
- unsigned int status;
- union acpi_object *r;
- if (battery_limit_use_wmbb) {
- r = lg_wmbb(WMBB_BATT_LIMIT, WM_GET, 0);
- if (!r)
- return -EIO;
- if (r->type != ACPI_TYPE_BUFFER) {
- kfree(r);
- return -EIO;
- }
- status = r->buffer.pointer[0x10];
- } else {
- r = lg_wmab(WM_BATT_LIMIT, WM_GET, 0);
- if (!r)
- return -EIO;
- if (r->type != ACPI_TYPE_INTEGER) {
- kfree(r);
- return -EIO;
- }
- status = r->integer.value;
- }
- kfree(r);
- if (status != 80 && status != 100)
- status = 0;
- return sysfs_emit(buf, "%d\n", status);
- }
- static ssize_t battery_care_limit_show(struct device *dev,
- struct device_attribute *attr,
- char *buffer)
- {
- return charge_control_end_threshold_show(dev, attr, buffer);
- }
- static ssize_t battery_care_limit_store(struct device *dev,
- struct device_attribute *attr,
- const char *buffer, size_t count)
- {
- return charge_control_end_threshold_store(dev, attr, buffer, count);
- }
- static DEVICE_ATTR_RW(fan_mode);
- static DEVICE_ATTR_RW(usb_charge);
- static DEVICE_ATTR_RW(reader_mode);
- static DEVICE_ATTR_RW(fn_lock);
- static DEVICE_ATTR_RW(charge_control_end_threshold);
- static DEVICE_ATTR_RW(battery_care_limit);
- static int lg_battery_add(struct power_supply *battery)
- {
- if (device_create_file(&battery->dev,
- &dev_attr_charge_control_end_threshold))
- return -ENODEV;
- return 0;
- }
- static int lg_battery_remove(struct power_supply *battery)
- {
- device_remove_file(&battery->dev,
- &dev_attr_charge_control_end_threshold);
- return 0;
- }
- static struct acpi_battery_hook battery_hook = {
- .add_battery = lg_battery_add,
- .remove_battery = lg_battery_remove,
- .name = "LG Battery Extension",
- };
- static struct attribute *dev_attributes[] = {
- &dev_attr_fan_mode.attr,
- &dev_attr_usb_charge.attr,
- &dev_attr_reader_mode.attr,
- &dev_attr_fn_lock.attr,
- &dev_attr_battery_care_limit.attr,
- NULL
- };
- static const struct attribute_group dev_attribute_group = {
- .attrs = dev_attributes,
- };
- static void tpad_led_set(struct led_classdev *cdev,
- enum led_brightness brightness)
- {
- union acpi_object *r;
- r = lg_wmab(WM_TLED, WM_SET, brightness > LED_OFF);
- kfree(r);
- }
- static enum led_brightness tpad_led_get(struct led_classdev *cdev)
- {
- return ggov(GOV_TLED) > 0 ? LED_ON : LED_OFF;
- }
- static LED_DEVICE(tpad_led, 1, 0);
- static void kbd_backlight_set(struct led_classdev *cdev,
- enum led_brightness brightness)
- {
- u32 val;
- union acpi_object *r;
- val = 0x22;
- if (brightness <= LED_OFF)
- val = 0;
- if (brightness >= LED_FULL)
- val = 0x24;
- r = lg_wmab(WM_KEY_LIGHT, WM_SET, val);
- kfree(r);
- }
- static enum led_brightness get_kbd_backlight_level(void)
- {
- union acpi_object *r;
- int val;
- r = lg_wmab(WM_KEY_LIGHT, WM_GET, 0);
- if (!r)
- return LED_OFF;
- if (r->type != ACPI_TYPE_BUFFER || r->buffer.pointer[1] != 0x05) {
- kfree(r);
- return LED_OFF;
- }
- switch (r->buffer.pointer[0] & 0x27) {
- case 0x24:
- val = LED_FULL;
- break;
- case 0x22:
- val = LED_HALF;
- break;
- default:
- val = LED_OFF;
- }
- kfree(r);
- return val;
- }
- static enum led_brightness kbd_backlight_get(struct led_classdev *cdev)
- {
- return get_kbd_backlight_level();
- }
- static LED_DEVICE(kbd_backlight, 255, LED_BRIGHT_HW_CHANGED);
- static void wmi_input_destroy(void)
- {
- if (inited & INIT_INPUT_WMI_2)
- wmi_remove_notify_handler(WMI_EVENT_GUID2);
- if (inited & INIT_INPUT_WMI_0)
- wmi_remove_notify_handler(WMI_EVENT_GUID0);
- if (inited & INIT_SPARSE_KEYMAP)
- input_unregister_device(wmi_input_dev);
- inited &= ~(INIT_INPUT_WMI_0 | INIT_INPUT_WMI_2 | INIT_SPARSE_KEYMAP);
- }
- static struct platform_driver pf_driver = {
- .driver = {
- .name = PLATFORM_NAME,
- }
- };
- static int acpi_add(struct acpi_device *device)
- {
- int ret;
- const char *product;
- int year = 2017;
- if (pf_device)
- return 0;
- ret = platform_driver_register(&pf_driver);
- if (ret)
- return ret;
- pf_device = platform_device_register_simple(PLATFORM_NAME,
- PLATFORM_DEVID_NONE,
- NULL, 0);
- if (IS_ERR(pf_device)) {
- ret = PTR_ERR(pf_device);
- pf_device = NULL;
- pr_err("unable to register platform device\n");
- goto out_platform_registered;
- }
- product = dmi_get_system_info(DMI_PRODUCT_NAME);
- if (product && strlen(product) > 4)
- switch (product[4]) {
- case '5':
- if (strlen(product) > 5)
- switch (product[5]) {
- case 'N':
- year = 2021;
- break;
- case '0':
- year = 2016;
- break;
- default:
- year = 2022;
- }
- break;
- case '6':
- year = 2016;
- break;
- case '7':
- year = 2017;
- break;
- case '8':
- year = 2018;
- break;
- case '9':
- year = 2019;
- break;
- case '0':
- if (strlen(product) > 5)
- switch (product[5]) {
- case 'N':
- year = 2020;
- break;
- case 'P':
- year = 2021;
- break;
- default:
- year = 2022;
- }
- break;
- default:
- year = 2019;
- }
- pr_info("product: %s year: %d\n", product, year);
- if (year >= 2019)
- battery_limit_use_wmbb = 1;
- ret = sysfs_create_group(&pf_device->dev.kobj, &dev_attribute_group);
- if (ret)
- goto out_platform_device;
- /* LEDs are optional */
- led_classdev_register(&pf_device->dev, &kbd_backlight);
- led_classdev_register(&pf_device->dev, &tpad_led);
- wmi_input_setup();
- battery_hook_register(&battery_hook);
- return 0;
- out_platform_device:
- platform_device_unregister(pf_device);
- out_platform_registered:
- platform_driver_unregister(&pf_driver);
- return ret;
- }
- static int acpi_remove(struct acpi_device *device)
- {
- sysfs_remove_group(&pf_device->dev.kobj, &dev_attribute_group);
- led_classdev_unregister(&tpad_led);
- led_classdev_unregister(&kbd_backlight);
- battery_hook_unregister(&battery_hook);
- wmi_input_destroy();
- platform_device_unregister(pf_device);
- pf_device = NULL;
- platform_driver_unregister(&pf_driver);
- return 0;
- }
- static const struct acpi_device_id device_ids[] = {
- {"LGEX0815", 0},
- {"", 0}
- };
- MODULE_DEVICE_TABLE(acpi, device_ids);
- static struct acpi_driver acpi_driver = {
- .name = "LG Gram Laptop Support",
- .class = "lg-laptop",
- .ids = device_ids,
- .ops = {
- .add = acpi_add,
- .remove = acpi_remove,
- .notify = acpi_notify,
- },
- .owner = THIS_MODULE,
- };
- static int __init acpi_init(void)
- {
- int result;
- result = acpi_bus_register_driver(&acpi_driver);
- if (result < 0) {
- pr_debug("Error registering driver\n");
- return -ENODEV;
- }
- return 0;
- }
- static void __exit acpi_exit(void)
- {
- acpi_bus_unregister_driver(&acpi_driver);
- }
- module_init(acpi_init);
- module_exit(acpi_exit);
|