1 /****************************************************************************** 2 * 3 * Module Name: utosi - Support for the _OSI predefined control method 4 * 5 *****************************************************************************/ 6 7 /* 8 * Copyright (C) 2000 - 2013, Intel Corp. 9 * All rights reserved. 10 * 11 * Redistribution and use in source and binary forms, with or without 12 * modification, are permitted provided that the following conditions 13 * are met: 14 * 1. Redistributions of source code must retain the above copyright 15 * notice, this list of conditions, and the following disclaimer, 16 * without modification. 17 * 2. Redistributions in binary form must reproduce at minimum a disclaimer 18 * substantially similar to the "NO WARRANTY" disclaimer below 19 * ("Disclaimer") and any redistribution must be conditioned upon 20 * including a substantially similar Disclaimer requirement for further 21 * binary redistribution. 22 * 3. Neither the names of the above-listed copyright holders nor the names 23 * of any contributors may be used to endorse or promote products derived 24 * from this software without specific prior written permission. 25 * 26 * Alternatively, this software may be distributed under the terms of the 27 * GNU General Public License ("GPL") version 2 as published by the Free 28 * Software Foundation. 29 * 30 * NO WARRANTY 31 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 32 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 33 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR 34 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT 35 * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 36 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 37 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 38 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, 39 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING 40 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 41 * POSSIBILITY OF SUCH DAMAGES. 42 */ 43 44 #define __UTOSI_C__ 45 46 #include <contrib/dev/acpica/include/acpi.h> 47 #include <contrib/dev/acpica/include/accommon.h> 48 49 50 #define _COMPONENT ACPI_UTILITIES 51 ACPI_MODULE_NAME ("utosi") 52 53 /* 54 * Strings supported by the _OSI predefined control method (which is 55 * implemented internally within this module.) 56 * 57 * March 2009: Removed "Linux" as this host no longer wants to respond true 58 * for this string. Basically, the only safe OS strings are windows-related 59 * and in many or most cases represent the only test path within the 60 * BIOS-provided ASL code. 61 * 62 * The last element of each entry is used to track the newest version of 63 * Windows that the BIOS has requested. 64 */ 65 static ACPI_INTERFACE_INFO AcpiDefaultSupportedInterfaces[] = 66 { 67 /* Operating System Vendor Strings */ 68 69 {"Windows 2000", NULL, 0, ACPI_OSI_WIN_2000}, /* Windows 2000 */ 70 {"Windows 2001", NULL, 0, ACPI_OSI_WIN_XP}, /* Windows XP */ 71 {"Windows 2001 SP1", NULL, 0, ACPI_OSI_WIN_XP_SP1}, /* Windows XP SP1 */ 72 {"Windows 2001.1", NULL, 0, ACPI_OSI_WINSRV_2003}, /* Windows Server 2003 */ 73 {"Windows 2001 SP2", NULL, 0, ACPI_OSI_WIN_XP_SP2}, /* Windows XP SP2 */ 74 {"Windows 2001.1 SP1", NULL, 0, ACPI_OSI_WINSRV_2003_SP1}, /* Windows Server 2003 SP1 - Added 03/2006 */ 75 {"Windows 2006", NULL, 0, ACPI_OSI_WIN_VISTA}, /* Windows Vista - Added 03/2006 */ 76 {"Windows 2006.1", NULL, 0, ACPI_OSI_WINSRV_2008}, /* Windows Server 2008 - Added 09/2009 */ 77 {"Windows 2006 SP1", NULL, 0, ACPI_OSI_WIN_VISTA_SP1}, /* Windows Vista SP1 - Added 09/2009 */ 78 {"Windows 2006 SP2", NULL, 0, ACPI_OSI_WIN_VISTA_SP2}, /* Windows Vista SP2 - Added 09/2010 */ 79 {"Windows 2009", NULL, 0, ACPI_OSI_WIN_7}, /* Windows 7 and Server 2008 R2 - Added 09/2009 */ 80 {"Windows 2012", NULL, 0, ACPI_OSI_WIN_8}, /* Windows 8 and Server 2012 - Added 08/2012 */ 81 82 /* Feature Group Strings */ 83 84 {"Extended Address Space Descriptor", NULL, ACPI_OSI_FEATURE, 0}, 85 86 /* 87 * All "optional" feature group strings (features that are implemented 88 * by the host) should be dynamically modified to VALID by the host via 89 * AcpiInstallInterface or AcpiUpdateInterfaces. Such optional feature 90 * group strings are set as INVALID by default here. 91 */ 92 93 {"Module Device", NULL, ACPI_OSI_OPTIONAL_FEATURE, 0}, 94 {"Processor Device", NULL, ACPI_OSI_OPTIONAL_FEATURE, 0}, 95 {"3.0 Thermal Model", NULL, ACPI_OSI_OPTIONAL_FEATURE, 0}, 96 {"3.0 _SCP Extensions", NULL, ACPI_OSI_OPTIONAL_FEATURE, 0}, 97 {"Processor Aggregator Device", NULL, ACPI_OSI_OPTIONAL_FEATURE, 0} 98 }; 99 100 101 /******************************************************************************* 102 * 103 * FUNCTION: AcpiUtInitializeInterfaces 104 * 105 * PARAMETERS: None 106 * 107 * RETURN: Status 108 * 109 * DESCRIPTION: Initialize the global _OSI supported interfaces list 110 * 111 ******************************************************************************/ 112 113 ACPI_STATUS 114 AcpiUtInitializeInterfaces ( 115 void) 116 { 117 ACPI_STATUS Status; 118 UINT32 i; 119 120 121 Status = AcpiOsAcquireMutex (AcpiGbl_OsiMutex, ACPI_WAIT_FOREVER); 122 if (ACPI_FAILURE (Status)) 123 { 124 return (Status); 125 } 126 127 AcpiGbl_SupportedInterfaces = AcpiDefaultSupportedInterfaces; 128 129 /* Link the static list of supported interfaces */ 130 131 for (i = 0; i < (ACPI_ARRAY_LENGTH (AcpiDefaultSupportedInterfaces) - 1); i++) 132 { 133 AcpiDefaultSupportedInterfaces[i].Next = 134 &AcpiDefaultSupportedInterfaces[(ACPI_SIZE) i + 1]; 135 } 136 137 AcpiOsReleaseMutex (AcpiGbl_OsiMutex); 138 return (AE_OK); 139 } 140 141 142 /******************************************************************************* 143 * 144 * FUNCTION: AcpiUtInterfaceTerminate 145 * 146 * PARAMETERS: None 147 * 148 * RETURN: Status 149 * 150 * DESCRIPTION: Delete all interfaces in the global list. Sets 151 * AcpiGbl_SupportedInterfaces to NULL. 152 * 153 ******************************************************************************/ 154 155 ACPI_STATUS 156 AcpiUtInterfaceTerminate ( 157 void) 158 { 159 ACPI_STATUS Status; 160 ACPI_INTERFACE_INFO *NextInterface; 161 162 163 Status = AcpiOsAcquireMutex (AcpiGbl_OsiMutex, ACPI_WAIT_FOREVER); 164 if (ACPI_FAILURE (Status)) 165 { 166 return (Status); 167 } 168 169 NextInterface = AcpiGbl_SupportedInterfaces; 170 while (NextInterface) 171 { 172 AcpiGbl_SupportedInterfaces = NextInterface->Next; 173 174 if (NextInterface->Flags & ACPI_OSI_DYNAMIC) 175 { 176 /* Only interfaces added at runtime can be freed */ 177 178 ACPI_FREE (NextInterface->Name); 179 ACPI_FREE (NextInterface); 180 } 181 else 182 { 183 /* Interface is in static list. Reset it to invalid or valid. */ 184 185 if (NextInterface->Flags & ACPI_OSI_DEFAULT_INVALID) 186 { 187 NextInterface->Flags |= ACPI_OSI_INVALID; 188 } 189 else 190 { 191 NextInterface->Flags &= ~ACPI_OSI_INVALID; 192 } 193 } 194 195 NextInterface = AcpiGbl_SupportedInterfaces; 196 } 197 198 AcpiOsReleaseMutex (AcpiGbl_OsiMutex); 199 return (AE_OK); 200 } 201 202 203 /******************************************************************************* 204 * 205 * FUNCTION: AcpiUtInstallInterface 206 * 207 * PARAMETERS: InterfaceName - The interface to install 208 * 209 * RETURN: Status 210 * 211 * DESCRIPTION: Install the interface into the global interface list. 212 * Caller MUST hold AcpiGbl_OsiMutex 213 * 214 ******************************************************************************/ 215 216 ACPI_STATUS 217 AcpiUtInstallInterface ( 218 ACPI_STRING InterfaceName) 219 { 220 ACPI_INTERFACE_INFO *InterfaceInfo; 221 222 223 /* Allocate info block and space for the name string */ 224 225 InterfaceInfo = ACPI_ALLOCATE_ZEROED (sizeof (ACPI_INTERFACE_INFO)); 226 if (!InterfaceInfo) 227 { 228 return (AE_NO_MEMORY); 229 } 230 231 InterfaceInfo->Name = ACPI_ALLOCATE_ZEROED (ACPI_STRLEN (InterfaceName) + 1); 232 if (!InterfaceInfo->Name) 233 { 234 ACPI_FREE (InterfaceInfo); 235 return (AE_NO_MEMORY); 236 } 237 238 /* Initialize new info and insert at the head of the global list */ 239 240 ACPI_STRCPY (InterfaceInfo->Name, InterfaceName); 241 InterfaceInfo->Flags = ACPI_OSI_DYNAMIC; 242 InterfaceInfo->Next = AcpiGbl_SupportedInterfaces; 243 244 AcpiGbl_SupportedInterfaces = InterfaceInfo; 245 return (AE_OK); 246 } 247 248 249 /******************************************************************************* 250 * 251 * FUNCTION: AcpiUtRemoveInterface 252 * 253 * PARAMETERS: InterfaceName - The interface to remove 254 * 255 * RETURN: Status 256 * 257 * DESCRIPTION: Remove the interface from the global interface list. 258 * Caller MUST hold AcpiGbl_OsiMutex 259 * 260 ******************************************************************************/ 261 262 ACPI_STATUS 263 AcpiUtRemoveInterface ( 264 ACPI_STRING InterfaceName) 265 { 266 ACPI_INTERFACE_INFO *PreviousInterface; 267 ACPI_INTERFACE_INFO *NextInterface; 268 269 270 PreviousInterface = NextInterface = AcpiGbl_SupportedInterfaces; 271 while (NextInterface) 272 { 273 if (!ACPI_STRCMP (InterfaceName, NextInterface->Name)) 274 { 275 /* Found: name is in either the static list or was added at runtime */ 276 277 if (NextInterface->Flags & ACPI_OSI_DYNAMIC) 278 { 279 /* Interface was added dynamically, remove and free it */ 280 281 if (PreviousInterface == NextInterface) 282 { 283 AcpiGbl_SupportedInterfaces = NextInterface->Next; 284 } 285 else 286 { 287 PreviousInterface->Next = NextInterface->Next; 288 } 289 290 ACPI_FREE (NextInterface->Name); 291 ACPI_FREE (NextInterface); 292 } 293 else 294 { 295 /* 296 * Interface is in static list. If marked invalid, then it 297 * does not actually exist. Else, mark it invalid. 298 */ 299 if (NextInterface->Flags & ACPI_OSI_INVALID) 300 { 301 return (AE_NOT_EXIST); 302 } 303 304 NextInterface->Flags |= ACPI_OSI_INVALID; 305 } 306 307 return (AE_OK); 308 } 309 310 PreviousInterface = NextInterface; 311 NextInterface = NextInterface->Next; 312 } 313 314 /* Interface was not found */ 315 316 return (AE_NOT_EXIST); 317 } 318 319 320 /******************************************************************************* 321 * 322 * FUNCTION: AcpiUtUpdateInterfaces 323 * 324 * PARAMETERS: Action - Actions to be performed during the 325 * update 326 * 327 * RETURN: Status 328 * 329 * DESCRIPTION: Update _OSI interface strings, disabling or enabling OS vendor 330 * strings or/and feature group strings. 331 * Caller MUST hold AcpiGbl_OsiMutex 332 * 333 ******************************************************************************/ 334 335 ACPI_STATUS 336 AcpiUtUpdateInterfaces ( 337 UINT8 Action) 338 { 339 ACPI_INTERFACE_INFO *NextInterface; 340 341 342 NextInterface = AcpiGbl_SupportedInterfaces; 343 while (NextInterface) 344 { 345 if (((NextInterface->Flags & ACPI_OSI_FEATURE) && 346 (Action & ACPI_FEATURE_STRINGS)) || 347 (!(NextInterface->Flags & ACPI_OSI_FEATURE) && 348 (Action & ACPI_VENDOR_STRINGS))) 349 { 350 if (Action & ACPI_DISABLE_INTERFACES) 351 { 352 /* Mark the interfaces as invalid */ 353 354 NextInterface->Flags |= ACPI_OSI_INVALID; 355 } 356 else 357 { 358 /* Mark the interfaces as valid */ 359 360 NextInterface->Flags &= ~ACPI_OSI_INVALID; 361 } 362 } 363 364 NextInterface = NextInterface->Next; 365 } 366 367 return (AE_OK); 368 } 369 370 371 /******************************************************************************* 372 * 373 * FUNCTION: AcpiUtGetInterface 374 * 375 * PARAMETERS: InterfaceName - The interface to find 376 * 377 * RETURN: ACPI_INTERFACE_INFO if found. NULL if not found. 378 * 379 * DESCRIPTION: Search for the specified interface name in the global list. 380 * Caller MUST hold AcpiGbl_OsiMutex 381 * 382 ******************************************************************************/ 383 384 ACPI_INTERFACE_INFO * 385 AcpiUtGetInterface ( 386 ACPI_STRING InterfaceName) 387 { 388 ACPI_INTERFACE_INFO *NextInterface; 389 390 391 NextInterface = AcpiGbl_SupportedInterfaces; 392 while (NextInterface) 393 { 394 if (!ACPI_STRCMP (InterfaceName, NextInterface->Name)) 395 { 396 return (NextInterface); 397 } 398 399 NextInterface = NextInterface->Next; 400 } 401 402 return (NULL); 403 } 404 405 406 /******************************************************************************* 407 * 408 * FUNCTION: AcpiUtOsiImplementation 409 * 410 * PARAMETERS: WalkState - Current walk state 411 * 412 * RETURN: Status 413 * 414 * DESCRIPTION: Implementation of the _OSI predefined control method. When 415 * an invocation of _OSI is encountered in the system AML, 416 * control is transferred to this function. 417 * 418 ******************************************************************************/ 419 420 ACPI_STATUS 421 AcpiUtOsiImplementation ( 422 ACPI_WALK_STATE *WalkState) 423 { 424 ACPI_OPERAND_OBJECT *StringDesc; 425 ACPI_OPERAND_OBJECT *ReturnDesc; 426 ACPI_INTERFACE_INFO *InterfaceInfo; 427 ACPI_INTERFACE_HANDLER InterfaceHandler; 428 ACPI_STATUS Status; 429 UINT32 ReturnValue; 430 431 432 ACPI_FUNCTION_TRACE (UtOsiImplementation); 433 434 435 /* Validate the string input argument (from the AML caller) */ 436 437 StringDesc = WalkState->Arguments[0].Object; 438 if (!StringDesc || 439 (StringDesc->Common.Type != ACPI_TYPE_STRING)) 440 { 441 return_ACPI_STATUS (AE_TYPE); 442 } 443 444 /* Create a return object */ 445 446 ReturnDesc = AcpiUtCreateInternalObject (ACPI_TYPE_INTEGER); 447 if (!ReturnDesc) 448 { 449 return_ACPI_STATUS (AE_NO_MEMORY); 450 } 451 452 /* Default return value is 0, NOT SUPPORTED */ 453 454 ReturnValue = 0; 455 Status = AcpiOsAcquireMutex (AcpiGbl_OsiMutex, ACPI_WAIT_FOREVER); 456 if (ACPI_FAILURE (Status)) 457 { 458 AcpiUtRemoveReference (ReturnDesc); 459 return_ACPI_STATUS (Status); 460 } 461 462 /* Lookup the interface in the global _OSI list */ 463 464 InterfaceInfo = AcpiUtGetInterface (StringDesc->String.Pointer); 465 if (InterfaceInfo && 466 !(InterfaceInfo->Flags & ACPI_OSI_INVALID)) 467 { 468 /* 469 * The interface is supported. 470 * Update the OsiData if necessary. We keep track of the latest 471 * version of Windows that has been requested by the BIOS. 472 */ 473 if (InterfaceInfo->Value > AcpiGbl_OsiData) 474 { 475 AcpiGbl_OsiData = InterfaceInfo->Value; 476 } 477 478 ReturnValue = ACPI_UINT32_MAX; 479 } 480 481 AcpiOsReleaseMutex (AcpiGbl_OsiMutex); 482 483 /* 484 * Invoke an optional _OSI interface handler. The host OS may wish 485 * to do some interface-specific handling. For example, warn about 486 * certain interfaces or override the true/false support value. 487 */ 488 InterfaceHandler = AcpiGbl_InterfaceHandler; 489 if (InterfaceHandler) 490 { 491 ReturnValue = InterfaceHandler ( 492 StringDesc->String.Pointer, ReturnValue); 493 } 494 495 ACPI_DEBUG_PRINT_RAW ((ACPI_DB_INFO, 496 "ACPI: BIOS _OSI(\"%s\") is %ssupported\n", 497 StringDesc->String.Pointer, ReturnValue == 0 ? "not " : "")); 498 499 /* Complete the return object */ 500 501 ReturnDesc->Integer.Value = ReturnValue; 502 WalkState->ReturnDesc = ReturnDesc; 503 return_ACPI_STATUS (AE_OK); 504 } 505