Release 4.7 drivers/acpi/button.c
/*
* button.c - ACPI Button Driver
*
* Copyright (C) 2001, 2002 Andy Grover <andrew.grover@intel.com>
* Copyright (C) 2001, 2002 Paul Diefenbaugh <paul.s.diefenbaugh@intel.com>
*
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
*
* 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.
*
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
*/
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h>
#include <linux/types.h>
#include <linux/proc_fs.h>
#include <linux/seq_file.h>
#include <linux/input.h>
#include <linux/slab.h>
#include <linux/acpi.h>
#include <acpi/button.h>
#define PREFIX "ACPI: "
#define ACPI_BUTTON_CLASS "button"
#define ACPI_BUTTON_FILE_INFO "info"
#define ACPI_BUTTON_FILE_STATE "state"
#define ACPI_BUTTON_TYPE_UNKNOWN 0x00
#define ACPI_BUTTON_NOTIFY_STATUS 0x80
#define ACPI_BUTTON_SUBCLASS_POWER "power"
#define ACPI_BUTTON_HID_POWER "PNP0C0C"
#define ACPI_BUTTON_DEVICE_NAME_POWER "Power Button"
#define ACPI_BUTTON_TYPE_POWER 0x01
#define ACPI_BUTTON_SUBCLASS_SLEEP "sleep"
#define ACPI_BUTTON_HID_SLEEP "PNP0C0E"
#define ACPI_BUTTON_DEVICE_NAME_SLEEP "Sleep Button"
#define ACPI_BUTTON_TYPE_SLEEP 0x03
#define ACPI_BUTTON_SUBCLASS_LID "lid"
#define ACPI_BUTTON_HID_LID "PNP0C0D"
#define ACPI_BUTTON_DEVICE_NAME_LID "Lid Switch"
#define ACPI_BUTTON_TYPE_LID 0x05
#define _COMPONENT ACPI_BUTTON_COMPONENT
ACPI_MODULE_NAME("button");
MODULE_AUTHOR("Paul Diefenbaugh");
MODULE_DESCRIPTION("ACPI Button Driver");
MODULE_LICENSE("GPL");
static const struct acpi_device_id button_device_ids[] = {
{ACPI_BUTTON_HID_LID, 0},
{ACPI_BUTTON_HID_SLEEP, 0},
{ACPI_BUTTON_HID_SLEEPF, 0},
{ACPI_BUTTON_HID_POWER, 0},
{ACPI_BUTTON_HID_POWERF, 0},
{"", 0},
};
MODULE_DEVICE_TABLE(acpi, button_device_ids);
static int acpi_button_add(struct acpi_device *device);
static int acpi_button_remove(struct acpi_device *device);
static void acpi_button_notify(struct acpi_device *device, u32 event);
#ifdef CONFIG_PM_SLEEP
static int acpi_button_suspend(struct device *dev);
static int acpi_button_resume(struct device *dev);
#else
#define acpi_button_suspend NULL
#define acpi_button_resume NULL
#endif
static SIMPLE_DEV_PM_OPS(acpi_button_pm, acpi_button_suspend, acpi_button_resume);
static struct acpi_driver acpi_button_driver = {
.name = "button",
.class = ACPI_BUTTON_CLASS,
.ids = button_device_ids,
.ops = {
.add = acpi_button_add,
.remove = acpi_button_remove,
.notify = acpi_button_notify,
},
.drv.pm = &acpi_button_pm,
};
struct acpi_button {
unsigned int type;
struct input_dev *input;
char phys[32]; /* for input device */
unsigned long pushed;
bool suspended;
};
static BLOCKING_NOTIFIER_HEAD(acpi_lid_notifier);
static struct acpi_device *lid_device;
/* --------------------------------------------------------------------------
FS Interface (/proc)
-------------------------------------------------------------------------- */
static struct proc_dir_entry *acpi_button_dir;
static struct proc_dir_entry *acpi_lid_dir;
static int acpi_button_state_seq_show(struct seq_file *seq, void *offset)
{
struct acpi_device *device = seq->private;
acpi_status status;
unsigned long long state;
status = acpi_evaluate_integer(device->handle, "_LID", NULL, &state);
seq_printf(seq, "state: %s\n",
ACPI_FAILURE(status) ? "unsupported" :
(state ? "open" : "closed"));
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
alexey starikovskiy | alexey starikovskiy | 59 | 80.82% | 1 | 16.67% |
dmitry torokhov | dmitry torokhov | 7 | 9.59% | 1 | 16.67% |
patrick mochel | patrick mochel | 4 | 5.48% | 2 | 33.33% |
bjorn helgaas | bjorn helgaas | 2 | 2.74% | 1 | 16.67% |
matthew wilcox | matthew wilcox | 1 | 1.37% | 1 | 16.67% |
| Total | 73 | 100.00% | 6 | 100.00% |
static int acpi_button_state_open_fs(struct inode *inode, struct file *file)
{
return single_open(file, acpi_button_state_seq_show, PDE_DATA(inode));
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
alexey starikovskiy | alexey starikovskiy | 28 | 96.55% | 1 | 50.00% |
al viro | al viro | 1 | 3.45% | 1 | 50.00% |
| Total | 29 | 100.00% | 2 | 100.00% |
static const struct file_operations acpi_button_state_fops = {
.owner = THIS_MODULE,
.open = acpi_button_state_open_fs,
.read = seq_read,
.llseek = seq_lseek,
.release = single_release,
};
static int acpi_button_add_fs(struct acpi_device *device)
{
struct acpi_button *button = acpi_driver_data(device);
struct proc_dir_entry *entry = NULL;
int ret = 0;
/* procfs I/F for ACPI lid device only */
if (button->type != ACPI_BUTTON_TYPE_LID)
return 0;
if (acpi_button_dir || acpi_lid_dir) {
printk(KERN_ERR PREFIX "More than one Lid device found!\n");
return -EEXIST;
}
/* create /proc/acpi/button */
acpi_button_dir = proc_mkdir(ACPI_BUTTON_CLASS, acpi_root_dir);
if (!acpi_button_dir)
return -ENODEV;
/* create /proc/acpi/button/lid */
acpi_lid_dir = proc_mkdir(ACPI_BUTTON_SUBCLASS_LID, acpi_button_dir);
if (!acpi_lid_dir) {
ret = -ENODEV;
goto remove_button_dir;
}
/* create /proc/acpi/button/lid/LID/ */
acpi_device_dir(device) = proc_mkdir(acpi_device_bid(device), acpi_lid_dir);
if (!acpi_device_dir(device)) {
ret = -ENODEV;
goto remove_lid_dir;
}
/* create /proc/acpi/button/lid/LID/state */
entry = proc_create_data(ACPI_BUTTON_FILE_STATE,
S_IRUGO, acpi_device_dir(device),
&acpi_button_state_fops, device);
if (!entry) {
ret = -ENODEV;
goto remove_dev_dir;
}
done:
return ret;
remove_dev_dir:
remove_proc_entry(acpi_device_bid(device),
acpi_lid_dir);
acpi_device_dir(device) = NULL;
remove_lid_dir:
remove_proc_entry(ACPI_BUTTON_SUBCLASS_LID, acpi_button_dir);
remove_button_dir:
remove_proc_entry(ACPI_BUTTON_CLASS, acpi_root_dir);
goto done;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
alexey starikovskiy | alexey starikovskiy | 101 | 45.70% | 1 | 16.67% |
rui zhang | rui zhang | 99 | 44.80% | 1 | 16.67% |
denis v. lunev | denis v. lunev | 9 | 4.07% | 1 | 16.67% |
bjorn helgaas | bjorn helgaas | 9 | 4.07% | 1 | 16.67% |
thomas renninger | thomas renninger | 2 | 0.90% | 1 | 16.67% |
patrick mochel | patrick mochel | 1 | 0.45% | 1 | 16.67% |
| Total | 221 | 100.00% | 6 | 100.00% |
static int acpi_button_remove_fs(struct acpi_device *device)
{
struct acpi_button *button = acpi_driver_data(device);
if (button->type != ACPI_BUTTON_TYPE_LID)
return 0;
remove_proc_entry(ACPI_BUTTON_FILE_STATE,
acpi_device_dir(device));
remove_proc_entry(acpi_device_bid(device),
acpi_lid_dir);
acpi_device_dir(device) = NULL;
remove_proc_entry(ACPI_BUTTON_SUBCLASS_LID, acpi_button_dir);
remove_proc_entry(ACPI_BUTTON_CLASS, acpi_root_dir);
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
alexey starikovskiy | alexey starikovskiy | 54 | 71.05% | 2 | 40.00% |
rui zhang | rui zhang | 19 | 25.00% | 1 | 20.00% |
patrick mochel | patrick mochel | 2 | 2.63% | 1 | 20.00% |
dmitry torokhov | dmitry torokhov | 1 | 1.32% | 1 | 20.00% |
| Total | 76 | 100.00% | 5 | 100.00% |
/* --------------------------------------------------------------------------
Driver Interface
-------------------------------------------------------------------------- */
int acpi_lid_notifier_register(struct notifier_block *nb)
{
return blocking_notifier_chain_register(&acpi_lid_notifier, nb);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
jesse barnes | jesse barnes | 19 | 100.00% | 1 | 100.00% |
| Total | 19 | 100.00% | 1 | 100.00% |
EXPORT_SYMBOL(acpi_lid_notifier_register);
int acpi_lid_notifier_unregister(struct notifier_block *nb)
{
return blocking_notifier_chain_unregister(&acpi_lid_notifier, nb);
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
jesse barnes | jesse barnes | 19 | 100.00% | 1 | 100.00% |
| Total | 19 | 100.00% | 1 | 100.00% |
EXPORT_SYMBOL(acpi_lid_notifier_unregister);
int acpi_lid_open(void)
{
acpi_status status;
unsigned long long state;
if (!lid_device)
return -ENODEV;
status = acpi_evaluate_integer(lid_device->handle, "_LID", NULL,
&state);
if (ACPI_FAILURE(status))
return -ENODEV;
return !!state;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
jesse barnes | jesse barnes | 56 | 100.00% | 2 | 100.00% |
| Total | 56 | 100.00% | 2 | 100.00% |
EXPORT_SYMBOL(acpi_lid_open);
static int acpi_lid_send_state(struct acpi_device *device)
{
struct acpi_button *button = acpi_driver_data(device);
unsigned long long state;
acpi_status status;
int ret;
status = acpi_evaluate_integer(device->handle, "_LID", NULL, &state);
if (ACPI_FAILURE(status))
return -ENODEV;
/* input layer checks if event is redundant */
input_report_switch(button->input, SW_LID, !state);
input_sync(button->input);
if (state)
pm_wakeup_event(&device->dev, 0);
ret = blocking_notifier_call_chain(&acpi_lid_notifier, state, device);
if (ret == NOTIFY_DONE)
ret = blocking_notifier_call_chain(&acpi_lid_notifier, state,
device);
if (ret == NOTIFY_DONE || ret == NOTIFY_OK) {
/*
* It is also regarded as success if the notifier_chain
* returns NOTIFY_OK or NOTIFY_DONE.
*/
ret = 0;
}
return ret;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
alexey starikovskiy | alexey starikovskiy | 58 | 40.56% | 1 | 14.29% |
jesse barnes | jesse barnes | 34 | 23.78% | 1 | 14.29% |
yakui zhao | yakui zhao | 17 | 11.89% | 1 | 14.29% |
rafael j. wysocki | rafael j. wysocki | 14 | 9.79% | 1 | 14.29% |
bjorn helgaas | bjorn helgaas | 12 | 8.39% | 1 | 14.29% |
guillem jover | guillem jover | 7 | 4.90% | 1 | 14.29% |
matthew wilcox | matthew wilcox | 1 | 0.70% | 1 | 14.29% |
| Total | 143 | 100.00% | 7 | 100.00% |
static void acpi_button_notify(struct acpi_device *device, u32 event)
{
struct acpi_button *button = acpi_driver_data(device);
struct input_dev *input;
switch (event) {
case ACPI_FIXED_HARDWARE_EVENT:
event = ACPI_BUTTON_NOTIFY_STATUS;
/* fall through */
case ACPI_BUTTON_NOTIFY_STATUS:
input = button->input;
if (button->type == ACPI_BUTTON_TYPE_LID) {
acpi_lid_send_state(device);
} else {
int keycode;
pm_wakeup_event(&device->dev, 0);
if (button->suspended)
break;
keycode = test_bit(KEY_SLEEP, input->keybit) ?
KEY_SLEEP : KEY_POWER;
input_report_key(input, keycode, 1);
input_sync(input);
input_report_key(input, keycode, 0);
input_sync(input);
acpi_bus_generate_netlink_event(
device->pnp.device_class,
dev_name(&device->dev),
event, ++button->pushed);
}
break;
default:
ACPI_DEBUG_PRINT((ACPI_DB_INFO,
"Unsupported event [0x%x]\n", event));
break;
}
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
dmitry torokhov | dmitry torokhov | 68 | 39.77% | 1 | 11.11% |
andy grover | andy grover | 38 | 22.22% | 1 | 11.11% |
lan tianyu | lan tianyu | 24 | 14.04% | 1 | 11.11% |
rafael j. wysocki | rafael j. wysocki | 21 | 12.28% | 1 | 11.11% |
bjorn helgaas | bjorn helgaas | 17 | 9.94% | 2 | 22.22% |
len brown | len brown | 1 | 0.58% | 1 | 11.11% |
guillem jover | guillem jover | 1 | 0.58% | 1 | 11.11% |
alexey starikovskiy | alexey starikovskiy | 1 | 0.58% | 1 | 11.11% |
| Total | 171 | 100.00% | 9 | 100.00% |
#ifdef CONFIG_PM_SLEEP
static int acpi_button_suspend(struct device *dev)
{
struct acpi_device *device = to_acpi_device(dev);
struct acpi_button *button = acpi_driver_data(device);
button->suspended = true;
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
rafael j. wysocki | rafael j. wysocki | 40 | 100.00% | 1 | 100.00% |
| Total | 40 | 100.00% | 1 | 100.00% |
static int acpi_button_resume(struct device *dev)
{
struct acpi_device *device = to_acpi_device(dev);
struct acpi_button *button = acpi_driver_data(device);
button->suspended = false;
if (button->type == ACPI_BUTTON_TYPE_LID)
return acpi_lid_send_state(device);
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
alexey starikovskiy | alexey starikovskiy | 33 | 61.11% | 1 | 20.00% |
rafael j. wysocki | rafael j. wysocki | 18 | 33.33% | 2 | 40.00% |
bjorn helgaas | bjorn helgaas | 3 | 5.56% | 2 | 40.00% |
| Total | 54 | 100.00% | 5 | 100.00% |
#endif
static int acpi_button_add(struct acpi_device *device)
{
struct acpi_button *button;
struct input_dev *input;
const char *hid = acpi_device_hid(device);
char *name, *class;
int error;
button = kzalloc(sizeof(struct acpi_button), GFP_KERNEL);
if (!button)
return -ENOMEM;
device->driver_data = button;
button->input = input = input_allocate_device();
if (!input) {
error = -ENOMEM;
goto err_free_button;
}
name = acpi_device_name(device);
class = acpi_device_class(device);
if (!strcmp(hid, ACPI_BUTTON_HID_POWER) ||
!strcmp(hid, ACPI_BUTTON_HID_POWERF)) {
button->type = ACPI_BUTTON_TYPE_POWER;
strcpy(name, ACPI_BUTTON_DEVICE_NAME_POWER);
sprintf(class, "%s/%s",
ACPI_BUTTON_CLASS, ACPI_BUTTON_SUBCLASS_POWER);
} else if (!strcmp(hid, ACPI_BUTTON_HID_SLEEP) ||
!strcmp(hid, ACPI_BUTTON_HID_SLEEPF)) {
button->type = ACPI_BUTTON_TYPE_SLEEP;
strcpy(name, ACPI_BUTTON_DEVICE_NAME_SLEEP);
sprintf(class, "%s/%s",
ACPI_BUTTON_CLASS, ACPI_BUTTON_SUBCLASS_SLEEP);
} else if (!strcmp(hid, ACPI_BUTTON_HID_LID)) {
button->type = ACPI_BUTTON_TYPE_LID;
strcpy(name, ACPI_BUTTON_DEVICE_NAME_LID);
sprintf(class, "%s/%s",
ACPI_BUTTON_CLASS, ACPI_BUTTON_SUBCLASS_LID);
} else {
printk(KERN_ERR PREFIX "Unsupported hid [%s]\n", hid);
error = -ENODEV;
goto err_free_input;
}
error = acpi_button_add_fs(device);
if (error)
goto err_free_input;
snprintf(button->phys, sizeof(button->phys), "%s/button/input0", hid);
input->name = name;
input->phys = button->phys;
input->id.bustype = BUS_HOST;
input->id.product = button->type;
input->dev.parent = &device->dev;
switch (button->type) {
case ACPI_BUTTON_TYPE_POWER:
input_set_capability(input, EV_KEY, KEY_POWER);
break;
case ACPI_BUTTON_TYPE_SLEEP:
input_set_capability(input, EV_KEY, KEY_SLEEP);
break;
case ACPI_BUTTON_TYPE_LID:
input_set_capability(input, EV_SW, SW_LID);
break;
}
error = input_register_device(input);
if (error)
goto err_remove_fs;
if (button->type == ACPI_BUTTON_TYPE_LID) {
acpi_lid_send_state(device);
/*
* This assumes there's only one lid device, or if there are
* more we only care about the last one...
*/
lid_device = device;
}
printk(KERN_INFO PREFIX "%s [%s]\n", name, acpi_device_bid(device));
return 0;
err_remove_fs:
acpi_button_remove_fs(device);
err_free_input:
input_free_device(input);
err_free_button:
kfree(button);
return error;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
dmitry torokhov | dmitry torokhov | 165 | 37.08% | 1 | 5.26% |
andy grover | andy grover | 156 | 35.06% | 2 | 10.53% |
bjorn helgaas | bjorn helgaas | 49 | 11.01% | 5 | 26.32% |
alexey starikovskiy | alexey starikovskiy | 23 | 5.17% | 2 | 10.53% |
lan tianyu | lan tianyu | 12 | 2.70% | 1 | 5.26% |
andrey borzenkov | andrey borzenkov | 11 | 2.47% | 1 | 5.26% |
thomas renninger | thomas renninger | 8 | 1.80% | 1 | 5.26% |
jesse barnes | jesse barnes | 7 | 1.57% | 1 | 5.26% |
len brown | len brown | 7 | 1.57% | 2 | 10.53% |
andrew morton | andrew morton | 3 | 0.67% | 1 | 5.26% |
pavel machek | pavel machek | 2 | 0.45% | 1 | 5.26% |
patrick mochel | patrick mochel | 2 | 0.45% | 1 | 5.26% |
| Total | 445 | 100.00% | 19 | 100.00% |
static int acpi_button_remove(struct acpi_device *device)
{
struct acpi_button *button = acpi_driver_data(device);
acpi_button_remove_fs(device);
input_unregister_device(button->input);
kfree(button);
return 0;
}
Contributors
| Person | Tokens | Prop | Commits | CommitProp |
andy grover | andy grover | 28 | 68.29% | 2 | 28.57% |
dmitry torokhov | dmitry torokhov | 5 | 12.20% | 1 | 14.29% |
alexey starikovskiy | alexey starikovskiy | 3 | 7.32% | 1 | 14.29% |
patrick mochel | patrick mochel | 3 | 7.32% | 2 | 28.57% |
bjorn helgaas | bjorn helgaas | 2 | 4.88% | 1 | 14.29% |
| Total | 41 | 100.00% | 7 | 100.00% |
module_acpi_driver(acpi_button_driver);
Overall Contributors
| Person | Tokens | Prop | Commits | CommitProp |
alexey starikovskiy | alexey starikovskiy | 478 | 26.47% | 3 | 6.12% |
dmitry torokhov | dmitry torokhov | 264 | 14.62% | 1 | 2.04% |
andy grover | andy grover | 249 | 13.79% | 4 | 8.16% |
jesse barnes | jesse barnes | 162 | 8.97% | 2 | 4.08% |
rui zhang | rui zhang | 150 | 8.31% | 1 | 2.04% |
rafael j. wysocki | rafael j. wysocki | 147 | 8.14% | 4 | 8.16% |
bjorn helgaas | bjorn helgaas | 106 | 5.87% | 5 | 10.20% |
patrick mochel | patrick mochel | 66 | 3.65% | 3 | 6.12% |
thomas renninger | thomas renninger | 63 | 3.49% | 3 | 6.12% |
lan tianyu | lan tianyu | 36 | 1.99% | 2 | 4.08% |
yakui zhao | yakui zhao | 17 | 0.94% | 1 | 2.04% |
len brown | len brown | 16 | 0.89% | 7 | 14.29% |
andrey borzenkov | andrey borzenkov | 11 | 0.61% | 1 | 2.04% |
denis v. lunev | denis v. lunev | 9 | 0.50% | 1 | 2.04% |
guillem jover | guillem jover | 8 | 0.44% | 1 | 2.04% |
shuah khan | shuah khan | 6 | 0.33% | 1 | 2.04% |
andy shevchenko | andy shevchenko | 3 | 0.17% | 1 | 2.04% |
tejun heo | tejun heo | 3 | 0.17% | 1 | 2.04% |
andrew morton | andrew morton | 3 | 0.17% | 1 | 2.04% |
matthew wilcox | matthew wilcox | 2 | 0.11% | 1 | 2.04% |
pavel machek | pavel machek | 2 | 0.11% | 1 | 2.04% |
mika westerberg | mika westerberg | 2 | 0.11% | 1 | 2.04% |
al viro | al viro | 1 | 0.06% | 1 | 2.04% |
jarkko nikula | jarkko nikula | 1 | 0.06% | 1 | 2.04% |
lv zheng | lv zheng | 1 | 0.06% | 1 | 2.04% |
| Total | 1806 | 100.00% | 49 | 100.00% |
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.