Re: [PATCH] ACPI: Platform driver to support App Hot Startup (PNP0C32)

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



-----BEGIN PGP SIGNED MESSAGE-----
Hash: SHA1

On Tue, 25 Mar 2008 14:59:33 +0800
"Zhang, Rui" <rui.zhang@xxxxxxxxx> wrote:

> Hi, Angelo,
> 
> On Fri, 2008-03-14 at 23:17 +0800, Ângelo Miguel Arrifano wrote:
> > Here is the patch made from the today linux-acpi tree.
> > 
> > Thanks,
> > Angelo Arrifano
> we'd better add the change log here.
> Some introductions about PNP0C32 device would be great.

Documentation (from README)
- -----------------------------------------------------------------
**********************************************************
* quickstart.c - ACPI Direct App Launch Driver for Linux *
*            Vista(TM) calls this Hotstart(TM)           *
*            http://quickstart.sourceforge.net           *
**********************************************************

WHAT IS IT?
  <<The Windows Vista(TM) operating system provides built-in support
  for a fast system startup experience that boots or resumes directly
  into media or other applications.
  This support, called direct application launch, is possible on PCs
  that are running Windows Vista by making simple changes to platform
  firmware and underlying platform wake circuitry.>>

* With the introduction of Vista Microsoft launched a new ACPI
  ?specification? to allow computers in sleep/off states to boot 
  directly into multimedia applications by only pressing buttons.
* This driver provides support to these devices allowing userspace
  scripts to launch multimedia applications based on which button
  was pressed.

HOW IT WORKS:
* When the system boots, the driver receives an ACPI event telling
  which button was used to power the system on. Then, the driver exports
  this information to sysfs allowing userspace scripts to know the
  button name. This also applies to memory suspend and other power
  saving states.

EXAMPLE:
* File buttons shows a list of supported hot buttons.
  $ cat /sys/devices/platform/quickstart/buttons
  QBTN
  DBTN
  MUBN
  PIBN
  WEBN
  LVBN
  VOBN

* File pressed_button shows the hot button name used to power/resume the system.
  You can also write "none" to it to reset.
  $ cat /sys/devices/platform/quickstart/pressed_button
  none

WHAT FOR?
* Mobile and Media Center Linux distributors could use this to enrich
  their linux experience.
* It's easy to make an alternate init script that checks if some button
  was pressed and launch the appropriate application without having to
  load the entire system.

REFERENCES
* http://download.microsoft.com/download/9/c/5/9c5b2167-8017-4bae-9fde-d599bac8184a/DirAppLaunch_Vista.doc


CHANGELOG
- -----------------------------------------------------------------
* version 1.00-r1 2008/03/06
  - Fixed Makefile for linux-2.6.24

* version 1.00 2008/03/03
  - Little changes to support linux-2.6.23 and newer.
  - Added example autostart scripts.
  - Going stable, version bump.

* version 0.04 2007/11/17
  - Added reporting of runtime key presses (hotkeys) for testing.

* version 0.03 2007/11/13
  - Method GHID is now called under its device scope.
  - Code cleanup.

* version 0.02 2007/11/12
  - Project renamed to quickstart to prevent conflict with Microsoft
    trademark Hotstart(TM).
  - Removed procfs support since /proc/acpi will be removed from kernel soon.
  - Added sysfs support.
  - Button states are now unified into a single attribute file which displays
    the button name used to power the system on.
  - Added attribute file to show all suported buttons.
  - Code cleanups.

* version 0.01 2007/11/10
  - Detects button devices and creates proc entries to each device.
  - Proc entries can be written.
  - Workaround to raise device 

PATCH
- -----------------------------------------------------------------
diff --git a/drivers/acpi/Kconfig b/drivers/acpi/Kconfig
index f688c21..fb096ee 100644
- --- a/drivers/acpi/Kconfig
+++ b/drivers/acpi/Kconfig
@@ -196,6 +196,14 @@ config ACPI_THERMAL
 	  recommended that this option be enabled, as your processor(s)
 	  may be damaged without it.
 
+config ACPI_QUICKSTART
+	tristate "Quickstart"
+	default y
+	help
+	  This driver add support for ACPI Direct Application Launch
+	  also known as Hotstart(TM). Say yes here to have a entry in
+	  sysfs telling which button was used to turn on the system.
+
 config ACPI_NUMA
 	bool "NUMA support"
 	depends on NUMA
diff --git a/drivers/acpi/Makefile b/drivers/acpi/Makefile
index f29812a..badb0a6 100644
- --- a/drivers/acpi/Makefile
+++ b/drivers/acpi/Makefile
@@ -62,3 +62,4 @@ obj-$(CONFIG_ACPI_HOTPLUG_MEMORY)	+= acpi_memhotplug.o
 obj-$(CONFIG_ACPI_PROCFS_POWER)	+= cm_sbs.o
 obj-$(CONFIG_ACPI_SBS)		+= sbs.o
 obj-$(CONFIG_ACPI_SBS)		+= sbshc.o
