Received: by 2002:a05:6a10:af89:0:0:0:0 with SMTP id iu9csp370155pxb; Fri, 28 Jan 2022 00:36:47 -0800 (PST) X-Google-Smtp-Source: ABdhPJzN4NzBZ3fWp55M5wItA5fazoBNb07RFFKT1RdcmOMBcIKZsHSF3qu898wx7cDdv3H23AFB X-Received: by 2002:a65:57cf:: with SMTP id q15mr5725893pgr.307.1643359007087; Fri, 28 Jan 2022 00:36:47 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1643359007; cv=none; d=google.com; s=arc-20160816; b=sZviDosGEl8OY3SIYUvmGs2ggRav7Ib+id2tFCvi2uiUCzGZiFkSITr55LmzH8wGFw J7bXN3aZuLUwFJDIvvmovcvhMvOnwWZZEWC1lkApLxQy5+FAzrIgRT/QSdxw0rh5a1bN lrosLjo4I3nJBfuV9M3k3+F7kAWBvzqhbQNogT7ndtD9d9QviCVJaFsUtrMZ23BYVTw4 k7BLPiGI7JOOv4mpIyc4XeKePCtRMWZ48IKW9XzFE7ksJwcIRoYxG1NNlsjJQLGrz4DG DmzT0ohIZM+1ld4VTHYEW+RQADcSA3McT63uPMQbVUQT0OslxfmAXCm97BKJwIcBvIJe l1Zg== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:content-language :in-reply-to:mime-version:user-agent:date:message-id:from:references :cc:to:subject:dkim-signature; bh=gW6K+kraV6392TNMeuD1F1XdINwonAzDtxz7AmALJ10=; b=hKSs78pw/V1R6evi76jU1543HeTVRMcd8CkBQAbTXl8drOQs4Ndt9gIPYGYsCGt+Rw YDJG/mUqQOMDKO3h+W7fl/lW5nKpoi4atwNkiWfqqAl86qhcUe5AKwzUhNka7RIYUYvH RJVl5fZC+7LIJIFmwiSyzN5mrtDWGuYIcyvPJsBLoWLFXOjry24N4yRffsTtrgCIn/IT eN/CGV8WxL/vkhtanGHBWIWJL9dZSQsiotXJtorSDnZcrfkkzd2Nw2gZKSQ+zdH0J3fG kSlFwAce4xB6yuoH+Lc5vq4Egba48YfPuSJJACi8INvzFTtHSXjZ2jHEArhRFMmosUTX YXLQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@collabora.com header.s=mail header.b=AaggjM82; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=collabora.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [23.128.96.18]) by mx.google.com with ESMTP id k10si3844040pfu.226.2022.01.28.00.36.31; Fri, 28 Jan 2022 00:36:47 -0800 (PST) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) client-ip=23.128.96.18; Authentication-Results: mx.google.com; dkim=pass header.i=@collabora.com header.s=mail header.b=AaggjM82; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 23.128.96.18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=collabora.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S243110AbiA0Pq0 (ORCPT + 99 others); Thu, 27 Jan 2022 10:46:26 -0500 Received: from bhuna.collabora.co.uk ([46.235.227.227]:54812 "EHLO bhuna.collabora.co.uk" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S242292AbiA0PqY (ORCPT ); Thu, 27 Jan 2022 10:46:24 -0500 Received: from [127.0.0.1] (localhost [127.0.0.1]) (Authenticated sender: kholk11) with ESMTPSA id 9A6CF1F45571 DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=collabora.com; s=mail; t=1643298383; bh=xdhZ7mIuhmOUfIV4ejxquCa70EjGgsF4qZseOX7OsOg=; h=Subject:To:Cc:References:From:Date:In-Reply-To:From; b=AaggjM82DIDDw6s8SNUESEltqNgqnMvk9Q7S6N2ef9s8V4mNNtY0A2z0J6uilwY5Z DsB6XQ2WO0xkaRO5U/nCoC3qTh1OXFVV/I2rN/r4mINxbROY5mPccOW5oHwI30Gmr3 lMoiPVpetdioicuOUHOXJqdfswGyEgxKHCo8DI2Mvu7Sz73mzKp/Tk2V/6yieMyWR5 nLF9BeuBC5HozBrAozCYJH4zqQOQZwhYL4XLirgrqqCys/H+Rp/x1PZVLS/5Ldy483 Whcdh6KMdaz5n1ZmLdwbcurco7Evm5nJwGfWgKnSxAn5xmlTq/jZ5NHeEWBcaMR9DP Z/DYnNH0JHL1A== Subject: Re: [PATCH v3] drm/mediatek: mtk_dsi: Avoid EPROBE_DEFER loop with external bridge To: Chun-Kuang Hu Cc: DRI Development , David Airlie , linux-kernel , andrzej.hajda@intel.com, "moderated list:ARM/Mediatek SoC support" , Jagan Teki , Matthias Brugger , Collabora Kernel ML , Linux ARM References: <20220127143623.123025-1-angelogioacchino.delregno@collabora.com> From: AngeloGioacchino Del Regno Message-ID: Date: Thu, 27 Jan 2022 16:46:19 +0100 User-Agent: Mozilla/5.0 (X11; Linux x86_64; rv:78.0) Gecko/20100101 Thunderbird/78.13.0 MIME-Version: 1.0 In-Reply-To: Content-Type: text/plain; charset=utf-8; format=flowed Content-Language: en-US Content-Transfer-Encoding: 8bit Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Il 27/01/22 16:21, Chun-Kuang Hu ha scritto: > Hi, Angelo: > > AngeloGioacchino Del Regno 於 > 2022年1月27日 週四 下午10:36寫道: >> >> DRM bridge drivers are now attaching their DSI device at probe time, >> which requires us to register our DSI host in order to let the bridge >> to probe: this recently started producing an endless -EPROBE_DEFER >> loop on some machines that are using external bridges, like the >> parade-ps8640, found on the ACER Chromebook R13. >> >> Now that the DSI hosts/devices probe sequence is documented, we can >> do adjustments to the mtk_dsi driver as to both fix now and make sure >> to avoid this situation in the future: for this, following what is >> documented in drm_bridge.c, move the mtk_dsi component_add() to the >> mtk_dsi_ops.attach callback and delete it in the detach callback; >> keeping in mind that we are registering a drm_bridge for our DSI, >> which is only used/attached if the DSI Host is bound, it wouldn't >> make sense to keep adding our bridge at probe time (as it would >> be useless to have it if mtk_dsi_ops.attach() fails!), so also move >> that one to the dsi host attach function (and remove it in detach). >> >> Fixes: 209264a85707 ("drm/bridge: Document the probe issue with MIPI-DSI bridges") > > The fixed tag should indicate the patch which cause the bug, but why a > patch just adding document would cause bug? > So no any patch cause bug? This patch just want to prevent a possible bug? > I think you've missed my previous message on v2, so I will paste it here: unfortunately I couldn't find a valid commit for a Fixes tag. This started being an issue at some point, when the DRM was changed to adhere to the documented probe sequence: the MediaTek DSI driver was not the only one that got broken/affected by these changes. If you have any advice on which commit should be tagged, I'm open for any kind of suggestion. I tried to check on other drivers which got fixed for the same behavior, for example drm/msm, but none of them had a Fixes tag. When the DRM got changed to adhere to this sequence, some drm/bridge drivers were also changed; this has created some incompatibilities with some drm drivers, including drm/msm and drm/mediatek. This commit is not fixing a latent bug that was introduced in drm/mediatek but rather one that was induced by the new, fixed, probe flow that got recently documented - and to which drivers should adhere; failing to adhere to that will produce an endless -EPROBE_DEFER loop, due to other drivers (as mentioned, for example drm/bridge drivers) having been changed to use that probe sequence. Regards, Angelo > Regards, > Chun-Kuang. > >> Signed-off-by: AngeloGioacchino Del Regno >> Reviewed-by: Andrzej Hajda >> Reviewed-by: Jagan Teki >> >> --- >> drivers/gpu/drm/mediatek/mtk_dsi.c | 167 +++++++++++++++-------------- >> 1 file changed, 84 insertions(+), 83 deletions(-) >> >> diff --git a/drivers/gpu/drm/mediatek/mtk_dsi.c b/drivers/gpu/drm/mediatek/mtk_dsi.c >> index 5d90d2eb0019..bced4c7d668e 100644 >> --- a/drivers/gpu/drm/mediatek/mtk_dsi.c >> +++ b/drivers/gpu/drm/mediatek/mtk_dsi.c >> @@ -786,18 +786,101 @@ void mtk_dsi_ddp_stop(struct device *dev) >> mtk_dsi_poweroff(dsi); >> } >> >> +static int mtk_dsi_encoder_init(struct drm_device *drm, struct mtk_dsi *dsi) >> +{ >> + int ret; >> + >> + ret = drm_simple_encoder_init(drm, &dsi->encoder, >> + DRM_MODE_ENCODER_DSI); >> + if (ret) { >> + DRM_ERROR("Failed to encoder init to drm\n"); >> + return ret; >> + } >> + >> + dsi->encoder.possible_crtcs = mtk_drm_find_possible_crtc_by_comp(drm, dsi->host.dev); >> + >> + ret = drm_bridge_attach(&dsi->encoder, &dsi->bridge, NULL, >> + DRM_BRIDGE_ATTACH_NO_CONNECTOR); >> + if (ret) >> + goto err_cleanup_encoder; >> + >> + dsi->connector = drm_bridge_connector_init(drm, &dsi->encoder); >> + if (IS_ERR(dsi->connector)) { >> + DRM_ERROR("Unable to create bridge connector\n"); >> + ret = PTR_ERR(dsi->connector); >> + goto err_cleanup_encoder; >> + } >> + drm_connector_attach_encoder(dsi->connector, &dsi->encoder); >> + >> + return 0; >> + >> +err_cleanup_encoder: >> + drm_encoder_cleanup(&dsi->encoder); >> + return ret; >> +} >> + >> +static int mtk_dsi_bind(struct device *dev, struct device *master, void *data) >> +{ >> + int ret; >> + struct drm_device *drm = data; >> + struct mtk_dsi *dsi = dev_get_drvdata(dev); >> + >> + ret = mtk_dsi_encoder_init(drm, dsi); >> + if (ret) >> + return ret; >> + >> + return device_reset_optional(dev); >> +} >> + >> +static void mtk_dsi_unbind(struct device *dev, struct device *master, >> + void *data) >> +{ >> + struct mtk_dsi *dsi = dev_get_drvdata(dev); >> + >> + drm_encoder_cleanup(&dsi->encoder); >> +} >> + >> +static const struct component_ops mtk_dsi_component_ops = { >> + .bind = mtk_dsi_bind, >> + .unbind = mtk_dsi_unbind, >> +}; >> + >> static int mtk_dsi_host_attach(struct mipi_dsi_host *host, >> struct mipi_dsi_device *device) >> { >> struct mtk_dsi *dsi = host_to_dsi(host); >> + struct device *dev = host->dev; >> + int ret; >> >> dsi->lanes = device->lanes; >> dsi->format = device->format; >> dsi->mode_flags = device->mode_flags; >> + dsi->next_bridge = devm_drm_of_get_bridge(dev, dev->of_node, 0, 0); >> + if (IS_ERR(dsi->next_bridge)) >> + return PTR_ERR(dsi->next_bridge); >> + >> + drm_bridge_add(&dsi->bridge); >> + >> + ret = component_add(host->dev, &mtk_dsi_component_ops); >> + if (ret) { >> + DRM_ERROR("failed to add dsi_host component: %d\n", ret); >> + drm_bridge_remove(&dsi->bridge); >> + return ret; >> + } >> >> return 0; >> } >> >> +static int mtk_dsi_host_detach(struct mipi_dsi_host *host, >> + struct mipi_dsi_device *device) >> +{ >> + struct mtk_dsi *dsi = host_to_dsi(host); >> + >> + component_del(host->dev, &mtk_dsi_component_ops); >> + drm_bridge_remove(&dsi->bridge); >> + return 0; >> +} >> + >> static void mtk_dsi_wait_for_idle(struct mtk_dsi *dsi) >> { >> int ret; >> @@ -938,73 +1021,14 @@ static ssize_t mtk_dsi_host_transfer(struct mipi_dsi_host *host, >> >> static const struct mipi_dsi_host_ops mtk_dsi_ops = { >> .attach = mtk_dsi_host_attach, >> + .detach = mtk_dsi_host_detach, >> .transfer = mtk_dsi_host_transfer, >> }; >> >> -static int mtk_dsi_encoder_init(struct drm_device *drm, struct mtk_dsi *dsi) >> -{ >> - int ret; >> - >> - ret = drm_simple_encoder_init(drm, &dsi->encoder, >> - DRM_MODE_ENCODER_DSI); >> - if (ret) { >> - DRM_ERROR("Failed to encoder init to drm\n"); >> - return ret; >> - } >> - >> - dsi->encoder.possible_crtcs = mtk_drm_find_possible_crtc_by_comp(drm, dsi->host.dev); >> - >> - ret = drm_bridge_attach(&dsi->encoder, &dsi->bridge, NULL, >> - DRM_BRIDGE_ATTACH_NO_CONNECTOR); >> - if (ret) >> - goto err_cleanup_encoder; >> - >> - dsi->connector = drm_bridge_connector_init(drm, &dsi->encoder); >> - if (IS_ERR(dsi->connector)) { >> - DRM_ERROR("Unable to create bridge connector\n"); >> - ret = PTR_ERR(dsi->connector); >> - goto err_cleanup_encoder; >> - } >> - drm_connector_attach_encoder(dsi->connector, &dsi->encoder); >> - >> - return 0; >> - >> -err_cleanup_encoder: >> - drm_encoder_cleanup(&dsi->encoder); >> - return ret; >> -} >> - >> -static int mtk_dsi_bind(struct device *dev, struct device *master, void *data) >> -{ >> - int ret; >> - struct drm_device *drm = data; >> - struct mtk_dsi *dsi = dev_get_drvdata(dev); >> - >> - ret = mtk_dsi_encoder_init(drm, dsi); >> - if (ret) >> - return ret; >> - >> - return device_reset_optional(dev); >> -} >> - >> -static void mtk_dsi_unbind(struct device *dev, struct device *master, >> - void *data) >> -{ >> - struct mtk_dsi *dsi = dev_get_drvdata(dev); >> - >> - drm_encoder_cleanup(&dsi->encoder); >> -} >> - >> -static const struct component_ops mtk_dsi_component_ops = { >> - .bind = mtk_dsi_bind, >> - .unbind = mtk_dsi_unbind, >> -}; >> - >> static int mtk_dsi_probe(struct platform_device *pdev) >> { >> struct mtk_dsi *dsi; >> struct device *dev = &pdev->dev; >> - struct drm_panel *panel; >> struct resource *regs; >> int irq_num; >> int ret; >> @@ -1021,19 +1045,6 @@ static int mtk_dsi_probe(struct platform_device *pdev) >> return ret; >> } >> >> - ret = drm_of_find_panel_or_bridge(dev->of_node, 0, 0, >> - &panel, &dsi->next_bridge); >> - if (ret) >> - goto err_unregister_host; >> - >> - if (panel) { >> - dsi->next_bridge = devm_drm_panel_bridge_add(dev, panel); >> - if (IS_ERR(dsi->next_bridge)) { >> - ret = PTR_ERR(dsi->next_bridge); >> - goto err_unregister_host; >> - } >> - } >> - >> dsi->driver_data = of_device_get_match_data(dev); >> >> dsi->engine_clk = devm_clk_get(dev, "engine"); >> @@ -1098,14 +1109,6 @@ static int mtk_dsi_probe(struct platform_device *pdev) >> dsi->bridge.of_node = dev->of_node; >> dsi->bridge.type = DRM_MODE_CONNECTOR_DSI; >> >> - drm_bridge_add(&dsi->bridge); >> - >> - ret = component_add(&pdev->dev, &mtk_dsi_component_ops); >> - if (ret) { >> - dev_err(&pdev->dev, "failed to add component: %d\n", ret); >> - goto err_unregister_host; >> - } >> - >> return 0; >> >> err_unregister_host: >> @@ -1118,8 +1121,6 @@ static int mtk_dsi_remove(struct platform_device *pdev) >> struct mtk_dsi *dsi = platform_get_drvdata(pdev); >> >> mtk_output_dsi_disable(dsi); >> - drm_bridge_remove(&dsi->bridge); >> - component_del(&pdev->dev, &mtk_dsi_component_ops); >> mipi_dsi_host_unregister(&dsi->host); >> >> return 0; >> -- >> 2.33.1 >> -- AngeloGioacchino Del Regno Software Engineer Collabora Ltd. Platinum Building, St John's Innovation Park, Cambridge CB4 0DS, UK Registered in England & Wales, no. 5513718