exoparg3.c 7.9 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8

/******************************************************************************
 *
 * Module Name: exoparg3 - AML execution - opcodes with 3 arguments
 *
 *****************************************************************************/

/*
9
 * Copyright (C) 2000 - 2011, Intel Corp.
L
Linus Torvalds 已提交
10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions, and the following disclaimer,
 *    without modification.
 * 2. Redistributions in binary form must reproduce at minimum a disclaimer
 *    substantially similar to the "NO WARRANTY" disclaimer below
 *    ("Disclaimer") and any redistribution must be conditioned upon
 *    including a substantially similar Disclaimer requirement for further
 *    binary redistribution.
 * 3. Neither the names of the above-listed copyright holders nor the names
 *    of any contributors may be used to endorse or promote products derived
 *    from this software without specific prior written permission.
 *
 * Alternatively, this software may be distributed under the terms of the
 * GNU General Public License ("GPL") version 2 as published by the Free
 * Software Foundation.
 *
 * NO WARRANTY
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTIBILITY AND FITNESS FOR
 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
 * HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
 * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
 * IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGES.
 */

#include <acpi/acpi.h>
L
Len Brown 已提交
46 47 48 49
#include "accommon.h"
#include "acinterp.h"
#include "acparser.h"
#include "amlcode.h"
L
Linus Torvalds 已提交
50 51

#define _COMPONENT          ACPI_EXECUTER
L
Len Brown 已提交
52
ACPI_MODULE_NAME("exoparg3")
L
Linus Torvalds 已提交
53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85

/*!
 * Naming convention for AML interpreter execution routines.
 *
 * The routines that begin execution of AML opcodes are named with a common
 * convention based upon the number of arguments, the number of target operands,
 * and whether or not a value is returned:
 *
 *      AcpiExOpcode_xA_yT_zR
 *
 * Where:
 *
 * xA - ARGUMENTS:    The number of arguments (input operands) that are
 *                    required for this opcode type (1 through 6 args).
 * yT - TARGETS:      The number of targets (output operands) that are required
 *                    for this opcode type (0, 1, or 2 targets).
 * zR - RETURN VALUE: Indicates whether this opcode type returns a value
 *                    as the function return (0 or 1).
 *
 * The AcpiExOpcode* functions are called via the Dispatcher component with
 * fully resolved operands.
!*/
/*******************************************************************************
 *
 * FUNCTION:    acpi_ex_opcode_3A_0T_0R
 *
 * PARAMETERS:  walk_state          - Current walk state
 *
 * RETURN:      Status
 *
 * DESCRIPTION: Execute Triadic operator (3 operands)
 *
 ******************************************************************************/
L
Len Brown 已提交
86
acpi_status acpi_ex_opcode_3A_0T_0R(struct acpi_walk_state *walk_state)
L
Linus Torvalds 已提交
87
{
L
Len Brown 已提交
88 89 90
	union acpi_operand_object **operand = &walk_state->operands[0];
	struct acpi_signal_fatal_info *fatal;
	acpi_status status = AE_OK;
L
Linus Torvalds 已提交
91

B
Bob Moore 已提交
92
	ACPI_FUNCTION_TRACE_STR(ex_opcode_3A_0T_0R,
L
Len Brown 已提交
93
				acpi_ps_get_opcode_name(walk_state->opcode));
L
Linus Torvalds 已提交
94 95

	switch (walk_state->opcode) {
L
Len Brown 已提交
96
	case AML_FATAL_OP:	/* Fatal (fatal_type fatal_code fatal_arg) */
L
Linus Torvalds 已提交
97

L
Len Brown 已提交
98
		ACPI_DEBUG_PRINT((ACPI_DB_INFO,
B
Bob Moore 已提交
99
				  "FatalOp: Type %X Code %X Arg %X <<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<\n",
L
Len Brown 已提交
100 101 102
				  (u32) operand[0]->integer.value,
				  (u32) operand[1]->integer.value,
				  (u32) operand[2]->integer.value));
L
Linus Torvalds 已提交
103

B
Bob Moore 已提交
104
		fatal = ACPI_ALLOCATE(sizeof(struct acpi_signal_fatal_info));
L
Linus Torvalds 已提交
105
		if (fatal) {
L
Len Brown 已提交
106 107
			fatal->type = (u32) operand[0]->integer.value;
			fatal->code = (u32) operand[1]->integer.value;
L
Linus Torvalds 已提交
108 109 110
			fatal->argument = (u32) operand[2]->integer.value;
		}

R
Robert Moore 已提交
111 112
		/* Always signal the OS! */

L
Len Brown 已提交
113
		status = acpi_os_signal(ACPI_SIGNAL_FATAL, fatal);
L
Linus Torvalds 已提交
114 115 116

		/* Might return while OS is shutting down, just continue */

B
Bob Moore 已提交
117
		ACPI_FREE(fatal);
L
Linus Torvalds 已提交
118 119 120 121
		break;

	default:

122
		ACPI_ERROR((AE_INFO, "Unknown AML opcode 0x%X",
B
Bob Moore 已提交
123
			    walk_state->opcode));
L
Linus Torvalds 已提交
124 125 126 127
		status = AE_AML_BAD_OPCODE;
		goto cleanup;
	}

L
Len Brown 已提交
128
      cleanup:
L
Linus Torvalds 已提交
129

L
Len Brown 已提交
130
	return_ACPI_STATUS(status);
L
Linus Torvalds 已提交
131 132 133 134 135 136 137 138 139 140 141 142 143 144
}

