2 /******************************************************************************
 
   4  * Module Name: exsystem - Interface to OS services
 
   6  *****************************************************************************/
 
   9  * Copyright (C) 2000 - 2006, R. Byron Moore
 
  10  * All rights reserved.
 
  12  * Redistribution and use in source and binary forms, with or without
 
  13  * modification, are permitted provided that the following conditions
 
  15  * 1. Redistributions of source code must retain the above copyright
 
  16  *    notice, this list of conditions, and the following disclaimer,
 
  17  *    without modification.
 
  18  * 2. Redistributions in binary form must reproduce at minimum a disclaimer
 
  19  *    substantially similar to the "NO WARRANTY" disclaimer below
 
  20  *    ("Disclaimer") and any redistribution must be conditioned upon
 
  21  *    including a substantially similar Disclaimer requirement for further
 
  22  *    binary redistribution.
 
  23  * 3. Neither the names of the above-listed copyright holders nor the names
 
  24  *    of any contributors may be used to endorse or promote products derived
 
  25  *    from this software without specific prior written permission.
 
  27  * Alternatively, this software may be distributed under the terms of the
 
  28  * GNU General Public License ("GPL") version 2 as published by the Free
 
  29  * Software Foundation.
 
  32  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 
  33  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 
  34  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
 
  35  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 
  36  * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 
  37  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 
  38  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 
  39  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 
  40  * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
 
  41  * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 
  42  * POSSIBILITY OF SUCH DAMAGES.
 
  45 #include <acpi/acpi.h>
 
  46 #include <acpi/acinterp.h>
 
  47 #include <acpi/acevents.h>
 
  49 #define _COMPONENT          ACPI_EXECUTER
 
  50 ACPI_MODULE_NAME("exsystem")
 
  52 /*******************************************************************************
 
  54  * FUNCTION:    acpi_ex_system_wait_semaphore
 
  56  * PARAMETERS:  Semaphore       - Semaphore to wait on
 
  57  *              Timeout         - Max time to wait
 
  61  * DESCRIPTION: Implements a semaphore wait with a check to see if the
 
  62  *              semaphore is available immediately.  If it is not, the
 
  63  *              interpreter is released.
 
  65  ******************************************************************************/
 
  66 acpi_status acpi_ex_system_wait_semaphore(acpi_handle semaphore, u16 timeout)
 
  71         ACPI_FUNCTION_TRACE("ex_system_wait_semaphore");
 
  73         status = acpi_os_wait_semaphore(semaphore, 1, 0);
 
  74         if (ACPI_SUCCESS(status)) {
 
  75                 return_ACPI_STATUS(status);
 
  78         if (status == AE_TIME) {
 
  79                 /* We must wait, so unlock the interpreter */
 
  81                 acpi_ex_exit_interpreter();
 
  83                 status = acpi_os_wait_semaphore(semaphore, 1, timeout);
 
  85                 ACPI_DEBUG_PRINT((ACPI_DB_EXEC,
 
  86                                   "*** Thread awake after blocking, %s\n",
 
  87                                   acpi_format_exception(status)));
 
  89                 /* Reacquire the interpreter */
 
  91                 status2 = acpi_ex_enter_interpreter();
 
  92                 if (ACPI_FAILURE(status2)) {
 
  93                         /* Report fatal error, could not acquire interpreter */
 
  95                         return_ACPI_STATUS(status2);
 
  99         return_ACPI_STATUS(status);
 
 102 /*******************************************************************************
 
 104  * FUNCTION:    acpi_ex_system_do_stall
 
 106  * PARAMETERS:  how_long        - The amount of time to stall,
 
 111  * DESCRIPTION: Suspend running thread for specified amount of time.
 
 112  *              Note: ACPI specification requires that Stall() does not
 
 113  *              relinquish the processor, and delays longer than 100 usec
 
 114  *              should use Sleep() instead.  We allow stalls up to 255 usec
 
 115  *              for compatibility with other interpreters and existing BIOSs.
 
 117  ******************************************************************************/
 
 119 acpi_status acpi_ex_system_do_stall(u32 how_long)
 
 121         acpi_status status = AE_OK;
 
 123         ACPI_FUNCTION_ENTRY();
 
 125         if (how_long > 255) {   /* 255 microseconds */
 
 127                  * Longer than 255 usec, this is an error
 
 129                  * (ACPI specifies 100 usec as max, but this gives some slack in
 
 130                  * order to support existing BIOSs)
 
 132                 ACPI_ERROR((AE_INFO, "Time parameter is too large (%d)",
 
 134                 status = AE_AML_OPERAND_VALUE;
 
 136                 acpi_os_stall(how_long);
 
 142 /*******************************************************************************
 
 144  * FUNCTION:    acpi_ex_system_do_suspend
 
 146  * PARAMETERS:  how_long        - The amount of time to suspend,
 
 151  * DESCRIPTION: Suspend running thread for specified amount of time.
 
 153  ******************************************************************************/
 
 155 acpi_status acpi_ex_system_do_suspend(acpi_integer how_long)
 
 159         ACPI_FUNCTION_ENTRY();
 
 161         /* Since this thread will sleep, we must release the interpreter */
 
 163         acpi_ex_exit_interpreter();
 
 165         acpi_os_sleep(how_long);
 
 167         /* And now we must get the interpreter again */
 
 169         status = acpi_ex_enter_interpreter();
 
 173 /*******************************************************************************
 
 175  * FUNCTION:    acpi_ex_system_acquire_mutex
 
 177  * PARAMETERS:  time_desc       - The 'time to delay' object descriptor
 
 178  *              obj_desc        - The object descriptor for this op
 
 182  * DESCRIPTION: Provides an access point to perform synchronization operations
 
 183  *              within the AML.  This function will cause a lock to be generated
 
 184  *              for the Mutex pointed to by obj_desc.
 
 186  ******************************************************************************/
 
 189 acpi_ex_system_acquire_mutex(union acpi_operand_object * time_desc,
 
 190                              union acpi_operand_object * obj_desc)
 
 192         acpi_status status = AE_OK;
 
 194         ACPI_FUNCTION_TRACE_PTR("ex_system_acquire_mutex", obj_desc);
 
 197                 return_ACPI_STATUS(AE_BAD_PARAMETER);
 
 200         /* Support for the _GL_ Mutex object -- go get the global lock */
 
 202         if (obj_desc->mutex.semaphore == acpi_gbl_global_lock_semaphore) {
 
 204                     acpi_ev_acquire_global_lock((u16) time_desc->integer.value);
 
 205                 return_ACPI_STATUS(status);
 
 208         status = acpi_ex_system_wait_semaphore(obj_desc->mutex.semaphore,
 
 209                                                (u16) time_desc->integer.value);
 
 210         return_ACPI_STATUS(status);
 
 213 /*******************************************************************************
 
 215  * FUNCTION:    acpi_ex_system_release_mutex
 
 217  * PARAMETERS:  obj_desc        - The object descriptor for this op
 
 221  * DESCRIPTION: Provides an access point to perform synchronization operations
 
 222  *              within the AML.  This operation is a request to release a
 
 223  *              previously acquired Mutex.  If the Mutex variable is set then
 
 224  *              it will be decremented.
 
 226  ******************************************************************************/
 
 228 acpi_status acpi_ex_system_release_mutex(union acpi_operand_object *obj_desc)
 
 230         acpi_status status = AE_OK;
 
 232         ACPI_FUNCTION_TRACE("ex_system_release_mutex");
 
 235                 return_ACPI_STATUS(AE_BAD_PARAMETER);
 
 238         /* Support for the _GL_ Mutex object -- release the global lock */
 
 240         if (obj_desc->mutex.semaphore == acpi_gbl_global_lock_semaphore) {
 
 241                 status = acpi_ev_release_global_lock();
 
 242                 return_ACPI_STATUS(status);
 
 245         status = acpi_os_signal_semaphore(obj_desc->mutex.semaphore, 1);
 
 246         return_ACPI_STATUS(status);
 
 249 /*******************************************************************************
 
 251  * FUNCTION:    acpi_ex_system_signal_event
 
 253  * PARAMETERS:  obj_desc        - The object descriptor for this op
 
 257  * DESCRIPTION: Provides an access point to perform synchronization operations
 
 260  ******************************************************************************/
 
 262 acpi_status acpi_ex_system_signal_event(union acpi_operand_object *obj_desc)
 
 264         acpi_status status = AE_OK;
 
 266         ACPI_FUNCTION_TRACE("ex_system_signal_event");
 
 269                 status = acpi_os_signal_semaphore(obj_desc->event.semaphore, 1);
 
 272         return_ACPI_STATUS(status);
 
 275 /*******************************************************************************
 
 277  * FUNCTION:    acpi_ex_system_wait_event
 
 279  * PARAMETERS:  time_desc       - The 'time to delay' object descriptor
 
 280  *              obj_desc        - The object descriptor for this op
 
 284  * DESCRIPTION: Provides an access point to perform synchronization operations
 
 285  *              within the AML.  This operation is a request to wait for an
 
 288  ******************************************************************************/
 
 291 acpi_ex_system_wait_event(union acpi_operand_object *time_desc,
 
 292                           union acpi_operand_object *obj_desc)
 
 294         acpi_status status = AE_OK;
 
 296         ACPI_FUNCTION_TRACE("ex_system_wait_event");
 
 300                     acpi_ex_system_wait_semaphore(obj_desc->event.semaphore,
 
 301                                                   (u16) time_desc->integer.
 
 305         return_ACPI_STATUS(status);
 
 308 /*******************************************************************************
 
 310  * FUNCTION:    acpi_ex_system_reset_event
 
 312  * PARAMETERS:  obj_desc        - The object descriptor for this op
 
 316  * DESCRIPTION: Reset an event to a known state.
 
 318  ******************************************************************************/
 
 320 acpi_status acpi_ex_system_reset_event(union acpi_operand_object *obj_desc)
 
 322         acpi_status status = AE_OK;
 
 323         void *temp_semaphore;
 
 325         ACPI_FUNCTION_ENTRY();
 
 328          * We are going to simply delete the existing semaphore and
 
 332             acpi_os_create_semaphore(ACPI_NO_UNIT_LIMIT, 0, &temp_semaphore);
 
 333         if (ACPI_SUCCESS(status)) {
 
 334                 (void)acpi_os_delete_semaphore(obj_desc->event.semaphore);
 
 335                 obj_desc->event.semaphore = temp_semaphore;