forked from mirrors/linux
		
	 c163f90cc8
			
		
	
	
		c163f90cc8
		
	
	
	
	
		
			
			Signed-off-by: Erik Schmauss <erik.schmauss@intel.com> Signed-off-by: Bob Moore <robert.moore@intel.com> Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
		
			
				
	
	
		
			345 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			345 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0
 | |
| /******************************************************************************
 | |
|  *
 | |
|  * Module Name: exserial - field_unit support for serial address spaces
 | |
|  *
 | |
|  * Copyright (C) 2000 - 2019, Intel Corp.
 | |
|  *
 | |
|  *****************************************************************************/
 | |
| 
 | |
| #include <acpi/acpi.h>
 | |
| #include "accommon.h"
 | |
| #include "acdispat.h"
 | |
| #include "acinterp.h"
 | |
| #include "amlcode.h"
 | |
| 
 | |
| #define _COMPONENT          ACPI_EXECUTER
 | |
| ACPI_MODULE_NAME("exserial")
 | |
| 
 | |
| /*******************************************************************************
 | |
|  *
 | |
|  * FUNCTION:    acpi_ex_read_gpio
 | |
|  *
 | |
|  * PARAMETERS:  obj_desc            - The named field to read
 | |
|  *              buffer              - Where the return data is returned
 | |
|  *
 | |
|  * RETURN:      Status
 | |
|  *
 | |
|  * DESCRIPTION: Read from a named field that references a Generic Serial Bus
 | |
|  *              field
 | |
|  *
 | |
|  ******************************************************************************/
 | |
| acpi_status acpi_ex_read_gpio(union acpi_operand_object *obj_desc, void *buffer)
 | |
| {
 | |
| 	acpi_status status;
 | |
| 
 | |
| 	ACPI_FUNCTION_TRACE_PTR(ex_read_gpio, obj_desc);
 | |
| 
 | |
| 	/*
 | |
| 	 * For GPIO (general_purpose_io), the Address will be the bit offset
 | |
| 	 * from the previous Connection() operator, making it effectively a
 | |
| 	 * pin number index. The bit_length is the length of the field, which
 | |
| 	 * is thus the number of pins.
 | |
| 	 */
 | |
| 	ACPI_DEBUG_PRINT((ACPI_DB_BFIELD,
 | |
| 			  "GPIO FieldRead [FROM]:  Pin %u Bits %u\n",
 | |
| 			  obj_desc->field.pin_number_index,
 | |
| 			  obj_desc->field.bit_length));
 | |
| 
 | |
| 	/* Lock entire transaction if requested */
 | |
| 
 | |
| 	acpi_ex_acquire_global_lock(obj_desc->common_field.field_flags);
 | |
| 
 | |
| 	/* Perform the read */
 | |
| 
 | |
| 	status = acpi_ex_access_region(obj_desc, 0, (u64 *)buffer, ACPI_READ);
 | |
| 
 | |
| 	acpi_ex_release_global_lock(obj_desc->common_field.field_flags);
 | |
| 	return_ACPI_STATUS(status);
 | |
| }
 | |
| 
 | |
| /*******************************************************************************
 | |
|  *
 | |
|  * FUNCTION:    acpi_ex_write_gpio
 | |
|  *
 | |
|  * PARAMETERS:  source_desc         - Contains data to write. Expect to be
 | |
|  *                                    an Integer object.
 | |
|  *              obj_desc            - The named field
 | |
|  *              result_desc         - Where the return value is returned, if any
 | |
|  *
 | |
|  * RETURN:      Status
 | |
|  *
 | |
|  * DESCRIPTION: Write to a named field that references a General Purpose I/O
 | |
|  *              field.
 | |
|  *
 | |
|  ******************************************************************************/
 | |
| 
 | |
| acpi_status
 | |
| acpi_ex_write_gpio(union acpi_operand_object *source_desc,
 | |
| 		   union acpi_operand_object *obj_desc,
 | |
| 		   union acpi_operand_object **return_buffer)
 | |
