Received: by 2002:a25:8b12:0:0:0:0:0 with SMTP id i18csp5235347ybl; Tue, 27 Aug 2019 01:17:00 -0700 (PDT) X-Google-Smtp-Source: APXvYqwkBKcaFUulz8wuBINFh8nf+mMMFM0RYhURs7TUO6QKQ3Ib1z7wJSeGaXGxdtfMoBK5MDpi X-Received: by 2002:a17:90a:18e:: with SMTP id 14mr23681743pjc.66.1566893820895; Tue, 27 Aug 2019 01:17:00 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1566893820; cv=none; d=google.com; s=arc-20160816; b=OiM9zAR2G9rkGhYmv1QqMIG7z77wat5udH0/lUVSdD8OXQlmwct60xkFo6dEjEySdt 3xvViGkze9ks8uH6hKhjFaMp0Kd5JhMQaTEelPN8IgXM6Rho5li1oxq3QWZvnb2JLKCd n16XMj76rdugg6Nd0p3+4KZ9XBNSWJX1b/meXPElWZ3Kk6jfitCsH/oXUgpuWtT/6PlV rnvcHilOmni0asYDXVc5VVWtHkBoemUeFUBjIXk4r6h84dq97xBs54c+bDp98mBCyamw aomXmPhKuBQUByefH5qXDj48Cwt8mQNuqxITQT7KEAXsDrvwR1Wqq3gn++wFhMhnUU+n xCJA== 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:message-id:date:subject:cc:to:from :dkim-signature; bh=D4e7iGHsnMCj/mwiWKJTzKLUGfzuq+WR9rmN1dnOPAE=; b=RrLm0m5tiPzr+F1BIoiG4wgx+tdYb26akrf+FW0WS2cFfekxMvbawaSmkVsL/lfizp mMjtWqdP8w6Khg47HzA3vzZ4uf9AjlBhxmTnwuGHmlwXbB00av2HWXDrtqs70/BcICww cnNHWkzF0rTo7dtlLd0LB74gbiXJ+fsz+HUGV+EGyCjtqkhlIm1m/d35qN1j7sGcWLfX 4JlNSa8IyVVt6JuHJBHoybQQVyLhxETcD+aAzXSFlUjsFTUAoNOTTOlevqlxwWwvyYa6 h39nSZF5gYk1mQ7gWEcF7Pw2sNtkaRok8KuldqirEfAGmNU+6f0/dbpVi2Z2YJ1Ms/r9 RAzQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@baylibre-com.20150623.gappssmtp.com header.s=20150623 header.b=KWmJrIF5; 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 az7si11951691plb.71.2019.08.27.01.16.45; Tue, 27 Aug 2019 01:17:00 -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 header.i=@baylibre-com.20150623.gappssmtp.com header.s=20150623 header.b=KWmJrIF5; 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 S1731498AbfH0IOy (ORCPT + 99 others); Tue, 27 Aug 2019 04:14:54 -0400 Received: from mail-wm1-f67.google.com ([209.85.128.67]:37155 "EHLO mail-wm1-f67.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1731132AbfH0IOd (ORCPT ); Tue, 27 Aug 2019 04:14:33 -0400 Received: by mail-wm1-f67.google.com with SMTP id d16so1993135wme.2 for ; Tue, 27 Aug 2019 01:14:31 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=baylibre-com.20150623.gappssmtp.com; s=20150623; h=from:to:cc:subject:date:message-id:in-reply-to:references :mime-version:content-transfer-encoding; bh=D4e7iGHsnMCj/mwiWKJTzKLUGfzuq+WR9rmN1dnOPAE=; b=KWmJrIF5DhACxVfxJuxqoiBamz86vy3kyWZvsEo2ZlNlEFVsiGIFodSwTWpLcX6Fdx 2VTUDNOipJSYOFR1VAQfxpxk9pGhaat/NPalWshlriMRzGHXauhgP9Wprs6gVW/OS2VB YfYrq+8zaQ8mwTufZOC0JrVP3i2RjQZRiBUa5ul3DdUIBra/NTzCZzalP9zNpuE1OJyq AKW5d5tq2dQlfXcLHn8DbO7oTNh840c3KKgo2moHwZqjCI6IuL5b05V5Aw/5x/LcNvLd 90U+TMMR/B61z9ZJZ/tp9q6Kt+dlKFBi9qZcHqZtjFxubtEPajNniiknzSiu6VsapRyJ 76jA== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:from:to:cc:subject:date:message-id:in-reply-to :references:mime-version:content-transfer-encoding; bh=D4e7iGHsnMCj/mwiWKJTzKLUGfzuq+WR9rmN1dnOPAE=; b=XyYQrPISJ24OeVcFg43MEejwASAKC0Je5PyJ7q9ODlQqPX54BjWcHqZ3nhL87MvIqG /+XPNz2YeTmpDiQDUhTrEnFx1ggHfkD58hgsnU2S7chYoV8VEK+Yy/4LNb1WpLHlCEdT S1W0yj+YpK/RBSOclasjDHGlf9bwxHEJTqQl4iw3qj55Nfnj3XIyjoRsqdiEZgm++xEd +mZechVQiDLmCD+wNMQgE3Pn/vw15DyMTyXD+iJHWIBJ9Hg3K61zMuYTbcd2Z0439lSp 9JCEhLi26br92nTGdQqB+Ix4hBetAOP8DUl0Q7x7w9u8bC0LdYshhEXs0kcFFl/K/utf Cx3Q== X-Gm-Message-State: APjAAAVbjqcrwqmUk1N4+1QZV79ol0rT0Zu14XcrzkgdUpgEPjCtbWfw gQMZFbIEX7F6ImKkh2bIAUzkyg== X-Received: by 2002:a1c:a514:: with SMTP id o20mr27565979wme.149.1566893670659; Tue, 27 Aug 2019 01:14:30 -0700 (PDT) Received: from bender.baylibre.local (lmontsouris-657-1-212-31.w90-63.abo.wanadoo.fr. [90.63.244.31]) by smtp.gmail.com with ESMTPSA id f18sm11911792wrx.85.2019.08.27.01.14.29 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Tue, 27 Aug 2019 01:14:30 -0700 (PDT) From: Neil Armstrong To: a.hajda@samsung.com, Laurent.pinchart@ideasonboard.com, jonas@kwiboo.se, jernej.skrabec@siol.net, boris.brezillon@collabora.com Cc: Neil Armstrong , linux-amlogic@lists.infradead.org, dri-devel@lists.freedesktop.org, linux-kernel@vger.kernel.org Subject: [PATCH RFC v2 3/8] drm/bridge: synopsys: dw-hdmi: add bus format negociation Date: Tue, 27 Aug 2019 10:14:20 +0200 Message-Id: <20190827081425.15011-4-narmstrong@baylibre.com> X-Mailer: git-send-email 2.22.0 In-Reply-To: <20190827081425.15011-1-narmstrong@baylibre.com> References: <20190827081425.15011-1-narmstrong@baylibre.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Add the atomic_get_output_bus_fmts, atomic_get_input_bus_fmts to negociate the possible output and input formats for the current mode and monitor, and use the negociated formats in a basic atomic_check callback. Signed-off-by: Neil Armstrong --- drivers/gpu/drm/bridge/synopsys/dw-hdmi.c | 245 +++++++++++++++++++++- 1 file changed, 241 insertions(+), 4 deletions(-) diff --git a/drivers/gpu/drm/bridge/synopsys/dw-hdmi.c b/drivers/gpu/drm/bridge/synopsys/dw-hdmi.c index bd65d0479683..00aacad51e29 100644 --- a/drivers/gpu/drm/bridge/synopsys/dw-hdmi.c +++ b/drivers/gpu/drm/bridge/synopsys/dw-hdmi.c @@ -1987,11 +1987,10 @@ static int dw_hdmi_setup(struct dw_hdmi *hdmi, struct drm_display_mode *mode) hdmi->hdmi_data.video_mode.mpixelrepetitionoutput = 0; hdmi->hdmi_data.video_mode.mpixelrepetitioninput = 0; - /* TOFIX: Get input format from plat data or fallback to RGB888 */ if (hdmi->plat_data->input_bus_format) hdmi->hdmi_data.enc_in_bus_format = hdmi->plat_data->input_bus_format; - else + else if (hdmi->hdmi_data.enc_in_bus_format == MEDIA_BUS_FMT_FIXED) hdmi->hdmi_data.enc_in_bus_format = MEDIA_BUS_FMT_RGB888_1X24; /* TOFIX: Get input encoding from plat data or fallback to none */ @@ -2001,8 +2000,8 @@ static int dw_hdmi_setup(struct dw_hdmi *hdmi, struct drm_display_mode *mode) else hdmi->hdmi_data.enc_in_encoding = V4L2_YCBCR_ENC_DEFAULT; - /* TOFIX: Default to RGB888 output format */ - hdmi->hdmi_data.enc_out_bus_format = MEDIA_BUS_FMT_RGB888_1X24; + if (hdmi->hdmi_data.enc_out_bus_format == MEDIA_BUS_FMT_FIXED) + hdmi->hdmi_data.enc_out_bus_format = MEDIA_BUS_FMT_RGB888_1X24; hdmi->hdmi_data.pix_repet_factor = 0; hdmi->hdmi_data.hdcp_enable = 0; @@ -2227,6 +2226,240 @@ static const struct drm_connector_helper_funcs dw_hdmi_connector_helper_funcs = .get_modes = dw_hdmi_connector_get_modes, }; +static void dw_hdmi_bridge_atomic_get_output_bus_fmts(struct drm_bridge *bridge, + struct drm_bridge_state *bridge_state, + struct drm_crtc_state *crtc_state, + struct drm_connector_state *conn_state, + unsigned int *num_output_fmts, + u32 *output_fmts) +{ + struct drm_connector *conn = conn_state->connector; + struct drm_display_info *info = &conn->display_info; + struct drm_display_mode *mode = &crtc_state->mode; + bool is_hdmi2_sink = info->hdmi.scdc.supported; + int i = 0; + + /* + * If the current mode enforces 4:2:0, force the output but format + * to 4:2:0 and do not add the YUV422/444/RGB formats + */ + if (drm_mode_is_420_only(info, mode) || + (!is_hdmi2_sink && drm_mode_is_420_also(info, mode))) { + + /* Order bus formats from 16bit to 8bit if supported */ + if (info->bpc == 16 && + (info->hdmi.y420_dc_modes & DRM_EDID_YCBCR420_DC_48)) { + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_UYYVYY16_0_5X48; + ++i; + } + + if (info->bpc >= 12 && + (info->hdmi.y420_dc_modes & DRM_EDID_YCBCR420_DC_36)) { + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_UYYVYY12_0_5X36; + ++i; + } + + if (info->bpc >= 10 && + (info->hdmi.y420_dc_modes & DRM_EDID_YCBCR420_DC_30)) { + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_UYYVYY10_0_5X30; + ++i; + } + + /* Default 8bit fallback */ + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_UYYVYY8_0_5X24; + ++i; + + *num_output_fmts = i; + + return; + } + + /* + * Order bus formats from 16bit to 8bit and from YUV422 to RGB + * if supported. In any case the default RGB888 format is added + */ + + if (info->bpc == 16) { + if (info->color_formats & DRM_COLOR_FORMAT_YCRCB444) { + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_YUV16_1X48; + ++i; + } + + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_RGB161616_1X48; + ++i; + } + + if (info->bpc >= 12) { + if (info->color_formats & DRM_COLOR_FORMAT_YCRCB422) { + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_UYVY12_1X24; + ++i; + } + + if (info->color_formats & DRM_COLOR_FORMAT_YCRCB444) { + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_YUV12_1X36; + ++i; + } + + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_RGB121212_1X36; + ++i; + } + + if (info->bpc >= 10) { + if (info->color_formats & DRM_COLOR_FORMAT_YCRCB422) { + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_UYVY10_1X20; + ++i; + } + + if (info->color_formats & DRM_COLOR_FORMAT_YCRCB444) { + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_YUV10_1X30; + ++i; + } + + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_RGB101010_1X30; + ++i; + } + + if (info->color_formats & DRM_COLOR_FORMAT_YCRCB422) { + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_UYVY8_1X16; + ++i; + } + + if (info->color_formats & DRM_COLOR_FORMAT_YCRCB444) { + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_YUV8_1X24; + ++i; + } + + /* Default 8bit RGB fallback */ + if (output_fmts) + output_fmts[i] = MEDIA_BUS_FMT_RGB888_1X24; + ++i; + + *num_output_fmts = i; +} + +static void dw_hdmi_bridge_atomic_get_input_bus_fmts(struct drm_bridge *bridge, + struct drm_bridge_state *bridge_state, + struct drm_crtc_state *crtc_state, + struct drm_connector_state *conn_state, + u32 output_fmt, + unsigned int *num_input_fmts, + u32 *input_fmts) +{ + int i = 0; + + switch (output_fmt) { + /* 8bit */ + case MEDIA_BUS_FMT_RGB888_1X24: + if (input_fmts) + input_fmts[i] = MEDIA_BUS_FMT_RGB888_1X24; + ++i; + /* Fallthrought */ + case MEDIA_BUS_FMT_YUV8_1X24: + if (input_fmts) + input_fmts[i] = MEDIA_BUS_FMT_YUV8_1X24; + ++i; + break; + case MEDIA_BUS_FMT_UYVY8_1X16: + /* Fallthrought */ + case MEDIA_BUS_FMT_UYYVYY8_0_5X24: + if (input_fmts) + input_fmts[i] = output_fmt; + ++i; + break; + + /* 10bit */ + case MEDIA_BUS_FMT_RGB101010_1X30: + if (input_fmts) + input_fmts[i] = output_fmt; + ++i; + /* Fallthrought */ + case MEDIA_BUS_FMT_YUV10_1X30: + if (input_fmts) + input_fmts[i] = MEDIA_BUS_FMT_YUV10_1X30; + ++i; + break; + case MEDIA_BUS_FMT_UYVY10_1X20: + /* Fallthrought */ + case MEDIA_BUS_FMT_UYYVYY10_0_5X30: + if (input_fmts) + input_fmts[i] = output_fmt; + ++i; + break; + + /* 12bit */ + case MEDIA_BUS_FMT_RGB121212_1X36: + if (input_fmts) + input_fmts[i] = output_fmt; + ++i; + /* Fallthrought */ + case MEDIA_BUS_FMT_YUV12_1X36: + if (input_fmts) + input_fmts[i] = MEDIA_BUS_FMT_YUV12_1X36; + ++i; + break; + case MEDIA_BUS_FMT_UYVY12_1X24: + /* Fallthrought */ + case MEDIA_BUS_FMT_UYYVYY12_0_5X36: + if (input_fmts) + input_fmts[i] = output_fmt; + ++i; + break; + + /* 16bit */ + case MEDIA_BUS_FMT_RGB161616_1X48: + if (input_fmts) + input_fmts[i] = output_fmt; + ++i; + /* Fallthrought */ + case MEDIA_BUS_FMT_YUV16_1X48: + if (input_fmts) + input_fmts[i] = MEDIA_BUS_FMT_YUV16_1X48; + ++i; + break; + case MEDIA_BUS_FMT_UYYVYY16_0_5X48: + if (input_fmts) + input_fmts[i] = output_fmt; + ++i; + break; + } + + *num_input_fmts = i; +} + +static int dw_hdmi_bridge_atomic_check(struct drm_bridge *bridge, + struct drm_bridge_state *bridge_state, + struct drm_crtc_state *crtc_state, + struct drm_connector_state *conn_state) +{ + struct dw_hdmi *hdmi = bridge->driver_private; + + dev_dbg(hdmi->dev, "selected output format %x\n", + bridge_state->output_bus_cfg.fmt); + + hdmi->hdmi_data.enc_out_bus_format = bridge_state->output_bus_cfg.fmt; + + dev_dbg(hdmi->dev, "selected input format %x\n", + bridge_state->input_bus_cfg.fmt); + + hdmi->hdmi_data.enc_in_bus_format = bridge_state->input_bus_cfg.fmt; + + return 0; +} + static int dw_hdmi_bridge_attach(struct drm_bridge *bridge) { struct dw_hdmi *hdmi = bridge->driver_private; @@ -2327,6 +2560,9 @@ static void dw_hdmi_bridge_enable(struct drm_bridge *bridge) static const struct drm_bridge_funcs dw_hdmi_bridge_funcs = { .attach = dw_hdmi_bridge_attach, .detach = dw_hdmi_bridge_detach, + .atomic_check = dw_hdmi_bridge_atomic_check, + .atomic_get_output_bus_fmts = dw_hdmi_bridge_atomic_get_output_bus_fmts, + .atomic_get_input_bus_fmts = dw_hdmi_bridge_atomic_get_input_bus_fmts, .enable = dw_hdmi_bridge_enable, .disable = dw_hdmi_bridge_disable, .mode_set = dw_hdmi_bridge_mode_set, @@ -2790,6 +3026,7 @@ __dw_hdmi_probe(struct platform_device *pdev, hdmi->bridge.driver_private = hdmi; hdmi->bridge.funcs = &dw_hdmi_bridge_funcs; + #ifdef CONFIG_OF hdmi->bridge.of_node = pdev->dev.of_node; #endif -- 2.22.0