Received: by 2002:a6b:500f:0:0:0:0:0 with SMTP id e15csp4797144iob; Mon, 9 May 2022 01:42:59 -0700 (PDT) X-Google-Smtp-Source: ABdhPJzHy5wABriFiq+6ooy4ixELYeRomaysb8U3we1FrPPFRtJ4NMbcYcQ80H81cyRBoWeWDz56 X-Received: by 2002:a05:6a00:1946:b0:4fe:309f:d612 with SMTP id s6-20020a056a00194600b004fe309fd612mr15400160pfk.10.1652085779012; Mon, 09 May 2022 01:42:59 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1652085779; cv=none; d=google.com; s=arc-20160816; b=mzR9Fvprycc8vHX/V1El4y/UFIboJ/VSJxii0jlTzySoq3boCGKyXY4B8fGpgDiwzA tjRpd3roy4+gZuEni5LiS5XnigD1l1iY+E6ASnCxjkHeKJtHQDxo3He/3gWXJ+SdqwCX Z9YXR+zCGw2AEG7oOvjJsWN2N8lPbVIsZFCgVWicsK+U7zKtmJkqcTT5xDZg31rB4k5b +ANAr+JQHsQDhOllM5p2jF0Dy6+Q5sZDmRjmFszTwx7eEVzpD7DTqF6OnQqraKOsl3t2 Ld1mmvZQgldSxDe4FYBdfpzWajoH4hdy3qdtcvkf3c8PVqeH1udopjSemVPRKpZ8QEaR uPpw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:content-transfer-encoding:in-reply-to:from :references:cc:to:content-language:subject:user-agent:mime-version :date:message-id:dkim-signature; bh=mbrFtqvXowI/g08jyHxbBzncqQqEwGurYhPKIhLlXMM=; b=MDjQufU0Dp4bxSXUDVEJNuNFCwIq4EAztXGd3Mr7hFFiqBTsYYbN/zdU09100D2wkC 1TU2JieJkXfik2F1QEmhF4akedE3aBGqonqiBlekG7vpr7sVc6ucNVlh2wQmZd+jj2zD E6EIBsbDOGCdSLIQC7CpUzGzFs5NbEhnE+93QtSMntgPWEzAYDFAqaRzLlX9W1ZSCVE+ 8KRqx4ipHVxfIF+u9zsybn9S2txn2N6k3MLT8mVk4xw3wy8Kg+zOyV11oVme08IvTEQ6 4VvmErgtSLDI1oVG9+Ure1CMYxypsV7u3etO0U9/wgIh8Qx3PpurAIn3WelSCvihRV2d eT0g== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@quicinc.com header.s=qcdkim header.b="lvZazIL/"; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=quicinc.com Return-Path: Received: from lindbergh.monkeyblade.net (lindbergh.monkeyblade.net. [2620:137:e000::1:18]) by mx.google.com with ESMTPS id h186-20020a636cc3000000b003c186ab9092si12954951pgc.538.2022.05.09.01.42.58 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Mon, 09 May 2022 01:42:58 -0700 (PDT) Received-SPF: pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:18 as permitted sender) client-ip=2620:137:e000::1:18; Authentication-Results: mx.google.com; dkim=pass header.i=@quicinc.com header.s=qcdkim header.b="lvZazIL/"; spf=pass (google.com: domain of linux-kernel-owner@vger.kernel.org designates 2620:137:e000::1:18 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=quicinc.com Received: from out1.vger.email (out1.vger.email [IPv6:2620:137:e000::1:20]) by lindbergh.monkeyblade.net (Postfix) with ESMTP id 94395132751; Mon, 9 May 2022 01:27:21 -0700 (PDT) Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1444320AbiEFVHn (ORCPT + 99 others); Fri, 6 May 2022 17:07:43 -0400 Received: from lindbergh.monkeyblade.net ([23.128.96.19]:60248 "EHLO lindbergh.monkeyblade.net" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1378271AbiEFVHh (ORCPT ); Fri, 6 May 2022 17:07:37 -0400 Received: from alexa-out.qualcomm.com (alexa-out.qualcomm.com [129.46.98.28]) by lindbergh.monkeyblade.net (Postfix) with ESMTPS id 895AE6EB14; Fri, 6 May 2022 14:03:52 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=quicinc.com; i=@quicinc.com; q=dns/txt; s=qcdkim; t=1651871032; x=1683407032; h=message-id:date:mime-version:subject:to:cc:references: from:in-reply-to:content-transfer-encoding; bh=mbrFtqvXowI/g08jyHxbBzncqQqEwGurYhPKIhLlXMM=; b=lvZazIL/W8KYX03DDimDxbtknXD/isiqY4qd2CAL/YGUMh1tn5YGjVdr URwUJBwbxszPYTOYsHDYINtmPBDqJJL+Kqsc5MQ5UzR3H3fYCDTdibSt8 3K9ts/7l3FIJ8r5pSq7LheCFdjiixK8y2Yd2FDhpDGytz1EUpVPIqs0+T s=; Received: from ironmsg07-lv.qualcomm.com ([10.47.202.151]) by alexa-out.qualcomm.com with ESMTP; 06 May 2022 14:03:52 -0700 X-QCInternal: smtphost Received: from nasanex01c.na.qualcomm.com ([10.47.97.222]) by ironmsg07-lv.qualcomm.com with ESMTP/TLS/ECDHE-RSA-AES256-GCM-SHA384; 06 May 2022 14:03:51 -0700 Received: from nalasex01a.na.qualcomm.com (10.47.209.196) by nasanex01c.na.qualcomm.com (10.47.97.222) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.2.986.22; Fri, 6 May 2022 14:03:50 -0700 Received: from [10.111.168.240] (10.80.80.8) by nalasex01a.na.qualcomm.com (10.47.209.196) with Microsoft SMTP Server (version=TLS1_2, cipher=TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384) id 15.2.986.22; Fri, 6 May 2022 14:03:46 -0700 Message-ID: <30e4f4db-9c4c-af90-f1d2-ec39019f95a5@quicinc.com> Date: Fri, 6 May 2022 14:03:43 -0700 MIME-Version: 1.0 User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:91.0) Gecko/20100101 Thunderbird/91.6.2 Subject: Re: [PATCH v10 1/4] drm/msm/dp: Add eDP support via aux_bus Content-Language: en-US To: Dmitry Baryshkov , Sankeerth Billakanti , , , , , CC: , , , , , , , , , , , , References: <1650887072-16652-1-git-send-email-quic_sbillaka@quicinc.com> <1650887072-16652-2-git-send-email-quic_sbillaka@quicinc.com> <8fe3f9cd-546d-539b-cba4-8479d780a4c5@linaro.org> From: Abhinav Kumar In-Reply-To: <8fe3f9cd-546d-539b-cba4-8479d780a4c5@linaro.org> Content-Type: text/plain; charset="UTF-8"; format=flowed Content-Transfer-Encoding: 8bit X-Originating-IP: [10.80.80.8] X-ClientProxiedBy: nasanex01a.na.qualcomm.com (10.52.223.231) To nalasex01a.na.qualcomm.com (10.47.209.196) X-Spam-Status: No, score=-3.1 required=5.0 tests=BAYES_00,DKIM_SIGNED, DKIM_VALID,DKIM_VALID_AU,HEADER_FROM_DIFFERENT_DOMAINS, MAILING_LIST_MULTI,NICE_REPLY_A,RDNS_NONE,SPF_HELO_NONE, T_SCC_BODY_TEXT_LINE autolearn=unavailable autolearn_force=no version=3.4.6 X-Spam-Checker-Version: SpamAssassin 3.4.6 (2021-04-09) on lindbergh.monkeyblade.net Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 5/6/2022 1:49 PM, Dmitry Baryshkov wrote: > On 25/04/2022 14:44, Sankeerth Billakanti wrote: >> This patch adds support for generic eDP sink through aux_bus. The eDP/DP >> controller driver should support aux transactions originating from the >> panel-edp driver and hence should be initialized and ready. >> >> The panel bridge supporting the panel should be ready before the bridge >> connector is initialized. The generic panel probe needs the controller >> resources to be enabled to support the aux transactions originating from >> the panel probe. >> >> Signed-off-by: Sankeerth Billakanti >> Reviewed-by: Douglas Anderson >> Reviewed-by: Stephen Boyd > > An additional side effect from this patch. Previously missing panel > would have caused the bind error. Now it is the dp_modeset_init error, > which translates to kms_hw_init returning -517. I kind ask to move the > next_bridge acquisition back to the dp_bind in one of the followup patches. > This is true. But the end result would be same isnt it? When dp_display_bind() failed earlier, it would cause master bind failure too due to component model. Even now, it causes the same result? >> --- >> Changes in v10: >>    - modify the error handling condition >>    - modify the kernel doc >> >> Changes in v9: >>    - add comments for panel probe >>    - modify the error handling checks >> >> Changes in v8: >>    - handle corner cases >>    - add comment for the bridge ops >> >> Changes in v7: >>    - aux_bus is mandatory for eDP >>    - connector type check modified to just check for eDP >> >> Changes in v6: >>    - Remove initialization >>    - Fix aux_bus node leak >>    - Split the patches >> >>   drivers/gpu/drm/msm/dp/dp_display.c | 72 >> ++++++++++++++++++++++++++++++++++--- >>   drivers/gpu/drm/msm/dp/dp_display.h |  1 + >>   drivers/gpu/drm/msm/dp/dp_drm.c     | 21 ++++++++--- >>   drivers/gpu/drm/msm/dp/dp_parser.c  | 23 ++---------- >>   drivers/gpu/drm/msm/dp/dp_parser.h  | 14 +++++++- >>   5 files changed, 101 insertions(+), 30 deletions(-) >> >> diff --git a/drivers/gpu/drm/msm/dp/dp_display.c >> b/drivers/gpu/drm/msm/dp/dp_display.c >> index d7a19d6..f772d84 100644 >> --- a/drivers/gpu/drm/msm/dp/dp_display.c >> +++ b/drivers/gpu/drm/msm/dp/dp_display.c >> @@ -10,6 +10,7 @@ >>   #include >>   #include >>   #include >> +#include >>   #include "msm_drv.h" >>   #include "msm_kms.h" >> @@ -259,14 +260,12 @@ static int dp_display_bind(struct device *dev, >> struct device *master, >>       dp->dp_display.drm_dev = drm; >>       priv->dp[dp->id] = &dp->dp_display; >> -    rc = dp->parser->parse(dp->parser, dp->dp_display.connector_type); >> +    rc = dp->parser->parse(dp->parser); >>       if (rc) { >>           DRM_ERROR("device tree parsing failed\n"); >>           goto end; >>       } >> -    dp->dp_display.next_bridge = dp->parser->next_bridge; >> - >>       dp->aux->drm_dev = drm; >>       rc = dp_aux_register(dp->aux); >>       if (rc) { >> @@ -1319,6 +1318,8 @@ static int dp_display_probe(struct >> platform_device *pdev) >>       dp->pdev = pdev; >>       dp->name = "drm_dp"; >>       dp->dp_display.connector_type = desc->connector_type; >> +    dp->dp_display.is_edp = >> +        (dp->dp_display.connector_type == DRM_MODE_CONNECTOR_eDP); >>       rc = dp_init_sub_modules(dp); >>       if (rc) { >> @@ -1508,7 +1509,8 @@ void msm_dp_irq_postinstall(struct msm_dp >> *dp_display) >>       dp_hpd_event_setup(dp); >> -    dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100); >> +    if (!dp_display->is_edp) >> +        dp_add_event(dp, EV_HPD_INIT_SETUP, 0, 100); >>   } >>   void msm_dp_debugfs_init(struct msm_dp *dp_display, struct drm_minor >> *minor) >> @@ -1530,6 +1532,64 @@ void msm_dp_debugfs_init(struct msm_dp >> *dp_display, struct drm_minor *minor) >>       } >>   } >> +static int dp_display_get_next_bridge(struct msm_dp *dp) >> +{ >> +    int rc; >> +    struct dp_display_private *dp_priv; >> +    struct device_node *aux_bus; >> +    struct device *dev; >> + >> +    dp_priv = container_of(dp, struct dp_display_private, dp_display); >> +    dev = &dp_priv->pdev->dev; >> +    aux_bus = of_get_child_by_name(dev->of_node, "aux-bus"); >> + >> +    if (aux_bus && dp->is_edp) { >> +        dp_display_host_init(dp_priv); >> +        dp_catalog_ctrl_hpd_config(dp_priv->catalog); >> +        dp_display_host_phy_init(dp_priv); >> +        enable_irq(dp_priv->irq); >> + >> +        /* >> +         * The code below assumes that the panel will finish probing >> +         * by the time devm_of_dp_aux_populate_ep_devices() returns. >> +         * This isn't a great assumption since it will fail if the >> +         * panel driver is probed asynchronously but is the best we >> +         * can do without a bigger driver reorganization. >> +         */ >> +        rc = devm_of_dp_aux_populate_ep_devices(dp_priv->aux); >> +        of_node_put(aux_bus); >> +        if (rc) >> +            goto error; >> +    } else if (dp->is_edp) { >> +        DRM_ERROR("eDP aux_bus not found\n"); >> +        return -ENODEV; >> +    } >> + >> +    /* >> +     * External bridges are mandatory for eDP interfaces: one has to >> +     * provide at least an eDP panel (which gets wrapped into >> panel-bridge). >> +     * >> +     * For DisplayPort interfaces external bridges are optional, so >> +     * silently ignore an error if one is not present (-ENODEV). >> +     */ >> +    rc = dp_parser_find_next_bridge(dp_priv->parser); >> +    if (!dp->is_edp && rc == -ENODEV) >> +        return 0; >> + >> +    if (!rc) { >> +        dp->next_bridge = dp_priv->parser->next_bridge; >> +        return 0; >> +    } >> + >> +error: >> +    if (dp->is_edp) { >> +        disable_irq(dp_priv->irq); >> +        dp_display_host_phy_exit(dp_priv); >> +        dp_display_host_deinit(dp_priv); >> +    } >> +    return rc; >> +} >> + >>   int msm_dp_modeset_init(struct msm_dp *dp_display, struct drm_device >> *dev, >>               struct drm_encoder *encoder) >>   { >> @@ -1553,6 +1613,10 @@ int msm_dp_modeset_init(struct msm_dp >> *dp_display, struct drm_device *dev, >>       dp_display->encoder = encoder; >> +    ret = dp_display_get_next_bridge(dp_display); >> +    if (ret) >> +        return ret; >> + >>       dp_display->bridge = dp_bridge_init(dp_display, dev, encoder); >>       if (IS_ERR(dp_display->bridge)) { >>           ret = PTR_ERR(dp_display->bridge); >> diff --git a/drivers/gpu/drm/msm/dp/dp_display.h >> b/drivers/gpu/drm/msm/dp/dp_display.h >> index 49a1d89..1377cc3 100644 >> --- a/drivers/gpu/drm/msm/dp/dp_display.h >> +++ b/drivers/gpu/drm/msm/dp/dp_display.h >> @@ -21,6 +21,7 @@ struct msm_dp { >>       bool audio_enabled; >>       bool power_on; >>       unsigned int connector_type; >> +    bool is_edp; >>       hdmi_codec_plugged_cb plugged_cb; >> diff --git a/drivers/gpu/drm/msm/dp/dp_drm.c >> b/drivers/gpu/drm/msm/dp/dp_drm.c >> index 7ce1aca..8a75c55 100644 >> --- a/drivers/gpu/drm/msm/dp/dp_drm.c >> +++ b/drivers/gpu/drm/msm/dp/dp_drm.c >> @@ -114,10 +114,23 @@ struct drm_bridge *dp_bridge_init(struct msm_dp >> *dp_display, struct drm_device * >>       bridge->funcs = &dp_bridge_ops; >>       bridge->type = dp_display->connector_type; >> -    bridge->ops = >> -        DRM_BRIDGE_OP_DETECT | >> -        DRM_BRIDGE_OP_HPD | >> -        DRM_BRIDGE_OP_MODES; >> +    /* >> +     * Many ops only make sense for DP. Why? >> +     * - Detect/HPD are used by DRM to know if a display is _physically_ >> +     *   there, not whether the display is powered on / finished >> initting. >> +     *   On eDP we assume the display is always there because you can't >> +     *   know until power is applied. If we don't implement the ops >> DRM will >> +     *   assume our display is always there. >> +     * - Currently eDP mode reading is driven by the panel driver. This >> +     *   allows the panel driver to properly power itself on to read the >> +     *   modes. >> +     */ >> +    if (!dp_display->is_edp) { >> +        bridge->ops = >> +            DRM_BRIDGE_OP_DETECT | >> +            DRM_BRIDGE_OP_HPD | >> +            DRM_BRIDGE_OP_MODES; >> +    } >>       rc = drm_bridge_attach(encoder, bridge, NULL, >> DRM_BRIDGE_ATTACH_NO_CONNECTOR); >>       if (rc) { >> diff --git a/drivers/gpu/drm/msm/dp/dp_parser.c >> b/drivers/gpu/drm/msm/dp/dp_parser.c >> index 1056b8d..4bdbf91 100644 >> --- a/drivers/gpu/drm/msm/dp/dp_parser.c >> +++ b/drivers/gpu/drm/msm/dp/dp_parser.c >> @@ -265,7 +265,7 @@ static int dp_parser_clock(struct dp_parser *parser) >>       return 0; >>   } >> -static int dp_parser_find_next_bridge(struct dp_parser *parser) >> +int dp_parser_find_next_bridge(struct dp_parser *parser) >>   { >>       struct device *dev = &parser->pdev->dev; >>       struct drm_bridge *bridge; >> @@ -279,7 +279,7 @@ static int dp_parser_find_next_bridge(struct >> dp_parser *parser) >>       return 0; >>   } >> -static int dp_parser_parse(struct dp_parser *parser, int connector_type) >> +static int dp_parser_parse(struct dp_parser *parser) >>   { >>       int rc = 0; >> @@ -300,25 +300,6 @@ static int dp_parser_parse(struct dp_parser >> *parser, int connector_type) >>       if (rc) >>           return rc; >> -    /* >> -     * External bridges are mandatory for eDP interfaces: one has to >> -     * provide at least an eDP panel (which gets wrapped into >> panel-bridge). >> -     * >> -     * For DisplayPort interfaces external bridges are optional, so >> -     * silently ignore an error if one is not present (-ENODEV). >> -     */ >> -    rc = dp_parser_find_next_bridge(parser); >> -    if (rc == -ENODEV) { >> -        if (connector_type == DRM_MODE_CONNECTOR_eDP) { >> -            DRM_ERROR("eDP: next bridge is not present\n"); >> -            return rc; >> -        } >> -    } else if (rc) { >> -        if (rc != -EPROBE_DEFER) >> -            DRM_ERROR("DP: error parsing next bridge: %d\n", rc); >> -        return rc; >> -    } >> - >>       /* Map the corresponding regulator information according to >>        * version. Currently, since we only have one supported platform, >>        * mapping the regulator directly. >> diff --git a/drivers/gpu/drm/msm/dp/dp_parser.h >> b/drivers/gpu/drm/msm/dp/dp_parser.h >> index d371bae..3a4d797 100644 >> --- a/drivers/gpu/drm/msm/dp/dp_parser.h >> +++ b/drivers/gpu/drm/msm/dp/dp_parser.h >> @@ -125,7 +125,7 @@ struct dp_parser { >>       u32 max_dp_lanes; >>       struct drm_bridge *next_bridge; >> -    int (*parse)(struct dp_parser *parser, int connector_type); >> +    int (*parse)(struct dp_parser *parser); >>   }; >>   /** >> @@ -141,4 +141,16 @@ struct dp_parser { >>    */ >>   struct dp_parser *dp_parser_get(struct platform_device *pdev); >> +/** >> + * dp_parser_find_next_bridge() - find an additional bridge to DP >> + * >> + * @parser: dp_parser data from client >> + * >> + * This function is used to find any additional bridge attached to >> + * the DP controller. The eDP interface requires a panel bridge. >> + * >> + * Return: 0 if able to get the bridge, otherwise negative errno for >> failure. >> + */ >> +int dp_parser_find_next_bridge(struct dp_parser *parser); >> + >>   #endif > >