| {
 | |
| 	acpi_status status;
 | |
| 	void *buffer;
 | |
| 
 | |
| 	ACPI_FUNCTION_TRACE_PTR(ex_write_gpio, obj_desc);
 | |
| 
 | |
| 	/*
 | |
| 	 * For GPIO (general_purpose_io), we will bypass the entire field
 | |
| 	 * mechanism and handoff the bit address and bit width directly to
 | |
| 	 * the handler. The Address will be the bit offset
 | |
| 	 * from the previous Connection() operator, making it effectively a
 | |
| 	 * pin number index. The bit_length is the length of the field, which
 | |
| 	 * is thus the number of pins.
 | |
| 	 */
 | |
| 	if (source_desc->common.type != ACPI_TYPE_INTEGER) {
 | |
| 		return_ACPI_STATUS(AE_AML_OPERAND_TYPE);
 | |
| 	}
 | |
| 
 | |
| 	ACPI_DEBUG_PRINT((ACPI_DB_BFIELD,
 | |
| 			  "GPIO FieldWrite [FROM]: (%s:%X), Value %.8X  [TO]: Pin %u Bits %u\n",
 | |
| 			  acpi_ut_get_type_name(source_desc->common.type),
 | |
| 			  source_desc->common.type,
 | |
| 			  (u32)source_desc->integer.value,
 | |
| 			  obj_desc->field.pin_number_index,
 | |
| 			  obj_desc->field.bit_length));
 | |
| 
 | |
| 	buffer = &source_desc->integer.value;
 | |
| 
 | |
| 	/* Lock entire transaction if requested */
 | |
| 
 | |
| 	acpi_ex_acquire_global_lock(obj_desc->common_field.field_flags);
 | |
| 
 | |
| 	/* Perform the write */
 | |
| 
 | |
| 	status = acpi_ex_access_region(obj_desc, 0, (u64 *)buffer, ACPI_WRITE);
 | |
| 	acpi_ex_release_global_lock(obj_desc->common_field.field_flags);
 | |
| 	return_ACPI_STATUS(status);
 | |
| }
 | |
| 
 | |
| /*******************************************************************************
 | |
|  *
 | |
|  * FUNCTION:    acpi_ex_read_serial_bus
 | |
|  *
 | |
|  * PARAMETERS:  obj_desc            - The named field to read
 | |
|  *              return_buffer       - Where the return value is returned, if any
 | |
|  *
 | |
|  * RETURN:      Status
 | |
|  *
 | |
|  * DESCRIPTION: Read from a named field that references a serial bus
 | |
|  *              (SMBus, IPMI, or GSBus).
 | |
|  *
 | |
|  ******************************************************************************/
 | |
| 
 | |
| acpi_status
 | |
| acpi_ex_read_serial_bus(union acpi_operand_object *obj_desc,
 | |
| 			union acpi_operand_object **return_buffer)
 | |
