Hi Hans, On Tue, Dec 03, 2024 at 10:07:31AM +0100, Hans Verkuil wrote: > Hi Alain, > > On 18/11/2024 14:35, Alain Volmat wrote: > > The stm32mp25 supports both parallel & csi inputs. > > An additional clock control is necessary. > > Skeleton of the subdev structures for the stm32mp25 is added, > > identical for the time being to the stm32mp13 however more subdeves > > will be added in further commits. > > > > Signed-off-by: Alain Volmat <alain.volmat@xxxxxxxxxxx> > > --- > > .../platform/st/stm32/stm32-dcmipp/dcmipp-core.c | 98 ++++++++++++++++++---- > > 1 file changed, 80 insertions(+), 18 deletions(-) > > > > diff --git a/drivers/media/platform/st/stm32/stm32-dcmipp/dcmipp-core.c b/drivers/media/platform/st/stm32/stm32-dcmipp/dcmipp-core.c > > index d2cc19bb40d77f67a1f5fe565bc62f45eff2d266..0087f9017f024ba6b918b99c1ef39212ad6b881a 100644 > > --- a/drivers/media/platform/st/stm32/stm32-dcmipp/dcmipp-core.c > > +++ b/drivers/media/platform/st/stm32/stm32-dcmipp/dcmipp-core.c > > @@ -40,6 +40,7 @@ struct dcmipp_device { > > > > /* Hardware resources */ > > void __iomem *regs; > > + struct clk *mclk; > > struct clk *kclk; > > > > /* The pipeline configuration */ > > @@ -132,6 +133,40 @@ static const struct dcmipp_pipeline_config stm32mp13_pipe_cfg = { > > .hw_revision = DCMIPP_STM32MP13_VERR > > }; > > > > +static const struct dcmipp_ent_config stm32mp25_ent_config[] = { > > + { > > + .name = "dcmipp_input", > > + .init = dcmipp_inp_ent_init, > > + .release = dcmipp_inp_ent_release, > > + }, > > + { > > + .name = "dcmipp_dump_postproc", > > + .init = dcmipp_byteproc_ent_init, > > + .release = dcmipp_byteproc_ent_release, > > + }, > > + { > > + .name = "dcmipp_dump_capture", > > + .init = dcmipp_bytecap_ent_init, > > + .release = dcmipp_bytecap_ent_release, > > + }, > > +}; > > + > > +static const struct dcmipp_ent_link stm32mp25_ent_links[] = { > > + DCMIPP_ENT_LINK(ID_INPUT, 1, ID_DUMP_BYTEPROC, 0, > > + MEDIA_LNK_FL_ENABLED | MEDIA_LNK_FL_IMMUTABLE), > > + DCMIPP_ENT_LINK(ID_DUMP_BYTEPROC, 1, ID_DUMP_CAPTURE, 0, > > + MEDIA_LNK_FL_ENABLED | MEDIA_LNK_FL_IMMUTABLE), > > +}; > > + > > +#define DCMIPP_STM32MP25_VERR 0x30 > > +static const struct dcmipp_pipeline_config stm32mp25_pipe_cfg = { > > + .ents = stm32mp25_ent_config, > > + .num_ents = ARRAY_SIZE(stm32mp25_ent_config), > > + .links = stm32mp25_ent_links, > > + .num_links = ARRAY_SIZE(stm32mp25_ent_links), > > + .hw_revision = DCMIPP_STM32MP25_VERR > > +}; > > + > > #define LINK_FLAG_TO_STR(f) ((f) == 0 ? "" :\ > > (f) == MEDIA_LNK_FL_ENABLED ? "ENABLED" :\ > > (f) == MEDIA_LNK_FL_IMMUTABLE ? "IMMUTABLE" :\ > > @@ -212,6 +247,7 @@ static int dcmipp_create_subdevs(struct dcmipp_device *dcmipp) > > > > static const struct of_device_id dcmipp_of_match[] = { > > { .compatible = "st,stm32mp13-dcmipp", .data = &stm32mp13_pipe_cfg }, > > + { .compatible = "st,stm32mp25-dcmipp", .data = &stm32mp25_pipe_cfg }, > > { /* end node */ }, > > }; > > MODULE_DEVICE_TABLE(of, dcmipp_of_match); > > @@ -261,13 +297,22 @@ static int dcmipp_graph_notify_bound(struct v4l2_async_notifier *notifier, > > { > > struct dcmipp_device *dcmipp = notifier_to_dcmipp(notifier); > > unsigned int ret; > > - int src_pad; > > + int src_pad, i; > > struct dcmipp_ent_device *sink; > > - struct v4l2_fwnode_endpoint vep = { .bus_type = V4L2_MBUS_PARALLEL }; > > + struct v4l2_fwnode_endpoint vep = { 0 }; > > struct fwnode_handle *ep; > > + enum v4l2_mbus_type supported_types[] = { > > + V4L2_MBUS_PARALLEL, V4L2_MBUS_BT656, V4L2_MBUS_CSI2_DPHY > > + }; > > + int supported_types_nb = ARRAY_SIZE(supported_types); > > > > dev_dbg(dcmipp->dev, "Subdev \"%s\" bound\n", subdev->name); > > > > + /* Only MP25 supports CSI input */ > > + if (!of_device_is_compatible(dcmipp->dev->of_node, > > + "st,stm32mp25-dcmipp")) > > + supported_types_nb--; > > + > > /* > > * Link this sub-device to DCMIPP, it could be > > * a parallel camera sensor or a CSI-2 to parallel bridge > > @@ -284,21 +329,23 @@ static int dcmipp_graph_notify_bound(struct v4l2_async_notifier *notifier, > > return -ENODEV; > > } > > > > - /* Check for parallel bus-type first, then bt656 */ > > - ret = v4l2_fwnode_endpoint_parse(ep, &vep); > > - if (ret) { > > - vep.bus_type = V4L2_MBUS_BT656; > > + /* Check for supported MBUS type */ > > + for (i = 0; i < supported_types_nb; i++) { > > + vep.bus_type = supported_types[i]; > > ret = v4l2_fwnode_endpoint_parse(ep, &vep); > > - if (ret) { > > - dev_err(dcmipp->dev, "Could not parse the endpoint\n"); > > - fwnode_handle_put(ep); > > - return ret; > > - } > > + if (!ret) > > + break; > > } > > > > fwnode_handle_put(ep); > > > > - if (vep.bus.parallel.bus_width == 0) { > > + if (ret) { > > + dev_err(dcmipp->dev, "Could not parse the endpoint\n"); > > + return ret; > > + } > > + > > + if (vep.bus_type != V4L2_MBUS_CSI2_DPHY && > > + vep.bus.parallel.bus_width == 0) { > > dev_err(dcmipp->dev, "Invalid parallel interface bus-width\n"); > > return -ENODEV; > > } > > @@ -311,11 +358,13 @@ static int dcmipp_graph_notify_bound(struct v4l2_async_notifier *notifier, > > return -ENODEV; > > } > > > > - /* Parallel input device detected, connect it to parallel subdev */ > > + /* Connect input device to the dcmipp_input subdev */ > > sink = dcmipp->entity[ID_INPUT]; > > - sink->bus.flags = vep.bus.parallel.flags; > > - sink->bus.bus_width = vep.bus.parallel.bus_width; > > - sink->bus.data_shift = vep.bus.parallel.data_shift; > > + if (vep.bus_type != V4L2_MBUS_CSI2_DPHY) { > > + sink->bus.flags = vep.bus.parallel.flags; > > + sink->bus.bus_width = vep.bus.parallel.bus_width; > > + sink->bus.data_shift = vep.bus.parallel.data_shift; > > + } > > sink->bus_type = vep.bus_type; > > ret = media_create_pad_link(&subdev->entity, src_pad, sink->ent, 0, > > MEDIA_LNK_FL_IMMUTABLE | > > @@ -414,7 +463,7 @@ static int dcmipp_graph_init(struct dcmipp_device *dcmipp) > > static int dcmipp_probe(struct platform_device *pdev) > > { > > struct dcmipp_device *dcmipp; > > - struct clk *kclk; > > + struct clk *kclk, *mclk; > > const struct dcmipp_pipeline_config *pipe_cfg; > > struct reset_control *rstc; > > int irq; > > @@ -474,12 +523,20 @@ static int dcmipp_probe(struct platform_device *pdev) > > return ret; > > } > > > > - kclk = devm_clk_get(&pdev->dev, NULL); > > + kclk = devm_clk_get(&pdev->dev, "kclk"); > > if (IS_ERR(kclk)) > > return dev_err_probe(&pdev->dev, PTR_ERR(kclk), > > "Unable to get kclk\n"); > > dcmipp->kclk = kclk; > > > > + if (!of_device_is_compatible(pdev->dev.of_node, "st,stm32mp13-dcmipp")) { > > + mclk = devm_clk_get(&pdev->dev, "mclk"); > > + if (IS_ERR(mclk)) > > + return dev_err_probe(&pdev->dev, PTR_ERR(mclk), > > + "Unable to get mclk\n"); > > + dcmipp->mclk = mclk; > > + } > > + > > dcmipp->entity = devm_kcalloc(&pdev->dev, dcmipp->pipe_cfg->num_ents, > > sizeof(*dcmipp->entity), GFP_KERNEL); > > if (!dcmipp->entity) > > @@ -542,6 +599,7 @@ static int dcmipp_runtime_suspend(struct device *dev) > > struct dcmipp_device *dcmipp = dev_get_drvdata(dev); > > > > clk_disable_unprepare(dcmipp->kclk); > > + clk_disable_unprepare(dcmipp->mclk); > > > > return 0; > > } > > @@ -551,6 +609,10 @@ static int dcmipp_runtime_resume(struct device *dev) > > struct dcmipp_device *dcmipp = dev_get_drvdata(dev); > > int ret; > > > > + ret = clk_prepare_enable(dcmipp->mclk); > > + if (ret) > > + dev_err(dev, "%s: Failed to prepare_enable mclk\n", __func__); > > + > > Shouldn't this return on error? Yes indeed. > > > ret = clk_prepare_enable(dcmipp->kclk); > > if (ret) > > dev_err(dev, "%s: Failed to prepare_enable kclk\n", __func__); > > And on error here, it should call clk_disable_unprepare for mclk. Ok. > > > > > This was reported by smatch: > > .../dcmipp-core.c:620 dcmipp_runtime_resume() warn: 'dcmipp->mclk' from clk_prepare_enable() not released on lines: 620. > > Everything else in this series looks good to me, so you can either post a fix for this > and I will fold it into this 13/15 patch, or you post an updated v3.1 13/15 patch. > Or of course a v4, if you prefer. I prepared a v4 with this fix in the 13/15 (that's the only difference between the v3 and the v4). Will post it right away. Regards, Alain > > Regards, > > Hans