/* * Copyright 2013 Matthew Garrett * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include #include #include #include MODULE_LICENSE("GPL"); static ssize_t irst_show_wakeup_events(struct device *dev, struct device_attribute *attr, char *buf) { struct acpi_device *acpi; struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL }; union acpi_object *result; acpi_status status; acpi = to_acpi_device(dev); status = acpi_evaluate_object(acpi->handle, "GFFS", NULL, &output); if (!ACPI_SUCCESS(status)) return -EINVAL; result = output.pointer; if (result->type != ACPI_TYPE_INTEGER) { kfree(result); return -EINVAL; } return sprintf(buf, "%lld\n", result->integer.value); } static ssize_t irst_store_wakeup_events(struct device *dev, struct device_attribute *attr, const char *buf, size_t count) { struct acpi_device *acpi; struct acpi_object_list input; union acpi_object param; acpi_status status; unsigned long value; int error; acpi = to_acpi_device(dev); error = kstrtoul(buf, 0, &value); if (error) return error; param.type = ACPI_TYPE_INTEGER; param.integer.value = value; input.count = 1; input.pointer = ¶m; status = acpi_evaluate_object(acpi->handle, "SFFS", &input, NULL); if (!ACPI_SUCCESS(status)) return -EINVAL; return count; } static struct device_attribute irst_wakeup_attr = { .attr = { .name = "wakeup_events", .mode = 0600 }, .show = irst_show_wakeup_events, .store = irst_store_wakeup_events }; static ssize_t irst_show_wakeup_time(struct device *dev, struct device_attribute *attr, char *buf) { struct acpi_device *acpi; struct acpi_buffer output = { ACPI_ALLOCATE_BUFFER, NULL }; union acpi_object *result; acpi_status status; acpi = to_acpi_device(dev); status = acpi_evaluate_object(acpi->handle, "GFTV", NULL, &output); if (!ACPI_SUCCESS(status)) return -EINVAL; result = output.pointer; if (result->type != ACPI_TYPE_INTEGER) { kfree(result); return -EINVAL; } return sprintf(buf, "%lld\n", result->integer.value); } static ssize_t irst_store_wakeup_time(struct device *dev, struct device_attribute *attr, const char *buf, size_t count) { struct acpi_device *acpi; struct acpi_object_list input; union acpi_object param; acpi_status status; unsigned long value; int error; acpi = to_acpi_device(dev); error = kstrtoul(buf, 0, &value); if (error) return error; param.type = ACPI_TYPE_INTEGER; param.integer.value = value; input.count = 1; input.pointer = ¶m; status = acpi_evaluate_object(acpi->handle, "SFTV", &input, NULL); if (!ACPI_SUCCESS(status)) return -EINVAL; return count; } static struct device_attribute irst_timeout_attr = { .attr = { .name = "wakeup_time", .mode = 0600 }, .show = irst_show_wakeup_time, .store = irst_store_wakeup_time }; static int irst_add(struct acpi_device *acpi) { int error = 0; error = device_create_file(&acpi->dev, &irst_timeout_attr); if (error) goto out; error = device_create_file(&acpi->dev, &irst_wakeup_attr); if (error) goto out_timeout; return 0; out_timeout: device_remove_file(&acpi->dev, &irst_timeout_attr); out: return error; } static int irst_remove(struct acpi_device *acpi) { device_remove_file(&acpi->dev, &irst_wakeup_attr); device_remove_file(&acpi->dev, &irst_timeout_attr); return 0; } static const struct acpi_device_id irst_ids[] = { {"INT3392", 0}, {"", 0} }; static struct acpi_driver irst_driver = { .owner = THIS_MODULE, .name = "intel_rapid_start", .class = "intel_rapid_start", .ids = irst_ids, .ops = { .add = irst_add, .remove = irst_remove, }, }; static int irst_init(void) { return acpi_bus_register_driver(&irst_driver); } static void irst_exit(void) { acpi_bus_unregister_driver(&irst_driver); } module_init(irst_init); module_exit(irst_exit); MODULE_DEVICE_TABLE(acpi, irst_ids);