| /******************************************************************************* |
| * |
| * Module Name: rsxface - Public interfaces to the resource manager |
| * |
| ******************************************************************************/ |
| |
| /* |
| * Copyright (C) 2000 - 2005, R. Byron Moore |
| * 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 <linux/module.h> |
| |
| #include <acpi/acpi.h> |
| #include <acpi/acresrc.h> |
| |
| #define _COMPONENT ACPI_RESOURCES |
| ACPI_MODULE_NAME("rsxface") |
| |
| /* Local macros for 16,32-bit to 64-bit conversion */ |
| #define ACPI_COPY_FIELD(out, in, field) ((out)->field = (in)->field) |
| #define ACPI_COPY_ADDRESS(out, in) \ |
| ACPI_COPY_FIELD(out, in, resource_type); \ |
| ACPI_COPY_FIELD(out, in, producer_consumer); \ |
| ACPI_COPY_FIELD(out, in, decode); \ |
| ACPI_COPY_FIELD(out, in, min_address_fixed); \ |
| ACPI_COPY_FIELD(out, in, max_address_fixed); \ |
| ACPI_COPY_FIELD(out, in, attribute); \ |
| ACPI_COPY_FIELD(out, in, granularity); \ |
| ACPI_COPY_FIELD(out, in, min_address_range); \ |
| ACPI_COPY_FIELD(out, in, max_address_range); \ |
| ACPI_COPY_FIELD(out, in, address_translation_offset); \ |
| ACPI_COPY_FIELD(out, in, address_length); \ |
| ACPI_COPY_FIELD(out, in, resource_source); |
| /******************************************************************************* |
| * |
| * FUNCTION: acpi_get_irq_routing_table |
| * |
| * PARAMETERS: device_handle - a handle to the Bus device we are querying |
| * ret_buffer - a pointer to a buffer to receive the |
| * current resources for the device |
| * |
| * RETURN: Status |
| * |
| * DESCRIPTION: This function is called to get the IRQ routing table for a |
| * specific bus. The caller must first acquire a handle for the |
| * desired bus. The routine table is placed in the buffer pointed |
| * to by the ret_buffer variable parameter. |
| * |
| * If the function fails an appropriate status will be returned |
| * and the value of ret_buffer is undefined. |
| * |
| * This function attempts to execute the _PRT method contained in |
| * the object indicated by the passed device_handle. |
| * |
| ******************************************************************************/ |
| acpi_status |
| acpi_get_irq_routing_table(acpi_handle device_handle, |
| struct acpi_buffer *ret_buffer) |
| { |
| acpi_status status; |
| |
| ACPI_FUNCTION_TRACE("acpi_get_irq_routing_table "); |
| |
| /* |
| * Must have a valid handle and buffer, So we have to have a handle |
| * and a return buffer structure, and if there is a non-zero buffer length |
| * we also need a valid pointer in the buffer. If it's a zero buffer length, |
| * we'll be returning the needed buffer size, so keep going. |
| */ |
| if (!device_handle) { |
| return_ACPI_STATUS(AE_BAD_PARAMETER); |
| } |
| |
| status = acpi_ut_validate_buffer(ret_buffer); |
| if (ACPI_FAILURE(status)) { |
| return_ACPI_STATUS(status); |
| } |
| |
| status = acpi_rs_get_prt_method_data(device_handle, ret_buffer); |
| return_ACPI_STATUS(status); |
| } |
| |
| /******************************************************************************* |
| * |
| * FUNCTION: acpi_get_current_resources |
| * |
| * PARAMETERS: device_handle - a handle to the device object for the |
| * device we are querying |
| * ret_buffer - a pointer to a buffer to receive the |
| * current resources for the device |
| * |
| * RETURN: Status |
| * |
| * DESCRIPTION: This function is called to get the current resources for a |
| * specific device. The caller must first acquire a handle for |
| * the desired device. The resource data is placed in the buffer |
| * pointed to by the ret_buffer variable parameter. |
| * |
| * If the function fails an appropriate status will be returned |
| * and the value of ret_buffer is undefined. |
| * |
| * This function attempts to execute the _CRS method contained in |
| * the object indicated by the passed device_handle. |
| * |
| ******************************************************************************/ |
| |
| acpi_status |
| acpi_get_current_resources(acpi_handle device_handle, |
| struct acpi_buffer *ret_buffer) |
| { |
| acpi_status status; |
| |
| ACPI_FUNCTION_TRACE("acpi_get_current_resources"); |
| |
| /* |
| * Must have a valid handle and buffer, So we have to have a handle |
| * and a return buffer structure, and if there is a non-zero buffer length |
| * we also need a valid pointer in the buffer. If it's a zero buffer length, |
| * we'll be returning the needed buffer size, so keep going. |
| */ |
| if (!device_handle) { |
| return_ACPI_STATUS(AE_BAD_PARAMETER); |
| } |
| |
| status = acpi_ut_validate_buffer(ret_buffer); |
| if (ACPI_FAILURE(status)) { |
| return_ACPI_STATUS(status); |
| } |
| |
| status = acpi_rs_get_crs_method_data(device_handle, ret_buffer); |
| return_ACPI_STATUS(status); |
| } |
| |
| EXPORT_SYMBOL(acpi_get_current_resources); |
| |
| /******************************************************************************* |
| * |
| * FUNCTION: acpi_get_possible_resources |
| * |
| * PARAMETERS: device_handle - a handle to the device object for the |
| * device we are querying |
| * ret_buffer - a pointer to a buffer to receive the |
| * resources for the device |
| * |
| * RETURN: Status |
| * |
| * DESCRIPTION: This function is called to get a list of the possible resources |
| * for a specific device. The caller must first acquire a handle |
| * for the desired device. The resource data is placed in the |
| * buffer pointed to by the ret_buffer variable. |
| * |
| * If the function fails an appropriate status will be returned |
| * and the value of ret_buffer is undefined. |
| * |
| ******************************************************************************/ |
| |
| #ifdef ACPI_FUTURE_USAGE |
| acpi_status |
| acpi_get_possible_resources(acpi_handle device_handle, |
| struct acpi_buffer *ret_buffer) |
| { |
| acpi_status status; |
| |
| ACPI_FUNCTION_TRACE("acpi_get_possible_resources"); |
| |
| /* |
| * Must have a valid handle and buffer, So we have to have a handle |
| * and a return buffer structure, and if there is a non-zero buffer length |
| * we also need a valid pointer in the buffer. If it's a zero buffer length, |
| * we'll be returning the needed buffer size, so keep going. |
| */ |
| if (!device_handle) { |
| return_ACPI_STATUS(AE_BAD_PARAMETER); |
| } |
| |
| status = acpi_ut_validate_buffer(ret_buffer); |
| if (ACPI_FAILURE(status)) { |
| return_ACPI_STATUS(status); |
| } |
| |
| status = acpi_rs_get_prs_method_data(device_handle, ret_buffer); |
| return_ACPI_STATUS(status); |
| } |
| |
| EXPORT_SYMBOL(acpi_get_possible_resources); |
| #endif /* ACPI_FUTURE_USAGE */ |
| |
| /******************************************************************************* |
| * |
| * FUNCTION: acpi_walk_resources |
| * |
| * PARAMETERS: device_handle - a handle to the device object for the |
| * device we are querying |
| * Path - method name of the resources we want |
| * (METHOD_NAME__CRS or METHOD_NAME__PRS) |
| * user_function - called for each resource |
| * Context - passed to user_function |
| * |
| * RETURN: Status |
| * |
| * DESCRIPTION: Retrieves the current or possible resource list for the |
| * specified device. The user_function is called once for |
| * each resource in the list. |
| * |
| ******************************************************************************/ |
| |
| acpi_status |
| acpi_walk_resources(acpi_handle device_handle, |
| char *path, |
| ACPI_WALK_RESOURCE_CALLBACK user_function, void *context) |
| { |
| acpi_status status; |
| struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL }; |
| struct acpi_resource *resource; |
| struct acpi_resource *buffer_end; |
| |
| ACPI_FUNCTION_TRACE("acpi_walk_resources"); |
| |
| if (!device_handle || |
| (ACPI_STRNCMP(path, METHOD_NAME__CRS, sizeof(METHOD_NAME__CRS)) && |
| ACPI_STRNCMP(path, METHOD_NAME__PRS, sizeof(METHOD_NAME__PRS)))) { |
| return_ACPI_STATUS(AE_BAD_PARAMETER); |
| } |
| |
| status = acpi_rs_get_method_data(device_handle, path, &buffer); |
| if (ACPI_FAILURE(status)) { |
| return_ACPI_STATUS(status); |
| } |
| |
| /* Setup pointers */ |
| |
| resource = (struct acpi_resource *)buffer.pointer; |
| buffer_end = ACPI_CAST_PTR(struct acpi_resource, |
| ((u8 *) buffer.pointer + buffer.length)); |
| |
| /* Walk the resource list */ |
| |
| for (;;) { |
| if (!resource || resource->id == ACPI_RSTYPE_END_TAG) { |
| break; |
| } |
| |
| status = user_function(resource, context); |
| |
| switch (status) { |
| case AE_OK: |
| case AE_CTRL_DEPTH: |
| |
| /* Just keep going */ |
| |
| status = AE_OK; |
| break; |
| |
| case AE_CTRL_TERMINATE: |
| |
| /* Exit now, with OK stats */ |
| |
| status = AE_OK; |
| goto cleanup; |
| |
| default: |
| |
| /* All others are valid exceptions */ |
| |
| goto cleanup; |
| } |
| |
| /* Get the next resource descriptor */ |
| |
| resource = ACPI_NEXT_RESOURCE(resource); |
| |
| /* Check for end-of-buffer */ |
| |
| if (resource >= buffer_end) { |
| goto cleanup; |
| } |
| } |
| |
| cleanup: |
| |
| acpi_os_free(buffer.pointer); |
| return_ACPI_STATUS(status); |
| } |
| |
| EXPORT_SYMBOL(acpi_walk_resources); |
| |
| /******************************************************************************* |
| * |
| * FUNCTION: acpi_set_current_resources |
| * |
| * PARAMETERS: device_handle - a handle to the device object for the |
| * device we are changing the resources of |
| * in_buffer - a pointer to a buffer containing the |
| * resources to be set for the device |
| * |
| * RETURN: Status |
| * |
| * DESCRIPTION: This function is called to set the current resources for a |
| * specific device. The caller must first acquire a handle for |
| * the desired device. The resource data is passed to the routine |
| * the buffer pointed to by the in_buffer variable. |
| * |
| ******************************************************************************/ |
| |
| acpi_status |
| acpi_set_current_resources(acpi_handle device_handle, |
| struct acpi_buffer *in_buffer) |
| { |
| acpi_status status; |
| |
| ACPI_FUNCTION_TRACE("acpi_set_current_resources"); |
| |
| /* Must have a valid handle and buffer */ |
| |
| if ((!device_handle) || |
| (!in_buffer) || (!in_buffer->pointer) || (!in_buffer->length)) { |
| return_ACPI_STATUS(AE_BAD_PARAMETER); |
| } |
| |
| status = acpi_rs_set_srs_method_data(device_handle, in_buffer); |
| return_ACPI_STATUS(status); |
| } |
| |
| EXPORT_SYMBOL(acpi_set_current_resources); |
| |
| /****************************************************************************** |
| * |
| * FUNCTION: acpi_resource_to_address64 |
| * |
| * PARAMETERS: resource - Pointer to a resource |
| * out - Pointer to the users's return |
| * buffer (a struct |
| * struct acpi_resource_address64) |
| * |
| * RETURN: Status |
| * |
| * DESCRIPTION: If the resource is an address16, address32, or address64, |
| * copy it to the address64 return buffer. This saves the |
| * caller from having to duplicate code for different-sized |
| * addresses. |
| * |
| ******************************************************************************/ |
| |
| acpi_status |
| acpi_resource_to_address64(struct acpi_resource *resource, |
| struct acpi_resource_address64 *out) |
| { |
| struct acpi_resource_address16 *address16; |
| struct acpi_resource_address32 *address32; |
| |
| switch (resource->id) { |
| case ACPI_RSTYPE_ADDRESS16: |
| |
| address16 = (struct acpi_resource_address16 *)&resource->data; |
| ACPI_COPY_ADDRESS(out, address16); |
| break; |
| |
| case ACPI_RSTYPE_ADDRESS32: |
| |
| address32 = (struct acpi_resource_address32 *)&resource->data; |
| ACPI_COPY_ADDRESS(out, address32); |
| break; |
| |
| case ACPI_RSTYPE_ADDRESS64: |
| |
| /* Simple copy for 64 bit source */ |
| |
| ACPI_MEMCPY(out, &resource->data, |
| sizeof(struct acpi_resource_address64)); |
| break; |
| |
| default: |
| return (AE_BAD_PARAMETER); |
| } |
| |
| return (AE_OK); |
| } |
| |
| EXPORT_SYMBOL(acpi_resource_to_address64); |