On Mon, Sep 11, 2017 at 02:29:51PM +0200, Hans Verkuil wrote: [...] > diff --git a/drivers/media/platform/tegra-cec/tegra_cec.c b/drivers/media/platform/tegra-cec/tegra_cec.c [...] > +static int tegra_cec_probe(struct platform_device *pdev) > +{ > + struct platform_device *hdmi_dev; > + struct device_node *np; > + struct tegra_cec *cec; > + struct resource *res; > + int ret = 0; > + > + np = of_parse_phandle(pdev->dev.of_node, "hdmi-phandle", 0); > + > + if (!np) { > + dev_err(&pdev->dev, "Failed to find hdmi node in device tree\n"); > + return -ENODEV; > + } > + hdmi_dev = of_find_device_by_node(np); > + if (hdmi_dev == NULL) > + return -EPROBE_DEFER; This seems a little awkward. Why exactly do we need to defer probe here? It seems to me like cec_notifier_get() should be able to deal with HDMI appearing at a later point. > + > + cec = devm_kzalloc(&pdev->dev, sizeof(struct tegra_cec), GFP_KERNEL); > + > + if (!cec) > + return -ENOMEM; > + > + res = platform_get_resource(pdev, IORESOURCE_MEM, 0); > + > + if (!res) { > + dev_err(&pdev->dev, > + "Unable to allocate resources for device\n"); > + ret = -EBUSY; > + goto cec_error; > + } > + > + if (!devm_request_mem_region(&pdev->dev, res->start, resource_size(res), > + pdev->name)) { > + dev_err(&pdev->dev, > + "Unable to request mem region for device\n"); > + ret = -EBUSY; > + goto cec_error; > + } > + > + cec->tegra_cec_irq = platform_get_irq(pdev, 0); > + > + if (cec->tegra_cec_irq <= 0) { > + ret = -EBUSY; > + goto cec_error; > + } > + > + cec->cec_base = devm_ioremap_nocache(&pdev->dev, res->start, > + resource_size(res)); > + > + if (!cec->cec_base) { > + dev_err(&pdev->dev, "Unable to grab IOs for device\n"); > + ret = -EBUSY; > + goto cec_error; > + } > + > + cec->clk = devm_clk_get(&pdev->dev, "cec"); > + > + if (IS_ERR_OR_NULL(cec->clk)) { > + dev_err(&pdev->dev, "Can't get clock for CEC\n"); > + ret = -ENOENT; > + goto clk_error; > + } > + > + clk_prepare_enable(cec->clk); > + > + /* set context info. */ > + cec->dev = &pdev->dev; > + > + platform_set_drvdata(pdev, cec); > + > + ret = devm_request_threaded_irq(&pdev->dev, cec->tegra_cec_irq, > + tegra_cec_irq_handler, tegra_cec_irq_thread_handler, > + 0, "cec_irq", &pdev->dev); > + > + if (ret) { > + dev_err(&pdev->dev, > + "Unable to request interrupt for device\n"); > + goto cec_error; > + } > + > + cec->notifier = cec_notifier_get(&hdmi_dev->dev); > + if (!cec->notifier) { > + ret = -ENOMEM; > + goto cec_error; > + } Ah... I see why we need the HDMI device right away. This seems a little brittle to me, for two reasons: what if the HDMI controller goes away? Will we be hanging on to a stale device? I mean, the device doesn't necessarily have to go away, but what's the effect on CEC if the driver unbinds from the HDMI controller? Secondly, this creates a circular dependency. It seems to me like it'd actually be simpler if the CEC controller was a "service provider" that HDMI could use and "request/release" as appropriate. In that case, the DT would look somewhat like this: hdmi@... { cec = <&cec>; }; cec: cec@... { ... }; And then the HDMI driver could do something like: cec = cec_get(&pdev->dev); /* register notifier, ... */ That way the dependency becomes unidirectional and it seems to me like that would allow interactions between HDMI and CEC would become simpler overall. Anyway, this is something that could always be changed after the fact (except maybe for some bits needed for backwards-compatibility with old device trees), and this seems to work well enough as it is, so: Acked-by: Thierry Reding <treding@xxxxxxxxxx>
Attachment:
signature.asc
Description: PGP signature
_______________________________________________ dri-devel mailing list dri-devel@xxxxxxxxxxxxxxxxxxxxx https://lists.freedesktop.org/mailman/listinfo/dri-devel