7a54f25cef
This makes it possible to build pci hotplug drivers outside of the main kernel tree, and Sam keeps telling me to move local header files to their proper places... Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
414 lines
11 KiB
C
414 lines
11 KiB
C
/*
|
|
* Common ACPI functions for hot plug platforms
|
|
*
|
|
* Copyright (C) 2006 Intel Corporation
|
|
*
|
|
* All rights reserved.
|
|
*
|
|
* 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, GOOD TITLE or
|
|
* NON INFRINGEMENT. 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., 675 Mass Ave, Cambridge, MA 02139, USA.
|
|
*
|
|
* Send feedback to <kristen.c.accardi@intel.com>
|
|
*
|
|
*/
|
|
|
|
#include <linux/module.h>
|
|
#include <linux/moduleparam.h>
|
|
#include <linux/kernel.h>
|
|
#include <linux/types.h>
|
|
#include <linux/pci.h>
|
|
#include <linux/pci_hotplug.h>
|
|
#include <acpi/acpi.h>
|
|
#include <acpi/acpi_bus.h>
|
|
#include <acpi/actypes.h>
|
|
|
|
#define MY_NAME "acpi_pcihp"
|
|
|
|
#define dbg(fmt, arg...) do { if (debug_acpi) printk(KERN_DEBUG "%s: %s: " fmt , MY_NAME , __FUNCTION__ , ## arg); } while (0)
|
|
#define err(format, arg...) printk(KERN_ERR "%s: " format , MY_NAME , ## arg)
|
|
#define info(format, arg...) printk(KERN_INFO "%s: " format , MY_NAME , ## arg)
|
|
#define warn(format, arg...) printk(KERN_WARNING "%s: " format , MY_NAME , ## arg)
|
|
|
|
#define METHOD_NAME__SUN "_SUN"
|
|
#define METHOD_NAME__HPP "_HPP"
|
|
#define METHOD_NAME_OSHP "OSHP"
|
|
|
|
static int debug_acpi;
|
|
|
|
static acpi_status
|
|
decode_type0_hpx_record(union acpi_object *record, struct hotplug_params *hpx)
|
|
{
|
|
int i;
|
|
union acpi_object *fields = record->package.elements;
|
|
u32 revision = fields[1].integer.value;
|
|
|
|
switch (revision) {
|
|
case 1:
|
|
if (record->package.count != 6)
|
|
return AE_ERROR;
|
|
for (i = 2; i < 6; i++)
|
|
if (fields[i].type != ACPI_TYPE_INTEGER)
|
|
return AE_ERROR;
|
|
hpx->t0 = &hpx->type0_data;
|
|
hpx->t0->revision = revision;
|
|
hpx->t0->cache_line_size = fields[2].integer.value;
|
|
hpx->t0->latency_timer = fields[3].integer.value;
|
|
hpx->t0->enable_serr = fields[4].integer.value;
|
|
hpx->t0->enable_perr = fields[5].integer.value;
|
|
break;
|
|
default:
|
|
printk(KERN_WARNING
|
|
"%s: Type 0 Revision %d record not supported\n",
|
|
__FUNCTION__, revision);
|
|
return AE_ERROR;
|
|
}
|
|
return AE_OK;
|
|
}
|
|
|
|
static acpi_status
|
|
decode_type1_hpx_record(union acpi_object *record, struct hotplug_params *hpx)
|
|
{
|
|
int i;
|
|
union acpi_object *fields = record->package.elements;
|
|
u32 revision = fields[1].integer.value;
|
|
|
|
switch (revision) {
|
|
case 1:
|
|
if (record->package.count != 5)
|
|
return AE_ERROR;
|
|
for (i = 2; i < 5; i++)
|
|
if (fields[i].type != ACPI_TYPE_INTEGER)
|
|
return AE_ERROR;
|
|
hpx->t1 = &hpx->type1_data;
|
|
hpx->t1->revision = revision;
|
|
hpx->t1->max_mem_read = fields[2].integer.value;
|
|
hpx->t1->avg_max_split = fields[3].integer.value;
|
|
hpx->t1->tot_max_split = fields[4].integer.value;
|
|
break;
|
|
default:
|
|
printk(KERN_WARNING
|
|
"%s: Type 1 Revision %d record not supported\n",
|
|
__FUNCTION__, revision);
|
|
return AE_ERROR;
|
|
}
|
|
return AE_OK;
|
|
}
|
|
|
|
static acpi_status
|
|
decode_type2_hpx_record(union acpi_object *record, struct hotplug_params *hpx)
|
|
{
|
|
int i;
|
|
union acpi_object *fields = record->package.elements;
|
|
u32 revision = fields[1].integer.value;
|
|
|
|
switch (revision) {
|
|
case 1:
|
|
if (record->package.count != 18)
|
|
return AE_ERROR;
|
|
for (i = 2; i < 18; i++)
|
|
if (fields[i].type != ACPI_TYPE_INTEGER)
|
|
return AE_ERROR;
|
|
hpx->t2 = &hpx->type2_data;
|
|
hpx->t2->revision = revision;
|
|
hpx->t2->unc_err_mask_and = fields[2].integer.value;
|
|
hpx->t2->unc_err_mask_or = fields[3].integer.value;
|
|
hpx->t2->unc_err_sever_and = fields[4].integer.value;
|
|
hpx->t2->unc_err_sever_or = fields[5].integer.value;
|
|
hpx->t2->cor_err_mask_and = fields[6].integer.value;
|
|
hpx->t2->cor_err_mask_or = fields[7].integer.value;
|
|
hpx->t2->adv_err_cap_and = fields[8].integer.value;
|
|
hpx->t2->adv_err_cap_or = fields[9].integer.value;
|
|
hpx->t2->pci_exp_devctl_and = fields[10].integer.value;
|
|
hpx->t2->pci_exp_devctl_or = fields[11].integer.value;
|
|
hpx->t2->pci_exp_lnkctl_and = fields[12].integer.value;
|
|
hpx->t2->pci_exp_lnkctl_or = fields[13].integer.value;
|
|
hpx->t2->sec_unc_err_sever_and = fields[14].integer.value;
|
|
hpx->t2->sec_unc_err_sever_or = fields[15].integer.value;
|
|
hpx->t2->sec_unc_err_mask_and = fields[16].integer.value;
|
|
hpx->t2->sec_unc_err_mask_or = fields[17].integer.value;
|
|
break;
|
|
default:
|
|
printk(KERN_WARNING
|
|
"%s: Type 2 Revision %d record not supported\n",
|
|
__FUNCTION__, revision);
|
|
return AE_ERROR;
|
|
}
|
|
return AE_OK;
|
|
}
|
|
|
|
static acpi_status
|
|
acpi_run_hpx(acpi_handle handle, struct hotplug_params *hpx)
|
|
{
|
|
acpi_status status;
|
|
struct acpi_buffer buffer = {ACPI_ALLOCATE_BUFFER, NULL};
|
|
union acpi_object *package, *record, *fields;
|
|
u32 type;
|
|
int i;
|
|
|
|
/* Clear the return buffer with zeros */
|
|
memset(hpx, 0, sizeof(struct hotplug_params));
|
|
|
|
status = acpi_evaluate_object(handle, "_HPX", NULL, &buffer);
|
|
if (ACPI_FAILURE(status))
|
|
return status;
|
|
|
|
package = (union acpi_object *)buffer.pointer;
|
|
if (package->type != ACPI_TYPE_PACKAGE) {
|
|
status = AE_ERROR;
|
|
goto exit;
|
|
}
|
|
|
|
for (i = 0; i < package->package.count; i++) {
|
|
record = &package->package.elements[i];
|
|
if (record->type != ACPI_TYPE_PACKAGE) {
|
|
status = AE_ERROR;
|
|
goto exit;
|
|
}
|
|
|
|
fields = record->package.elements;
|
|
if (fields[0].type != ACPI_TYPE_INTEGER ||
|
|
fields[1].type != ACPI_TYPE_INTEGER) {
|
|
status = AE_ERROR;
|
|
goto exit;
|
|
}
|
|
|
|
type = fields[0].integer.value;
|
|
switch (type) {
|
|
case 0:
|
|
status = decode_type0_hpx_record(record, hpx);
|
|
if (ACPI_FAILURE(status))
|
|
goto exit;
|
|
break;
|
|
case 1:
|
|
status = decode_type1_hpx_record(record, hpx);
|
|
if (ACPI_FAILURE(status))
|
|
goto exit;
|
|
break;
|
|
case 2:
|
|
status = decode_type2_hpx_record(record, hpx);
|
|
if (ACPI_FAILURE(status))
|
|
goto exit;
|
|
break;
|
|
default:
|
|
printk(KERN_ERR "%s: Type %d record not supported\n",
|
|
__FUNCTION__, type);
|
|
status = AE_ERROR;
|
|
goto exit;
|
|
}
|
|
}
|
|
exit:
|
|
kfree(buffer.pointer);
|
|
return status;
|
|
}
|
|
|
|
static acpi_status
|
|
acpi_run_hpp(acpi_handle handle, struct hotplug_params *hpp)
|
|
{
|
|
acpi_status status;
|
|
u8 nui[4];
|
|
struct acpi_buffer ret_buf = { 0, NULL};
|
|
struct acpi_buffer string = { ACPI_ALLOCATE_BUFFER, NULL };
|
|
union acpi_object *ext_obj, *package;
|
|
int i, len = 0;
|
|
|
|
acpi_get_name(handle, ACPI_FULL_PATHNAME, &string);
|
|
|
|
/* Clear the return buffer with zeros */
|
|
memset(hpp, 0, sizeof(struct hotplug_params));
|
|
|
|
/* get _hpp */
|
|
status = acpi_evaluate_object(handle, METHOD_NAME__HPP, NULL, &ret_buf);
|
|
switch (status) {
|
|
case AE_BUFFER_OVERFLOW:
|
|
ret_buf.pointer = kmalloc (ret_buf.length, GFP_KERNEL);
|
|
if (!ret_buf.pointer) {
|
|
printk(KERN_ERR "%s:%s alloc for _HPP fail\n",
|
|
__FUNCTION__, (char *)string.pointer);
|
|
kfree(string.pointer);
|
|
return AE_NO_MEMORY;
|
|
}
|
|
status = acpi_evaluate_object(handle, METHOD_NAME__HPP,
|
|
NULL, &ret_buf);
|
|
if (ACPI_SUCCESS(status))
|
|
break;
|
|
default:
|
|
if (ACPI_FAILURE(status)) {
|
|
pr_debug("%s:%s _HPP fail=0x%x\n", __FUNCTION__,
|
|
(char *)string.pointer, status);
|
|
kfree(string.pointer);
|
|
return status;
|
|
}
|
|
}
|
|
|
|
ext_obj = (union acpi_object *) ret_buf.pointer;
|
|
if (ext_obj->type != ACPI_TYPE_PACKAGE) {
|
|
printk(KERN_ERR "%s:%s _HPP obj not a package\n", __FUNCTION__,
|
|
(char *)string.pointer);
|
|
status = AE_ERROR;
|
|
goto free_and_return;
|
|
}
|
|
|
|
len = ext_obj->package.count;
|
|
package = (union acpi_object *) ret_buf.pointer;
|
|
for ( i = 0; (i < len) || (i < 4); i++) {
|
|
ext_obj = (union acpi_object *) &package->package.elements[i];
|
|
switch (ext_obj->type) {
|
|
case ACPI_TYPE_INTEGER:
|
|
nui[i] = (u8)ext_obj->integer.value;
|
|
break;
|
|
default:
|
|
printk(KERN_ERR "%s:%s _HPP obj type incorrect\n",
|
|
__FUNCTION__, (char *)string.pointer);
|
|
status = AE_ERROR;
|
|
goto free_and_return;
|
|
}
|
|
}
|
|
|
|
hpp->t0 = &hpp->type0_data;
|
|
hpp->t0->cache_line_size = nui[0];
|
|
hpp->t0->latency_timer = nui[1];
|
|
hpp->t0->enable_serr = nui[2];
|
|
hpp->t0->enable_perr = nui[3];
|
|
|
|
pr_debug(" _HPP: cache_line_size=0x%x\n", hpp->t0->cache_line_size);
|
|
pr_debug(" _HPP: latency timer =0x%x\n", hpp->t0->latency_timer);
|
|
pr_debug(" _HPP: enable SERR =0x%x\n", hpp->t0->enable_serr);
|
|
pr_debug(" _HPP: enable PERR =0x%x\n", hpp->t0->enable_perr);
|
|
|
|
free_and_return:
|
|
kfree(string.pointer);
|
|
kfree(ret_buf.pointer);
|
|
return status;
|
|
}
|
|
|
|
|
|
|
|
/* acpi_run_oshp - get control of hotplug from the firmware
|
|
*
|
|
* @handle - the handle of the hotplug controller.
|
|
*/
|
|
acpi_status acpi_run_oshp(acpi_handle handle)
|
|
{
|
|
acpi_status status;
|
|
struct acpi_buffer string = { ACPI_ALLOCATE_BUFFER, NULL };
|
|
|
|
acpi_get_name(handle, ACPI_FULL_PATHNAME, &string);
|
|
|
|
/* run OSHP */
|
|
status = acpi_evaluate_object(handle, METHOD_NAME_OSHP, NULL, NULL);
|
|
if (ACPI_FAILURE(status))
|
|
if (status != AE_NOT_FOUND)
|
|
printk(KERN_ERR "%s:%s OSHP fails=0x%x\n",
|
|
__FUNCTION__, (char *)string.pointer, status);
|
|
else
|
|
dbg("%s:%s OSHP not found\n",
|
|
__FUNCTION__, (char *)string.pointer);
|
|
else
|
|
pr_debug("%s:%s OSHP passes\n", __FUNCTION__,
|
|
(char *)string.pointer);
|
|
|
|
kfree(string.pointer);
|
|
return status;
|
|
}
|
|
EXPORT_SYMBOL_GPL(acpi_run_oshp);
|
|
|
|
|
|
|
|
/* acpi_get_hp_params_from_firmware
|
|
*
|
|
* @bus - the pci_bus of the bus on which the device is newly added
|
|
* @hpp - allocated by the caller
|
|
*/
|
|
acpi_status acpi_get_hp_params_from_firmware(struct pci_bus *bus,
|
|
struct hotplug_params *hpp)
|
|
{
|
|
acpi_status status = AE_NOT_FOUND;
|
|
acpi_handle handle, phandle;
|
|
struct pci_bus *pbus = bus;
|
|
struct pci_dev *pdev;
|
|
|
|
do {
|
|
pdev = pbus->self;
|
|
if (!pdev) {
|
|
handle = acpi_get_pci_rootbridge_handle(
|
|
pci_domain_nr(pbus), pbus->number);
|
|
break;
|
|
}
|
|
handle = DEVICE_ACPI_HANDLE(&(pdev->dev));
|
|
pbus = pbus->parent;
|
|
} while (!handle);
|
|
|
|
/*
|
|
* _HPP settings apply to all child buses, until another _HPP is
|
|
* encountered. If we don't find an _HPP for the input pci dev,
|
|
* look for it in the parent device scope since that would apply to
|
|
* this pci dev. If we don't find any _HPP, use hardcoded defaults
|
|
*/
|
|
while (handle) {
|
|
status = acpi_run_hpx(handle, hpp);
|
|
if (ACPI_SUCCESS(status))
|
|
break;
|
|
status = acpi_run_hpp(handle, hpp);
|
|
if (ACPI_SUCCESS(status))
|
|
break;
|
|
if (acpi_root_bridge(handle))
|
|
break;
|
|
status = acpi_get_parent(handle, &phandle);
|
|
if (ACPI_FAILURE(status))
|
|
break;
|
|
handle = phandle;
|
|
}
|
|
return status;
|
|
}
|
|
EXPORT_SYMBOL_GPL(acpi_get_hp_params_from_firmware);
|
|
|
|
|
|
/* acpi_root_bridge - check to see if this acpi object is a root bridge
|
|
*
|
|
* @handle - the acpi object in question.
|
|
*/
|
|
int acpi_root_bridge(acpi_handle handle)
|
|
{
|
|
acpi_status status;
|
|
struct acpi_device_info *info;
|
|
struct acpi_buffer buffer = {ACPI_ALLOCATE_BUFFER, NULL};
|
|
int i;
|
|
|
|
status = acpi_get_object_info(handle, &buffer);
|
|
if (ACPI_SUCCESS(status)) {
|
|
info = buffer.pointer;
|
|
if ((info->valid & ACPI_VALID_HID) &&
|
|
!strcmp(PCI_ROOT_HID_STRING,
|
|
info->hardware_id.value)) {
|
|
kfree(buffer.pointer);
|
|
return 1;
|
|
}
|
|
if (info->valid & ACPI_VALID_CID) {
|
|
for (i=0; i < info->compatibility_id.count; i++) {
|
|
if (!strcmp(PCI_ROOT_HID_STRING,
|
|
info->compatibility_id.id[i].value)) {
|
|
kfree(buffer.pointer);
|
|
return 1;
|
|
}
|
|
}
|
|
}
|
|
kfree(buffer.pointer);
|
|
}
|
|
return 0;
|
|
}
|
|
EXPORT_SYMBOL_GPL(acpi_root_bridge);
|
|
|
|
module_param(debug_acpi, bool, 0644);
|
|
MODULE_PARM_DESC(debug_acpi, "Debugging mode for ACPI enabled or not");
|