/* * ec_sys.c * * Copyright (C) 2010 SUSE Products GmbH/Novell * Author: * Thomas Renninger <trenn@suse.de> * * This work is licensed under the terms of the GNU GPL, version 2. */ #include <linux/kernel.h> #include <linux/acpi.h> #include <linux/debugfs.h> #include <linux/module.h> #include <linux/uaccess.h> #include "internal.h" MODULE_AUTHOR("Thomas Renninger <trenn@suse.de>"); MODULE_DESCRIPTION("ACPI EC sysfs access driver"); MODULE_LICENSE("GPL"); static bool write_support; module_param(write_support, bool, 0644); MODULE_PARM_DESC(write_support, "Dangerous, reboot and removal of battery may " "be needed."); #define EC_SPACE_SIZE 256 static struct dentry *acpi_ec_debugfs_dir; static ssize_t acpi_ec_read_io(struct file *f, char __user *buf, size_t count, loff_t *off) { /* Use this if support reading/writing multiple ECs exists in ec.c: * struct acpi_ec *ec = ((struct seq_file *)f->private_data)->private; */ unsigned int size = EC_SPACE_SIZE; loff_t init_off = *off; int err = 0; if (*off >= size) return 0; if (*off + count >= size) { size -= *off; count = size; } else size = count; while (size) { u8 byte_read; err = ec_read(*off, &byte_read); if (err) return err; if (put_user(byte_read, buf + *off - init_off)) { if (*off - init_off) return *off - init_off; /* partial read */ return -EFAULT; } *off += 1; size--; } return count; } static ssize_t acpi_ec_write_io(struct file *f, const char __user *buf, size_t count, loff_t *off) { /* Use this if support reading/writing multiple ECs exists in ec.c: * struct acpi_ec *ec = ((struct seq_file *)f->private_data)->private; */ unsigned int size = count; loff_t init_off = *off; int err = 0; if (*off >= EC_SPACE_SIZE) return 0; if (*off + count >= EC_SPACE_SIZE) { size = EC_SPACE_SIZE - *off; count = size; } while (size) { u8 byte_write; if (get_user(byte_write, buf + *off - init_off)) { if (*off - init_off) return *off - init_off; /* partial write */ return -EFAULT; } err = ec_write(*off, byte_write); if (err) return err; *off += 1; size--; } return count; } static const struct file_operations acpi_ec_io_ops = { .owner = THIS_MODULE, .open = simple_open, .read = acpi_ec_read_io, .write = acpi_ec_write_io, .llseek = default_llseek, }; static int acpi_ec_add_debugfs(struct acpi_ec *ec, unsigned int ec_device_count) { struct dentry *dev_dir; char name[64]; umode_t mode = 0400; if (ec_device_count == 0) { acpi_ec_debugfs_dir = debugfs_create_dir("ec", NULL); if (!acpi_ec_debugfs_dir) return -ENOMEM; } sprintf(name, "ec%u", ec_device_count); dev_dir = debugfs_create_dir(name, acpi_ec_debugfs_dir); if (!dev_dir) { if (ec_device_count != 0) goto error; return -ENOMEM; } if (!debugfs_create_x32("gpe", 0444, dev_dir, (u32 *)&first_ec->gpe)) goto error; if (!debugfs_create_bool("use_global_lock", 0444, dev_dir, &first_ec->global_lock)) goto error; if (write_support) mode = 0600; if (!debugfs_create_file("io", mode, dev_dir, ec, &acpi_ec_io_ops)) goto error; return 0; error: debugfs_remove_recursive(acpi_ec_debugfs_dir); return -ENOMEM; } static int __init acpi_ec_sys_init(void) { int err = 0; if (first_ec) err = acpi_ec_add_debugfs(first_ec, 0); else err = -ENODEV; return err; } static void __exit acpi_ec_sys_exit(void) { debugfs_remove_recursive(acpi_ec_debugfs_dir); } module_init(acpi_ec_sys_init); module_exit(acpi_ec_sys_exit);