mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 b09796d528
			
		
	
	
		b09796d528
		
	
	
	
	
		
			
			Now that fw_devlink=on and fw_devlink.strict=1 by default and fw_devlink supports iommu DT properties, the execution will never get to the point where driver_deferred_probe_check_state() is called before the supplier has probed successfully or before deferred probe timeout has expired. So, delete the call and replace it with -ENODEV. Tested-by: Geert Uytterhoeven <geert+renesas@glider.be> Signed-off-by: Saravana Kannan <saravanak@google.com> Link: https://lore.kernel.org/r/20220601070707.3946847-9-saravanak@google.com Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
		
			
				
	
	
		
			175 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			175 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0-only
 | |
| /*
 | |
|  * OF helpers for IOMMU
 | |
|  *
 | |
|  * Copyright (c) 2012, NVIDIA CORPORATION.  All rights reserved.
 | |
|  */
 | |
| 
 | |
| #include <linux/export.h>
 | |
| #include <linux/iommu.h>
 | |
| #include <linux/limits.h>
 | |
| #include <linux/module.h>
 | |
| #include <linux/msi.h>
 | |
| #include <linux/of.h>
 | |
| #include <linux/of_iommu.h>
 | |
| #include <linux/of_pci.h>
 | |
| #include <linux/pci.h>
 | |
| #include <linux/slab.h>
 | |
| #include <linux/fsl/mc.h>
 | |
| 
 | |
| #define NO_IOMMU	1
 | |
| 
 | |
| static int of_iommu_xlate(struct device *dev,
 | |
| 			  struct of_phandle_args *iommu_spec)
 | |
