* [PATCH] toshiba_acpi 0.17
@ 2004-01-27 14:55 John Belmonte
[not found] ` <40167BD8.9000107-wanGne27zNesTnJN9+BGXg@public.gmane.org>
0 siblings, 1 reply; 2+ messages in thread
From: John Belmonte @ 2004-01-27 14:55 UTC (permalink / raw)
To: Brown, Len; +Cc: acpi-devel-5NWGOfrQmneRv+LV9MX5uipxlwaOVQ5f
[-- Attachment #1: Type: text/plain, Size: 616 bytes --]
Len,
Please apply this patch, which will yield version 0.17 of the
toshiba_acpi driver against the head of the Linux 2.4 and 2.5
development trees.
Changelog:
* Fix remote chance of invalid buffer access in write_video.
* Support alternate HCI method path (recent "Phoenix BIOS" Toshiba's).
* Signal more proc-write errors.
* On proc-reads, report errors via printk instead of proc output.
* Add log level and driver name prefix to all printk's.
* Add missing __init and __exit function attributes.
* Be explicit about vars for which code relies on zero-init of BSS.
Regards,
-John
[-- Attachment #2: toshiba_acpi_0.17-linux_head.patch --]
[-- Type: text/x-patch, Size: 5893 bytes --]
diff -urN old/drivers/acpi/toshiba_acpi.c new/drivers/acpi/toshiba_acpi.c
--- old/drivers/acpi/toshiba_acpi.c 2004-01-27 09:09:12.000000000 -0500
+++ new/drivers/acpi/toshiba_acpi.c 2004-01-27 09:16:22.000000000 -0500
@@ -2,7 +2,7 @@
* toshiba_acpi.c - Toshiba Laptop ACPI Extras
*
*
- * Copyright (C) 2002-2003 John Belmonte
+ * Copyright (C) 2002-2004 John Belmonte
*
* 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
@@ -33,7 +33,7 @@
*
*/
-#define TOSHIBA_ACPI_VERSION "0.16"
+#define TOSHIBA_ACPI_VERSION "0.17"
#define PROC_INTERFACE_VERSION 1
#include <linux/kernel.h>
@@ -48,9 +48,15 @@
MODULE_DESCRIPTION("Toshiba Laptop ACPI Extras Driver");
MODULE_LICENSE("GPL");
+#define MY_LOGPREFIX "toshiba_acpi: "
+#define MY_ERR KERN_ERR MY_LOGPREFIX
+#define MY_NOTICE KERN_NOTICE MY_LOGPREFIX
+#define MY_INFO KERN_INFO MY_LOGPREFIX
+
/* Toshiba ACPI method paths */
#define METHOD_LCD_BRIGHTNESS "\\_SB_.PCI0.VGA_.LCD_._BCM"
-#define METHOD_HCI "\\_SB_.VALD.GHCI"
+#define METHOD_HCI_1 "\\_SB_.VALD.GHCI"
+#define METHOD_HCI_2 "\\_SB_.VALZ.GHCI"
#define METHOD_VIDEO_OUT "\\_SB_.VALX.DSSX"
/* Toshiba HCI interface definitions
@@ -121,6 +127,16 @@
*/
static int
+is_valid_acpi_path(const char* methodName)
+{
+ acpi_handle handle;
+ acpi_status status;
+
+ status = acpi_get_handle(0, (char*)methodName, &handle);
+ return !ACPI_FAILURE(status);
+}
+
+static int
write_acpi_int(const char* methodName, int val)
{
struct acpi_object_list params;
@@ -154,6 +170,8 @@
}
#endif
+static const char* method_hci /*= 0*/;
+
/* Perform a raw HCI call. Here we don't care about input or output buffer
* format.
*/
@@ -177,7 +195,7 @@
results.length = sizeof(out_objs);
results.pointer = out_objs;
- status = acpi_evaluate_object(0, METHOD_HCI, ¶ms,
+ status = acpi_evaluate_object(0, (char*)method_hci, ¶ms,
&results);
if ((status == AE_OK) && (out_objs->package.count <= HCI_WORDS)) {
for (i = 0; i < out_objs->package.count; ++i) {
@@ -215,7 +233,7 @@
return status;
}
-static struct proc_dir_entry* toshiba_proc_dir;
+static struct proc_dir_entry* toshiba_proc_dir /*= 0*/;
static int force_fan;
static int last_key_event;
static int key_event_valid;
@@ -270,7 +288,7 @@
p += sprintf(p, "brightness_levels: %d\n",
HCI_LCD_BRIGHTNESS_LEVELS);
} else {
- p += sprintf(p, "ERROR\n");
+ printk(MY_ERR "Error reading LCD brightness\n");
}
return p;
@@ -310,7 +328,7 @@
p += sprintf(p, "crt_out: %d\n", is_crt);
p += sprintf(p, "tv_out: %d\n", is_tv);
} else {
- p += sprintf(p, "ERROR\n");
+ printk(MY_ERR "Error reading video out status\n");
}
return p;
@@ -320,25 +338,31 @@
write_video(const char* buffer, unsigned long count)
{
int value;
- const char* buffer_end = buffer + count;
+ int remain = count;
int lcd_out = -1;
int crt_out = -1;
int tv_out = -1;
u32 hci_result;
int video_out;
- /* scan expression. Multiple expressions may be delimited with ; */
- do {
- if (snscanf(buffer, count, " lcd_out : %i", &value) == 1)
+ /* scan expression. Multiple expressions may be delimited with ;
+ *
+ * NOTE: to keep scanning simple, invalid fields are ignored
+ */
+ while (remain) {
+ if (snscanf(buffer, remain, " lcd_out : %i", &value) == 1)
lcd_out = value & 1;
- else if (snscanf(buffer, count, " crt_out : %i", &value) == 1)
+ else if (snscanf(buffer, remain, " crt_out : %i", &value) == 1)
crt_out = value & 1;
- else if (snscanf(buffer, count, " tv_out : %i", &value) == 1)
+ else if (snscanf(buffer, remain, " tv_out : %i", &value) == 1)
tv_out = value & 1;
/* advance to one character past the next ; */
- do ++buffer;
- while ((buffer < buffer_end) && (*(buffer-1) != ';'));
- } while (buffer < buffer_end);
+ do {
+ ++buffer;
+ --remain;
+ }
+ while (remain && *(buffer-1) != ';');
+ }
hci_read1(HCI_VIDEO_OUT, &video_out, &hci_result);
if (hci_result == HCI_SUCCESS) {
@@ -353,6 +377,8 @@
* video setting if something changed. */
if (new_video_out != video_out)
write_acpi_int(METHOD_VIDEO_OUT, new_video_out);
+ } else {
+ return -EFAULT;
}
return count;
@@ -369,7 +395,7 @@
p += sprintf(p, "running: %d\n", (value > 0));
p += sprintf(p, "force_on: %d\n", force_fan);
} else {
- p += sprintf(p, "ERROR\n");
+ printk(MY_ERR "Error reading fan status\n");
}
return p;
@@ -413,8 +439,9 @@
* some machines where system events sporadically
* become disabled. */
hci_write1(HCI_SYSTEM_EVENT, 1, &hci_result);
+ printk(MY_NOTICE "Re-enabled hotkeys\n");
} else {
- p += sprintf(p, "ERROR\n");
+ printk(MY_ERR "Error reading hotkey status\n");
goto end;
}
}
@@ -465,7 +492,7 @@
{ 0 , 0 , 0 },
};
-static acpi_status
+static acpi_status __init
add_device(void)
{
struct proc_dir_entry* proc;
@@ -483,7 +510,7 @@
return(AE_OK);
}
-static acpi_status
+static acpi_status __exit
remove_device(void)
{
ProcItem* item;
@@ -497,15 +524,19 @@
toshiba_acpi_init(void)
{
acpi_status status = AE_OK;
- int value;
u32 hci_result;
- /* simple device detection: try reading an HCI register */
- hci_read1(HCI_LCD_BRIGHTNESS, &value, &hci_result);
- if (hci_result != HCI_SUCCESS)
+ /* simple device detection: look for HCI method */
+ if (is_valid_acpi_path(METHOD_HCI_1))
+ method_hci = METHOD_HCI_1;
+ else if (is_valid_acpi_path(METHOD_HCI_2))
+ method_hci = METHOD_HCI_2;
+ else
return -ENODEV;
- printk("Toshiba Laptop ACPI Extras version %s\n", TOSHIBA_ACPI_VERSION);
+ printk(MY_INFO "Toshiba Laptop ACPI Extras version %s\n",
+ TOSHIBA_ACPI_VERSION);
+ printk(MY_INFO " HCI method: %s\n", method_hci);
force_fan = 0;
key_event_valid = 0;
^ permalink raw reply [flat|nested] 2+ messages in thread
* Re: [PATCH] toshiba_acpi 0.17
[not found] ` <40167BD8.9000107-wanGne27zNesTnJN9+BGXg@public.gmane.org>
@ 2004-02-07 7:19 ` Len Brown
0 siblings, 0 replies; 2+ messages in thread
From: Len Brown @ 2004-02-07 7:19 UTC (permalink / raw)
To: John Belmonte; +Cc: ACPI Developers
Accepted.
thanks,
-Len
On Tue, 2004-01-27 at 09:55, John Belmonte wrote:
> Len,
>
> Please apply this patch, which will yield version 0.17 of the
> toshiba_acpi driver against the head of the Linux 2.4 and 2.5
> development trees.
>
> Changelog:
>
> * Fix remote chance of invalid buffer access in write_video.
>
> * Support alternate HCI method path (recent "Phoenix BIOS" Toshiba's).
>
> * Signal more proc-write errors.
>
> * On proc-reads, report errors via printk instead of proc output.
>
> * Add log level and driver name prefix to all printk's.
>
> * Add missing __init and __exit function attributes.
>
> * Be explicit about vars for which code relies on zero-init of BSS.
>
>
> Regards,
> -John
>
> ______________________________________________________________________
>
> diff -urN old/drivers/acpi/toshiba_acpi.c new/drivers/acpi/toshiba_acpi.c
> --- old/drivers/acpi/toshiba_acpi.c 2004-01-27 09:09:12.000000000 -0500
> +++ new/drivers/acpi/toshiba_acpi.c 2004-01-27 09:16:22.000000000 -0500
> @@ -2,7 +2,7 @@
> * toshiba_acpi.c - Toshiba Laptop ACPI Extras
> *
> *
> - * Copyright (C) 2002-2003 John Belmonte
> + * Copyright (C) 2002-2004 John Belmonte
> *
> * 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
> @@ -33,7 +33,7 @@
> *
> */
>
> -#define TOSHIBA_ACPI_VERSION "0.16"
> +#define TOSHIBA_ACPI_VERSION "0.17"
> #define PROC_INTERFACE_VERSION 1
>
> #include <linux/kernel.h>
> @@ -48,9 +48,15 @@
> MODULE_DESCRIPTION("Toshiba Laptop ACPI Extras Driver");
> MODULE_LICENSE("GPL");
>
> +#define MY_LOGPREFIX "toshiba_acpi: "
> +#define MY_ERR KERN_ERR MY_LOGPREFIX
> +#define MY_NOTICE KERN_NOTICE MY_LOGPREFIX
> +#define MY_INFO KERN_INFO MY_LOGPREFIX
> +
> /* Toshiba ACPI method paths */
> #define METHOD_LCD_BRIGHTNESS "\\_SB_.PCI0.VGA_.LCD_._BCM"
> -#define METHOD_HCI "\\_SB_.VALD.GHCI"
> +#define METHOD_HCI_1 "\\_SB_.VALD.GHCI"
> +#define METHOD_HCI_2 "\\_SB_.VALZ.GHCI"
> #define METHOD_VIDEO_OUT "\\_SB_.VALX.DSSX"
>
> /* Toshiba HCI interface definitions
> @@ -121,6 +127,16 @@
> */
>
> static int
> +is_valid_acpi_path(const char* methodName)
> +{
> + acpi_handle handle;
> + acpi_status status;
> +
> + status = acpi_get_handle(0, (char*)methodName, &handle);
> + return !ACPI_FAILURE(status);
> +}
> +
> +static int
> write_acpi_int(const char* methodName, int val)
> {
> struct acpi_object_list params;
> @@ -154,6 +170,8 @@
> }
> #endif
>
> +static const char* method_hci /*= 0*/;
> +
> /* Perform a raw HCI call. Here we don't care about input or output buffer
> * format.
> */
> @@ -177,7 +195,7 @@
> results.length = sizeof(out_objs);
> results.pointer = out_objs;
>
> - status = acpi_evaluate_object(0, METHOD_HCI, ¶ms,
> + status = acpi_evaluate_object(0, (char*)method_hci, ¶ms,
> &results);
> if ((status == AE_OK) && (out_objs->package.count <= HCI_WORDS)) {
> for (i = 0; i < out_objs->package.count; ++i) {
> @@ -215,7 +233,7 @@
> return status;
> }
>
> -static struct proc_dir_entry* toshiba_proc_dir;
> +static struct proc_dir_entry* toshiba_proc_dir /*= 0*/;
> static int force_fan;
> static int last_key_event;
> static int key_event_valid;
> @@ -270,7 +288,7 @@
> p += sprintf(p, "brightness_levels: %d\n",
> HCI_LCD_BRIGHTNESS_LEVELS);
> } else {
> - p += sprintf(p, "ERROR\n");
> + printk(MY_ERR "Error reading LCD brightness\n");
> }
>
> return p;
> @@ -310,7 +328,7 @@
> p += sprintf(p, "crt_out: %d\n", is_crt);
> p += sprintf(p, "tv_out: %d\n", is_tv);
> } else {
> - p += sprintf(p, "ERROR\n");
> + printk(MY_ERR "Error reading video out status\n");
> }
>
> return p;
> @@ -320,25 +338,31 @@
> write_video(const char* buffer, unsigned long count)
> {
> int value;
> - const char* buffer_end = buffer + count;
> + int remain = count;
> int lcd_out = -1;
> int crt_out = -1;
> int tv_out = -1;
> u32 hci_result;
> int video_out;
>
> - /* scan expression. Multiple expressions may be delimited with ; */
> - do {
> - if (snscanf(buffer, count, " lcd_out : %i", &value) == 1)
> + /* scan expression. Multiple expressions may be delimited with ;
> + *
> + * NOTE: to keep scanning simple, invalid fields are ignored
> + */
> + while (remain) {
> + if (snscanf(buffer, remain, " lcd_out : %i", &value) == 1)
> lcd_out = value & 1;
> - else if (snscanf(buffer, count, " crt_out : %i", &value) == 1)
> + else if (snscanf(buffer, remain, " crt_out : %i", &value) == 1)
> crt_out = value & 1;
> - else if (snscanf(buffer, count, " tv_out : %i", &value) == 1)
> + else if (snscanf(buffer, remain, " tv_out : %i", &value) == 1)
> tv_out = value & 1;
> /* advance to one character past the next ; */
> - do ++buffer;
> - while ((buffer < buffer_end) && (*(buffer-1) != ';'));
> - } while (buffer < buffer_end);
> + do {
> + ++buffer;
> + --remain;
> + }
> + while (remain && *(buffer-1) != ';');
> + }
>
> hci_read1(HCI_VIDEO_OUT, &video_out, &hci_result);
> if (hci_result == HCI_SUCCESS) {
> @@ -353,6 +377,8 @@
> * video setting if something changed. */
> if (new_video_out != video_out)
> write_acpi_int(METHOD_VIDEO_OUT, new_video_out);
> + } else {
> + return -EFAULT;
> }
>
> return count;
> @@ -369,7 +395,7 @@
> p += sprintf(p, "running: %d\n", (value > 0));
> p += sprintf(p, "force_on: %d\n", force_fan);
> } else {
> - p += sprintf(p, "ERROR\n");
> + printk(MY_ERR "Error reading fan status\n");
> }
>
> return p;
> @@ -413,8 +439,9 @@
> * some machines where system events sporadically
> * become disabled. */
> hci_write1(HCI_SYSTEM_EVENT, 1, &hci_result);
> + printk(MY_NOTICE "Re-enabled hotkeys\n");
> } else {
> - p += sprintf(p, "ERROR\n");
> + printk(MY_ERR "Error reading hotkey status\n");
> goto end;
> }
> }
> @@ -465,7 +492,7 @@
> { 0 , 0 , 0 },
> };
>
> -static acpi_status
> +static acpi_status __init
> add_device(void)
> {
> struct proc_dir_entry* proc;
> @@ -483,7 +510,7 @@
> return(AE_OK);
> }
>
> -static acpi_status
> +static acpi_status __exit
> remove_device(void)
> {
> ProcItem* item;
> @@ -497,15 +524,19 @@
> toshiba_acpi_init(void)
> {
> acpi_status status = AE_OK;
> - int value;
> u32 hci_result;
>
> - /* simple device detection: try reading an HCI register */
> - hci_read1(HCI_LCD_BRIGHTNESS, &value, &hci_result);
> - if (hci_result != HCI_SUCCESS)
> + /* simple device detection: look for HCI method */
> + if (is_valid_acpi_path(METHOD_HCI_1))
> + method_hci = METHOD_HCI_1;
> + else if (is_valid_acpi_path(METHOD_HCI_2))
> + method_hci = METHOD_HCI_2;
> + else
> return -ENODEV;
>
> - printk("Toshiba Laptop ACPI Extras version %s\n", TOSHIBA_ACPI_VERSION);
> + printk(MY_INFO "Toshiba Laptop ACPI Extras version %s\n",
> + TOSHIBA_ACPI_VERSION);
> + printk(MY_INFO " HCI method: %s\n", method_hci);
>
> force_fan = 0;
> key_event_valid = 0;
-------------------------------------------------------
The SF.Net email is sponsored by EclipseCon 2004
Premiere Conference on Open Tools Development and Integration
See the breadth of Eclipse activity. February 3-5 in Anaheim, CA.
http://www.eclipsecon.org/osdn
^ permalink raw reply [flat|nested] 2+ messages in thread
end of thread, other threads:[~2004-02-07 7:19 UTC | newest]
Thread overview: 2+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2004-01-27 14:55 [PATCH] toshiba_acpi 0.17 John Belmonte
[not found] ` <40167BD8.9000107-wanGne27zNesTnJN9+BGXg@public.gmane.org>
2004-02-07 7:19 ` Len Brown
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox