0001
0002
0003
0004
0005
0006
0007
0008 #include <acpi/acpi.h>
0009 #include "accommon.h"
0010
0011 #define _COMPONENT ACPI_NAMESPACE
0012 ACPI_MODULE_NAME("hwpci")
0013
0014
0015 #define PCI_CFG_HEADER_TYPE_REG 0x0E
0016 #define PCI_CFG_PRIMARY_BUS_NUMBER_REG 0x18
0017 #define PCI_CFG_SECONDARY_BUS_NUMBER_REG 0x19
0018
0019 #define PCI_HEADER_TYPE_MASK 0x7F
0020 #define PCI_TYPE_BRIDGE 0x01
0021 #define PCI_TYPE_CARDBUS_BRIDGE 0x02
0022 typedef struct acpi_pci_device {
0023 acpi_handle device;
0024 struct acpi_pci_device *next;
0025
0026 } acpi_pci_device;
0027
0028
0029
0030 static acpi_status
0031 acpi_hw_build_pci_list(acpi_handle root_pci_device,
0032 acpi_handle pci_region,
0033 struct acpi_pci_device **return_list_head);
0034
0035 static acpi_status
0036 acpi_hw_process_pci_list(struct acpi_pci_id *pci_id,
0037 struct acpi_pci_device *list_head);
0038
0039 static void acpi_hw_delete_pci_list(struct acpi_pci_device *list_head);
0040
0041 static acpi_status
0042 acpi_hw_get_pci_device_info(struct acpi_pci_id *pci_id,
0043 acpi_handle pci_device,
0044 u16 *bus_number, u8 *is_bridge);
0045
0046
0047
0048
0049
0050
0051
0052
0053
0054
0055
0056
0057
0058
0059
0060
0061
0062
0063
0064
0065
0066
0067
0068
0069
0070
0071
0072
0073
0074
0075
0076
0077
0078
0079
0080
0081
0082
0083
0084
0085 acpi_status
0086 acpi_hw_derive_pci_id(struct acpi_pci_id *pci_id,
0087 acpi_handle root_pci_device, acpi_handle pci_region)
0088 {
0089 acpi_status status;
0090 struct acpi_pci_device *list_head;
0091
0092 ACPI_FUNCTION_TRACE(hw_derive_pci_id);
0093
0094 if (!pci_id) {
0095 return_ACPI_STATUS(AE_BAD_PARAMETER);
0096 }
0097
0098
0099
0100 status =
0101 acpi_hw_build_pci_list(root_pci_device, pci_region, &list_head);
0102 if (ACPI_SUCCESS(status)) {
0103
0104
0105
0106 status = acpi_hw_process_pci_list(pci_id, list_head);
0107
0108
0109
0110 acpi_hw_delete_pci_list(list_head);
0111 }
0112
0113 return_ACPI_STATUS(status);
0114 }
0115
0116
0117
0118
0119
0120
0121
0122
0123
0124
0125
0126
0127
0128
0129
0130
0131
0132
0133
0134
0135 static acpi_status
0136 acpi_hw_build_pci_list(acpi_handle root_pci_device,
0137 acpi_handle pci_region,
0138 struct acpi_pci_device **return_list_head)
0139 {
0140 acpi_handle current_device;
0141 acpi_handle parent_device;
0142 acpi_status status;
0143 struct acpi_pci_device *list_element;
0144
0145
0146
0147
0148
0149
0150 *return_list_head = NULL;
0151 current_device = pci_region;
0152 while (1) {
0153 status = acpi_get_parent(current_device, &parent_device);
0154 if (ACPI_FAILURE(status)) {
0155
0156
0157
0158 acpi_hw_delete_pci_list(*return_list_head);
0159 return (status);
0160 }
0161
0162
0163
0164 if (parent_device == root_pci_device) {
0165 return (AE_OK);
0166 }
0167
0168 list_element = ACPI_ALLOCATE(sizeof(struct acpi_pci_device));
0169 if (!list_element) {
0170
0171
0172
0173 acpi_hw_delete_pci_list(*return_list_head);
0174 return (AE_NO_MEMORY);
0175 }
0176
0177
0178
0179 list_element->next = *return_list_head;
0180 list_element->device = parent_device;
0181 *return_list_head = list_element;
0182
0183 current_device = parent_device;
0184 }
0185 }
0186
0187
0188
0189
0190
0191
0192
0193
0194
0195
0196
0197
0198
0199
0200
0201
0202
0203
0204 static acpi_status
0205 acpi_hw_process_pci_list(struct acpi_pci_id *pci_id,
0206 struct acpi_pci_device *list_head)
0207 {
0208 acpi_status status = AE_OK;
0209 struct acpi_pci_device *info;
0210 u16 bus_number;
0211 u8 is_bridge = TRUE;
0212
0213 ACPI_FUNCTION_NAME(hw_process_pci_list);
0214
0215 ACPI_DEBUG_PRINT((ACPI_DB_OPREGION,
0216 "Input PciId: Seg %4.4X Bus %4.4X Dev %4.4X Func %4.4X\n",
0217 pci_id->segment, pci_id->bus, pci_id->device,
0218 pci_id->function));
0219
0220 bus_number = pci_id->bus;
0221
0222
0223
0224
0225
0226
0227
0228
0229 info = list_head;
0230 while (info) {
0231 status = acpi_hw_get_pci_device_info(pci_id, info->device,
0232 &bus_number, &is_bridge);
0233 if (ACPI_FAILURE(status)) {
0234 return (status);
0235 }
0236
0237 info = info->next;
0238 }
0239
0240 ACPI_DEBUG_PRINT((ACPI_DB_OPREGION,
0241 "Output PciId: Seg %4.4X Bus %4.4X Dev %4.4X Func %4.4X "
0242 "Status %X BusNumber %X IsBridge %X\n",
0243 pci_id->segment, pci_id->bus, pci_id->device,
0244 pci_id->function, status, bus_number, is_bridge));
0245
0246 return (AE_OK);
0247 }
0248
0249
0250
0251
0252
0253
0254
0255
0256
0257
0258
0259
0260
0261
0262 static void acpi_hw_delete_pci_list(struct acpi_pci_device *list_head)
0263 {
0264 struct acpi_pci_device *next;
0265 struct acpi_pci_device *previous;
0266
0267 next = list_head;
0268 while (next) {
0269 previous = next;
0270 next = previous->next;
0271 ACPI_FREE(previous);
0272 }
0273 }
0274
0275
0276
0277
0278
0279
0280
0281
0282
0283
0284
0285
0286
0287
0288
0289
0290
0291
0292
0293
0294
0295 static acpi_status
0296 acpi_hw_get_pci_device_info(struct acpi_pci_id *pci_id,
0297 acpi_handle pci_device,
0298 u16 *bus_number, u8 *is_bridge)
0299 {
0300 acpi_status status;
0301 acpi_object_type object_type;
0302 u64 return_value;
0303 u64 pci_value;
0304
0305
0306
0307 status = acpi_get_type(pci_device, &object_type);
0308 if (ACPI_FAILURE(status)) {
0309 return (status);
0310 }
0311
0312 if (object_type != ACPI_TYPE_DEVICE) {
0313 return (AE_OK);
0314 }
0315
0316
0317
0318 status = acpi_ut_evaluate_numeric_object(METHOD_NAME__ADR,
0319 pci_device, &return_value);
0320 if (ACPI_FAILURE(status)) {
0321 return (AE_OK);
0322 }
0323
0324
0325
0326
0327
0328 pci_id->device = ACPI_HIWORD(ACPI_LODWORD(return_value));
0329 pci_id->function = ACPI_LOWORD(ACPI_LODWORD(return_value));
0330
0331
0332
0333
0334
0335 if (*is_bridge) {
0336 pci_id->bus = *bus_number;
0337 }
0338
0339
0340
0341
0342
0343
0344 *is_bridge = FALSE;
0345 status = acpi_os_read_pci_configuration(pci_id,
0346 PCI_CFG_HEADER_TYPE_REG,
0347 &pci_value, 8);
0348 if (ACPI_FAILURE(status)) {
0349 return (status);
0350 }
0351
0352
0353
0354 pci_value &= PCI_HEADER_TYPE_MASK;
0355
0356 if ((pci_value != PCI_TYPE_BRIDGE) &&
0357 (pci_value != PCI_TYPE_CARDBUS_BRIDGE)) {
0358 return (AE_OK);
0359 }
0360
0361
0362
0363 status = acpi_os_read_pci_configuration(pci_id,
0364 PCI_CFG_PRIMARY_BUS_NUMBER_REG,
0365 &pci_value, 8);
0366 if (ACPI_FAILURE(status)) {
0367 return (status);
0368 }
0369
0370 *is_bridge = TRUE;
0371 pci_id->bus = (u16)pci_value;
0372
0373
0374
0375 status = acpi_os_read_pci_configuration(pci_id,
0376 PCI_CFG_SECONDARY_BUS_NUMBER_REG,
0377 &pci_value, 8);
0378 if (ACPI_FAILURE(status)) {
0379 return (status);
0380 }
0381
0382 *bus_number = (u16)pci_value;
0383 return (AE_OK);
0384 }