patch-2.4.0-test3 linux/drivers/acpi/interpreter/amdyadic.c
Next file: linux/drivers/acpi/interpreter/amfield.c
Previous file: linux/drivers/acpi/interpreter/amcreate.c
Back to the patch index
Back to the overall index
- Lines: 751
- Date:
Wed Jul 5 11:23:12 2000
- Orig file:
v2.4.0-test2/linux/drivers/acpi/interpreter/amdyadic.c
- Orig date:
Wed Dec 31 16:00:00 1969
diff -u --recursive --new-file v2.4.0-test2/linux/drivers/acpi/interpreter/amdyadic.c linux/drivers/acpi/interpreter/amdyadic.c
@@ -0,0 +1,750 @@
+
+/******************************************************************************
+ *
+ * Module Name: amdyadic - ACPI AML (p-code) execution for dyadic operators
+ *
+ *****************************************************************************/
+
+/*
+ * Copyright (C) 2000 R. Byron Moore
+ *
+ * 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
+ */
+
+
+#include "acpi.h"
+#include "parser.h"
+#include "namesp.h"
+#include "interp.h"
+#include "events.h"
+#include "amlcode.h"
+#include "dispatch.h"
+
+
+#define _COMPONENT INTERPRETER
+ MODULE_NAME ("amdyadic");
+
+
+/*****************************************************************************
+ *
+ * FUNCTION: Acpi_aml_exec_dyadic1
+ *
+ * PARAMETERS: Opcode - The opcode to be executed
+ *
+ * RETURN: Status
+ *
+ * DESCRIPTION: Execute Type 1 dyadic operator with numeric operands:
+ * Notify_op
+ *
+ * ALLOCATION: Deletes both operands
+ *
+ ****************************************************************************/
+
+ACPI_STATUS
+acpi_aml_exec_dyadic1 (
+ u16 opcode,
+ ACPI_WALK_STATE *walk_state)
+{
+ ACPI_OBJECT_INTERNAL *obj_desc = NULL;
+ ACPI_OBJECT_INTERNAL *val_desc = NULL;
+ ACPI_NAMED_OBJECT *entry;
+ ACPI_STATUS status = AE_OK;
+
+
+ /* Resolve all operands */
+
+ status = acpi_aml_resolve_operands (opcode, WALK_OPERANDS);
+ /* Get the operands */
+
+ status |= acpi_ds_obj_stack_pop_object (&val_desc, walk_state);
+ status |= acpi_ds_obj_stack_pop_object (&obj_desc, walk_state);
+ if (ACPI_FAILURE (status)) {
+ /* Invalid parameters on object stack */
+
+ acpi_aml_append_operand_diag (_THIS_MODULE, __LINE__, opcode,
+ WALK_OPERANDS, 2);
+ goto cleanup;
+ }
+
+
+ /* Examine the opcode */
+
+ switch (opcode)
+ {
+
+ /* Def_notify := Notify_op Notify_object Notify_value */
+
+ case AML_NOTIFY_OP:
+
+ /* The Obj_desc is actually an NTE */
+
+ entry = (ACPI_NAMED_OBJECT*) obj_desc;
+ obj_desc = NULL;
+
+ /* Object must be a device or thermal zone */
+
+ if (entry && val_desc) {
+ switch (entry->type)
+ {
+ case ACPI_TYPE_DEVICE:
+ case ACPI_TYPE_THERMAL:
+
+ /*
+ * Requires that Device and Thermal_zone be compatible
+ * mappings
+ */
+
+ /* Dispatch the notify to the appropriate handler */
+
+ acpi_ev_notify_dispatch (entry, val_desc->number.value);
+ break;
+
+ default:
+ status = AE_AML_OPERAND_TYPE;
+ }
+ }
+ break;
+
+ default:
+ status = AE_AML_BAD_OPCODE;
+ }
+
+
+cleanup:
+
+ /* Always delete both operands */
+
+ acpi_cm_remove_reference (val_desc);
+ acpi_cm_remove_reference (obj_desc);
+
+
+ return (status);
+}
+
+
+/*****************************************************************************
+ *
+ * FUNCTION: Acpi_aml_exec_dyadic2_r
+ *
+ * PARAMETERS: Opcode - The opcode to be executed
+ *
+ * RETURN: Status
+ *
+ * DESCRIPTION: Execute Type 2 dyadic operator with numeric operands and
+ * one or two result operands.
+ *
+ * ALLOCATION: Deletes one operand descriptor -- other remains on stack
+ *
+ ****************************************************************************/
+
+ACPI_STATUS
+acpi_aml_exec_dyadic2_r (
+ u16 opcode,
+ ACPI_WALK_STATE *walk_state,
+ ACPI_OBJECT_INTERNAL **return_desc)
+{
+ ACPI_OBJECT_INTERNAL *obj_desc = NULL;
+ ACPI_OBJECT_INTERNAL *obj_desc2 = NULL;
+ ACPI_OBJECT_INTERNAL *res_desc = NULL;
+ ACPI_OBJECT_INTERNAL *res_desc2 = NULL;
+ ACPI_OBJECT_INTERNAL *ret_desc = NULL;
+ ACPI_OBJECT_INTERNAL *ret_desc2 = NULL;
+ ACPI_STATUS status = AE_OK;
+ u32 remainder;
+ s32 num_operands = 3;
+ char *new_buf;
+
+
+ /* Resolve all operands */
+
+ status = acpi_aml_resolve_operands (opcode, WALK_OPERANDS);
+ /* Get all operands */
+
+ if (AML_DIVIDE_OP == opcode) {
+ num_operands = 4;
+ status |= acpi_ds_obj_stack_pop_object (&res_desc2, walk_state);
+ }
+
+ status |= acpi_ds_obj_stack_pop_object (&res_desc, walk_state);
+ status |= acpi_ds_obj_stack_pop_object (&obj_desc2, walk_state);
+ status |= acpi_ds_obj_stack_pop_object (&obj_desc, walk_state);
+ if (status != AE_OK) {
+ acpi_aml_append_operand_diag (_THIS_MODULE, __LINE__, opcode,
+ &(walk_state->operands [walk_state->num_operands -1]),
+ num_operands);
+ goto cleanup;
+ }
+
+
+ /* Create an internal return object if necessary */
+
+ switch (opcode)
+ {
+ case AML_ADD_OP:
+ case AML_BIT_AND_OP:
+ case AML_BIT_NAND_OP:
+ case AML_BIT_OR_OP:
+ case AML_BIT_NOR_OP:
+ case AML_BIT_XOR_OP:
+ case AML_DIVIDE_OP:
+ case AML_MULTIPLY_OP:
+ case AML_SHIFT_LEFT_OP:
+ case AML_SHIFT_RIGHT_OP:
+ case AML_SUBTRACT_OP:
+
+ ret_desc = acpi_cm_create_internal_object (ACPI_TYPE_NUMBER);
+ if (!ret_desc) {
+ status = AE_NO_MEMORY;
+ goto cleanup;
+ }
+
+ break;
+ }
+
+
+ /*
+ * Execute the opcode
+ */
+
+ switch (opcode)
+ {
+
+ /* Def_add := Add_op Operand1 Operand2 Result */
+
+ case AML_ADD_OP:
+
+ ret_desc->number.value = obj_desc->number.value +
+ obj_desc2->number.value;
+ break;
+
+
+ /* Def_and := And_op Operand1 Operand2 Result */
+
+ case AML_BIT_AND_OP:
+
+ ret_desc->number.value = obj_desc->number.value &
+ obj_desc2->number.value;
+ break;
+
+
+ /* Def_nAnd := NAnd_op Operand1 Operand2 Result */
+
+ case AML_BIT_NAND_OP:
+
+ ret_desc->number.value = ~(obj_desc->number.value &
+ obj_desc2->number.value);
+ break;
+
+
+ /* Def_or := Or_op Operand1 Operand2 Result */
+
+ case AML_BIT_OR_OP:
+
+ ret_desc->number.value = obj_desc->number.value |
+ obj_desc2->number.value;
+ break;
+
+
+ /* Def_nOr := NOr_op Operand1 Operand2 Result */
+
+ case AML_BIT_NOR_OP:
+
+ ret_desc->number.value = ~(obj_desc->number.value |
+ obj_desc2->number.value);
+ break;
+
+
+ /* Def_xOr := XOr_op Operand1 Operand2 Result */
+
+ case AML_BIT_XOR_OP:
+
+ ret_desc->number.value = obj_desc->number.value ^
+ obj_desc2->number.value;
+ break;
+
+
+ /* Def_divide := Divide_op Dividend Divisor Remainder Quotient */
+
+ case AML_DIVIDE_OP:
+
+ if ((u32) 0 == obj_desc2->number.value) {
+ REPORT_ERROR ("Aml_exec_dyadic2_r/Divide_op: Divide by zero");
+
+ status = AE_AML_DIVIDE_BY_ZERO;
+ goto cleanup;
+ }
+
+ ret_desc2 = acpi_cm_create_internal_object (ACPI_TYPE_NUMBER);
+ if (!ret_desc2) {
+ status = AE_NO_MEMORY;
+ goto cleanup;
+ }
+
+ remainder = obj_desc->number.value %
+ obj_desc2->number.value;
+ ret_desc->number.value = remainder;
+
+ /* Result (what we used to call the quotient) */
+
+ ret_desc2->number.value = obj_desc->number.value /
+ obj_desc2->number.value;
+ break;
+
+
+ /* Def_multiply := Multiply_op Operand1 Operand2 Result */
+
+ case AML_MULTIPLY_OP:
+
+ ret_desc->number.value = obj_desc->number.value *
+ obj_desc2->number.value;
+ break;
+
+
+ /* Def_shift_left := Shift_left_op Operand Shift_count Result */
+
+ case AML_SHIFT_LEFT_OP:
+
+ ret_desc->number.value = obj_desc->number.value <<
+ obj_desc2->number.value;
+ break;
+
+
+ /* Def_shift_right := Shift_right_op Operand Shift_count Result */
+
+ case AML_SHIFT_RIGHT_OP:
+
+ ret_desc->number.value = obj_desc->number.value >>
+ obj_desc2->number.value;
+ break;
+
+
+ /* Def_subtract := Subtract_op Operand1 Operand2 Result */
+
+ case AML_SUBTRACT_OP:
+
+ ret_desc->number.value = obj_desc->number.value -
+ obj_desc2->number.value;
+ break;
+
+
+ /* Def_concat := Concat_op Data1 Data2 Result */
+
+ case AML_CONCAT_OP:
+
+ if (obj_desc2->common.type != obj_desc->common.type) {
+ status = AE_AML_OPERAND_TYPE;
+ goto cleanup;
+ }
+
+ /* Both operands are now known to be the same */
+
+ if (ACPI_TYPE_STRING == obj_desc->common.type) {
+ ret_desc = acpi_cm_create_internal_object (ACPI_TYPE_STRING);
+ if (!ret_desc) {
+ status = AE_NO_MEMORY;
+ goto cleanup;
+ }
+
+ /* Operand1 is string */
+
+ new_buf = acpi_cm_allocate (obj_desc->string.length +
+ obj_desc2->string.length + 1);
+ if (!new_buf) {
+ REPORT_ERROR
+ ("Aml_exec_dyadic2_r/Concat_op: String allocation failure");
+ status = AE_NO_MEMORY;
+ goto cleanup;
+ }
+
+ STRCPY (new_buf, (char *) obj_desc->string.pointer);
+ STRCPY (new_buf + obj_desc->string.length,
+ (char *) obj_desc2->string.pointer);
+
+ /* Point the return object to the new string */
+
+ ret_desc->string.pointer = new_buf;
+ ret_desc->string.length = obj_desc->string.length +=
+ obj_desc2->string.length;
+ }
+
+ else {
+ /* Operand1 is not a string ==> must be a buffer */
+
+ ret_desc = acpi_cm_create_internal_object (ACPI_TYPE_BUFFER);
+ if (!ret_desc) {
+ status = AE_NO_MEMORY;
+ goto cleanup;
+ }
+
+ new_buf = acpi_cm_allocate (obj_desc->buffer.length +
+ obj_desc2->buffer.length);
+ if (!new_buf) {
+ /* Only bail out if the buffer is small */
+
+ /* TBD: [Investigate] what is the point of this code? */
+
+ if (obj_desc->buffer.length + obj_desc2->buffer.length < 1024) {
+ REPORT_ERROR
+ ("Aml_exec_dyadic2_r/Concat_op: Buffer allocation failure");
+ return (AE_NO_MEMORY);
+ }
+
+ status = AE_NO_MEMORY;
+ goto cleanup;
+ }
+
+ MEMCPY (new_buf, obj_desc->buffer.pointer,
+ obj_desc->buffer.length);
+ MEMCPY (new_buf + obj_desc->buffer.length, obj_desc2->buffer.pointer,
+ obj_desc2->buffer.length);
+
+ /*
+ * Point the return object to the new buffer
+ */
+
+ ret_desc->buffer.pointer = (u8 *) new_buf;
+ ret_desc->buffer.length = obj_desc->buffer.length +
+ obj_desc2->buffer.length;
+ }
+ break;
+
+
+ default:
+
+ status = AE_AML_BAD_OPCODE;
+ goto cleanup;
+ }
+
+
+ /*
+ * Store the result of the operation (which is now in Obj_desc) into
+ * the result descriptor, or the location pointed to by the result
+ * descriptor (Res_desc).
+ */
+
+ if ((status = acpi_aml_exec_store (ret_desc, res_desc)) != AE_OK) {
+ goto cleanup;
+ }
+
+ if (AML_DIVIDE_OP == opcode) {
+ status = acpi_aml_exec_store (ret_desc2, res_desc2);
+
+ /*
+ * Since the remainder is not returned, remove a reference to
+ * the object we created earlier
+ */
+
+ acpi_cm_remove_reference (ret_desc2);
+ }
+
+
+cleanup:
+
+ /* Always delete the operands */
+
+ acpi_cm_remove_reference (obj_desc);
+ acpi_cm_remove_reference (obj_desc2);
+
+
+ /* Delete return object on error */
+
+ if (ACPI_FAILURE (status)) {
+ /* On failure, delete the result ops */
+
+ acpi_cm_remove_reference (res_desc);
+ acpi_cm_remove_reference (res_desc2);
+
+ if (ret_desc) {
+ /* And delete the internal return object */
+
+ acpi_cm_remove_reference (ret_desc);
+ ret_desc = NULL;
+ }
+ }
+
+ /* Set the return object and exit */
+
+ *return_desc = ret_desc;
+ return (status);
+}
+
+
+/*****************************************************************************
+ *
+ * FUNCTION: Acpi_aml_exec_dyadic2_s
+ *
+ * PARAMETERS: Opcode - The opcode to be executed
+ *
+ * RETURN: Status
+ *
+ * DESCRIPTION: Execute Type 2 dyadic synchronization operator
+ *
+ * ALLOCATION: Deletes one operand descriptor -- other remains on stack
+ *
+ ****************************************************************************/
+
+ACPI_STATUS
+acpi_aml_exec_dyadic2_s (
+ u16 opcode,
+ ACPI_WALK_STATE *walk_state,
+ ACPI_OBJECT_INTERNAL **return_desc)
+{
+ ACPI_OBJECT_INTERNAL *obj_desc;
+ ACPI_OBJECT_INTERNAL *time_desc;
+ ACPI_OBJECT_INTERNAL *ret_desc = NULL;
+ ACPI_STATUS status;
+
+
+ /* Resolve all operands */
+
+ status = acpi_aml_resolve_operands (opcode, WALK_OPERANDS);
+ /* Get all operands */
+
+ status |= acpi_ds_obj_stack_pop_object (&time_desc, walk_state);
+ status |= acpi_ds_obj_stack_pop_object (&obj_desc, walk_state);
+ if (status != AE_OK) {
+ /* Invalid parameters on object stack */
+
+ acpi_aml_append_operand_diag (_THIS_MODULE, __LINE__, opcode,
+ WALK_OPERANDS, 2);
+ goto cleanup;
+ }
+
+
+ /* Create the internal return object */
+
+ ret_desc = acpi_cm_create_internal_object (ACPI_TYPE_NUMBER);
+ if (!ret_desc) {
+ status = AE_NO_MEMORY;
+ goto cleanup;
+ }
+
+ /* Default return value is FALSE, operation did not time out */
+
+ ret_desc->number.value = 0;
+
+
+ /* Examine the opcode */
+
+ switch (opcode)
+ {
+
+ /* Def_acquire := Acquire_op Mutex_object Timeout */
+
+ case AML_ACQUIRE_OP:
+
+ status = acpi_aml_system_acquire_mutex (time_desc, obj_desc);
+ break;
+
+
+ /* Def_wait := Wait_op Acpi_event_object Timeout */
+
+ case AML_WAIT_OP:
+
+ status = acpi_aml_system_wait_event (time_desc, obj_desc);
+ break;
+
+
+ default:
+
+ status = AE_AML_BAD_OPCODE;
+ goto cleanup;
+ }
+
+
+ /*
+ * Return a boolean indicating if operation timed out
+ * (TRUE) or not (FALSE)
+ */
+
+ if (status == AE_TIME) {
+ ret_desc->number.value = (u32)(-1); /* TRUE, op timed out */
+ status = AE_OK;
+ }
+
+
+cleanup:
+
+ /* Delete params */
+
+ acpi_cm_remove_reference (time_desc);
+ acpi_cm_remove_reference (obj_desc);
+
+ /* Delete return object on error */
+
+ if (ACPI_FAILURE (status) &&
+ (ret_desc))
+ {
+ acpi_cm_remove_reference (ret_desc);
+ ret_desc = NULL;
+ }
+
+
+ /* Set the return object and exit */
+
+ *return_desc = ret_desc;
+ return (status);
+}
+
+
+/*****************************************************************************
+ *
+ * FUNCTION: Acpi_aml_exec_dyadic2
+ *
+ * PARAMETERS: Opcode - The opcode to be executed
+ *
+ * RETURN: Status
+ *
+ * DESCRIPTION: Execute Type 2 dyadic operator with numeric operands and
+ * no result operands
+ *
+ * ALLOCATION: Deletes one operand descriptor -- other remains on stack
+ * containing result value
+ *
+ ****************************************************************************/
+
+ACPI_STATUS
+acpi_aml_exec_dyadic2 (
+ u16 opcode,
+ ACPI_WALK_STATE *walk_state,
+ ACPI_OBJECT_INTERNAL **return_desc)
+{
+ ACPI_OBJECT_INTERNAL *obj_desc;
+ ACPI_OBJECT_INTERNAL *obj_desc2;
+ ACPI_OBJECT_INTERNAL *ret_desc = NULL;
+ ACPI_STATUS status;
+ u8 lboolean;
+
+
+ /* Resolve all operands */
+
+ status = acpi_aml_resolve_operands (opcode, WALK_OPERANDS);
+ /* Get all operands */
+
+ status |= acpi_ds_obj_stack_pop_object (&obj_desc2, walk_state);
+ status |= acpi_ds_obj_stack_pop_object (&obj_desc, walk_state);
+ if (status != AE_OK) {
+ /* Invalid parameters on object stack */
+
+ acpi_aml_append_operand_diag (_THIS_MODULE, __LINE__, opcode,
+ WALK_OPERANDS, 2);
+ goto cleanup;
+ }
+
+
+ /* Create the internal return object */
+
+ ret_desc = acpi_cm_create_internal_object (ACPI_TYPE_NUMBER);
+ if (!ret_desc) {
+ status = AE_NO_MEMORY;
+ goto cleanup;
+ }
+
+ /*
+ * Execute the Opcode
+ */
+
+ lboolean = FALSE;
+ switch (opcode)
+ {
+
+ /* Def_lAnd := LAnd_op Operand1 Operand2 */
+
+ case AML_LAND_OP:
+
+ lboolean = (u8) (obj_desc->number.value &&
+ obj_desc2->number.value);
+ break;
+
+
+ /* Def_lEqual := LEqual_op Operand1 Operand2 */
+
+ case AML_LEQUAL_OP:
+
+ lboolean = (u8) (obj_desc->number.value ==
+ obj_desc2->number.value);
+ break;
+
+
+ /* Def_lGreater := LGreater_op Operand1 Operand2 */
+
+ case AML_LGREATER_OP:
+
+ lboolean = (u8) (obj_desc->number.value >
+ obj_desc2->number.value);
+ break;
+
+
+ /* Def_lLess := LLess_op Operand1 Operand2 */
+
+ case AML_LLESS_OP:
+
+ lboolean = (u8) (obj_desc->number.value <
+ obj_desc2->number.value);
+ break;
+
+
+ /* Def_lOr := LOr_op Operand1 Operand2 */
+
+ case AML_LOR_OP:
+
+ lboolean = (u8) (obj_desc->number.value ||
+ obj_desc2->number.value);
+ break;
+
+
+ default:
+
+ status = AE_AML_BAD_OPCODE;
+ goto cleanup;
+ break;
+ }
+
+
+ /* Set return value to logical TRUE (all ones) or FALSE (zero) */
+
+ if (lboolean) {
+ ret_desc->number.value = 0xffffffff;
+ }
+ else {
+ ret_desc->number.value = 0;
+ }
+
+
+cleanup:
+
+ /* Always delete operands */
+
+ acpi_cm_remove_reference (obj_desc);
+ acpi_cm_remove_reference (obj_desc2);
+
+
+ /* Delete return object on error */
+
+ if (ACPI_FAILURE (status) &&
+ (ret_desc))
+ {
+ acpi_cm_remove_reference (ret_desc);
+ ret_desc = NULL;
+ }
+
+
+ /* Set the return object and exit */
+
+ *return_desc = ret_desc;
+ return (status);
+}
+
+
FUNET's LINUX-ADM group, linux-adm@nic.funet.fi
TCL-scripts by Sam Shen (who was at: slshen@lbl.gov)