diff options
Diffstat (limited to 'arch/um/drivers/virt-pci.c')
| -rw-r--r-- | arch/um/drivers/virt-pci.c | 139 | 
1 files changed, 132 insertions, 7 deletions
diff --git a/arch/um/drivers/virt-pci.c b/arch/um/drivers/virt-pci.c index 3ac220dafec4..7699ca5f35d4 100644 --- a/arch/um/drivers/virt-pci.c +++ b/arch/um/drivers/virt-pci.c @@ -8,6 +8,7 @@  #include <linux/virtio.h>  #include <linux/virtio_config.h>  #include <linux/logic_iomem.h> +#include <linux/of_platform.h>  #include <linux/irqdomain.h>  #include <linux/virtio_pcidev.h>  #include <linux/virtio-uml.h> @@ -39,6 +40,8 @@ struct um_pci_device {  	unsigned long status;  	int irq; + +	bool platform;  };  struct um_pci_device_reg { @@ -48,13 +51,15 @@ struct um_pci_device_reg {  static struct pci_host_bridge *bridge;  static DEFINE_MUTEX(um_pci_mtx); +static struct um_pci_device *um_pci_platform_device;  static struct um_pci_device_reg um_pci_devices[MAX_DEVICES];  static struct fwnode_handle *um_pci_fwnode;  static struct irq_domain *um_pci_inner_domain;  static struct irq_domain *um_pci_msi_domain;  static unsigned long um_pci_msi_used[BITS_TO_LONGS(MAX_MSI_VECTORS)]; -#define UM_VIRT_PCI_MAXDELAY 40000 +static unsigned int um_pci_max_delay_us = 40000; +module_param_named(max_delay_us, um_pci_max_delay_us, uint, 0644);  struct um_pci_message_buffer {  	struct virtio_pcidev_msg hdr; @@ -132,8 +137,11 @@ static int um_pci_send_cmd(struct um_pci_device *dev,  				out ? 1 : 0,  				posted ? cmd : HANDLE_NO_FREE(cmd),  				GFP_ATOMIC); -	if (ret) +	if (ret) { +		if (posted) +			kfree(cmd);  		goto out; +	}  	if (posted) {  		virtqueue_kick(dev->cmd_vq); @@ -155,7 +163,7 @@ static int um_pci_send_cmd(struct um_pci_device *dev,  			kfree(completed);  		if (WARN_ONCE(virtqueue_is_broken(dev->cmd_vq) || -			      ++delay_count > UM_VIRT_PCI_MAXDELAY, +			      ++delay_count > um_pci_max_delay_us,  			      "um virt-pci delay: %d", delay_count)) {  			ret = -EIO;  			break; @@ -480,6 +488,9 @@ static void um_pci_handle_irq_message(struct virtqueue *vq,  	struct virtio_device *vdev = vq->vdev;  	struct um_pci_device *dev = vdev->priv; +	if (!dev->irq) +		return; +  	/* we should properly chain interrupts, but on ARCH=um we don't care */  	switch (msg->op) { @@ -533,6 +544,25 @@ static void um_pci_irq_vq_cb(struct virtqueue *vq)  	}  } +/* Copied from arch/x86/kernel/devicetree.c */ +struct device_node *pcibios_get_phb_of_node(struct pci_bus *bus) +{ +	struct device_node *np; + +	for_each_node_by_type(np, "pci") { +		const void *prop; +		unsigned int bus_min; + +		prop = of_get_property(np, "bus-range", NULL); +		if (!prop) +			continue; +		bus_min = be32_to_cpup(prop); +		if (bus->number == bus_min) +			return np; +	} +	return NULL; +} +  static int um_pci_init_vqs(struct um_pci_device *dev)  {  	struct virtqueue *vqs[2]; @@ -561,6 +591,55 @@ static int um_pci_init_vqs(struct um_pci_device *dev)  	return 0;  } +static void __um_pci_virtio_platform_remove(struct virtio_device *vdev, +					    struct um_pci_device *dev) +{ +	virtio_reset_device(vdev); +	vdev->config->del_vqs(vdev); + +	mutex_lock(&um_pci_mtx); +	um_pci_platform_device = NULL; +	mutex_unlock(&um_pci_mtx); + +	kfree(dev); +} + +static int um_pci_virtio_platform_probe(struct virtio_device *vdev, +					struct um_pci_device *dev) +{ +	int ret; + +	dev->platform = true; + +	mutex_lock(&um_pci_mtx); + +	if (um_pci_platform_device) { +		mutex_unlock(&um_pci_mtx); +		ret = -EBUSY; +		goto out_free; +	} + +	ret = um_pci_init_vqs(dev); +	if (ret) { +		mutex_unlock(&um_pci_mtx); +		goto out_free; +	} + +	um_pci_platform_device = dev; + +	mutex_unlock(&um_pci_mtx); + +	ret = of_platform_default_populate(vdev->dev.of_node, NULL, &vdev->dev); +	if (ret) +		__um_pci_virtio_platform_remove(vdev, dev); + +	return ret; + +out_free: +	kfree(dev); +	return ret; +} +  static int um_pci_virtio_probe(struct virtio_device *vdev)  {  	struct um_pci_device *dev; @@ -574,6 +653,9 @@ static int um_pci_virtio_probe(struct virtio_device *vdev)  	dev->vdev = vdev;  	vdev->priv = dev; +	if (of_device_is_compatible(vdev->dev.of_node, "simple-bus")) +		return um_pci_virtio_platform_probe(vdev, dev); +  	mutex_lock(&um_pci_mtx);  	for (i = 0; i < MAX_DEVICES; i++) {  		if (um_pci_devices[i].dev) @@ -623,9 +705,11 @@ static void um_pci_virtio_remove(struct virtio_device *vdev)  	struct um_pci_device *dev = vdev->priv;  	int i; -        /* Stop all virtqueues */ -        virtio_reset_device(vdev); -        vdev->config->del_vqs(vdev); +	if (dev->platform) { +		of_platform_depopulate(&vdev->dev); +		__um_pci_virtio_platform_remove(vdev, dev); +		return; +	}  	device_set_wakeup_enable(&vdev->dev, false); @@ -633,12 +717,27 @@ static void um_pci_virtio_remove(struct virtio_device *vdev)  	for (i = 0; i < MAX_DEVICES; i++) {  		if (um_pci_devices[i].dev != dev)  			continue; +  		um_pci_devices[i].dev = NULL;  		irq_free_desc(dev->irq); + +		break;  	}  	mutex_unlock(&um_pci_mtx); -	um_pci_rescan(); +	if (i < MAX_DEVICES) { +		struct pci_dev *pci_dev; + +		pci_dev = pci_get_slot(bridge->bus, i); +		if (pci_dev) +			pci_stop_and_remove_bus_device_locked(pci_dev); +	} + +	/* Stop all virtqueues */ +	virtio_reset_device(vdev); +	dev->cmd_vq = NULL; +	dev->irq_vq = NULL; +	vdev->config->del_vqs(vdev);  	kfree(dev);  } @@ -860,6 +959,30 @@ void *pci_root_bus_fwnode(struct pci_bus *bus)  	return um_pci_fwnode;  } +static long um_pci_map_platform(unsigned long offset, size_t size, +				const struct logic_iomem_ops **ops, +				void **priv) +{ +	if (!um_pci_platform_device) +		return -ENOENT; + +	*ops = &um_pci_device_bar_ops; +	*priv = &um_pci_platform_device->resptr[0]; + +	return 0; +} + +static const struct logic_iomem_region_ops um_pci_platform_ops = { +	.map = um_pci_map_platform, +}; + +static struct resource virt_platform_resource = { +	.name = "platform", +	.start = 0x10000000, +	.end = 0x1fffffff, +	.flags = IORESOURCE_MEM, +}; +  static int __init um_pci_init(void)  {  	int err, i; @@ -868,6 +991,8 @@ static int __init um_pci_init(void)  				       &um_pci_cfgspace_ops));  	WARN_ON(logic_iomem_add_region(&virt_iomem_resource,  				       &um_pci_iomem_ops)); +	WARN_ON(logic_iomem_add_region(&virt_platform_resource, +				       &um_pci_platform_ops));  	if (WARN(CONFIG_UML_PCI_OVER_VIRTIO_DEVICE_ID < 0,  		 "No virtio device ID configured for PCI - no PCI support\n"))  |