blob: 30f581b8791f5f84c6ab72a1f76bf1de8a5a2c87 [file] [log] [blame]
Zhang, Yanmin6c2b3742006-07-31 15:21:33 +08001/*
2 * Access ACPI _OSC method
3 *
4 * Copyright (C) 2006 Intel Corp.
5 * Tom Long Nguyen (tom.l.nguyen@intel.com)
6 * Zhang Yanmin (yanmin.zhang@intel.com)
7 *
8 */
9
10#include <linux/module.h>
11#include <linux/pci.h>
12#include <linux/kernel.h>
13#include <linux/errno.h>
14#include <linux/pm.h>
15#include <linux/suspend.h>
16#include <linux/acpi.h>
17#include <linux/pci-acpi.h>
18#include <linux/delay.h>
19#include "aerdrv.h"
20
21/**
22 * aer_osc_setup - run ACPI _OSC method
Zhang, Yanmin8d29bfb2007-06-06 11:44:16 +080023 * @pciedev: pcie_device which AER is being enabled on
Zhang, Yanmin6c2b3742006-07-31 15:21:33 +080024 *
Zhang, Yanmin8d29bfb2007-06-06 11:44:16 +080025 * @return: Zero on success. Nonzero otherwise.
Zhang, Yanmin6c2b3742006-07-31 15:21:33 +080026 *
27 * Invoked when PCIE bus loads AER service driver. To avoid conflict with
28 * BIOS AER support requires BIOS to yield AER control to OS native driver.
29 **/
Zhang, Yanmin8d29bfb2007-06-06 11:44:16 +080030int aer_osc_setup(struct pcie_device *pciedev)
Zhang, Yanmin6c2b3742006-07-31 15:21:33 +080031{
Zhang, Yanmin8d29bfb2007-06-06 11:44:16 +080032 acpi_status status = AE_NOT_FOUND;
33 struct pci_dev *pdev = pciedev->port;
Harvey Harrisona01e0352008-04-28 16:50:04 -070034 acpi_handle handle = NULL;
Zhang, Yanmin6c2b3742006-07-31 15:21:33 +080035
Yinghai Lu4c44bac2008-02-02 22:24:47 -080036 if (acpi_pci_disabled)
37 return -1;
38
Andrew Patterson3c75e232008-01-22 17:18:27 -070039 /* Find root host bridge */
Adrian Bunkd75b3052008-03-31 01:41:01 +030040 while (pdev->bus->self)
Andrew Patterson3c75e232008-01-22 17:18:27 -070041 pdev = pdev->bus->self;
42 handle = acpi_get_pci_rootbridge_handle(
43 pci_domain_nr(pdev->bus), pdev->bus->number);
Zhang, Yanmin6c2b3742006-07-31 15:21:33 +080044
Zhang, Yanmin8d29bfb2007-06-06 11:44:16 +080045 if (handle) {
Andrew Pattersond8634dd2008-01-22 17:18:17 -070046 pcie_osc_support_set(OSC_EXT_PCI_CONFIG_SUPPORT);
Zhang, Yanmin8d29bfb2007-06-06 11:44:16 +080047 status = pci_osc_control_set(handle,
48 OSC_PCI_EXPRESS_AER_CONTROL |
49 OSC_PCI_EXPRESS_CAP_STRUCTURE_CONTROL);
Zhang, Yanmin6c2b3742006-07-31 15:21:33 +080050 }
51
Zhang, Yanmin8d29bfb2007-06-06 11:44:16 +080052 if (ACPI_FAILURE(status)) {
Bjorn Helgaas531f2542008-06-13 10:52:12 -060053 dev_printk(KERN_DEBUG, &pciedev->device, "AER service couldn't "
54 "init device: %s\n",
55 (status == AE_SUPPORT || status == AE_NOT_FOUND) ?
56 "no _OSC support" : "_OSC failed");
Zhang, Yanmin8d29bfb2007-06-06 11:44:16 +080057 return -1;
58 }
Zhang, Yanmin6c2b3742006-07-31 15:21:33 +080059
Zhang, Yanmin8d29bfb2007-06-06 11:44:16 +080060 return 0;
61}