| {
 | |
| 	acpi_status status;
 | |
| 	u32 buffer_length;
 | |
| 	union acpi_operand_object *buffer_desc;
 | |
| 	u32 function;
 | |
| 	u16 accessor_type;
 | |
| 
 | |
| 	ACPI_FUNCTION_TRACE_PTR(ex_read_serial_bus, obj_desc);
 | |
| 
 | |
| 	/*
 | |
| 	 * This is an SMBus, GSBus or IPMI read. We must create a buffer to
 | |
| 	 * hold the data and then directly access the region handler.
 | |
| 	 *
 | |
| 	 * Note: SMBus and GSBus protocol value is passed in upper 16-bits
 | |
| 	 * of Function
 | |
| 	 *
 | |
| 	 * Common buffer format:
 | |
| 	 *     Status;    (Byte 0 of the data buffer)
 | |
| 	 *     Length;    (Byte 1 of the data buffer)
 | |
| 	 *     Data[x-1]: (Bytes 2-x of the arbitrary length data buffer)
 | |
| 	 */
 | |
| 	switch (obj_desc->field.region_obj->region.space_id) {
 | |
| 	case ACPI_ADR_SPACE_SMBUS:
 | |
| 
 | |
| 		buffer_length = ACPI_SMBUS_BUFFER_SIZE;
 | |
| 		function = ACPI_READ | (obj_desc->field.attribute << 16);
 | |
| 		break;
 | |
| 
 | |
| 	case ACPI_ADR_SPACE_IPMI:
 | |
| 
 | |
| 		buffer_length = ACPI_IPMI_BUFFER_SIZE;
 | |
| 		function = ACPI_READ;
 | |
| 		break;
 | |
| 
 | |
| 	case ACPI_ADR_SPACE_GSBUS:
 | |
| 
 | |
| 		accessor_type = obj_desc->field.attribute;
 | |
| 		if (accessor_type == AML_FIELD_ATTRIB_RAW_PROCESS_BYTES) {
 | |
| 			ACPI_ERROR((AE_INFO,
 | |
| 				    "Invalid direct read using bidirectional write-then-read protocol"));
 | |
| 
 | |
| 			return_ACPI_STATUS(AE_AML_PROTOCOL);
 | |
| 		}
 | |
| 
 | |
| 		status =
 | |
| 		    acpi_ex_get_protocol_buffer_length(accessor_type,
 | |
| 						       &buffer_length);
 | |
| 		if (ACPI_FAILURE(status)) {
 | |
| 			ACPI_ERROR((AE_INFO,
 | |
| 				    "Invalid protocol ID for GSBus: 0x%4.4X",
 | |
| 				    accessor_type));
 | |
| 
 | |
| 			return_ACPI_STATUS(status);
 | |
| 		}
 | |
| 
 | |
| 		/* Add header length to get the full size of the buffer */
 | |
| 
 | |
| 		buffer_length += ACPI_SERIAL_HEADER_SIZE;
 | |
| 		function = ACPI_READ | (accessor_type << 16);
 | |
| 		break;
 | |
| 
 | |
| 	default:
 | |
| 		return_ACPI_STATUS(AE_AML_INVALID_SPACE_ID);
 | |
| 	}
 | |
| 
 | |
| 	/* Create the local transfer buffer that is returned to the caller */
 | |
| 
 | |
| 	buffer_desc = acpi_ut_create_buffer_object(buffer_length);
 | |
| 	if (!buffer_desc) {
 | |
| 		return_ACPI_STATUS(AE_NO_MEMORY);
 | |
| 	}
 | |
| 
 | |
| 	/* Lock entire transaction if requested */
 | |
| 
 | |
| 	acpi_ex_acquire_global_lock(obj_desc->common_field.field_flags);
 | |
| 
 | |
| 	/* Call the region handler for the write-then-read */
 | |
| 
 | |
| 	status = acpi_ex_access_region(obj_desc, 0,
 | |
| 				       ACPI_CAST_PTR(u64,
 | |
| 						     buffer_desc->buffer.
 | |
| 						     pointer), function);
 | |
| 	acpi_ex_release_global_lock(obj_desc->common_field.field_flags);
 | |
| 
 | |
| 	*return_buffer = buffer_desc;
 | |
| 	return_ACPI_STATUS(status);
 | |
| }
 | |
| 
 | |
| /*******************************************************************************
 | |
|  *
 | |
|  * FUNCTION:    acpi_ex_write_serial_bus
 | |
|  *
 | |
|  * PARAMETERS:  source_desc         - Contains data to write
 | |
|  *              obj_desc            - The named field
 | |
|  *              return_buffer       - Where the return value is returned, if any
 | |
|  *
 | |
|  * RETURN:      Status
 | |
|  *
 | |
|  * DESCRIPTION: Write to a named field that references a serial bus
 | |
|  *              (SMBus, IPMI, GSBus).
 | |
|  *
 | |
|  ******************************************************************************/
 | |
| 
 | |
| acpi_status
 | |
| acpi_ex_write_serial_bus(union acpi_operand_object *source_desc,
 | |
| 			 union acpi_operand_object *obj_desc,
 | |
| 			 union acpi_operand_object **return_buffer)
 | |
