[PATCH] Allow acpi_osi="Winows XY" overrides

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

 



This is an ASL example from a HP nx9420.
The first part tries to evaluate the OS with the old
\_OS method (you can already fake the OS with acpi_os_name="Microsoft
Windows"). The second part makes use of the new _OSI function.
Overriding the _OSI provided string by OS was not yet implemented, that
is what the patch does.

For this BIOS it already was possible to fake the OS string for
"Microsoft Windows", "Microsoft WindowsME: Millennium Edition",
"Microsoft Windows NT" by e.g.: 
acpi_osi="" acpi_os_name="Microsoft Windows"

now you also can (currently it would use "Windows 2001 SP2") also fake
e.g.:
acpi_osi="Windows 2006"      (acpi_os_name="" -> not explicitly needed 
                              here, maybe on other AML implementations)


------------------------------------------------------------
    Name (C015, Package (0x03)
    {
        "Microsoft Windows",
        "Microsoft WindowsME: Millennium Edition",
        "Microsoft Windows NT"
    })
...

            Store (\_OS, Local0)
            Store (SizeOf (C015), Local7)
            Store (0x00, Local1)
            While (LAnd (LLess (Local1, Local7), LEqual (C014, 0xFFFFFFFD)))
            {
                Store (\C011 (Local0, DerefOf (Index (C015, Local1))), Local2)
                If (Local2)
                {
                    Add (Local1, 0x01, C014)
                }
                Else
                {
                    Increment (Local1)
                }
            }

            If (LEqual (C014, 0xFFFFFFFD))
            {
                Store (0x00, C014)
            }

            If (LOr (LEqual (C014, 0x00), LEqual (C014, 0x03)))
            {
                If (CondRefOf (\_OSI, Local0))
                {
                    If (\_OSI ("Windows 2001"))
                    {
                        Store (0x04, C014)
                    }

                    If (\_OSI ("Windows 2001 SP1"))
                    {
                        Store (0x04, C014)
                    }

                    If (\_OSI ("Windows 2001 SP2"))
                    {
                        Store (0x05, C014)
                    }

                    If (\_OSI ("Windows 2006"))
                    {
                        Store (0x06, C014)
                    }
                }
            }
------------------------------------------------------------

The patch is based on a SUSE 2.6.16.21 kernel. There shouldn't have
changed that much. It needs manual modifications as it touches ACPICA
code anyway.



Allow acpi_osi="Winows XY" overrides

Nearly all BIOSes have special code paths for special Windows
OS verions. This patch tells ACPI BIOS parts which OS it should
assume and might workaround BIOS issues.

Signed-off-by: Thomas Renninger <mail@xxxxxxxxxxxx>


 drivers/acpi/osl.c              |   19 +++++++++++++------
 drivers/acpi/utilities/uteval.c |   22 +++++++++++++++-------
 include/acpi/acglobal.h         |    3 +++
 3 files changed, 31 insertions(+), 13 deletions(-)

Index: linux-2.6.16-SLES10_GA_BRANCH/drivers/acpi/osl.c
===================================================================
--- linux-2.6.16-SLES10_GA_BRANCH.orig/drivers/acpi/osl.c
+++ linux-2.6.16-SLES10_GA_BRANCH/drivers/acpi/osl.c
@@ -228,8 +228,6 @@ acpi_os_get_physical_address(void *virt,
 }
 #endif
 
-#define ACPI_MAX_OVERRIDE_LEN 100
-
 static char acpi_os_name[ACPI_MAX_OVERRIDE_LEN];
 
 acpi_status
@@ -1067,17 +1065,26 @@ __setup("acpi_os_name=", acpi_os_name_se
 /*
  * _OSI control
  * empty string disables _OSI
- * TBD additional string adds to _OSI
  */
+char acpi_osi_override[ACPI_MAX_OVERRIDE_LEN] = "\0";
+
 static int __init acpi_osi_setup(char *str)
 {
+	int len;
+
 	if (str == NULL || *str == '\0') {
 		printk(KERN_INFO PREFIX "_OSI method disabled\n");
 		acpi_gbl_create_osi_method = FALSE;
 	} else {
-		/* TBD */
-		printk(KERN_ERR PREFIX "_OSI additional string ignored -- %s\n",
-		       str);
+		len = strlen(str);
+		if(len > ACPI_MAX_OVERRIDE_LEN)
+			printk(KERN_ERR PREFIX "acpi_osi_name too long"
+			       "(%d, max: %d)\n", len, ACPI_MAX_OVERRIDE_LEN);
+		else{
+			strcpy(acpi_osi_override, str);
+			printk(KERN_INFO PREFIX "_OSI string set to: %s\n",
+			       acpi_osi_override);
+		}
 	}
 
 	return 1;
Index: linux-2.6.16-SLES10_GA_BRANCH/drivers/acpi/utilities/uteval.c
===================================================================
--- linux-2.6.16-SLES10_GA_BRANCH.orig/drivers/acpi/utilities/uteval.c
+++ linux-2.6.16-SLES10_GA_BRANCH/drivers/acpi/utilities/uteval.c
@@ -93,17 +93,25 @@ acpi_status acpi_ut_osi_implementation(s
 
 	/* Compare input string to table of supported strings */
 
-	for (i = 0; i < ACPI_NUM_OSI_STRINGS; i++) {
-		if (!ACPI_STRCMP(string_desc->string.pointer,
-				 ACPI_CAST_PTR(char,
+	if (acpi_osi_override[0] == '\0'){
+		for (i = 0; i < ACPI_NUM_OSI_STRINGS; i++) {
+			if (!ACPI_STRCMP(string_desc->string.pointer,
+					 ACPI_CAST_PTR(char,
 					       acpi_gbl_valid_osi_strings[i])))
-		{
-			/* This string is supported */
+			{
+				/* This string is supported */
 
-			return_desc->integer.value = 0xFFFFFFFF;
-			break;
+				return_desc->integer.value = 0xFFFFFFFF;
+				break;
+			}
 		}
 	}
+	else{
+		if (!ACPI_STRCMP(string_desc->string.pointer,
+				 acpi_osi_override))
+			/* This string is supported */
+			return_desc->integer.value = 0xFFFFFFFF;
+	}
 
 	walk_state->return_desc = return_desc;
 	return_ACPI_STATUS(AE_CTRL_TERMINATE);
Index: linux-2.6.16-SLES10_GA_BRANCH/include/acpi/acglobal.h
===================================================================
--- linux-2.6.16-SLES10_GA_BRANCH.orig/include/acpi/acglobal.h
+++ linux-2.6.16-SLES10_GA_BRANCH/include/acpi/acglobal.h
@@ -246,6 +246,9 @@ extern const struct acpi_opcode_info acp
 extern const char *acpi_gbl_region_types[ACPI_NUM_PREDEFINED_REGIONS];
 extern const char *acpi_gbl_valid_osi_strings[ACPI_NUM_OSI_STRINGS];
 
+#define ACPI_MAX_OVERRIDE_LEN 100
+extern char acpi_osi_override[ACPI_MAX_OVERRIDE_LEN];
+
 /*****************************************************************************
  *
  * Namespace globals

Allow acpi_osi="Winows XY" overrides

Nearly all BIOSes have special code paths for special Windows
OS verions. This patch tells ACPI BIOS parts which OS it should
assume and might workaround BIOS issues.

Signed-off-by: Thomas Renninger <mail@xxxxxxxxxxxx>


 drivers/acpi/osl.c              |   19 +++++++++++++------
 drivers/acpi/utilities/uteval.c |   22 +++++++++++++++-------
 include/acpi/acglobal.h         |    3 +++
 3 files changed, 31 insertions(+), 13 deletions(-)

Index: linux-2.6.16-SLES10_GA_BRANCH/drivers/acpi/osl.c
===================================================================
--- linux-2.6.16-SLES10_GA_BRANCH.orig/drivers/acpi/osl.c
+++ linux-2.6.16-SLES10_GA_BRANCH/drivers/acpi/osl.c
@@ -228,8 +228,6 @@ acpi_os_get_physical_address(void *virt,
 }
 #endif
 
-#define ACPI_MAX_OVERRIDE_LEN 100
-
 static char acpi_os_name[ACPI_MAX_OVERRIDE_LEN];
 
 acpi_status
@@ -1067,17 +1065,26 @@ __setup("acpi_os_name=", acpi_os_name_se
 /*
  * _OSI control
  * empty string disables _OSI
- * TBD additional string adds to _OSI
  */
+char acpi_osi_override[ACPI_MAX_OVERRIDE_LEN] = "\0";
+
 static int __init acpi_osi_setup(char *str)
 {
+	int len;
+
 	if (str == NULL || *str == '\0') {
 		printk(KERN_INFO PREFIX "_OSI method disabled\n");
 		acpi_gbl_create_osi_method = FALSE;
 	} else {
-		/* TBD */
-		printk(KERN_ERR PREFIX "_OSI additional string ignored -- %s\n",
-		       str);
+		len = strlen(str);
+		if(len > ACPI_MAX_OVERRIDE_LEN)
+			printk(KERN_ERR PREFIX "acpi_osi_name too long"
+			       "(%d, max: %d)\n", len, ACPI_MAX_OVERRIDE_LEN);
+		else{
+			strcpy(acpi_osi_override, str);
+			printk(KERN_INFO PREFIX "_OSI string set to: %s\n",
+			       acpi_osi_override);
+		}
 	}
 
 	return 1;
Index: linux-2.6.16-SLES10_GA_BRANCH/drivers/acpi/utilities/uteval.c
===================================================================
--- linux-2.6.16-SLES10_GA_BRANCH.orig/drivers/acpi/utilities/uteval.c
+++ linux-2.6.16-SLES10_GA_BRANCH/drivers/acpi/utilities/uteval.c
@@ -93,17 +93,25 @@ acpi_status acpi_ut_osi_implementation(s
 
 	/* Compare input string to table of supported strings */
 
-	for (i = 0; i < ACPI_NUM_OSI_STRINGS; i++) {
-		if (!ACPI_STRCMP(string_desc->string.pointer,
-				 ACPI_CAST_PTR(char,
+	if (acpi_osi_override[0] == '\0'){
+		for (i = 0; i < ACPI_NUM_OSI_STRINGS; i++) {
+			if (!ACPI_STRCMP(string_desc->string.pointer,
+					 ACPI_CAST_PTR(char,
 					       acpi_gbl_valid_osi_strings[i])))
-		{
-			/* This string is supported */
+			{
+				/* This string is supported */
 
-			return_desc->integer.value = 0xFFFFFFFF;
-			break;
+				return_desc->integer.value = 0xFFFFFFFF;
+				break;
+			}
 		}
 	}
+	else{
+		if (!ACPI_STRCMP(string_desc->string.pointer,
+				 acpi_osi_override))
+			/* This string is supported */
+			return_desc->integer.value = 0xFFFFFFFF;
+	}
 
 	walk_state->return_desc = return_desc;
 	return_ACPI_STATUS(AE_CTRL_TERMINATE);
Index: linux-2.6.16-SLES10_GA_BRANCH/include/acpi/acglobal.h
===================================================================
--- linux-2.6.16-SLES10_GA_BRANCH.orig/include/acpi/acglobal.h
+++ linux-2.6.16-SLES10_GA_BRANCH/include/acpi/acglobal.h
@@ -246,6 +246,9 @@ extern const struct acpi_opcode_info acp
 extern const char *acpi_gbl_region_types[ACPI_NUM_PREDEFINED_REGIONS];
 extern const char *acpi_gbl_valid_osi_strings[ACPI_NUM_OSI_STRINGS];
 
+#define ACPI_MAX_OVERRIDE_LEN 100
+extern char acpi_osi_override[ACPI_MAX_OVERRIDE_LEN];
+
 /*****************************************************************************
  *
  * Namespace globals

[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