| 
									
										
										
										
											2019-06-04 10:11:33 +02:00
										 |  |  | // SPDX-License-Identifier: GPL-2.0-only
 | 
					
						
							| 
									
										
										
										
											2014-09-29 15:29:31 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Code borrowed from powerpc/kernel/pci-common.c | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2003 Anton Blanchard <anton@au.ibm.com>, IBM | 
					
						
							|  |  |  |  * Copyright (C) 2014 ARM Ltd. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-24 14:02:40 +00:00
										 |  |  | #include <linux/acpi.h>
 | 
					
						
							| 
									
										
										
										
											2014-09-29 15:29:31 +01:00
										 |  |  | #include <linux/init.h>
 | 
					
						
							|  |  |  | #include <linux/io.h>
 | 
					
						
							|  |  |  | #include <linux/kernel.h>
 | 
					
						
							|  |  |  | #include <linux/mm.h>
 | 
					
						
							|  |  |  | #include <linux/of_pci.h>
 | 
					
						
							|  |  |  | #include <linux/of_platform.h>
 | 
					
						
							| 
									
										
										
										
											2016-06-10 15:36:26 -05:00
										 |  |  | #include <linux/pci.h>
 | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | #include <linux/pci-acpi.h>
 | 
					
						
							|  |  |  | #include <linux/pci-ecam.h>
 | 
					
						
							| 
									
										
										
										
											2014-09-29 15:29:31 +01:00
										 |  |  | #include <linux/slab.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-28 15:14:12 -05:00
										 |  |  | #ifdef CONFIG_ACPI
 | 
					
						
							| 
									
										
										
										
											2014-09-29 15:29:31 +01:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:17 +02:00
										 |  |  |  * Try to assign the IRQ number when probing a new device | 
					
						
							| 
									
										
										
										
											2014-09-29 15:29:31 +01:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:17 +02:00
										 |  |  | int pcibios_alloc_irq(struct pci_dev *dev) | 
					
						
							| 
									
										
										
										
											2014-09-29 15:29:31 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-06-28 15:14:12 -05:00
										 |  |  | 	if (!acpi_disabled) | 
					
						
							|  |  |  | 		acpi_pci_irq_enable(dev); | 
					
						
							| 
									
										
										
										
											2014-09-29 15:29:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-06-28 15:14:12 -05:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-03-24 14:02:40 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * raw_pci_read/write - Platform-specific PCI config space access. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | int raw_pci_read(unsigned int domain, unsigned int bus, | 
					
						
							|  |  |  | 		  unsigned int devfn, int reg, int len, u32 *val) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:18 +02:00
										 |  |  | 	struct pci_bus *b = pci_find_bus(domain, bus); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!b) | 
					
						
							|  |  |  | 		return PCIBIOS_DEVICE_NOT_FOUND; | 
					
						
							|  |  |  | 	return b->ops->read(b, devfn, reg, len, val); | 
					
						
							| 
									
										
										
										
											2015-03-24 14:02:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int raw_pci_write(unsigned int domain, unsigned int bus, | 
					
						
							|  |  |  | 		unsigned int devfn, int reg, int len, u32 val) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:18 +02:00
										 |  |  | 	struct pci_bus *b = pci_find_bus(domain, bus); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!b) | 
					
						
							|  |  |  | 		return PCIBIOS_DEVICE_NOT_FOUND; | 
					
						
							|  |  |  | 	return b->ops->write(b, devfn, reg, len, val); | 
					
						
							| 
									
										
										
										
											2015-03-24 14:02:40 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-08 15:50:27 -07:00
										 |  |  | #ifdef CONFIG_NUMA
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int pcibus_to_node(struct pci_bus *bus) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return dev_to_node(&bus->dev); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | EXPORT_SYMBOL(pcibus_to_node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-24 14:02:40 +00:00
										 |  |  | #ifdef CONFIG_ACPI
 | 
					
						
							| 
									
										
										
										
											2016-06-10 15:36:26 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | struct acpi_pci_generic_root_info { | 
					
						
							|  |  |  | 	struct acpi_pci_root_info	common; | 
					
						
							|  |  |  | 	struct pci_config_window	*cfg;	/* config space mapping */ | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-10 15:36:26 -05:00
										 |  |  | int acpi_pci_bus_find_domain_nr(struct pci_bus *bus) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 	struct pci_config_window *cfg = bus->sysdata; | 
					
						
							|  |  |  | 	struct acpi_device *adev = to_acpi_device(cfg->parent); | 
					
						
							|  |  |  | 	struct acpi_pci_root *root = acpi_driver_data(adev); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return root->segment; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int pcibios_root_bridge_prepare(struct pci_host_bridge *bridge) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if (!acpi_disabled) { | 
					
						
							|  |  |  | 		struct pci_config_window *cfg = bridge->bus->sysdata; | 
					
						
							|  |  |  | 		struct acpi_device *adev = to_acpi_device(cfg->parent); | 
					
						
							| 
									
										
										
										
											2017-05-24 18:22:19 +01:00
										 |  |  | 		struct device *bus_dev = &bridge->bus->dev; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 		ACPI_COMPANION_SET(&bridge->dev, adev); | 
					
						
							| 
									
										
										
										
											2017-05-24 18:22:19 +01:00
										 |  |  | 		set_dev_node(bus_dev, acpi_get_node(acpi_device_handle(adev))); | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-10 15:36:26 -05:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-02 17:25:54 -06:00
										 |  |  | static int pci_acpi_root_prepare_resources(struct acpi_pci_root_info *ci) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct resource_entry *entry, *tmp; | 
					
						
							|  |  |  | 	int status; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	status = acpi_pci_probe_root_resources(ci); | 
					
						
							|  |  |  | 	resource_list_for_each_entry_safe(entry, tmp, &ci->resources) { | 
					
						
							|  |  |  | 		if (!(entry->res->flags & IORESOURCE_WINDOW)) | 
					
						
							|  |  |  | 			resource_list_destroy_entry(entry); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return status; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Lookup the bus range for the domain in MCFG, and set up config space | 
					
						
							|  |  |  |  * mapping. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static struct pci_config_window * | 
					
						
							|  |  |  | pci_acpi_setup_ecam_mapping(struct acpi_pci_root *root) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-12-01 11:33:57 -06:00
										 |  |  | 	struct device *dev = &root->device->dev; | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 	struct resource *bus_res = &root->secondary; | 
					
						
							|  |  |  | 	u16 seg = root->segment; | 
					
						
							| 
									
										
										
										
											2020-04-09 17:49:21 -06:00
										 |  |  | 	const struct pci_ecam_ops *ecam_ops; | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 	struct resource cfgres; | 
					
						
							| 
									
										
										
										
											2016-11-30 14:48:33 -06:00
										 |  |  | 	struct acpi_device *adev; | 
					
						
							|  |  |  | 	struct pci_config_window *cfg; | 
					
						
							| 
									
										
										
										
											2016-09-09 21:24:03 +02:00
										 |  |  | 	int ret; | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-09 21:24:03 +02:00
										 |  |  | 	ret = pci_mcfg_lookup(root, &cfgres, &ecam_ops); | 
					
						
							|  |  |  | 	if (ret) { | 
					
						
							| 
									
										
										
										
											2016-12-01 11:33:57 -06:00
										 |  |  | 		dev_err(dev, "%04x:%pR ECAM region not found\n", seg, bus_res); | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-30 14:48:33 -06:00
										 |  |  | 	adev = acpi_resource_consumer(&cfgres); | 
					
						
							|  |  |  | 	if (adev) | 
					
						
							|  |  |  | 		dev_info(dev, "ECAM area %pR reserved by %s\n", &cfgres, | 
					
						
							|  |  |  | 			 dev_name(&adev->dev)); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		dev_warn(dev, FW_BUG "ECAM area %pR not reserved in ACPI namespace\n", | 
					
						
							|  |  |  | 			 &cfgres); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-09 21:24:03 +02:00
										 |  |  | 	cfg = pci_ecam_create(dev, &cfgres, bus_res, ecam_ops); | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 	if (IS_ERR(cfg)) { | 
					
						
							| 
									
										
										
										
											2016-12-01 11:33:57 -06:00
										 |  |  | 		dev_err(dev, "%04x:%pR error %ld mapping ECAM\n", seg, bus_res, | 
					
						
							|  |  |  | 			PTR_ERR(cfg)); | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return cfg; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* release_info: free resources allocated by init_info */ | 
					
						
							|  |  |  | static void pci_acpi_generic_release_info(struct acpi_pci_root_info *ci) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct acpi_pci_generic_root_info *ri; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ri = container_of(ci, struct acpi_pci_generic_root_info, common); | 
					
						
							|  |  |  | 	pci_ecam_free(ri->cfg); | 
					
						
							| 
									
										
										
										
											2016-11-24 12:05:23 +01:00
										 |  |  | 	kfree(ci->ops); | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 	kfree(ri); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Interface called from ACPI code to setup PCI host controller */ | 
					
						
							| 
									
										
										
										
											2015-03-24 14:02:40 +00:00
										 |  |  | struct pci_bus *pci_acpi_scan_root(struct acpi_pci_root *root) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 	struct acpi_pci_generic_root_info *ri; | 
					
						
							|  |  |  | 	struct pci_bus *bus, *child; | 
					
						
							| 
									
										
										
										
											2016-11-24 12:05:23 +01:00
										 |  |  | 	struct acpi_pci_root_ops *root_ops; | 
					
						
							| 
									
										
										
										
											2019-06-15 10:23:59 +10:00
										 |  |  | 	struct pci_host_bridge *host; | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 16:05:12 +01:00
										 |  |  | 	ri = kzalloc(sizeof(*ri), GFP_KERNEL); | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 	if (!ri) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-28 16:05:12 +01:00
										 |  |  | 	root_ops = kzalloc(sizeof(*root_ops), GFP_KERNEL); | 
					
						
							| 
									
										
										
										
											2017-05-22 16:48:28 +01:00
										 |  |  | 	if (!root_ops) { | 
					
						
							|  |  |  | 		kfree(ri); | 
					
						
							| 
									
										
										
										
											2016-11-24 12:05:23 +01:00
										 |  |  | 		return NULL; | 
					
						
							| 
									
										
										
										
											2017-05-22 16:48:28 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-11-24 12:05:23 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 	ri->cfg = pci_acpi_setup_ecam_mapping(root); | 
					
						
							|  |  |  | 	if (!ri->cfg) { | 
					
						
							|  |  |  | 		kfree(ri); | 
					
						
							| 
									
										
										
										
											2016-11-24 12:05:23 +01:00
										 |  |  | 		kfree(root_ops); | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 		return NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-24 12:05:23 +01:00
										 |  |  | 	root_ops->release_info = pci_acpi_generic_release_info; | 
					
						
							| 
									
										
										
										
											2016-12-02 17:25:54 -06:00
										 |  |  | 	root_ops->prepare_resources = pci_acpi_root_prepare_resources; | 
					
						
							| 
									
										
										
										
											2020-04-09 17:49:21 -06:00
										 |  |  | 	root_ops->pci_ops = (struct pci_ops *)&ri->cfg->ops->pci_ops; | 
					
						
							| 
									
										
										
										
											2016-11-24 12:05:23 +01:00
										 |  |  | 	bus = acpi_pci_root_create(root, root_ops, &ri->common, ri->cfg); | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 	if (!bus) | 
					
						
							|  |  |  | 		return NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-06-15 10:23:59 +10:00
										 |  |  | 	/* If we must preserve the resource configuration, claim now */ | 
					
						
							|  |  |  | 	host = pci_find_host_bridge(bus); | 
					
						
							|  |  |  | 	if (host->preserve_config) | 
					
						
							|  |  |  | 		pci_bus_claim_resources(bus); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * Assign whatever was left unassigned. If we didn't claim above, | 
					
						
							|  |  |  | 	 * this will reassign everything. | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2019-06-15 10:23:56 +10:00
										 |  |  | 	pci_assign_unassigned_root_bus_resources(bus); | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	list_for_each_entry(child, &bus->children, node) | 
					
						
							|  |  |  | 		pcie_bus_configure_settings(child); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return bus; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void pcibios_add_bus(struct pci_bus *bus) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	acpi_pci_add_bus(bus); | 
					
						
							| 
									
										
										
										
											2015-03-24 14:02:40 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-06-10 21:55:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | void pcibios_remove_bus(struct pci_bus *bus) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	acpi_pci_remove_bus(bus); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-24 14:02:40 +00:00
										 |  |  | #endif
 |