Commit | Line | Data |
---|---|---|
736759ef | 1 | // SPDX-License-Identifier: GPL-2.0+ |
1da177e4 | 2 | /* |
783c49fc | 3 | * Common ACPI functions for hot plug platforms |
1da177e4 | 4 | * |
783c49fc | 5 | * Copyright (C) 2006 Intel Corporation |
1da177e4 LT |
6 | * |
7 | * All rights reserved. | |
8 | * | |
8cf4c195 | 9 | * Send feedback to <kristen.c.accardi@intel.com> |
1da177e4 LT |
10 | */ |
11 | ||
1da177e4 | 12 | #include <linux/module.h> |
aad20cab | 13 | #include <linux/moduleparam.h> |
1da177e4 LT |
14 | #include <linux/kernel.h> |
15 | #include <linux/types.h> | |
16 | #include <linux/pci.h> | |
7a54f25c | 17 | #include <linux/pci_hotplug.h> |
9f5404d8 | 18 | #include <linux/acpi.h> |
ac9c052d | 19 | #include <linux/pci-acpi.h> |
5a0e3ad6 | 20 | #include <linux/slab.h> |
1da177e4 | 21 | |
aad20cab KK |
22 | #define MY_NAME "acpi_pcihp" |
23 | ||
ff3ce480 BS |
24 | #define dbg(fmt, arg...) do { if (debug_acpi) printk(KERN_DEBUG "%s: %s: " fmt, MY_NAME, __func__, ## arg); } while (0) |
25 | #define err(format, arg...) printk(KERN_ERR "%s: " format, MY_NAME, ## arg) | |
26 | #define info(format, arg...) printk(KERN_INFO "%s: " format, MY_NAME, ## arg) | |
27 | #define warn(format, arg...) printk(KERN_WARNING "%s: " format, MY_NAME, ## arg) | |
aad20cab | 28 | |
1da177e4 | 29 | #define METHOD_NAME__SUN "_SUN" |
1da177e4 LT |
30 | #define METHOD_NAME_OSHP "OSHP" |
31 | ||
90ab5ee9 | 32 | static bool debug_acpi; |
aad20cab | 33 | |
783c49fc KA |
34 | /* acpi_run_oshp - get control of hotplug from the firmware |
35 | * | |
36 | * @handle - the handle of the hotplug controller. | |
37 | */ | |
ac9c052d | 38 | static acpi_status acpi_run_oshp(acpi_handle handle) |
1da177e4 LT |
39 | { |
40 | acpi_status status; | |
b2e6e3ba MT |
41 | struct acpi_buffer string = { ACPI_ALLOCATE_BUFFER, NULL }; |
42 | ||
43 | acpi_get_name(handle, ACPI_FULL_PATHNAME, &string); | |
1da177e4 LT |
44 | |
45 | /* run OSHP */ | |
a8a2be94 | 46 | status = acpi_evaluate_object(handle, METHOD_NAME_OSHP, NULL, NULL); |
783c49fc | 47 | if (ACPI_FAILURE(status)) |
aad20cab KK |
48 | if (status != AE_NOT_FOUND) |
49 | printk(KERN_ERR "%s:%s OSHP fails=0x%x\n", | |
66bef8c0 | 50 | __func__, (char *)string.pointer, status); |
aad20cab KK |
51 | else |
52 | dbg("%s:%s OSHP not found\n", | |
66bef8c0 | 53 | __func__, (char *)string.pointer); |
783c49fc | 54 | else |
66bef8c0 | 55 | pr_debug("%s:%s OSHP passes\n", __func__, |
b2e6e3ba MT |
56 | (char *)string.pointer); |
57 | ||
81b26bca | 58 | kfree(string.pointer); |
783c49fc KA |
59 | return status; |
60 | } | |
783c49fc | 61 | |
ac9c052d KK |
62 | /** |
63 | * acpi_get_hp_hw_control_from_firmware | |
64 | * @dev: the pci_dev of the bridge that has a hotplug controller | |
ac9c052d KK |
65 | * |
66 | * Attempt to take hotplug control from firmware. | |
67 | */ | |
6f77fa49 | 68 | int acpi_get_hp_hw_control_from_firmware(struct pci_dev *pdev) |
ac9c052d | 69 | { |
aa6be07c MW |
70 | const struct pci_host_bridge *host; |
71 | const struct acpi_pci_root *root; | |
ac9c052d | 72 | acpi_status status; |
e0d94bee | 73 | acpi_handle chandle, handle; |
ac9c052d KK |
74 | struct acpi_buffer string = { ACPI_ALLOCATE_BUFFER, NULL }; |
75 | ||
ac9c052d KK |
76 | /* |
77 | * Per PCI firmware specification, we should run the ACPI _OSC | |
78 | * method to get control of hotplug hardware before using it. If | |
79 | * an _OSC is missing, we look for an OSHP to do the same thing. | |
e0d94bee JS |
80 | * To handle different BIOS behavior, we look for _OSC on a root |
81 | * bridge preferentially (according to PCI fw spec). Later for | |
82 | * OSHP within the scope of the hotplug controller and its parents, | |
25985edc | 83 | * up to the host bridge under which this controller exists. |
ac9c052d | 84 | */ |
90cc0c3c | 85 | if (shpchp_is_native(pdev)) |
aa6be07c MW |
86 | return 0; |
87 | ||
88 | /* If _OSC exists, we should not evaluate OSHP */ | |
6f6f4246 BH |
89 | |
90 | /* | |
91 | * If there's no ACPI host bridge (i.e., ACPI support is compiled | |
92 | * into the kernel but the hardware platform doesn't support ACPI), | |
93 | * there's nothing to do here. | |
94 | */ | |
90cc0c3c | 95 | host = pci_find_host_bridge(pdev->bus); |
aa6be07c | 96 | root = acpi_pci_find_root(ACPI_HANDLE(&host->dev)); |
6f6f4246 BH |
97 | if (!root) |
98 | return 0; | |
99 | ||
aa6be07c MW |
100 | if (root->osc_support_set) |
101 | goto no_control; | |
e0d94bee | 102 | |
3a83f992 | 103 | handle = ACPI_HANDLE(&pdev->dev); |
d391f00f | 104 | if (!handle) { |
ac9c052d KK |
105 | /* |
106 | * This hotplug controller was not listed in the ACPI name | |
f2b775f5 | 107 | * space at all. Try to get ACPI handle of parent PCI bus. |
ac9c052d | 108 | */ |
d391f00f KK |
109 | struct pci_bus *pbus; |
110 | for (pbus = pdev->bus; pbus; pbus = pbus->parent) { | |
111 | handle = acpi_pci_get_bridge_handle(pbus); | |
112 | if (handle) | |
113 | break; | |
114 | } | |
ac9c052d KK |
115 | } |
116 | ||
117 | while (handle) { | |
118 | acpi_get_name(handle, ACPI_FULL_PATHNAME, &string); | |
f2b775f5 BH |
119 | pci_info(pdev, "Requesting control of SHPC hotplug via OSHP (%s)\n", |
120 | (char *)string.pointer); | |
e0d94bee JS |
121 | status = acpi_run_oshp(handle); |
122 | if (ACPI_SUCCESS(status)) | |
123 | goto got_one; | |
27558203 | 124 | if (acpi_is_root_bridge(handle)) |
ac9c052d KK |
125 | break; |
126 | chandle = handle; | |
127 | status = acpi_get_parent(chandle, &handle); | |
128 | if (ACPI_FAILURE(status)) | |
129 | break; | |
130 | } | |
9b536e0b | 131 | no_control: |
f2b775f5 | 132 | pci_info(pdev, "Cannot get control of SHPC hotplug\n"); |
ac9c052d KK |
133 | kfree(string.pointer); |
134 | return -ENODEV; | |
e0d94bee | 135 | got_one: |
f2b775f5 BH |
136 | pci_info(pdev, "Gained control of SHPC hotplug (%s)\n", |
137 | (char *)string.pointer); | |
e0d94bee JS |
138 | kfree(string.pointer); |
139 | return 0; | |
ac9c052d KK |
140 | } |
141 | EXPORT_SYMBOL(acpi_get_hp_hw_control_from_firmware); | |
1da177e4 | 142 | |
efe6d727 | 143 | static int pcihp_is_ejectable(acpi_handle handle) |
e8c331e9 KK |
144 | { |
145 | acpi_status status; | |
e8c331e9 | 146 | unsigned long long removable; |
2fd6f9c3 | 147 | if (!acpi_has_method(handle, "_ADR")) |
e8c331e9 | 148 | return 0; |
2fd6f9c3 | 149 | if (acpi_has_method(handle, "_EJ0")) |
e8c331e9 KK |
150 | return 1; |
151 | status = acpi_evaluate_integer(handle, "_RMV", NULL, &removable); | |
152 | if (ACPI_SUCCESS(status) && removable) | |
153 | return 1; | |
154 | return 0; | |
155 | } | |
156 | ||
157 | /** | |
158 | * acpi_pcihp_check_ejectable - check if handle is ejectable ACPI PCI slot | |
159 | * @pbus: the PCI bus of the PCI slot corresponding to 'handle' | |
160 | * @handle: ACPI handle to check | |
161 | * | |
162 | * Return 1 if handle is ejectable PCI slot, 0 otherwise. | |
163 | */ | |
164 | int acpi_pci_check_ejectable(struct pci_bus *pbus, acpi_handle handle) | |
165 | { | |
166 | acpi_handle bridge_handle, parent_handle; | |
167 | ||
79e50e72 QL |
168 | bridge_handle = acpi_pci_get_bridge_handle(pbus); |
169 | if (!bridge_handle) | |
e8c331e9 KK |
170 | return 0; |
171 | if ((ACPI_FAILURE(acpi_get_parent(handle, &parent_handle)))) | |
172 | return 0; | |
173 | if (bridge_handle != parent_handle) | |
174 | return 0; | |
efe6d727 | 175 | return pcihp_is_ejectable(handle); |
e8c331e9 KK |
176 | } |
177 | EXPORT_SYMBOL_GPL(acpi_pci_check_ejectable); | |
178 | ||
179 | static acpi_status | |
180 | check_hotplug(acpi_handle handle, u32 lvl, void *context, void **rv) | |
181 | { | |
182 | int *found = (int *)context; | |
efe6d727 | 183 | if (pcihp_is_ejectable(handle)) { |
e8c331e9 KK |
184 | *found = 1; |
185 | return AE_CTRL_TERMINATE; | |
186 | } | |
187 | return AE_OK; | |
188 | } | |
189 | ||
190 | /** | |
191 | * acpi_pci_detect_ejectable - check if the PCI bus has ejectable slots | |
7f538669 | 192 | * @handle - handle of the PCI bus to scan |
e8c331e9 KK |
193 | * |
194 | * Returns 1 if the PCI bus has ACPI based ejectable slots, 0 otherwise. | |
195 | */ | |
7f538669 | 196 | int acpi_pci_detect_ejectable(acpi_handle handle) |
e8c331e9 | 197 | { |
e8c331e9 KK |
198 | int found = 0; |
199 | ||
7f538669 AC |
200 | if (!handle) |
201 | return found; | |
202 | ||
203 | acpi_walk_namespace(ACPI_TYPE_DEVICE, handle, 1, | |
2263576c | 204 | check_hotplug, NULL, (void *)&found, NULL); |
e8c331e9 KK |
205 | return found; |
206 | } | |
207 | EXPORT_SYMBOL_GPL(acpi_pci_detect_ejectable); | |
208 | ||
aad20cab KK |
209 | module_param(debug_acpi, bool, 0644); |
210 | MODULE_PARM_DESC(debug_acpi, "Debugging mode for ACPI enabled or not"); |