| {
 | |
| 	const struct iommu_ops *ops;
 | |
| 	struct fwnode_handle *fwnode = &iommu_spec->np->fwnode;
 | |
| 	int ret;
 | |
| 
 | |
| 	ops = iommu_ops_from_fwnode(fwnode);
 | |
| 	if ((ops && !ops->of_xlate) ||
 | |
| 	    !of_device_is_available(iommu_spec->np))
 | |
| 		return NO_IOMMU;
 | |
| 
 | |
| 	ret = iommu_fwspec_init(dev, &iommu_spec->np->fwnode, ops);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 	/*
 | |
| 	 * The otherwise-empty fwspec handily serves to indicate the specific
 | |
| 	 * IOMMU device we're waiting for, which will be useful if we ever get
 | |
| 	 * a proper probe-ordering dependency mechanism in future.
 | |
| 	 */
 | |
| 	if (!ops)
 | |
| 		return -ENODEV;
 | |
| 
 | |
| 	if (!try_module_get(ops->owner))
 | |
| 		return -ENODEV;
 | |
| 
 | |
| 	ret = ops->of_xlate(dev, iommu_spec);
 | |
| 	module_put(ops->owner);
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static int of_iommu_configure_dev_id(struct device_node *master_np,
 | |
| 				     struct device *dev,
 | |
| 				     const u32 *id)
 | |
| {
 | |
| 	struct of_phandle_args iommu_spec = { .args_count = 1 };
 | |
| 	int err;
 | |
| 
 | |
| 	err = of_map_id(master_np, *id, "iommu-map",
 | |
| 			 "iommu-map-mask", &iommu_spec.np,
 | |
| 			 iommu_spec.args);
 | |
| 	if (err)
 | |
| 		return err == -ENODEV ? NO_IOMMU : err;
 | |
| 
 | |
| 	err = of_iommu_xlate(dev, &iommu_spec);
 | |
| 	of_node_put(iommu_spec.np);
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| static int of_iommu_configure_dev(struct device_node *master_np,
 | |
| 				  struct device *dev)
 | |
| {
 | |
| 	struct of_phandle_args iommu_spec;
 | |
| 	int err = NO_IOMMU, idx = 0;
 | |
| 
 | |
| 	while (!of_parse_phandle_with_args(master_np, "iommus",
 | |
| 					   "#iommu-cells",
 | |
| 					   idx, &iommu_spec)) {
 | |
| 		err = of_iommu_xlate(dev, &iommu_spec);
 | |
| 		of_node_put(iommu_spec.np);
 | |
| 		idx++;
 | |
| 		if (err)
 | |
| 			break;
 | |
| 	}
 | |
| 
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| struct of_pci_iommu_alias_info {
 | |
| 	struct device *dev;
 | |
| 	struct device_node *np;
 | |
| };
 | |
| 
 | |
| static int of_pci_iommu_init(struct pci_dev *pdev, u16 alias, void *data)
 | |
| {
 | |
| 	struct of_pci_iommu_alias_info *info = data;
 | |
| 	u32 input_id = alias;
 | |
| 
 | |
| 	return of_iommu_configure_dev_id(info->np, info->dev, &input_id);
 | |
| }
 | |
| 
 | |
| static int of_iommu_configure_device(struct device_node *master_np,
 | |
| 				     struct device *dev, const u32 *id)
 | |
| {
 | |
| 	return (id) ? of_iommu_configure_dev_id(master_np, dev, id) :
 | |
| 		      of_iommu_configure_dev(master_np, dev);
 | |
| }
 | |
| 
 | |
| const struct iommu_ops *of_iommu_configure(struct device *dev,
 | |
| 					   struct device_node *master_np,
 | |
| 					   const u32 *id)
 | |
| {
 | |
| 	const struct iommu_ops *ops = NULL;
 | |
| 	struct iommu_fwspec *fwspec = dev_iommu_fwspec_get(dev);
 | |
| 	int err = NO_IOMMU;
 | |
| 
 | |
| 	if (!master_np)
 | |
| 		return NULL;
 | |
| 
 | |
| 	if (fwspec) {
 | |
| 		if (fwspec->ops)
 | |
| 			return fwspec->ops;
 | |
| 
 | |
| 		/* In the deferred case, start again from scratch */
 | |
| 		iommu_fwspec_free(dev);
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 * We don't currently walk up the tree looking for a parent IOMMU.
 | |
| 	 * See the `Notes:' section of
 | |
| 	 * Documentation/devicetree/bindings/iommu/iommu.txt
 | |
| 	 */
 | |
| 	if (dev_is_pci(dev)) {
 | |
| 		struct of_pci_iommu_alias_info info = {
 | |
| 			.dev = dev,
 | |
| 			.np = master_np,
 | |
| 		};
 | |
| 
 | |
| 		pci_request_acs();
 | |
| 		err = pci_for_each_dma_alias(to_pci_dev(dev),
 | |
| 					     of_pci_iommu_init, &info);
 | |
| 	} else {
 | |
| 		err = of_iommu_configure_device(master_np, dev, id);
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	 * Two success conditions can be represented by non-negative err here:
 | |
| 	 * >0 : there is no IOMMU, or one was unavailable for non-fatal reasons
 | |
| 	 *  0 : we found an IOMMU, and dev->fwspec is initialised appropriately
 | |
| 	 * <0 : any actual error
 | |
| 	 */
 | |
| 	if (!err) {
 | |
| 		/* The fwspec pointer changed, read it again */
 | |
| 		fwspec = dev_iommu_fwspec_get(dev);
 | |
| 		ops    = fwspec->ops;
 | |
| 	}
 | |
| 	/*
 | |
| 	 * If we have reason to believe the IOMMU driver missed the initial
 | |
| 	 * probe for dev, replay it to get things in order.
 | |
| 	 */
 | |
| 	if (!err && dev->bus && !device_iommu_mapped(dev))
 | |
| 		err = iommu_probe_device(dev);
 | |
| 
 | |
| 	/* Ignore all other errors apart from EPROBE_DEFER */
 | |
| 	if (err == -EPROBE_DEFER) {
 | |
| 		ops = ERR_PTR(err);
 | |
| 	} else if (err < 0) {
 | |
| 		dev_dbg(dev, "Adding to IOMMU failed: %d\n", err);
 | |
| 		ops = NULL;
 | |
| 	}
 | |
| 
 | |
| 	return ops;
 | |
| }
 |