exfield.c

来自「是关于linux2.5.1的完全源码」· C语言 代码 · 共 307 行

C
307
字号
/****************************************************************************** * * Module Name: exfield - ACPI AML (p-code) execution - field manipulation *              $Revision: 108 $ * *****************************************************************************//* *  Copyright (C) 2000 - 2002, 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 "acdispat.h"#include "acinterp.h"#include "amlcode.h"#include "acnamesp.h"#include "achware.h"#include "acevents.h"#define _COMPONENT          ACPI_EXECUTER	 ACPI_MODULE_NAME    ("exfield")/******************************************************************************* * * FUNCTION:    Acpi_ex_read_data_from_field * * PARAMETERS:  Walk_state          - Current execution state *              Obj_desc            - The named field *              Ret_buffer_desc     - Where the return data object is stored * * RETURN:      Status * * DESCRIPTION: Read from a named field.  Returns either an Integer or a *              Buffer, depending on the size of the field. * ******************************************************************************/acpi_statusacpi_ex_read_data_from_field (	acpi_walk_state         *walk_state,	acpi_operand_object     *obj_desc,	acpi_operand_object     **ret_buffer_desc){	acpi_status             status;	acpi_operand_object     *buffer_desc;	u32                     length;	u32                     integer_size;	void                    *buffer;	u8                      locked;	ACPI_FUNCTION_TRACE_PTR ("Ex_read_data_from_field", obj_desc);	/* Parameter validation */	if (!obj_desc) {		return_ACPI_STATUS (AE_AML_NO_OPERAND);	}	if (obj_desc->common.type == ACPI_TYPE_BUFFER_FIELD) {		/*		 * If the Buffer_field arguments have not been previously evaluated,		 * evaluate them now and save the results.		 */		if (!(obj_desc->common.flags & AOPOBJ_DATA_VALID)) {			status = acpi_ds_get_buffer_field_arguments (obj_desc);			if (ACPI_FAILURE (status)) {				return_ACPI_STATUS (status);			}		}	}	/*	 * Allocate a buffer for the contents of the field.	 *	 * If the field is larger than the size of an acpi_integer, create	 * a BUFFER to hold it.  Otherwise, use an INTEGER.  This allows	 * the use of arithmetic operators on the returned value if the	 * field size is equal or smaller than an Integer.	 *	 * Note: Field.length is in bits.	 */	length = ACPI_ROUND_BITS_UP_TO_BYTES (obj_desc->field.bit_length);	/* Handle both ACPI 1.0 and ACPI 2.0 Integer widths */	integer_size = sizeof (acpi_integer);	if (walk_state->method_node->flags & ANOBJ_DATA_WIDTH_32) {		/*		 * We are running a method that exists in a 32-bit ACPI table.		 * Integer size is 4.		 */		integer_size = sizeof (u32);	}	if (length > integer_size) {		/* Field is too large for an Integer, create a Buffer instead */		buffer_desc = acpi_ut_create_internal_object (ACPI_TYPE_BUFFER);		if (!buffer_desc) {			return_ACPI_STATUS (AE_NO_MEMORY);		}		/* Create the actual read buffer */		buffer_desc->buffer.pointer = ACPI_MEM_CALLOCATE (length);		if (!buffer_desc->buffer.pointer) {			acpi_ut_remove_reference (buffer_desc);			return_ACPI_STATUS (AE_NO_MEMORY);		}		buffer_desc->common.flags = AOPOBJ_DATA_VALID;		buffer_desc->buffer.length = length;		buffer = buffer_desc->buffer.pointer;	}	else {		/* Field will fit within an Integer (normal case) */		buffer_desc = acpi_ut_create_internal_object (ACPI_TYPE_INTEGER);		if (!buffer_desc) {			return_ACPI_STATUS (AE_NO_MEMORY);		}		length = integer_size;		buffer_desc->integer.value = 0;		buffer = &buffer_desc->integer.value;	}	ACPI_DEBUG_PRINT ((ACPI_DB_BFIELD,		"Obj=%p Type=%X Buf=%p Len=%X\n",		obj_desc, obj_desc->common.type, buffer, length));	ACPI_DEBUG_PRINT ((ACPI_DB_BFIELD,		"Field_write: Bit_len=%X Bit_off=%X Byte_off=%X\n",		obj_desc->common_field.bit_length,		obj_desc->common_field.start_field_bit_offset,		obj_desc->common_field.base_byte_offset));	locked = acpi_ex_acquire_global_lock (obj_desc->common_field.field_flags);	/* Read from the field */	status = acpi_ex_extract_from_field (obj_desc, buffer, length);	/*	 * Release global lock if we acquired it earlier	 */	acpi_ex_release_global_lock (locked);	if (ACPI_FAILURE (status)) {		acpi_ut_remove_reference (buffer_desc);	}	else if (ret_buffer_desc) {		*ret_buffer_desc = buffer_desc;	}	return_ACPI_STATUS (status);}/******************************************************************************* * * FUNCTION:    Acpi_ex_write_data_to_field * * PARAMETERS:  Source_desc         - Contains data to write *              Obj_desc            - The named field * * RETURN:      Status * * DESCRIPTION: Write to a named field * ******************************************************************************/acpi_statusacpi_ex_write_data_to_field (	acpi_operand_object     *source_desc,	acpi_operand_object     *obj_desc){	acpi_status             status;	u32                     length;	u32                     required_length;	void                    *buffer;	void                    *new_buffer;	u8                      locked;	ACPI_FUNCTION_TRACE_PTR ("Ex_write_data_to_field", obj_desc);	/* Parameter validation */	if (!source_desc || !obj_desc) {		return_ACPI_STATUS (AE_AML_NO_OPERAND);	}	if (obj_desc->common.type == ACPI_TYPE_BUFFER_FIELD) {		/*		 * If the Buffer_field arguments have not been previously evaluated,		 * evaluate them now and save the results.		 */		if (!(obj_desc->common.flags & AOPOBJ_DATA_VALID)) {			status = acpi_ds_get_buffer_field_arguments (obj_desc);			if (ACPI_FAILURE (status)) {				return_ACPI_STATUS (status);			}		}	}	/*	 * Get a pointer to the data to be written	 */	switch (source_desc->common.type) {	case ACPI_TYPE_INTEGER:		buffer = &source_desc->integer.value;		length = sizeof (source_desc->integer.value);		break;	case ACPI_TYPE_BUFFER:		buffer = source_desc->buffer.pointer;		length = source_desc->buffer.length;		break;	case ACPI_TYPE_STRING:		buffer = source_desc->string.pointer;		length = source_desc->string.length;		break;	default:		return_ACPI_STATUS (AE_AML_OPERAND_TYPE);	}	/*	 * We must have a buffer that is at least as long as the field	 * we are writing to.  This is because individual fields are	 * indivisible and partial writes are not supported -- as per	 * the ACPI specification.	 */	new_buffer = NULL;	required_length = ACPI_ROUND_BITS_UP_TO_BYTES (obj_desc->common_field.bit_length);	if (length < required_length) {		/* We need to create a new buffer */		new_buffer = ACPI_MEM_CALLOCATE (required_length);		if (!new_buffer) {			return_ACPI_STATUS (AE_NO_MEMORY);		}		/*		 * Copy the original data to the new buffer, starting		 * at Byte zero.  All unused (upper) bytes of the		 * buffer will be 0.		 */		ACPI_MEMCPY ((char *) new_buffer, (char *) buffer, length);		buffer = new_buffer;		length = required_length;	}	ACPI_DEBUG_PRINT ((ACPI_DB_BFIELD,		"Obj=%p Type=%X Buf=%p Len=%X\n",		obj_desc, obj_desc->common.type, buffer, length));	ACPI_DEBUG_PRINT ((ACPI_DB_BFIELD,		"Field_read: Bit_len=%X Bit_off=%X Byte_off=%X\n",		obj_desc->common_field.bit_length,		obj_desc->common_field.start_field_bit_offset,		obj_desc->common_field.base_byte_offset));	locked = acpi_ex_acquire_global_lock (obj_desc->common_field.field_flags);	/*	 * Write to the field	 */	status = acpi_ex_insert_into_field (obj_desc, buffer, length);	/*	 * Release global lock if we acquired it earlier	 */	acpi_ex_release_global_lock (locked);	/* Free temporary buffer if we used one */	if (new_buffer) {		ACPI_MEM_FREE (new_buffer);	}	return_ACPI_STATUS (status);}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?