mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	drm: convert drivers to use of_graph_get_remote_node
Convert drivers to use the new of_graph_get_remote_node() helper instead of parsing the endpoint node and then getting the remote device node. Now drivers can just specify the device node and which port/endpoint and get back the connected remote device node. The details of the graph binding are nicely abstracted into the core OF graph code. This changes some error messages to debug messages (in the graph core). Graph connections are often "no connects" depending on the particular board, so we want to avoid spurious messages. Plus the kernel is not a DT validator. Signed-off-by: Rob Herring <robh@kernel.org> Acked-by: Neil Armstrong <narmstrong@baylibre.com> Tested-by: Liviu Dudau <liviu.dudau@arm.com> Tested-by: Eric Anholt <eric@anholt.net> Tested-by: Jyri Sarha <jsarha@ti.com> Tested by: Archit Taneja <architt@codeaurora.org> Signed-off-by: Sean Paul <seanpaul@chromium.org>
This commit is contained in:
		
							parent
							
								
									1f2db3034c
								
							
						
					
					
						commit
						86418f90a4
					
				| @ -392,30 +392,14 @@ static int compare_dev(struct device *dev, void *data) | |||||||
| 
 | 
 | ||||||
| static int hdlcd_probe(struct platform_device *pdev) | static int hdlcd_probe(struct platform_device *pdev) | ||||||
| { | { | ||||||
| 	struct device_node *port, *ep; | 	struct device_node *port; | ||||||
| 	struct component_match *match = NULL; | 	struct component_match *match = NULL; | ||||||
| 
 | 
 | ||||||
| 	if (!pdev->dev.of_node) |  | ||||||
| 		return -ENODEV; |  | ||||||
| 
 |  | ||||||
| 	/* there is only one output port inside each device, find it */ | 	/* there is only one output port inside each device, find it */ | ||||||
| 	ep = of_graph_get_next_endpoint(pdev->dev.of_node, NULL); | 	port = of_graph_get_remote_node(pdev->dev.of_node, 0, 0); | ||||||
| 	if (!ep) | 	if (!port) | ||||||
| 		return -ENODEV; | 		return -ENODEV; | ||||||
| 
 | 
 | ||||||
| 	if (!of_device_is_available(ep)) { |  | ||||||
| 		of_node_put(ep); |  | ||||||
| 		return -ENODEV; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	/* add the remote encoder port as component */ |  | ||||||
| 	port = of_graph_get_remote_port_parent(ep); |  | ||||||
| 	of_node_put(ep); |  | ||||||
| 	if (!port || !of_device_is_available(port)) { |  | ||||||
| 		of_node_put(port); |  | ||||||
| 		return -EAGAIN; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	drm_of_component_match_add(&pdev->dev, &match, compare_dev, port); | 	drm_of_component_match_add(&pdev->dev, &match, compare_dev, port); | ||||||
| 	of_node_put(port); | 	of_node_put(port); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -283,7 +283,6 @@ static int malidp_bind(struct device *dev) | |||||||
| { | { | ||||||
| 	struct resource *res; | 	struct resource *res; | ||||||
| 	struct drm_device *drm; | 	struct drm_device *drm; | ||||||
| 	struct device_node *ep; |  | ||||||
| 	struct malidp_drm *malidp; | 	struct malidp_drm *malidp; | ||||||
| 	struct malidp_hw_device *hwdev; | 	struct malidp_hw_device *hwdev; | ||||||
| 	struct platform_device *pdev = to_platform_device(dev); | 	struct platform_device *pdev = to_platform_device(dev); | ||||||
| @ -398,12 +397,7 @@ static int malidp_bind(struct device *dev) | |||||||
| 		goto init_fail; | 		goto init_fail; | ||||||
| 
 | 
 | ||||||
| 	/* Set the CRTC's port so that the encoder component can find it */ | 	/* Set the CRTC's port so that the encoder component can find it */ | ||||||
| 	ep = of_graph_get_next_endpoint(dev->of_node, NULL); | 	malidp->crtc.port = of_graph_get_port_by_id(dev->of_node, 0); | ||||||
| 	if (!ep) { |  | ||||||
| 		ret = -EINVAL; |  | ||||||
| 		goto port_fail; |  | ||||||
| 	} |  | ||||||
| 	malidp->crtc.port = of_get_next_parent(ep); |  | ||||||
| 
 | 
 | ||||||
| 	ret = component_bind_all(dev, drm); | 	ret = component_bind_all(dev, drm); | ||||||
| 	if (ret) { | 	if (ret) { | ||||||
| @ -458,7 +452,6 @@ irq_init_fail: | |||||||
| bind_fail: | bind_fail: | ||||||
| 	of_node_put(malidp->crtc.port); | 	of_node_put(malidp->crtc.port); | ||||||
| 	malidp->crtc.port = NULL; | 	malidp->crtc.port = NULL; | ||||||
| port_fail: |  | ||||||
| 	malidp_fini(drm); | 	malidp_fini(drm); | ||||||
| init_fail: | init_fail: | ||||||
| 	drm->dev_private = NULL; | 	drm->dev_private = NULL; | ||||||
| @ -516,30 +509,17 @@ static int malidp_compare_dev(struct device *dev, void *data) | |||||||
| 
 | 
 | ||||||
| static int malidp_platform_probe(struct platform_device *pdev) | static int malidp_platform_probe(struct platform_device *pdev) | ||||||
| { | { | ||||||
| 	struct device_node *port, *ep; | 	struct device_node *port; | ||||||
| 	struct component_match *match = NULL; | 	struct component_match *match = NULL; | ||||||
| 
 | 
 | ||||||
| 	if (!pdev->dev.of_node) | 	if (!pdev->dev.of_node) | ||||||
| 		return -ENODEV; | 		return -ENODEV; | ||||||
| 
 | 
 | ||||||
| 	/* there is only one output port inside each device, find it */ | 	/* there is only one output port inside each device, find it */ | ||||||
| 	ep = of_graph_get_next_endpoint(pdev->dev.of_node, NULL); | 	port = of_graph_get_remote_node(pdev->dev.of_node, 0, 0); | ||||||
| 	if (!ep) | 	if (!port) | ||||||
| 		return -ENODEV; | 		return -ENODEV; | ||||||
| 
 | 
 | ||||||
| 	if (!of_device_is_available(ep)) { |  | ||||||
| 		of_node_put(ep); |  | ||||||
| 		return -ENODEV; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	/* add the remote encoder port as component */ |  | ||||||
| 	port = of_graph_get_remote_port_parent(ep); |  | ||||||
| 	of_node_put(ep); |  | ||||||
| 	if (!port || !of_device_is_available(port)) { |  | ||||||
| 		of_node_put(port); |  | ||||||
| 		return -EAGAIN; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	drm_of_component_match_add(&pdev->dev, &match, malidp_compare_dev, | 	drm_of_component_match_add(&pdev->dev, &match, malidp_compare_dev, | ||||||
| 				   port); | 				   port); | ||||||
| 	of_node_put(port); | 	of_node_put(port); | ||||||
|  | |||||||
| @ -232,7 +232,6 @@ void adv7533_detach_dsi(struct adv7511 *adv) | |||||||
| int adv7533_parse_dt(struct device_node *np, struct adv7511 *adv) | int adv7533_parse_dt(struct device_node *np, struct adv7511 *adv) | ||||||
| { | { | ||||||
| 	u32 num_lanes; | 	u32 num_lanes; | ||||||
| 	struct device_node *endpoint; |  | ||||||
| 
 | 
 | ||||||
| 	of_property_read_u32(np, "adi,dsi-lanes", &num_lanes); | 	of_property_read_u32(np, "adi,dsi-lanes", &num_lanes); | ||||||
| 
 | 
 | ||||||
| @ -241,17 +240,10 @@ int adv7533_parse_dt(struct device_node *np, struct adv7511 *adv) | |||||||
| 
 | 
 | ||||||
| 	adv->num_dsi_lanes = num_lanes; | 	adv->num_dsi_lanes = num_lanes; | ||||||
| 
 | 
 | ||||||
| 	endpoint = of_graph_get_next_endpoint(np, NULL); | 	adv->host_node = of_graph_get_remote_node(np, 0, 0); | ||||||
| 	if (!endpoint) | 	if (!adv->host_node) | ||||||
| 		return -ENODEV; | 		return -ENODEV; | ||||||
| 
 | 
 | ||||||
| 	adv->host_node = of_graph_get_remote_port_parent(endpoint); |  | ||||||
| 	if (!adv->host_node) { |  | ||||||
| 		of_node_put(endpoint); |  | ||||||
| 		return -ENODEV; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	of_node_put(endpoint); |  | ||||||
| 	of_node_put(adv->host_node); | 	of_node_put(adv->host_node); | ||||||
| 
 | 
 | ||||||
| 	adv->use_timing_gen = !of_property_read_bool(np, | 	adv->use_timing_gen = !of_property_read_bool(np, | ||||||
|  | |||||||
| @ -154,21 +154,12 @@ static const struct drm_bridge_funcs dumb_vga_bridge_funcs = { | |||||||
| 
 | 
 | ||||||
| static struct i2c_adapter *dumb_vga_retrieve_ddc(struct device *dev) | static struct i2c_adapter *dumb_vga_retrieve_ddc(struct device *dev) | ||||||
| { | { | ||||||
| 	struct device_node *end_node, *phandle, *remote; | 	struct device_node *phandle, *remote; | ||||||
| 	struct i2c_adapter *ddc; | 	struct i2c_adapter *ddc; | ||||||
| 
 | 
 | ||||||
| 	end_node = of_graph_get_endpoint_by_regs(dev->of_node, 1, -1); | 	remote = of_graph_get_remote_node(dev->of_node, 1, -1); | ||||||
| 	if (!end_node) { | 	if (!remote) | ||||||
| 		dev_err(dev, "Missing connector endpoint\n"); |  | ||||||
| 		return ERR_PTR(-ENODEV); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	remote = of_graph_get_remote_port_parent(end_node); |  | ||||||
| 	of_node_put(end_node); |  | ||||||
| 	if (!remote) { |  | ||||||
| 		dev_err(dev, "Enable to parse remote node\n"); |  | ||||||
| 		return ERR_PTR(-EINVAL); | 		return ERR_PTR(-EINVAL); | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	phandle = of_parse_phandle(remote, "ddc-i2c-bus", 0); | 	phandle = of_parse_phandle(remote, "ddc-i2c-bus", 0); | ||||||
| 	of_node_put(remote); | 	of_node_put(remote); | ||||||
|  | |||||||
| @ -165,18 +165,13 @@ static irqreturn_t tfp410_hpd_irq_thread(int irq, void *arg) | |||||||
| 
 | 
 | ||||||
| static int tfp410_get_connector_properties(struct tfp410 *dvi) | static int tfp410_get_connector_properties(struct tfp410 *dvi) | ||||||
| { | { | ||||||
| 	struct device_node *ep = NULL, *connector_node = NULL; | 	struct device_node *connector_node, *ddc_phandle; | ||||||
| 	struct device_node *ddc_phandle = NULL; |  | ||||||
| 	int ret = 0; | 	int ret = 0; | ||||||
| 
 | 
 | ||||||
| 	/* port@1 is the connector node */ | 	/* port@1 is the connector node */ | ||||||
| 	ep = of_graph_get_endpoint_by_regs(dvi->dev->of_node, 1, -1); | 	connector_node = of_graph_get_remote_node(dvi->dev->of_node, 1, -1); | ||||||
| 	if (!ep) |  | ||||||
| 		goto fail; |  | ||||||
| 
 |  | ||||||
| 	connector_node = of_graph_get_remote_port_parent(ep); |  | ||||||
| 	if (!connector_node) | 	if (!connector_node) | ||||||
| 		goto fail; | 		return -ENODEV; | ||||||
| 
 | 
 | ||||||
| 	dvi->hpd = fwnode_get_named_gpiod(&connector_node->fwnode, | 	dvi->hpd = fwnode_get_named_gpiod(&connector_node->fwnode, | ||||||
| 					"hpd-gpios", 0, GPIOD_IN, "hpd"); | 					"hpd-gpios", 0, GPIOD_IN, "hpd"); | ||||||
| @ -199,10 +194,10 @@ static int tfp410_get_connector_properties(struct tfp410 *dvi) | |||||||
| 	else | 	else | ||||||
| 		ret = -EPROBE_DEFER; | 		ret = -EPROBE_DEFER; | ||||||
| 
 | 
 | ||||||
| fail: |  | ||||||
| 	of_node_put(ep); |  | ||||||
| 	of_node_put(connector_node); |  | ||||||
| 	of_node_put(ddc_phandle); | 	of_node_put(ddc_phandle); | ||||||
|  | 
 | ||||||
|  | fail: | ||||||
|  | 	of_node_put(connector_node); | ||||||
| 	return ret; | 	return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -163,27 +163,13 @@ enum { | |||||||
| 	FIMD_PORT_WRB, | 	FIMD_PORT_WRB, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| static struct device_node *exynos_dpi_of_find_panel_node(struct device *dev) |  | ||||||
| { |  | ||||||
| 	struct device_node *np, *ep; |  | ||||||
| 
 |  | ||||||
| 	ep = of_graph_get_endpoint_by_regs(dev->of_node, FIMD_PORT_RGB, 0); |  | ||||||
| 	if (!ep) |  | ||||||
| 		return NULL; |  | ||||||
| 
 |  | ||||||
| 	np = of_graph_get_remote_port_parent(ep); |  | ||||||
| 	of_node_put(ep); |  | ||||||
| 
 |  | ||||||
| 	return np; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static int exynos_dpi_parse_dt(struct exynos_dpi *ctx) | static int exynos_dpi_parse_dt(struct exynos_dpi *ctx) | ||||||
| { | { | ||||||
| 	struct device *dev = ctx->dev; | 	struct device *dev = ctx->dev; | ||||||
| 	struct device_node *dn = dev->of_node; | 	struct device_node *dn = dev->of_node; | ||||||
| 	struct device_node *np; | 	struct device_node *np; | ||||||
| 
 | 
 | ||||||
| 	ctx->panel_node = exynos_dpi_of_find_panel_node(dev); | 	ctx->panel_node = of_graph_get_remote_node(dn, FIMD_PORT_RGB, 0); | ||||||
| 
 | 
 | ||||||
| 	np = of_get_child_by_name(dn, "display-timings"); | 	np = of_get_child_by_name(dn, "display-timings"); | ||||||
| 	if (np) { | 	if (np) { | ||||||
|  | |||||||
| @ -1659,17 +1659,10 @@ static int exynos_dsi_parse_dt(struct exynos_dsi *dsi) | |||||||
| 
 | 
 | ||||||
| 	of_node_put(ep); | 	of_node_put(ep); | ||||||
| 
 | 
 | ||||||
| 	ep = of_graph_get_next_endpoint(node, NULL); | 	dsi->bridge_node = of_graph_get_remote_node(node, DSI_PORT_OUT, 0); | ||||||
| 	if (!ep) { | 	if (!dsi->bridge_node) | ||||||
| 		ret = -EINVAL; | 		return -EINVAL; | ||||||
| 		goto end; |  | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	dsi->bridge_node = of_graph_get_remote_port_parent(ep); |  | ||||||
| 	if (!dsi->bridge_node) { |  | ||||||
| 		ret = -EINVAL; |  | ||||||
| 		goto end; |  | ||||||
| 	} |  | ||||||
| end: | end: | ||||||
| 	of_node_put(ep); | 	of_node_put(ep); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -229,29 +229,6 @@ static void mic_set_reg_on(struct exynos_mic *mic, bool enable) | |||||||
| 	writel(reg, mic->reg + MIC_OP); | 	writel(reg, mic->reg + MIC_OP); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static struct device_node *get_remote_node(struct device_node *from, int reg) |  | ||||||
| { |  | ||||||
| 	struct device_node *endpoint = NULL, *remote_node = NULL; |  | ||||||
| 
 |  | ||||||
| 	endpoint = of_graph_get_endpoint_by_regs(from, reg, -1); |  | ||||||
| 	if (!endpoint) { |  | ||||||
| 		DRM_ERROR("mic: Failed to find remote port from %s", |  | ||||||
| 				from->full_name); |  | ||||||
| 		goto exit; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	remote_node = of_graph_get_remote_port_parent(endpoint); |  | ||||||
| 	if (!remote_node) { |  | ||||||
| 		DRM_ERROR("mic: Failed to find remote port parent from %s", |  | ||||||
| 							from->full_name); |  | ||||||
| 		goto exit; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| exit: |  | ||||||
| 	of_node_put(endpoint); |  | ||||||
| 	return remote_node; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static int parse_dt(struct exynos_mic *mic) | static int parse_dt(struct exynos_mic *mic) | ||||||
| { | { | ||||||
| 	int ret = 0, i, j; | 	int ret = 0, i, j; | ||||||
| @ -263,7 +240,7 @@ static int parse_dt(struct exynos_mic *mic) | |||||||
| 	 * The first node must be for decon and the second one must be for dsi. | 	 * The first node must be for decon and the second one must be for dsi. | ||||||
| 	 */ | 	 */ | ||||||
| 	for (i = 0, j = 0; i < NUM_ENDPOINTS; i++) { | 	for (i = 0, j = 0; i < NUM_ENDPOINTS; i++) { | ||||||
| 		remote_node = get_remote_node(mic->dev->of_node, i); | 		remote_node = of_graph_get_remote_node(mic->dev->of_node, i, 0); | ||||||
| 		if (!remote_node) { | 		if (!remote_node) { | ||||||
| 			ret = -EPIPE; | 			ret = -EPIPE; | ||||||
| 			goto exit; | 			goto exit; | ||||||
|  | |||||||
| @ -230,34 +230,6 @@ static const struct component_master_ops kirin_drm_ops = { | |||||||
| 	.unbind = kirin_drm_unbind, | 	.unbind = kirin_drm_unbind, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| static struct device_node *kirin_get_remote_node(struct device_node *np) |  | ||||||
| { |  | ||||||
| 	struct device_node *endpoint, *remote; |  | ||||||
| 
 |  | ||||||
| 	/* get the first endpoint, in our case only one remote node
 |  | ||||||
| 	 * is connected to display controller. |  | ||||||
| 	 */ |  | ||||||
| 	endpoint = of_graph_get_next_endpoint(np, NULL); |  | ||||||
| 	if (!endpoint) { |  | ||||||
| 		DRM_ERROR("no valid endpoint node\n"); |  | ||||||
| 		return ERR_PTR(-ENODEV); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	remote = of_graph_get_remote_port_parent(endpoint); |  | ||||||
| 	of_node_put(endpoint); |  | ||||||
| 	if (!remote) { |  | ||||||
| 		DRM_ERROR("no valid remote node\n"); |  | ||||||
| 		return ERR_PTR(-ENODEV); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	if (!of_device_is_available(remote)) { |  | ||||||
| 		DRM_ERROR("not available for remote node\n"); |  | ||||||
| 		return ERR_PTR(-ENODEV); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	return remote; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static int kirin_drm_platform_probe(struct platform_device *pdev) | static int kirin_drm_platform_probe(struct platform_device *pdev) | ||||||
| { | { | ||||||
| 	struct device *dev = &pdev->dev; | 	struct device *dev = &pdev->dev; | ||||||
| @ -271,7 +243,7 @@ static int kirin_drm_platform_probe(struct platform_device *pdev) | |||||||
| 		return -EINVAL; | 		return -EINVAL; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	remote = kirin_get_remote_node(np); | 	remote = of_graph_get_remote_node(np, 0, 0); | ||||||
| 	if (IS_ERR(remote)) | 	if (IS_ERR(remote)) | ||||||
| 		return PTR_ERR(remote); | 		return PTR_ERR(remote); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -661,7 +661,7 @@ static int mtk_dpi_probe(struct platform_device *pdev) | |||||||
| 	struct device *dev = &pdev->dev; | 	struct device *dev = &pdev->dev; | ||||||
| 	struct mtk_dpi *dpi; | 	struct mtk_dpi *dpi; | ||||||
| 	struct resource *mem; | 	struct resource *mem; | ||||||
| 	struct device_node *ep, *bridge_node = NULL; | 	struct device_node *bridge_node; | ||||||
| 	int comp_id; | 	int comp_id; | ||||||
| 	int ret; | 	int ret; | ||||||
| 
 | 
 | ||||||
| @ -706,15 +706,9 @@ static int mtk_dpi_probe(struct platform_device *pdev) | |||||||
| 		return -EINVAL; | 		return -EINVAL; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	ep = of_graph_get_next_endpoint(dev->of_node, NULL); | 	bridge_node = of_graph_get_remote_node(dev->of_node, 0, 0); | ||||||
| 	if (ep) { | 	if (!bridge_node) | ||||||
| 		bridge_node = of_graph_get_remote_port_parent(ep); |  | ||||||
| 		of_node_put(ep); |  | ||||||
| 	} |  | ||||||
| 	if (!bridge_node) { |  | ||||||
| 		dev_err(dev, "Failed to find bridge node\n"); |  | ||||||
| 		return -ENODEV; | 		return -ENODEV; | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	dev_info(dev, "Found bridge node: %s\n", bridge_node->full_name); | 	dev_info(dev, "Found bridge node: %s\n", bridge_node->full_name); | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -1434,7 +1434,7 @@ static int mtk_hdmi_dt_parse_pdata(struct mtk_hdmi *hdmi, | |||||||
| { | { | ||||||
| 	struct device *dev = &pdev->dev; | 	struct device *dev = &pdev->dev; | ||||||
| 	struct device_node *np = dev->of_node; | 	struct device_node *np = dev->of_node; | ||||||
| 	struct device_node *cec_np, *port, *ep, *remote, *i2c_np; | 	struct device_node *cec_np, *remote, *i2c_np; | ||||||
| 	struct platform_device *cec_pdev; | 	struct platform_device *cec_pdev; | ||||||
| 	struct regmap *regmap; | 	struct regmap *regmap; | ||||||
| 	struct resource *mem; | 	struct resource *mem; | ||||||
| @ -1486,29 +1486,9 @@ static int mtk_hdmi_dt_parse_pdata(struct mtk_hdmi *hdmi, | |||||||
| 	if (IS_ERR(hdmi->regs)) | 	if (IS_ERR(hdmi->regs)) | ||||||
| 		return PTR_ERR(hdmi->regs); | 		return PTR_ERR(hdmi->regs); | ||||||
| 
 | 
 | ||||||
| 	port = of_graph_get_port_by_id(np, 1); | 	remote = of_graph_get_remote_node(np, 1, 0); | ||||||
| 	if (!port) { | 	if (!remote) | ||||||
| 		dev_err(dev, "Missing output port node\n"); |  | ||||||
| 		return -EINVAL; | 		return -EINVAL; | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	ep = of_get_child_by_name(port, "endpoint"); |  | ||||||
| 	if (!ep) { |  | ||||||
| 		dev_err(dev, "Missing endpoint node in port %s\n", |  | ||||||
| 			port->full_name); |  | ||||||
| 		of_node_put(port); |  | ||||||
| 		return -EINVAL; |  | ||||||
| 	} |  | ||||||
| 	of_node_put(port); |  | ||||||
| 
 |  | ||||||
| 	remote = of_graph_get_remote_port_parent(ep); |  | ||||||
| 	if (!remote) { |  | ||||||
| 		dev_err(dev, "Missing connector/bridge node for endpoint %s\n", |  | ||||||
| 			ep->full_name); |  | ||||||
| 		of_node_put(ep); |  | ||||||
| 		return -EINVAL; |  | ||||||
| 	} |  | ||||||
| 	of_node_put(ep); |  | ||||||
| 
 | 
 | ||||||
| 	if (!of_device_is_compatible(remote, "hdmi-connector")) { | 	if (!of_device_is_compatible(remote, "hdmi-connector")) { | ||||||
| 		hdmi->next_bridge = of_drm_find_bridge(remote); | 		hdmi->next_bridge = of_drm_find_bridge(remote); | ||||||
|  | |||||||
| @ -224,25 +224,14 @@ static const struct drm_encoder_helper_funcs | |||||||
| 
 | 
 | ||||||
| static bool meson_venc_cvbs_connector_is_available(struct meson_drm *priv) | static bool meson_venc_cvbs_connector_is_available(struct meson_drm *priv) | ||||||
| { | { | ||||||
| 	struct device_node *ep, *remote; | 	struct device_node *remote; | ||||||
| 
 | 
 | ||||||
| 	/* CVBS VDAC output is on the first port, first endpoint */ | 	remote = of_graph_get_remote_node(priv->dev->of_node, 0, 0); | ||||||
| 	ep = of_graph_get_endpoint_by_regs(priv->dev->of_node, 0, 0); | 	if (!remote) | ||||||
| 	if (!ep) |  | ||||||
| 		return false; | 		return false; | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| 	/* If the endpoint node exists, consider it enabled */ |  | ||||||
| 	remote = of_graph_get_remote_port(ep); |  | ||||||
| 	if (remote) { |  | ||||||
| 		of_node_put(ep); |  | ||||||
| 		return true; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	of_node_put(ep); |  | ||||||
| 	of_node_put(remote); | 	of_node_put(remote); | ||||||
| 
 | 	return true; | ||||||
| 	return false; |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| int meson_venc_cvbs_create(struct meson_drm *priv) | int meson_venc_cvbs_create(struct meson_drm *priv) | ||||||
|  | |||||||
| @ -1635,7 +1635,7 @@ static int dsi_host_parse_dt(struct msm_dsi_host *msm_host) | |||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/* Get panel node from the output port's endpoint data */ | 	/* Get panel node from the output port's endpoint data */ | ||||||
| 	device_node = of_graph_get_remote_port_parent(endpoint); | 	device_node = of_graph_get_remote_node(np, 1, 0); | ||||||
| 	if (!device_node) { | 	if (!device_node) { | ||||||
| 		dev_dbg(dev, "%s: no valid device\n", __func__); | 		dev_dbg(dev, "%s: no valid device\n", __func__); | ||||||
| 		goto err; | 		goto err; | ||||||
|  | |||||||
| @ -225,32 +225,6 @@ int mdp4_enable(struct mdp4_kms *mdp4_kms) | |||||||
| 	return 0; | 	return 0; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static struct device_node *mdp4_detect_lcdc_panel(struct drm_device *dev) |  | ||||||
| { |  | ||||||
| 	struct device_node *endpoint, *panel_node; |  | ||||||
| 	struct device_node *np = dev->dev->of_node; |  | ||||||
| 
 |  | ||||||
| 	/*
 |  | ||||||
| 	 * LVDS/LCDC is the first port described in the list of ports in the |  | ||||||
| 	 * MDP4 DT node. |  | ||||||
| 	 */ |  | ||||||
| 	endpoint = of_graph_get_endpoint_by_regs(np, 0, -1); |  | ||||||
| 	if (!endpoint) { |  | ||||||
| 		DBG("no LVDS remote endpoint\n"); |  | ||||||
| 		return NULL; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	panel_node = of_graph_get_remote_port_parent(endpoint); |  | ||||||
| 	if (!panel_node) { |  | ||||||
| 		DBG("no valid panel node in LVDS endpoint\n"); |  | ||||||
| 		of_node_put(endpoint); |  | ||||||
| 		return NULL; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	of_node_put(endpoint); |  | ||||||
| 
 |  | ||||||
| 	return panel_node; |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| static int mdp4_modeset_init_intf(struct mdp4_kms *mdp4_kms, | static int mdp4_modeset_init_intf(struct mdp4_kms *mdp4_kms, | ||||||
| 				  int intf_type) | 				  int intf_type) | ||||||
| @ -269,7 +243,7 @@ static int mdp4_modeset_init_intf(struct mdp4_kms *mdp4_kms, | |||||||
| 		 * bail out early if there is no panel node (no need to | 		 * bail out early if there is no panel node (no need to | ||||||
| 		 * initialize LCDC encoder and LVDS connector) | 		 * initialize LCDC encoder and LVDS connector) | ||||||
| 		 */ | 		 */ | ||||||
| 		panel_node = mdp4_detect_lcdc_panel(dev); | 		panel_node = of_graph_get_remote_node(dev->dev->of_node, 0, 0); | ||||||
| 		if (!panel_node) | 		if (!panel_node) | ||||||
| 			return 0; | 			return 0; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -23,6 +23,7 @@ | |||||||
| #include <linux/workqueue.h> | #include <linux/workqueue.h> | ||||||
| #include <linux/completion.h> | #include <linux/completion.h> | ||||||
| #include <linux/dma-mapping.h> | #include <linux/dma-mapping.h> | ||||||
|  | #include <linux/of_graph.h> | ||||||
| 
 | 
 | ||||||
| #include "tilcdc_drv.h" | #include "tilcdc_drv.h" | ||||||
| #include "tilcdc_regs.h" | #include "tilcdc_regs.h" | ||||||
| @ -1035,16 +1036,7 @@ int tilcdc_crtc_create(struct drm_device *dev) | |||||||
| 	drm_crtc_helper_add(crtc, &tilcdc_crtc_helper_funcs); | 	drm_crtc_helper_add(crtc, &tilcdc_crtc_helper_funcs); | ||||||
| 
 | 
 | ||||||
| 	if (priv->is_componentized) { | 	if (priv->is_componentized) { | ||||||
| 		struct device_node *ports = | 		crtc->port = of_graph_get_port_by_id(dev->dev->of_node, 0); | ||||||
| 			of_get_child_by_name(dev->dev->of_node, "ports"); |  | ||||||
| 
 |  | ||||||
| 		if (ports) { |  | ||||||
| 			crtc->port = of_get_child_by_name(ports, "port"); |  | ||||||
| 			of_node_put(ports); |  | ||||||
| 		} else { |  | ||||||
| 			crtc->port = |  | ||||||
| 				of_get_child_by_name(dev->dev->of_node, "port"); |  | ||||||
| 		} |  | ||||||
| 		if (!crtc->port) { /* This should never happen */ | 		if (!crtc->port) { /* This should never happen */ | ||||||
| 			dev_err(dev->dev, "Port node not found in %s\n", | 			dev_err(dev->dev, "Port node not found in %s\n", | ||||||
| 				dev->dev->of_node->full_name); | 				dev->dev->of_node->full_name); | ||||||
|  | |||||||
| @ -185,39 +185,6 @@ int tilcdc_attach_bridge(struct drm_device *ddev, struct drm_bridge *bridge) | |||||||
| 	return ret; | 	return ret; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static int tilcdc_node_has_port(struct device_node *dev_node) |  | ||||||
| { |  | ||||||
| 	struct device_node *node; |  | ||||||
| 
 |  | ||||||
| 	node = of_get_child_by_name(dev_node, "ports"); |  | ||||||
| 	if (!node) |  | ||||||
| 		node = of_get_child_by_name(dev_node, "port"); |  | ||||||
| 	if (!node) |  | ||||||
| 		return 0; |  | ||||||
| 	of_node_put(node); |  | ||||||
| 
 |  | ||||||
| 	return 1; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| static |  | ||||||
| struct device_node *tilcdc_get_remote_node(struct device_node *node) |  | ||||||
| { |  | ||||||
| 	struct device_node *ep; |  | ||||||
| 	struct device_node *parent; |  | ||||||
| 
 |  | ||||||
| 	if (!tilcdc_node_has_port(node)) |  | ||||||
| 		return NULL; |  | ||||||
| 
 |  | ||||||
| 	ep = of_graph_get_next_endpoint(node, NULL); |  | ||||||
| 	if (!ep) |  | ||||||
| 		return NULL; |  | ||||||
| 
 |  | ||||||
| 	parent = of_graph_get_remote_port_parent(ep); |  | ||||||
| 	of_node_put(ep); |  | ||||||
| 
 |  | ||||||
| 	return parent; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| int tilcdc_attach_external_device(struct drm_device *ddev) | int tilcdc_attach_external_device(struct drm_device *ddev) | ||||||
| { | { | ||||||
| 	struct tilcdc_drm_private *priv = ddev->dev_private; | 	struct tilcdc_drm_private *priv = ddev->dev_private; | ||||||
| @ -225,7 +192,7 @@ int tilcdc_attach_external_device(struct drm_device *ddev) | |||||||
| 	struct drm_bridge *bridge; | 	struct drm_bridge *bridge; | ||||||
| 	int ret; | 	int ret; | ||||||
| 
 | 
 | ||||||
| 	remote_node = tilcdc_get_remote_node(ddev->dev->of_node); | 	remote_node = of_graph_get_remote_node(ddev->dev->of_node, 0, 0); | ||||||
| 	if (!remote_node) | 	if (!remote_node) | ||||||
| 		return 0; | 		return 0; | ||||||
| 
 | 
 | ||||||
| @ -264,35 +231,16 @@ int tilcdc_get_external_components(struct device *dev, | |||||||
| 				   struct component_match **match) | 				   struct component_match **match) | ||||||
| { | { | ||||||
| 	struct device_node *node; | 	struct device_node *node; | ||||||
| 	struct device_node *ep = NULL; |  | ||||||
| 	int count = 0; |  | ||||||
| 	int ret = 0; |  | ||||||
| 
 | 
 | ||||||
| 	if (!tilcdc_node_has_port(dev->of_node)) | 	node = of_graph_get_remote_node(dev->of_node, 0, 0); | ||||||
| 		return 0; |  | ||||||
| 
 |  | ||||||
| 	while ((ep = of_graph_get_next_endpoint(dev->of_node, ep))) { |  | ||||||
| 		node = of_graph_get_remote_port_parent(ep); |  | ||||||
| 		if (!node || !of_device_is_available(node)) { |  | ||||||
| 			of_node_put(node); |  | ||||||
| 			continue; |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		dev_dbg(dev, "Subdevice node '%s' found\n", node->name); |  | ||||||
| 
 |  | ||||||
| 		if (of_device_is_compatible(node, "nxp,tda998x")) { |  | ||||||
| 			if (match) |  | ||||||
| 				drm_of_component_match_add(dev, match, |  | ||||||
| 							   dev_match_of, node); |  | ||||||
| 			ret = 1; |  | ||||||
| 		} |  | ||||||
| 
 | 
 | ||||||
|  | 	if (!of_device_is_compatible(node, "nxp,tda998x")) { | ||||||
| 		of_node_put(node); | 		of_node_put(node); | ||||||
| 		if (count++ > 1) { | 		return 0; | ||||||
| 			dev_err(dev, "Only one port is supported\n"); |  | ||||||
| 			return -EINVAL; |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	return ret; | 	if (match) | ||||||
|  | 		drm_of_component_match_add(dev, match, dev_match_of, node); | ||||||
|  | 	of_node_put(node); | ||||||
|  | 	return 1; | ||||||
| } | } | ||||||
|  | |||||||
| @ -356,23 +356,14 @@ static const struct of_device_id vc4_dpi_dt_match[] = { | |||||||
|  */ |  */ | ||||||
| static struct drm_panel *vc4_dpi_get_panel(struct device *dev) | static struct drm_panel *vc4_dpi_get_panel(struct device *dev) | ||||||
| { | { | ||||||
| 	struct device_node *endpoint, *panel_node; | 	struct device_node *panel_node; | ||||||
| 	struct device_node *np = dev->of_node; | 	struct device_node *np = dev->of_node; | ||||||
| 	struct drm_panel *panel; | 	struct drm_panel *panel; | ||||||
| 
 | 
 | ||||||
| 	endpoint = of_graph_get_next_endpoint(np, NULL); |  | ||||||
| 	if (!endpoint) { |  | ||||||
| 		dev_err(dev, "no endpoint to fetch DPI panel\n"); |  | ||||||
| 		return NULL; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	/* don't proceed if we have an endpoint but no panel_node tied to it */ | 	/* don't proceed if we have an endpoint but no panel_node tied to it */ | ||||||
| 	panel_node = of_graph_get_remote_port_parent(endpoint); | 	panel_node = of_graph_get_remote_node(np, 0, 0); | ||||||
| 	of_node_put(endpoint); | 	if (!panel_node) | ||||||
| 	if (!panel_node) { |  | ||||||
| 		dev_err(dev, "no valid panel node\n"); |  | ||||||
| 		return NULL; | 		return NULL; | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	panel = of_drm_find_panel(panel_node); | 	panel = of_drm_find_panel(panel_node); | ||||||
| 	of_node_put(panel_node); | 	of_node_put(panel_node); | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Rob Herring
						Rob Herring