+obj-$(CONFIG_ACPI_QUICKSTART)	+= quickstart.o
diff --git a/drivers/acpi/quickstart.c b/drivers/acpi/quickstart.c
new file mode 100644
index 0000000..05c98f2
- --- /dev/null
+++ b/drivers/acpi/quickstart.c
@@ -0,0 +1,416 @@
+/*
+ *  quickstart.c - ACPI Direct App Launch driver
+ *
+ *
+ *  Copyright (C) 2007 Angelo Arrifano <miknix@xxxxxxxxx>
+ *
+ *  Information gathered from disassebled dsdt and from here:
+ *  "http://download.microsoft.com/download/9/c/5/
+ *  9c5b2167-8017-4bae-9fde-d599bac8184a/DirAppLaunch_Vista.doc"
+ *
+ *  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., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ *
+ */
+
+#define QUICKSTART_VERSION "1.01"
+
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/init.h>
+#include <linux/types.h>
+#include <acpi/acpi_drivers.h>
+#include <linux/platform_device.h>
+
+MODULE_AUTHOR("Angelo Arrifano");
+MODULE_DESCRIPTION("ACPI Direct App Launch driver");
+MODULE_LICENSE("GPL");
+
+#define QUICKSTART_ACPI_DEVICE_NAME   "quickstart"
+#define QUICKSTART_ACPI_CLASS         "quickstart"
+#define QUICKSTART_ACPI_HID           "PNP0C32"
+
+#define QUICKSTART_PF_DRIVER_NAME     "quickstart"
+#define QUICKSTART_PF_DEVICE_NAME     "quickstart"
+#define QUICKSTART_PF_DEVATTR_NAME    "pressed_button"
+
+#define QUICKSTART_MAX_BTN_NAME_LEN   16
+
+/* There will be two events:
+	 * 0x02 - A hot button was pressed while device was off/sleeping.
+	 * 0x08 - A hot button was pressed while device was up. */
+#define QUICKSTART_EVENT_WAKE         0x02
+#define QUICKSTART_EVENT_RUNTIME      0x80
+
+struct quickstart_btn {
+	char *name;
+	struct quickstart_btn *next;
+};
+
+static struct quickstart_driver_data {
+	struct quickstart_btn *btn_lst;
+	struct quickstart_btn *pressed;
+} quickstart_data = {
+	.btn_lst = NULL,
+	.pressed = NULL,
+};
+
+/*
+ * ACPI driver Structs
+ */
+
+struct quickstart_acpi {
+	struct acpi_device *device;
+	struct quickstart_btn *btn;
+};
+static int quickstart_acpi_add(struct acpi_device *device);
+static int quickstart_acpi_remove(struct acpi_device *device, int type);
+static const struct acpi_device_id  quickstart_device_ids[] = {
+	{QUICKSTART_ACPI_HID, 0},
+	{"", 0},
+};
+
+static struct acpi_driver quickstart_acpi_driver = {
+	.name = "quickstart",
+	.class = QUICKSTART_ACPI_CLASS,
+	.ids = quickstart_device_ids,
+	.ops = {
+			.add = quickstart_acpi_add,
+			.remove = quickstart_acpi_remove,
+		},
+};
+
+/*
+ * Platform driver structs
+ */
+static ssize_t buttons_show(struct device *dev,
+					struct device_attribute *attr,
+					char *buf);
+static ssize_t pressed_button_show(struct device *dev,
+					struct device_attribute *attr,
+					char *buf);
+static ssize_t pressed_button_store(struct device *dev,
+					struct device_attribute *attr,
+					 const char *buf,
+					 size_t count);
+static DEVICE_ATTR(pressed_button, 0666, pressed_button_show,
+					 pressed_button_store);
+static DEVICE_ATTR(buttons, 0444, buttons_show, NULL);
+static struct platform_device *pf_device;
+static struct platform_driver pf_driver = {
+	.driver = {
+		.name = QUICKSTART_PF_DRIVER_NAME,
+		.owner = THIS_MODULE,
+	}
+};
+
+/*
+ * Platform driver functions
+ */
+static ssize_t buttons_show(struct device *dev,
+					 struct device_attribute *attr,
+					 char *buf)
+{
+	int count = 0;
+	char tmpbuf[QUICKSTART_MAX_BTN_NAME_LEN + 2];
+	struct quickstart_btn *ptr = quickstart_data.btn_lst;
+
+	if (!ptr)
+		return snprintf(buf, PAGE_SIZE, "none");
+
+	while (ptr && (count < PAGE_SIZE)) {
+		if (ptr->name) {
+			strncpy(tmpbuf, ptr->name, QUICKSTART_MAX_BTN_NAME_LEN);
+			strcat(tmpbuf, "\n");
+			count += snprintf(buf + count,
+					PAGE_SIZE - count,
+					tmpbuf);
+		}
+		ptr = ptr->next;
+	}
+
+	return count;
+}
+
+static ssize_t pressed_button_show(struct device *dev,
+					struct device_attribute *attr,
+					char *buf)
+{
+	return snprintf(buf, PAGE_SIZE, "%s\n",
+		(quickstart_data.pressed?quickstart_data.pressed->name:"none"));
+}
+
+
+static ssize_t pressed_button_store(struct device *dev,
+					 struct device_attribute *attr,
+					 const char *buf, size_t count)
+{
+	if (count < 2)
+		return -EINVAL;
+
+	if (strncasecmp(buf, "none", 4) != 0)
+		return -EINVAL;
+
+	quickstart_data.pressed = NULL;
+	return count;
+}
+
+/* Hotstart Helper functions */
+static int quickstart_btnlst_add(struct quickstart_btn **data)
+{
+	struct quickstart_btn **ptr = &quickstart_data.btn_lst;
+
+	while (*ptr)
+		ptr = &((*ptr)->next);
+
+	*ptr = kzalloc(sizeof(struct quickstart_btn), GFP_KERNEL);
+	if (!*ptr) {
+		*data = NULL;
+		return(-ENOMEM);
+	}
+	*data = *ptr;
+
+	return 0;
+}
+
+static void quickstart_btnlst_free(void)
+{
+	struct quickstart_btn *ptr = quickstart_data.btn_lst;
+	struct quickstart_btn *lptr = NULL;
+
+	if (!ptr)
+		return;
+
+	while (ptr) {
+		lptr = ptr;
+		ptr = ptr->next;
+		kfree(lptr->name);
+		kfree(lptr);
+	}
+
+	return;
+}
+
+/* ACPI Driver functions */
+static void quickstart_acpi_notify(acpi_handle handle, u32 event, void *data)
+{
+	struct quickstart_acpi *quickstart = data;
+
+	if (!quickstart)
+		return;
+
+	if (event == QUICKSTART_EVENT_WAKE) {
+		quickstart_data.pressed = quickstart->btn;
+		printk(KERN_ERR "quickstart: Quickbutton %s pressed.\n",
+					acpi_device_bid(quickstart->device));
+	} else if (event == QUICKSTART_EVENT_RUNTIME) {
+		printk(KERN_ERR "quickstart: Runtime button %s pressed.\n\t"
+		       "please report on linux-acpi@xxxxxxxxxxxxxxx",
+					acpi_device_bid(quickstart->device));
+	}
+
+	return;
+}
+
+static void quickstart_acpi_raise_notify(struct quickstart_acpi *quickstart)
+{
+	acpi_status status;
+	struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL };
+
+	if (!quickstart)
+		return;
+
+	/* This returns a buffer telling the  button usage ID (useless),
+	 * we dont care about it now. The important is to trigger
+	 * pending notify events (The ones before booting). */
+	status = acpi_evaluate_object(quickstart->device->handle,
+					"GHID", NULL, &buffer);
+	if (ACPI_FAILURE(status) || !buffer.pointer) {
+		printk(KERN_ERR "quickstart: %s GHID method failed.\n",
+		       quickstart->btn->name);
+		return;
+	}
+
+	kfree(buffer.pointer);
+	return;
+}
+
+static int quickstart_acpi_config(struct quickstart_acpi *quickstart, char *bid)
+{
+	int len = strlen(bid);
+	int ret;
+
+	/* Add button to list */
+	ret = quickstart_btnlst_add(&quickstart->btn);
+	if (ret)
+		return ret;
+
+	quickstart->btn->name = kzalloc(len + 1, GFP_KERNEL);
+	if (!quickstart->btn->name) {
+		quickstart_btnlst_free();
+		return -ENOMEM;
+	}
+	strcpy(quickstart->btn->name, bid);
+
+	return 0;
+}
+
+static int quickstart_acpi_add(struct acpi_device *device)
+{
+	int ret = 0;
+	acpi_status status = AE_OK;
+	struct quickstart_acpi *quickstart = NULL;
+
+	if (!device)
+		return -EINVAL;
+
+	quickstart = kzalloc(sizeof(struct quickstart_acpi), GFP_KERNEL);
+	if (!quickstart)
+		return -ENOMEM;
+
+	quickstart->device = device;
+	strcpy(acpi_device_name(device), QUICKSTART_ACPI_DEVICE_NAME);
+	strcpy(acpi_device_class(device), QUICKSTART_ACPI_CLASS);
+	acpi_driver_data(device) = quickstart;
+
+	/* Add button to list and initialize some stuff */
+	ret = quickstart_acpi_config(quickstart, acpi_device_bid(device));
+	if (ret)
+		goto fail_config;
+
+	status = acpi_install_notify_handler(device->handle,
+						ACPI_ALL_NOTIFY,
+						quickstart_acpi_notify,
+						quickstart);
+	if (ACPI_FAILURE(status)) {
+		printk(KERN_ERR "quickstart: Notify handler install error\n");
+		ret = -ENODEV;
+		goto fail_installnotify;
+	}
+
+	quickstart_acpi_raise_notify(quickstart);
+
+	return 0;
+
+fail_installnotify:
+fail_config:
+
+	kfree(quickstart);
+
+	return ret;
+}
+
+static int quickstart_acpi_remove(struct acpi_device *device, int type)
+{
+	acpi_status status = 0;
+	struct quickstart_acpi *quickstart = NULL;
+
+	if (!device || !acpi_driver_data(device))
+		return -EINVAL;
+
+	quickstart = acpi_driver_data(device);
+
+	status = acpi_remove_notify_handler(device->handle,
+						 ACPI_ALL_NOTIFY,
+					    quickstart_acpi_notify);
+	if (ACPI_FAILURE(status))
+		printk(KERN_ERR "quickstart: Error removing notify handler\n");
+
+
+	kfree(quickstart);
+
+	return 0;
+}
+
+/* Module functions */
+
+static void quickstart_exit(void)
+{
+	device_remove_file(&pf_device->dev, &dev_attr_pressed_button);
+	device_remove_file(&pf_device->dev, &dev_attr_buttons);
+
+	platform_device_unregister(pf_device);
+
+	platform_driver_unregister(&pf_driver);
+
+	acpi_bus_unregister_driver(&quickstart_acpi_driver);
+
+	quickstart_btnlst_free();
+
+	return;
+}
+
+static int __init quickstart_init(void)
+{
+	int ret;
+	acpi_status status = 0;
+
+	/* ACPI Check */
+	if (acpi_disabled)
+		return -ENODEV;
+
+	/* ACPI driver register */
+	status = acpi_bus_register_driver(&quickstart_acpi_driver);
+	if (!quickstart_data.btn_lst || status < 0)
+		return -ENODEV;
+
+	/* Platform driver register */
+	ret = platform_driver_register(&pf_driver);
+	if (ret)
+		goto fail_pfdrv_reg;
+
+	/* Platform device register */
+	pf_device = platform_device_alloc(QUICKSTART_PF_DEVICE_NAME, -1);
+	if (!pf_device) {
+		ret = -ENOMEM;
+		goto fail_pfdev_alloc;
+	}
+	ret = platform_device_register(pf_device);
+	if (ret)
+		goto fail_pfdev_reg;
+
+	/* Create device sysfs file */
+	ret = device_create_file(&pf_device->dev, &dev_attr_pressed_button);
+	if (ret)
+		goto fail_dev_file;
+
+	ret = device_create_file(&pf_device->dev, &dev_attr_buttons);
+	if (ret)
+		goto fail_dev_file2;
+
+	printk(KERN_ERR "quickstart: ACPI Direct App Launch ver %s\n",
+						QUICKSTART_VERSION);
+
+	return 0;
+
+fail_dev_file2:
+	device_remove_file(&pf_device->dev, &dev_attr_pressed_button);
+
+fail_dev_file:
+	platform_device_del(pf_device);
+
+fail_pfdev_reg:
+	platform_device_put(pf_device);
+
+fail_pfdev_alloc:
+	platform_driver_unregister(&pf_driver);
+
+fail_pfdrv_reg:
+	acpi_bus_unregister_driver(&quickstart_acpi_driver);
+
+	return ret;
+}
+
+module_init(quickstart_init);
+module_exit(quickstart_exit);