/*******************************************************************************
 *
 * FUNCTION:    acpi_ex_opcode_3A_1T_1R
 *
 * PARAMETERS:  walk_state          - Current walk state
 *
 * RETURN:      Status
 *
 * DESCRIPTION: Execute Triadic operator (3 operands)
 *
 ******************************************************************************/

L
Len Brown 已提交
145
acpi_status acpi_ex_opcode_3A_1T_1R(struct acpi_walk_state *walk_state)
L
Linus Torvalds 已提交
146
{
L
Len Brown 已提交
147 148 149 150
	union acpi_operand_object **operand = &walk_state->operands[0];
	union acpi_operand_object *return_desc = NULL;
	char *buffer = NULL;
	acpi_status status = AE_OK;
151
	u64 index;
L
Len Brown 已提交
152
	acpi_size length;
L
Linus Torvalds 已提交
153

B
Bob Moore 已提交
154
	ACPI_FUNCTION_TRACE_STR(ex_opcode_3A_1T_1R,
L
Len Brown 已提交
155
				acpi_ps_get_opcode_name(walk_state->opcode));
L
Linus Torvalds 已提交
156 157

	switch (walk_state->opcode) {
L
Len Brown 已提交
158
	case AML_MID_OP:	/* Mid (Source[0], Index[1], Length[2], Result[3]) */
L
Linus Torvalds 已提交
159 160 161 162 163

		/*
		 * Create the return object.  The Source operand is guaranteed to be
		 * either a String or a Buffer, so just use its type.
		 */
164 165
		return_desc = acpi_ut_create_internal_object((operand[0])->
							     common.type);
L
Linus Torvalds 已提交
166 167 168 169 170 171 172
		if (!return_desc) {
			status = AE_NO_MEMORY;
			goto cleanup;
		}

		/* Get the Integer values from the objects */

R
Robert Moore 已提交
173
		index = operand[1]->integer.value;
L
Linus Torvalds 已提交
174 175 176 177 178 179
		length = (acpi_size) operand[2]->integer.value;

		/*
		 * If the index is beyond the length of the String/Buffer, or if the
		 * requested length is zero, return a zero-length String/Buffer
		 */
180 181 182 183 184 185 186 187
		if (index >= operand[0]->string.length) {
			length = 0;
		}

		/* Truncate request if larger than the actual String/Buffer */

		else if ((index + length) > operand[0]->string.length) {
			length = (acpi_size) operand[0]->string.length -
L
Len Brown 已提交
188
			    (acpi_size) index;
189
		}
L
Linus Torvalds 已提交
190

191 192
		/* Strings always have a sub-pointer, not so for buffers */

193
		switch ((operand[0])->common.type) {
194 195 196
		case ACPI_TYPE_STRING:

			/* Always allocate a new buffer for the String */
L
Linus Torvalds 已提交
197

B
Bob Moore 已提交
198
			buffer = ACPI_ALLOCATE_ZEROED((acpi_size) length + 1);
L
Linus Torvalds 已提交
199 200 201 202
			if (!buffer) {
				status = AE_NO_MEMORY;
				goto cleanup;
			}
203 204 205 206 207 208 209
			break;

		case ACPI_TYPE_BUFFER:

			/* If the requested length is zero, don't allocate a buffer */

			if (length > 0) {
B
Bob Moore 已提交
210

211 212
				/* Allocate a new buffer for the Buffer */

B
Bob Moore 已提交
213
				buffer = ACPI_ALLOCATE_ZEROED(length);
214 215 216 217 218 219
				if (!buffer) {
					status = AE_NO_MEMORY;
					goto cleanup;
				}
			}
			break;
L
Linus Torvalds 已提交
220

L
Len Brown 已提交
221
		default:	/* Should not happen */
222 223 224 225 226

			status = AE_AML_OPERAND_TYPE;
			goto cleanup;
		}

B
Bob Moore 已提交
227
		if (buffer) {
B
Bob Moore 已提交
228

B
Bob Moore 已提交
229
			/* We have a buffer, copy the portion requested */
L
Linus Torvalds 已提交
230

L
Len Brown 已提交
231 232
			ACPI_MEMCPY(buffer, operand[0]->string.pointer + index,
				    length);
233
		}
L
Linus Torvalds 已提交
234

235
		/* Set the length of the new String/Buffer */
L
Linus Torvalds 已提交
236

237 238
		return_desc->string.pointer = buffer;
		return_desc->string.length = (u32) length;
L
Linus Torvalds 已提交
239 240 241 242 243 244 245 246

		/* Mark buffer initialized */

		return_desc->buffer.flags |= AOPOBJ_DATA_VALID;
		break;

	default:

247
		ACPI_ERROR((AE_INFO, "Unknown AML opcode 0x%X",
B
Bob Moore 已提交
248
			    walk_state->opcode));
L
Linus Torvalds 已提交
249 250 251 252 253 254
		status = AE_AML_BAD_OPCODE;
		goto cleanup;
	}

	/* Store the result in the target */

L
Len Brown 已提交
255
	status = acpi_ex_store(return_desc, operand[3], walk_state);
L
Linus Torvalds 已提交
256

L
Len Brown 已提交
257
      cleanup:
L
Linus Torvalds 已提交
258 259 260

	/* Delete return object on error */

L
Len Brown 已提交
261 262
	if (ACPI_FAILURE(status) || walk_state->result_obj) {
		acpi_ut_remove_reference(return_desc);
263
		walk_state->result_obj = NULL;
L
Linus Torvalds 已提交
264 265 266 267
	}

	/* Set the return object and exit */

268
	else {
L
Linus Torvalds 已提交
269 270
		walk_state->result_obj = return_desc;
	}
L
Len Brown 已提交
271
	return_ACPI_STATUS(status);
L
Linus Torvalds 已提交
272
}