| {
 | |
| 	acpi_status status;
 | |
| 	u32 buffer_length;
 | |
| 	u32 data_length;
 | |
| 	void *buffer;
 | |
| 	union acpi_operand_object *buffer_desc;
 | |
| 	u32 function;
 | |
| 	u16 accessor_type;
 | |
| 
 | |
| 	ACPI_FUNCTION_TRACE_PTR(ex_write_serial_bus, obj_desc);
 | |
| 
 | |
| 	/*
 | |
| 	 * This is an SMBus, GSBus or IPMI write. We will bypass the entire
 | |
| 	 * field mechanism and handoff the buffer directly to the handler.
 | |
| 	 * For these address spaces, the buffer is bidirectional; on a
 | |
| 	 * write, return data is returned in the same buffer.
 | |
| 	 *
 | |
| 	 * Source must be a buffer of sufficient size, these are fixed size:
 | |
| 	 * ACPI_SMBUS_BUFFER_SIZE, or ACPI_IPMI_BUFFER_SIZE.
 | |
| 	 *
 | |
| 	 * Note: SMBus and GSBus protocol type is passed in upper 16-bits
 | |
| 	 * of Function
 | |
| 	 *
 | |
| 	 * Common buffer format:
 | |
| 	 *     Status;    (Byte 0 of the data buffer)
 | |
| 	 *     Length;    (Byte 1 of the data buffer)
 | |
| 	 *     Data[x-1]: (Bytes 2-x of the arbitrary length data buffer)
 | |
| 	 */
 | |
| 	if (source_desc->common.type != ACPI_TYPE_BUFFER) {
 | |
| 		ACPI_ERROR((AE_INFO,
 | |
| 			    "SMBus/IPMI/GenericSerialBus write requires "
 | |
| 			    "Buffer, found type %s",
 | |
| 			    acpi_ut_get_object_type_name(source_desc)));
 | |
| 
 | |
| 		return_ACPI_STATUS(AE_AML_OPERAND_TYPE);
 | |
| 	}
 | |
| 
 | |
| 	switch (obj_desc->field.region_obj->region.space_id) {
 | |
| 	case ACPI_ADR_SPACE_SMBUS:
 | |
| 
 | |
| 		buffer_length = ACPI_SMBUS_BUFFER_SIZE;
 | |
| 		function = ACPI_WRITE | (obj_desc->field.attribute << 16);
 | |
| 		break;
 | |
| 
 | |
| 	case ACPI_ADR_SPACE_IPMI:
 | |
| 
 | |
| 		buffer_length = ACPI_IPMI_BUFFER_SIZE;
 | |
| 		function = ACPI_WRITE;
 | |
| 		break;
 | |
| 
 | |
| 	case ACPI_ADR_SPACE_GSBUS:
 | |
| 
 | |
| 		accessor_type = obj_desc->field.attribute;
 | |
| 		status =
 | |
| 		    acpi_ex_get_protocol_buffer_length(accessor_type,
 | |
| 						       &buffer_length);
 | |
| 		if (ACPI_FAILURE(status)) {
 | |
| 			ACPI_ERROR((AE_INFO,
 | |
| 				    "Invalid protocol ID for GSBus: 0x%4.4X",
 | |
| 				    accessor_type));
 | |
| 
 | |
| 			return_ACPI_STATUS(status);
 | |
| 		}
 | |
| 
 | |
| 		/* Add header length to get the full size of the buffer */
 | |
| 
 | |
| 		buffer_length += ACPI_SERIAL_HEADER_SIZE;
 | |
| 		function = ACPI_WRITE | (accessor_type << 16);
 | |
| 		break;
 | |
| 
 | |
| 	default:
 | |
| 		return_ACPI_STATUS(AE_AML_INVALID_SPACE_ID);
 | |
| 	}
 | |
| 
 | |
| 	/* Create the transfer/bidirectional/return buffer */
 | |
| 
 | |
| 	buffer_desc = acpi_ut_create_buffer_object(buffer_length);
 | |
| 	if (!buffer_desc) {
 | |
| 		return_ACPI_STATUS(AE_NO_MEMORY);
 | |
| 	}
 | |
| 
 | |
| 	/* Copy the input buffer data to the transfer buffer */
 | |
| 
 | |
| 	buffer = buffer_desc->buffer.pointer;
 | |
| 	data_length = (buffer_length < source_desc->buffer.length ?
 | |
| 		       buffer_length : source_desc->buffer.length);
 | |
| 	memcpy(buffer, source_desc->buffer.pointer, data_length);
 | |
| 
 | |
| 	/* Lock entire transaction if requested */
 | |
| 
 | |
| 	acpi_ex_acquire_global_lock(obj_desc->common_field.field_flags);
 | |
| 
 | |
| 	/*
 | |
| 	 * Perform the write (returns status and perhaps data in the
 | |
| 	 * same buffer)
 | |
| 	 */
 | |
| 	status = acpi_ex_access_region(obj_desc, 0, (u64 *)buffer, function);
 | |
| 	acpi_ex_release_global_lock(obj_desc->common_field.field_flags);
 | |
| 
 | |
| 	*return_buffer = buffer_desc;
 | |
| 	return_ACPI_STATUS(status);
 | |
| }
 |