Received: by 10.213.65.68 with SMTP id h4csp630981imn; Fri, 6 Apr 2018 06:27:41 -0700 (PDT) X-Google-Smtp-Source: AIpwx4/lzrUb+sm6zaoOSmf6E5QpmuAXGopFQsxxfe17P1timpEqJPRav0Cd4Q4zQ443t3+tUNzH X-Received: by 2002:a17:902:28e3:: with SMTP id f90-v6mr27398426plb.250.1523021261910; Fri, 06 Apr 2018 06:27:41 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1523021261; cv=none; d=google.com; s=arc-20160816; b=Sxg5C8mRyt2hcx1ACdqakChDA/3ecKBxpPAuw/Zxoow6uVB6SO2cuyVsdHW1FhERCd Fa8HeS9ll5XblGOBUGv/jUnMTP3LRLVk26cEWSpAlMtQOURy6io8Pge/4EpYLE9oDv37 kPeDdBV2w1LsKcji9aXRDVZKSnSVgHO1SjHZmQSG0BHusV1eoXU7SP8n4svbrMqIdHON FbAbNgdLZZgr1c+/sXswGFRcd06pezWXiOCnj6cRCNWRbVSIBzOBa4eHtVGUvoCHmTHW PXq9wZwPYLGlQuaabxZtj+Iqm067Zfu3HezFLHKpSEj+FdAndaPwuUiyhZM42CVJXoEX xmjA== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :references:in-reply-to:organization:message-id:date:subject:cc:to :from:dkim-signature:arc-authentication-results; bh=qbZrIDbVLpWf5wPQJ8HWrSMYp+Zt63cUYkwFajbMe7U=; b=lFs+W5gQpqc999z29tLXyNryXSxNfrc24ARGIj2Go0ajTN8XhP2HsLUe3hD0NJvkWl 7DcptX02krwxv0jGWfbsJmBsE9Kzxq1IzjLwMVCRcrlqwwfllYczgFmcLj3zr/Mssc16 5mHLa/Veqw242S9dLPkdCCUV6dKsUgf0+neL8V2h92w+0L7tlSepuZqVEevHd/NKpTmu uUO2m58odS2CqRF4gUk6/XX3bZLtpPxRxpXSKqL0bLMUUw7QCjYxMSwv8oVkjBkrYfER OJ+vFrpeLauVpjgGqCZGPo/si1rVa/rqqnrkX75v4f/WDVnxUKHBIE1WEscx9M4oZTuM s72A== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass (test mode) header.i=@ideasonboard.com header.s=mail header.b=UGzFfF0C; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id w9si7997954pfk.14.2018.04.06.06.27.27; Fri, 06 Apr 2018 06:27:41 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass (test mode) header.i=@ideasonboard.com header.s=mail header.b=UGzFfF0C; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1753089AbeDFNYu (ORCPT + 99 others); Fri, 6 Apr 2018 09:24:50 -0400 Received: from galahad.ideasonboard.com ([185.26.127.97]:34925 "EHLO galahad.ideasonboard.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1752517AbeDFNYr (ORCPT ); Fri, 6 Apr 2018 09:24:47 -0400 Received: from avalon.localnet (dfj612ybrt5fhg77mgycy-3.rev.dnainternet.fi [IPv6:2001:14ba:21f5:5b00:2e86:4862:ef6a:2804]) by galahad.ideasonboard.com (Postfix) with ESMTPSA id 7D97920124; Fri, 6 Apr 2018 15:22:12 +0200 (CEST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/simple; d=ideasonboard.com; s=mail; t=1523020933; bh=GgtHYdsCdvlyvY5JScCOxwGXlhqyNYQ9enU++sSlbwE=; h=From:To:Cc:Subject:Date:In-Reply-To:References:From; b=UGzFfF0CfiMd+CftgEBQ+Srg92vIePy92n57NFgrpcHL49k2Pft1JAivZTYuTM/3d VsS8PcZ/zIkHOTTP7dQjjV86ZYcxMRpOpGPnS0nOg7TH3y8asBSh96xJyub2sc7ee6 rC9KHvLfIetzVPYRroF6yQVBSTtDd5DDfH8O1yfk= From: Laurent Pinchart To: Jacopo Mondi Cc: architt@codeaurora.org, a.hajda@samsung.com, airlied@linux.ie, vladimir_zapolskiy@mentor.com, horms@verge.net.au, magnus.damm@gmail.com, geert@linux-m68k.org, niklas.soderlund@ragnatech.se, sergei.shtylyov@cogentembedded.com, robh+dt@kernel.org, mark.rutland@arm.com, dri-devel@lists.freedesktop.org, linux-renesas-soc@vger.kernel.org, devicetree@vger.kernel.org, linux-kernel@vger.kernel.org Subject: Re: [PATCH v7 2/2] drm: bridge: Add thc63lvd1024 LVDS decoder driver Date: Fri, 06 Apr 2018 16:24:44 +0300 Message-ID: <3367141.XOMI5ldQnc@avalon> Organization: Ideas on Board Oy In-Reply-To: <1523018517-24121-3-git-send-email-jacopo+renesas@jmondi.org> References: <1523018517-24121-1-git-send-email-jacopo+renesas@jmondi.org> <1523018517-24121-3-git-send-email-jacopo+renesas@jmondi.org> MIME-Version: 1.0 Content-Transfer-Encoding: quoted-printable Content-Type: text/plain; charset="iso-8859-1" Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Hi Jacopo, Thank you for the patch. On Friday, 6 April 2018 15:41:57 EEST Jacopo Mondi wrote: > Add DRM bridge driver for Thine THC63LVD1024 LVDS to digital parallel > output converter. >=20 > Signed-off-by: Jacopo Mondi > Reviewed-by: Andrzej Hajda > Reviewed-by: Niklas S=F6derlund > --- > drivers/gpu/drm/bridge/Kconfig | 6 + > drivers/gpu/drm/bridge/Makefile | 1 + > drivers/gpu/drm/bridge/thc63lvd1024.c | 212 ++++++++++++++++++++++++++++= +++ > 3 files changed, 219 insertions(+) > create mode 100644 drivers/gpu/drm/bridge/thc63lvd1024.c >=20 > diff --git a/drivers/gpu/drm/bridge/Kconfig b/drivers/gpu/drm/bridge/Kcon= fig > index 3aa65bd..42c9c2d 100644 > --- a/drivers/gpu/drm/bridge/Kconfig > +++ b/drivers/gpu/drm/bridge/Kconfig > @@ -93,6 +93,12 @@ config DRM_SII9234 > It is an I2C driver, that detects connection of MHL bridge > and starts encapsulation of HDMI signal. >=20 > +config DRM_THINE_THC63LVD1024 > + tristate "Thine THC63LVD1024 LVDS decoder bridge" > + depends on OF > + ---help--- > + Thine THC63LVD1024 LVDS/parallel converter driver. > + > config DRM_TOSHIBA_TC358767 > tristate "Toshiba TC358767 eDP bridge" > depends on OF > diff --git a/drivers/gpu/drm/bridge/Makefile > b/drivers/gpu/drm/bridge/Makefile index 373eb28..fd90b16 100644 > --- a/drivers/gpu/drm/bridge/Makefile > +++ b/drivers/gpu/drm/bridge/Makefile > @@ -8,6 +8,7 @@ obj-$(CONFIG_DRM_PARADE_PS8622) +=3D parade-ps8622.o > obj-$(CONFIG_DRM_SIL_SII8620) +=3D sil-sii8620.o > obj-$(CONFIG_DRM_SII902X) +=3D sii902x.o > obj-$(CONFIG_DRM_SII9234) +=3D sii9234.o > +obj-$(CONFIG_DRM_THINE_THC63LVD1024) +=3D thc63lvd1024.o > obj-$(CONFIG_DRM_TOSHIBA_TC358767) +=3D tc358767.o > obj-$(CONFIG_DRM_ANALOGIX_DP) +=3D analogix/ > obj-$(CONFIG_DRM_I2C_ADV7511) +=3D adv7511/ > diff --git a/drivers/gpu/drm/bridge/thc63lvd1024.c > b/drivers/gpu/drm/bridge/thc63lvd1024.c new file mode 100644 > index 0000000..c8fad9c > --- /dev/null > +++ b/drivers/gpu/drm/bridge/thc63lvd1024.c > @@ -0,0 +1,212 @@ > +// SPDX-License-Identifier: GPL-2.0 > +/* > + * THC63LVD1024 LVDS to parallel data DRM bridge driver. > + * > + * Copyright (C) 2018 Jacopo Mondi > + */ > + > +#include > +#include > +#include > + > +#include > +#include > +#include You should include slab.h as you're using devm_kzalloc(). > + > +enum thc63_ports { > + THC63_LVDS_IN0, > + THC63_LVDS_IN1, > + THC63_RGB_OUT0, > + THC63_RGB_OUT1, > +}; > + > +struct thc63_dev { > + struct device *dev; > + > + struct regulator *vcc; > + > + struct gpio_desc *pdwn; pwdn ? > + struct gpio_desc *oe; > + > + struct drm_bridge bridge; > + struct drm_bridge *next; > +}; > + > +static inline struct thc63_dev *to_thc63(struct drm_bridge *bridge) > +{ > + return container_of(bridge, struct thc63_dev, bridge); > +} > + > +static int thc63_attach(struct drm_bridge *bridge) > +{ > + struct thc63_dev *thc63 =3D to_thc63(bridge); > + > + return drm_bridge_attach(bridge->encoder, thc63->next, bridge); > +} > + > +static void thc63_enable(struct drm_bridge *bridge) > +{ > + struct thc63_dev *thc63 =3D to_thc63(bridge); > + int ret; > + > + ret =3D regulator_enable(thc63->vcc); > + if (ret) { > + dev_err(thc63->dev, > + "Failed to enable regulator \"vcc\": %d\n", ret); > + return; > + } > + > + if (thc63->pdwn) > + gpiod_set_value(thc63->pdwn, 0); > + > + if (thc63->oe) > + gpiod_set_value(thc63->oe, 1); You don't need to check the gpio_desc pointers first, gpiod_set_value() is = a=20 no-op if the pointer is NULL. > +} > + > +static void thc63_disable(struct drm_bridge *bridge) > +{ > + struct thc63_dev *thc63 =3D to_thc63(bridge); > + int ret; > + > + if (thc63->oe) > + gpiod_set_value(thc63->oe, 0); > + > + if (thc63->pdwn) > + gpiod_set_value(thc63->pdwn, 1); Same here. > + ret =3D regulator_disable(thc63->vcc); > + if (ret) > + dev_err(thc63->dev, > + "Failed to disable regulator \"vcc\": %d\n", ret); > +} > + > +static const struct drm_bridge_funcs thc63_bridge_func =3D { > + .attach =3D thc63_attach, > + .enable =3D thc63_enable, > + .disable =3D thc63_disable, > +}; > + > +static int thc63_parse_dt(struct thc63_dev *thc63) > +{ > + struct device_node *thc63_out; > + struct device_node *remote; > + > + thc63_out =3D of_graph_get_endpoint_by_regs(thc63->dev->of_node, > + THC63_RGB_OUT0, -1); > + if (!thc63_out) { > + dev_err(thc63->dev, "Missing endpoint in port@%u\n", > + THC63_RGB_OUT0); > + return -ENODEV; > + } > + > + remote =3D of_graph_get_remote_port_parent(thc63_out); > + of_node_put(thc63_out); > + if (!remote) { > + dev_err(thc63->dev, "Endpoint in port@%u unconnected\n", > + THC63_RGB_OUT0); > + return -ENODEV; > + } > + > + if (!of_device_is_available(remote)) { > + dev_err(thc63->dev, "port@%u remote endpoint is disabled\n", > + THC63_RGB_OUT0); > + of_node_put(remote); > + return -ENODEV; > + } > + > + thc63->next =3D of_drm_find_bridge(remote); > + if (!thc63->next) { > + of_node_put(remote); > + return -EPROBE_DEFER; > + } > + > + return 0; > +} > + > +static int thc63_gpio_init(struct thc63_dev *thc63) > +{ > + thc63->oe =3D devm_gpiod_get_optional(thc63->dev, "enable", GPIOD_OUT_L= OW); If you agree with my comment regarding the DT bindings, this should be "oe". > + if (IS_ERR(thc63->oe)) { > + dev_err(thc63->dev, "Unable to get \"enable-gpios\": %ld\n", > + PTR_ERR(thc63->oe)); And the message should be update here too. With those small issues fixed, Reviewed-by: Laurent Pinchart > + return PTR_ERR(thc63->oe); > + } > + > + thc63->pdwn =3D devm_gpiod_get_optional(thc63->dev, "powerdown", > + GPIOD_OUT_HIGH); > + if (IS_ERR(thc63->pdwn)) { > + dev_err(thc63->dev, "Unable to get \"powerdown-gpios\": %ld\n", > + PTR_ERR(thc63->pdwn)); > + return PTR_ERR(thc63->pdwn); > + } > + > + return 0; > +} > + > +static int thc63_probe(struct platform_device *pdev) > +{ > + struct thc63_dev *thc63; > + int ret; > + > + thc63 =3D devm_kzalloc(&pdev->dev, sizeof(*thc63), GFP_KERNEL); > + if (!thc63) > + return -ENOMEM; > + > + thc63->dev =3D &pdev->dev; > + platform_set_drvdata(pdev, thc63); > + > + thc63->vcc =3D devm_regulator_get(thc63->dev, "vcc"); > + if (IS_ERR(thc63->vcc)) { > + if (PTR_ERR(thc63->vcc) =3D=3D -EPROBE_DEFER) > + return -EPROBE_DEFER; > + > + dev_err(thc63->dev, "Unable to get \"vcc\" supply: %ld\n", > + PTR_ERR(thc63->vcc)); > + return PTR_ERR(thc63->vcc); > + } > + > + ret =3D thc63_gpio_init(thc63); > + if (ret) > + return ret; > + > + ret =3D thc63_parse_dt(thc63); > + if (ret) > + return ret; > + > + thc63->bridge.driver_private =3D thc63; > + thc63->bridge.of_node =3D pdev->dev.of_node; > + thc63->bridge.funcs =3D &thc63_bridge_func; > + > + drm_bridge_add(&thc63->bridge); > + > + return 0; > +} > + > +static int thc63_remove(struct platform_device *pdev) > +{ > + struct thc63_dev *thc63 =3D platform_get_drvdata(pdev); > + > + drm_bridge_remove(&thc63->bridge); > + > + return 0; > +} > + > +static const struct of_device_id thc63_match[] =3D { > + { .compatible =3D "thine,thc63lvd1024", }, > + { }, > +}; > +MODULE_DEVICE_TABLE(of, thc63_match); > + > +static struct platform_driver thc63_driver =3D { > + .probe =3D thc63_probe, > + .remove =3D thc63_remove, > + .driver =3D { > + .name =3D "thc63lvd1024", > + .of_match_table =3D thc63_match, > + }, > +}; > +module_platform_driver(thc63_driver); > + > +MODULE_AUTHOR("Jacopo Mondi "); > +MODULE_DESCRIPTION("Thine THC63LVD1024 LVDS decoder DRM bridge driver"); > +MODULE_LICENSE("GPL v2"); =2D-=20 Regards, Laurent Pinchart