> 
> > diff --git a/drivers/acpi/Kconfig b/drivers/acpi/Kconfig
> > index f688c21..fb096ee 100644
> > --- a/drivers/acpi/Kconfig
> > +++ b/drivers/acpi/Kconfig
> > @@ -196,6 +196,14 @@ config ACPI_THERMAL
> >           recommended that this option be enabled, as your
> > processor(s)
> >           may be damaged without it.
> > 
> > +config ACPI_QUICKSTART
> > +       tristate "Quickstart"
> > +       default y
> > +       help
> > +         This driver add support for ACPI Direct Application Launch
> > +         also known as Hotstart(TM). Say yes here to have a entry in
> > +         sysfs telling which button was used to turn on the system.
> > +
> >  config ACPI_NUMA
> >         bool "NUMA support"
> >         depends on NUMA
> > diff --git a/drivers/acpi/Makefile b/drivers/acpi/Makefile
> > index f29812a..badb0a6 100644
> > --- a/drivers/acpi/Makefile
> > +++ b/drivers/acpi/Makefile
> > @@ -62,3 +62,4 @@ obj-$(CONFIG_ACPI_HOTPLUG_MEMORY)     +=
> > acpi_memhotplug.o
> >  obj-$(CONFIG_ACPI_PROCFS_POWER)        += cm_sbs.o
> >  obj-$(CONFIG_ACPI_SBS)         += sbs.o
> >  obj-$(CONFIG_ACPI_SBS)         += sbshc.o
> > +obj-$(CONFIG_ACPI_QUICKSTART)  += quickstart.o
> > diff --git a/drivers/acpi/quickstart.c b/drivers/acpi/quickstart.c
> > new file mode 100644
> > index 0000000..a4726f8
> > --- /dev/null
> > +++ b/drivers/acpi/quickstart.c
> > @@ -0,0 +1,418 @@
> > +/*
> > + *  quickstart.c - ACPI Direct App Launch driver
> > + *
> > + *
> > + *  Copyright (C) 2007 Angelo Arrifano <miknix@xxxxxxxxx>
> > + *
> > + *  Information gathered from disassebled dsdt and from here:
> > + *  "http://download.microsoft.com/download/9/c/5/
> > + *  9c5b2167-8017-4bae-9fde-d599bac8184a/DirAppLaunch_Vista.doc"
> > + *
> > + *  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., 59 Temple Place, Suite 330, Boston, MA
> > 02111-1307  USA
> > + *
> > + */
> > +
> > +#define QUICKSTART_VERSION "1.00"
> > +
> > +#include <linux/kernel.h>
> > +#include <linux/module.h>
> > +#include <linux/init.h>
> > +#include <linux/types.h>
> > +#include <acpi/acpi_drivers.h>
> > +#include <linux/platform_device.h>
> > +
> > +MODULE_AUTHOR("Angelo Arrifano");
> > +MODULE_DESCRIPTION("ACPI Direct App Launch driver");
> > +MODULE_LICENSE("GPL");
> > +
> > +static int enable_runtime_btns;
> > +module_param(enable_runtime_btns, int, 0);
> > +MODULE_PARM_DESC(enable_runtime_btns,
> > +                       "Enable reporting of runtime button presses
> > (hotkeys).\n");
> "enable_runtime_btns" is useless in this patch.
FIXED on the patch above
> 
> > +
> > +#define QUICKSTART_ACPI_DEVICE_NAME   "quickstart"
> > +#define QUICKSTART_ACPI_CLASS         "quickstart"
> > +#define QUICKSTART_ACPI_HID           "PNP0C32"
> > +
> > +#define QUICKSTART_PF_DRIVER_NAME     "quickstart"
> > +#define QUICKSTART_PF_DEVICE_NAME     "quickstart"
> > +#define QUICKSTART_PF_DEVATTR_NAME    "pressed_button"
> > +
> > +#define QUICKSTART_MAX_BTN_NAME_LEN   16
> > +
> > +struct quickstart_btn_list {
> > +       char *name;
> > +       struct quickstart_btn_list *next;
> > +};
> > +
> > +static struct quickstart_driver_data {
> > +       struct quickstart_btn_list *btn_lst;
> > +       struct quickstart_btn_list *pressed;
> > +} quickstart_data = {
> > +       .btn_lst = NULL,
> > +       .pressed = NULL,
> > +};
> > +
> > +/*
> > + * ACPI driver Structs
> > + */
> > +
> > +struct quickstart_acpi {
> > +       struct acpi_device *device;
> > +       struct quickstart_btn_list *btn;
> > +};
> > +static int quickstart_acpi_add(struct acpi_device *device);
> > +static int quickstart_acpi_remove(struct acpi_device *device, int
> > type);
> > +static const struct acpi_device_id  quickstart_device_ids[] = {
> > +       {QUICKSTART_ACPI_HID, 0},
> > +       {"", 0},
> > +};
> > +
> > +static struct acpi_driver quickstart_acpi_driver = {
> > +       .name = "quickstart",
> > +       .class = QUICKSTART_ACPI_CLASS,
> > +       .ids = quickstart_device_ids,
> > +       .ops = {
> > +                       .add = quickstart_acpi_add,
> > +                       .remove = quickstart_acpi_remove,
> > +               },
> > +};
> > +
> > +/*
> > + * Platform driver structs
> > + */
> > +static ssize_t buttons_show(struct device *dev,
> > +                                       struct device_attribute *attr,
> > +                                       char *buf);
> > +static ssize_t pressed_button_show(struct device *dev,
> > +                                       struct device_attribute *attr,
> > +                                       char *buf);
> > +static ssize_t pressed_button_store(struct device *dev,
> > +                                       struct device_attribute *attr,
> > +                                        const char *buf,
> > +                                        size_t count);
> > +static DEVICE_ATTR(pressed_button, 0666, pressed_button_show,
> > +                                        pressed_button_store);
> > +static DEVICE_ATTR(buttons, 0444, buttons_show, NULL);
> > +static struct platform_device *pf_device;
> > +static struct platform_driver pf_driver = {
> > +       .driver = {
> > +               .name = QUICKSTART_PF_DRIVER_NAME,
> > +               .owner = THIS_MODULE,
> > +       }
> > +};
> > +
> > +/*
> > + * Platform driver functions
> > + */
> > +static ssize_t buttons_show(struct device *dev,
> > +                                        struct device_attribute
> > *attr,
> > +                                        char *buf)
> > +{
> > +       int count = 0;
> > +       char tmpbuf[QUICKSTART_MAX_BTN_NAME_LEN + 2];
> > +       struct quickstart_btn_list *ptr = quickstart_data.btn_lst;
> > +      
> > +       if(!ptr)
> > +               return snprintf(buf, PAGE_SIZE, "none");
> > +
> > +       while(ptr && (count < PAGE_SIZE)){
> > +               if(ptr->name){
> > +                       strncpy(tmpbuf, ptr->name,
> > QUICKSTART_MAX_BTN_NAME_LEN);
> > +                       strcat(tmpbuf, "\n");
> > +                       count += snprintf(buf + count,
> > +                                       PAGE_SIZE - count,
> > +                                       tmpbuf);
> > +               }
> > +               ptr = ptr->next;
> > +       }
> > +      
> > +       return count;
> > +}
> > +
> > +static ssize_t pressed_button_show(struct device *dev,
> > +                                       struct device_attribute *attr,
> > +                                       char *buf)
> > +{
> > +       return snprintf(buf, PAGE_SIZE, "%s\n",
> > +
> > (quickstart_data.pressed?quickstart_data.pressed->name:"none"));
> > +}
> > +
> > +
> > +static ssize_t pressed_button_store(struct device *dev,
> > +                                        struct device_attribute
> > *attr,
> > +                                        const char *buf, size_t
> > count)
> > +{
> > +       if(count < 2)
> > +               return -EINVAL;
> > +
> > +       if(strncasecmp(buf, "none", 4) != 0)
> > +               return -EINVAL;
> > +
> > +       quickstart_data.pressed = NULL;
> > +       return count;
> > +}
> > +
> > +/* Hotstart Helper functions */
> > +static int quickstart_btnlst_add(struct quickstart_btn_list **data)
> > +{
> > +       struct quickstart_btn_list **ptr = &quickstart_data.btn_lst;
> > +      
> > +       while(*ptr)
> > +               ptr = &((*ptr)->next);
> > +              
> > +       *ptr = kzalloc(sizeof(struct quickstart_btn_list),
> > GFP_KERNEL);
> > +       if(!*ptr){
> > +               *data = NULL;
> > +               return(-ENOMEM);
> > +       }
> > +       *data = *ptr;
> > +
> > +       return 0;
> > +}
> > +
> > +static void quickstart_btnlst_free(void)
> > +{
> > +       struct quickstart_btn_list *ptr = quickstart_data.btn_lst;
> > +       struct quickstart_btn_list *lptr = NULL;
> > +      
> > +       if(!ptr)
> > +               return;
> > +
> > +       while(ptr){
> > +               lptr = ptr;
> > +               ptr = ptr->next;
> > +               if(lptr->name)
> > +                       kfree(lptr->name);
> > +               kfree(lptr);
> > +       }
> > +      
> > +       return;
> > +}
> > +
> > +/* ACPI Driver functions */
> > +static void quickstart_acpi_notify(acpi_handle handle, u32 event,
> > void *data)
> > +{
> > +       struct quickstart_acpi *quickstart = data;
> > +
> > +       if (!quickstart)
> > +               return;
> > +
> > +       /* There will be two events:
> > +        * 0x02 - A quickstart button was pressed while device was
> > off/sleeping.
> > +        * 0x08 - A quickstart button was pressed while device was up.
> > */
> > +       if((event|0x02) == event){
> This is wrong.
> If only two events are supported, it should be either 0x02 or 0x08.
> Checking bit 1 doesn't make sense here.
FIXED on the patch above
> 
> > +               quickstart_data.pressed = quickstart->btn;
> > +               printk("quickstart: Quickbutton %s pressed.\n",
> > +
> > acpi_device_bid(quickstart->device));
> > +       }
> > +
> > +       if(enable_runtime_btns && (event|0x08) == event){
> ...
> 
> > +               printk("quickstart: Hotkey %s pressed.\n",
> > +
> > acpi_device_bid(quickstart->device));
> > +       }
> > +
> >       
> > +       return;
> > +}
> > +
> at least we should export a notification to user space in the notify
> handler.

According to the Microsoft paper, runtime hot button reporting is optional. And
my laptop doesn't report them, so I don't have a way to test this..

> 
> > +static void quickstart_acpi_raise_notify(struct quickstart_acpi
> > *quickstart)
> > +{
> > +       acpi_status status;
> > +
> > +       /* TODO:
> > +        * We should receive a buffer telling the kind of button,
> > +        * eg: dvd, printer, www, ...
> > +        * but we dont care about it now. The important is to trigger
> > +        * pending notify events (The ones before booting). */
> > +       status = acpi_evaluate_object(quickstart->device->handle,
> > +                                       "GHID", NULL, NULL);
> It's wrong as GHID returns a buffer.

FIXED on the patch above
The returned buffer is a BYTE/WORD/DWORD with a decimal number telling the button
usage ID.
IMHO, the button usage ID is useless.

> 
> > +       if(ACPI_FAILURE(status))
> > +               printk("quickstart: Warning: GHID method failed,
> > events of pressed buttons may have been lost.\n");
> > +
> > +       return;
> > +}
> > +
> > +static int quickstart_acpi_config(struct quickstart_acpi *quickstart,
> > char *bid)
> > +{
> > +       int len = strlen(bid);
> > +       int ret;
> > +
> > +       /* Add button to list */
> > +       ret = quickstart_btnlst_add(&quickstart->btn);
> > +       if(ret)
> > +               return ret;
> > +      
> > +       quickstart->btn->name = kzalloc(len + 1, GFP_KERNEL);
> > +       if(!quickstart->btn->name){
> > +               quickstart_btnlst_free();
> > +               return -ENOMEM;
> > +       }
> > +       strcpy(quickstart->btn->name, bid);
> > +
> > +       return 0;
> > +}
> > +
> > +static int quickstart_acpi_add(struct acpi_device *device)
> > +{
> > +       int ret = 0;
> > +       acpi_status status = AE_OK;
> > +       struct quickstart_acpi *quickstart = NULL;     
> > +
> > +       if (!device)
> > +               return -EINVAL;
> > +
> > +       quickstart = kzalloc(sizeof(struct quickstart_acpi),
> > GFP_KERNEL);
> > +       if (!quickstart)
> > +               return -ENOMEM;
> > +
> > +       quickstart->device = device;
> > +       strcpy(acpi_device_name(device), QUICKSTART_ACPI_DEVICE_NAME);
> > +       strcpy(acpi_device_class(device), QUICKSTART_ACPI_CLASS);
> > +       acpi_driver_data(device) = quickstart;
> > +      
> > +       /* Add button to list and initialize some stuff */
> > +       ret = quickstart_acpi_config(quickstart,
> > acpi_device_bid(device));
> > +       if(ret)
> > +               goto fail_config;
> > +
> > +       status = acpi_install_notify_handler(device->handle,
> > +                                                ACPI_ALL_NOTIFY,
> > +
> > quickstart_acpi_notify, quickstart);
> > +       if(ACPI_FAILURE(status)){
> > +               printk("quickstart: Error installing notify handler
> > \n");
> > +               ret = -ENODEV;
> > +               goto fail_installnotify;
> > +       }      
> > +
> > +       /* Call device GHID method. This is a workaround to get a
> > notify
> > +        * event when hot button is pressed while laptop is powered
> > off.
> > +        * This works, even if the method fails. */
> > +       quickstart_acpi_raise_notify(quickstart);
> > +
> > +       return 0;
> > +
> > +fail_installnotify:
> > +fail_config:
> > +
> > +       kfree(quickstart);
> > +
> > +       return ret;
> > +}
> > +
> > +static int quickstart_acpi_remove(struct acpi_device *device, int
> > type)
> > +{
> > +       acpi_status status = 0;
> > +       struct quickstart_acpi *quickstart = NULL;
> > +
> > +       if (!device || !acpi_driver_data(device))
> > +               return -EINVAL;
> > +
> > +       quickstart = acpi_driver_data(device);
> > +
> > +       status = acpi_remove_notify_handler(device->handle,
> > +                                                ACPI_ALL_NOTIFY,
> > +                                           quickstart_acpi_notify);
> > +       if (ACPI_FAILURE(status))
> > +               printk("quickstart: Error removing notify handler\n");
> > +
> > +
> > +       kfree(quickstart);
> > +
> > +       return 0;
> > +}
> > +
> > +/* Module functions */
> > +
> > +static void quickstart_exit(void)
> > +{
> > +       device_remove_file(&pf_device->dev, &dev_attr_pressed_button);
> > +       device_remove_file(&pf_device->dev, &dev_attr_buttons);
> > +
> > +       platform_device_unregister(pf_device);
> > +
> > +       platform_driver_unregister(&pf_driver);
> > +
> > +       acpi_bus_unregister_driver(&quickstart_acpi_driver);
> > +      
> > +       quickstart_btnlst_free();
> > +
> > +       return;
> > +}
> > +
> > +static int __init quickstart_init(void)
> > +{
> > +       int ret;
> > +       acpi_status status = 0;
> > +
> > +       /* ACPI Check */
> > +       if(acpi_disabled)
> > +               return -ENODEV;
> > +
> > +       /* ACPI driver register */
> > +       status = acpi_bus_register_driver(&quickstart_acpi_driver);
> > +       if (!quickstart_data.btn_lst || status < 0)
> > +               return -ENODEV;
> > +
> we should not register platform devices and sys attributes if no PNP0C32
> device is available.
> A simple way is to set a flag in the acpi driver's .add method which
> means PNP0C32 device is found.
> And return -ENODEV if the flag is not set after
> acpi_bus_register_driver.

That is exactly whats going on here.
acpi_bus_register_driver() is called with quickstart_data.btn_lst=NULL . If no
devices are detected, the quickstart_data.btn_lst is not touched and continues
to be null.
So we can consider quickstart_data.btn_lst as the flag you were talking.
> 
> > +       /* Platform driver register */
> > +       ret = platform_driver_register(&pf_driver);
> > +       if(ret)
> > +               goto fail_pfdrv_reg;
> > + 
> > +       /* Platform device register */
> > +       pf_device = platform_device_alloc(QUICKSTART_PF_DEVICE_NAME,
> > -1);
> > +       if(!pf_device){
> > +               ret = -ENOMEM;
> > +               goto fail_pfdev_alloc;
> > +       }
> > +
> > +       ret = platform_device_register(pf_device);
> kobject (f7808880): tried to init an initialized object, something is
> seriously wrong.
> Pid: 1, comm: swapper Not tainted 2.6.25-rc6 #1
>  [<c04dad4f>] kobject_init+0x29/0x53
>  [<c05475fc>] device_initialize+0x1b/0x8b
>  [<c054ab62>] platform_device_register+0xb/0x13
>  [<c073dd50>] quickstart_init+0x5d/0xff
>  [<c0725427>] kernel_init+0x12b/0x265
>  [<c041eb8d>] schedule_tail+0x16/0x43
>  [<c040481e>] ret_from_fork+0x6/0x1c
>  [<c07252fc>] kernel_init+0x0/0x265
>  [<c07252fc>] kernel_init+0x0/0x265
>  [<c0405523>] kernel_thread_helper+0x7/0x10
>  =======================
> quickstart: ACPI Direct App Launch ver 1.00
> 
> platform_device_add should be called here.
> I suspect the patch is not well tested.

I don't understand this one.
pf_device is initialized twice? How?

> 
> > +       if(ret)
> > +               goto fail_pfdev_reg;
> > +
> > +       /* Create device sysfs file */
> > +       ret = device_create_file(&pf_device->dev,
> > &dev_attr_pressed_button);
> > +       if(ret)
> > +               goto fail_dev_file;
> > +
> > +       ret = device_create_file(&pf_device->dev, &dev_attr_buttons);
> > +       if(ret)
> > +               goto fail_dev_file2;
> > +
> > +       printk("quickstart: ACPI Direct App Launch ver %s\n",
> > QUICKSTART_VERSION);
> > +       if(enable_runtime_btns)
> > +               printk("quickstart: Runtime button reporting is
> > enabled.");
> > +      
> > +       return 0;
> > +
> > +fail_dev_file2:
> > +       device_remove_file(&pf_device->dev, &dev_attr_pressed_button);
> > +
> > +fail_dev_file:
> > +       platform_device_del(pf_device);
> > +
> > +fail_pfdev_reg:
> > +       platform_device_put(pf_device);
> > +
> > +fail_pfdev_alloc:
> > +       platform_driver_unregister(&pf_driver);
> > +
> > +fail_pfdrv_reg:
> > +       acpi_bus_unregister_driver(&quickstart_acpi_driver);
> > +
> > +       return ret;
> > +}
> > +
> > +module_init(quickstart_init);
> > +module_exit(quickstart_exit);
> There are some coding style problems.
> Please run checkpatch before submitting the patch.

FIXED on the patch above

> 
> thanks,
> rui
> 
> 

Thanks,
- -- 
Angelo Arrifano AKA MiKNiX
CSE Student at UBI, Portugal
Gentoo Linux AMD64 Arch Tester
Linwizard Developer
http://miknix.homelinux.com
PGP Pubkey 0x3D92BB0B
-----BEGIN PGP SIGNATURE-----
Version: GnuPG v2.0.7 (GNU/Linux)

iD8DBQFH6o0rNahyoD2SuwsRAvhDAJ9NJoi5c5tEOHaj+wlRi9S4lEhHrwCfdyhr
IAShcu6win9qeoSrhi6lYc8=
=7Uap
-----END PGP SIGNATURE-----
--
To unsubscribe from this list: send the line "unsubscribe linux-acpi" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at  http://vger.kernel.org/majordomo-info.html

[Index of Archives]     [Linux IBM ACPI]     [Linux Power Management]     [Linux Kernel]     [Linux Laptop]     [Kernel Newbies]     [Share Photos]     [Security]     [Netfilter]     [Bugtraq]     [Yosemite News]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Samba]     [Video 4 Linux]     [Device Mapper]     [Linux Resources]

  Powered by Linux