Received: by 2002:a25:8b91:0:0:0:0:0 with SMTP id j17csp1457638ybl; Thu, 5 Dec 2019 01:35:20 -0800 (PST) X-Google-Smtp-Source: APXvYqx5UBPyG+KExdPI5+LsomE2qfzq8bsURV0PnkQL0GSfyOO2CJjck416G9G7SpXlC0FEvHBy X-Received: by 2002:a9d:6395:: with SMTP id w21mr5956807otk.3.1575538520754; Thu, 05 Dec 2019 01:35:20 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1575538520; cv=none; d=google.com; s=arc-20160816; b=JRUJlKECtIgbGlZHXQBa22AfLvirjXYalPAAd4jBhdezi38/iysSBR1tdJ9aj8KKLe e+L49GmWP/ngpCFYwyqfFiqZv/YbUdXkoBMZn0ya3+McuuLJ4DdyGvrJlmt37vSm+Bbp 4qp7aXbxqbnJhHAcmnv3FotRyV37M0qJGUzBWLngHPPp7uB8s31E1NrdO5EQwhUaSd1b 1mYLYpi3MS9BpI7dpcvXlCQT6pid90H6D+apBvA+u/gWbjxqG2M1npXPZDkFeKkw1QdR m9RX2o9V5AYsOtKSTzWwRTFn5dOykyd2UQYw7rpsf7bcB4V4Fi//sC21Qr4UnwcwP9in aIzQ== 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:cc:to:from :subject:references:mime-version:message-id:in-reply-to:date :dkim-signature; bh=UDw98xQ3l8jb/G4I/C4jLU3wltugH4Xugs74OzkFeik=; b=n1MrXmgXAt7RQjzkd1UPghV8T/VL8SzSMwS47xAwmjRqVPyFZgZ1+kDE8woJohoN0K qlFB7kjXH7paTqCxLy21vrBFj1zGTzOE23x4/eCn6Nw9zkcugVIaZcvpre+bBWD+nF1v xFoQklrLTiPr9ee9XhfG92+PfMMwCUQP14kk/bjclpQ4uu1OcxsAr+VGcTxerNMHQlJS tsF8p1lQch4fJvSoGbVdgKweKAjp7Wrr8r0Wp2BlZggc2Q8D2/+GOdfdel4Q+mcyf+y2 DP+vw5jozzNOEEZXhdZSUebA/XGv/HoaxP6PJJRd83dVlDvBlufygN3/AxgnIEnIR4iR cu3Q== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@google.com header.s=20161025 header.b=V41nfIyY; 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; dmarc=pass (p=REJECT sp=REJECT dis=NONE) header.from=google.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id x137si77959oif.42.2019.12.05.01.35.07; Thu, 05 Dec 2019 01:35:20 -0800 (PST) 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=@google.com header.s=20161025 header.b=V41nfIyY; 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; dmarc=pass (p=REJECT sp=REJECT dis=NONE) header.from=google.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1729126AbfLEJeC (ORCPT + 99 others); Thu, 5 Dec 2019 04:34:02 -0500 Received: from mail-pj1-f73.google.com ([209.85.216.73]:45730 "EHLO mail-pj1-f73.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1726096AbfLEJeB (ORCPT ); Thu, 5 Dec 2019 04:34:01 -0500 Received: by mail-pj1-f73.google.com with SMTP id t7so1469197pjg.12 for ; Thu, 05 Dec 2019 01:34:00 -0800 (PST) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=20161025; h=date:in-reply-to:message-id:mime-version:references:subject:from:to :cc:content-transfer-encoding; bh=UDw98xQ3l8jb/G4I/C4jLU3wltugH4Xugs74OzkFeik=; b=V41nfIyY4WmVZX/575il83HExfBAxo/Q3JYTo3utJdOEXmiDj3Xwhdwi/MDsL2fZfV VVhz5KV7V/IAn5WoaZ9IDVMNYTGGYcNLX1vVWOBb7LIYoYuZc2HiHy56ayFwSFaTkK1Z Gn/AYVreeDlqJ6v9IGPFzh3Zh//N/NNDeIWUHY/jqmnAoH94loJaQAOeycLfh3sKM2Jt iIrDEqvX6rhG5/8ey78/D5wl2RSAyXfc+gFqnVBWkBF7idf/NyZw0As1ITrVu/WI9Krq aFXwgludRWLWqu/ZCs1aXgDb9wQcpBLnirBkxDV1VIjyRGPlAeOWvnv8+H/2vA2IVH5z 77ZA== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:date:in-reply-to:message-id:mime-version :references:subject:from:to:cc:content-transfer-encoding; bh=UDw98xQ3l8jb/G4I/C4jLU3wltugH4Xugs74OzkFeik=; b=QUmUzp8txWWys2CjlsS0xwbZSGcKgcZ5gR2pI2gdE7pNRdOZvTioCi3pwmX0OW6O6d ii7TCHGGqKm+QU2uW5XaBWxnvh0oWhMFtEfknos6ARfrFra6QMKvcgryrjI0iRzghWAk 8gK8+bQKeD1bbocOgFZ+WVWbSJWwr2LTlfmLeBbvGMldgizGwxMVBehPsbaVWz6QqTMf x7eYlCkkKoQ00cFdvExhFvCdFCeZ2T9iBCdf10lmv0GBszt2LzuVOIyaYPry8C532PjI JgIPYXrl+feEvBrB6Z+OqQ0RUOuWANDw0l1Zjdsdn/EOS2lM3QPyFvuajbEGkBAkMgpK 32IA== X-Gm-Message-State: APjAAAX1tqcxQIy9v723MSQ9OMn7ZTCBNwkVSXXNZzfZSYbvifyuIKam xjqcuzIv1gg3Re6cMO/YKs0fHhumu5x8 X-Received: by 2002:a63:ed4a:: with SMTP id m10mr7430508pgk.99.1575538440322; Thu, 05 Dec 2019 01:34:00 -0800 (PST) Date: Thu, 5 Dec 2019 01:33:46 -0800 In-Reply-To: <20191205093346.57930-1-rajatja@google.com> Message-Id: <20191205093346.57930-3-rajatja@google.com> Mime-Version: 1.0 References: <20191205093346.57930-1-rajatja@google.com> X-Mailer: git-send-email 2.24.0.393.g34dc348eaf-goog Subject: [PATCH v3 3/3] drm/i915: Add support for integrated privacy screens From: Rajat Jain To: Maarten Lankhorst , Maxime Ripard , Sean Paul , David Airlie , Daniel Vetter , Jani Nikula , Joonas Lahtinen , Rodrigo Vivi , "=?UTF-8?q?Ville=20Syrj=C3=A4l=C3=A4?=" , Chris Wilson , Imre Deak , "=?UTF-8?q?Jos=C3=A9=20Roberto=20de=20Souza?=" , linux-kernel@vger.kernel.org, dri-devel@lists.freedesktop.org, intel-gfx@lists.freedesktop.org, gregkh@linuxfoundation.org, mathewk@google.com, Daniel Thompson , Jonathan Corbet , Pavel Machek , seanpaul@google.com, Duncan Laurie , jsbarnes@google.com, Thierry Reding Cc: Rajat Jain , rajatxjain@gmail.com Content-Type: text/plain; charset="UTF-8" Content-Transfer-Encoding: quoted-printable Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org Certain laptops now come with panels that have integrated privacy screens on them. This patch adds support for such panels by adding a privacy-screen property to the intel_connector for the panel, that the userspace can then use to control and check the status. Identifying the presence of privacy screen, and controlling it, is done via ACPI _DSM methods. Currently, this is done only for the Intel display ports. But in future, this can be done for any other ports if the hardware becomes available (e.g. external monitors supporting integrated privacy screens?). Signed-off-by: Rajat Jain --- v3: * Change license to GPL-2.0 OR MIT * Move privacy screen enum from UAPI to intel_display_types.h * Rename parameter name and some other minor changes. v2: Formed by splitting the original patch into multiple patches. - All code has been moved into i915 now. - Privacy screen is a i915 property - Have a local state variable to store the prvacy screen. Don't read it from hardware. drivers/gpu/drm/i915/Makefile | 3 +- drivers/gpu/drm/i915/display/intel_atomic.c | 13 +++- .../gpu/drm/i915/display/intel_connector.c | 35 +++++++++ .../gpu/drm/i915/display/intel_connector.h | 1 + .../drm/i915/display/intel_display_types.h | 18 +++++ drivers/gpu/drm/i915/display/intel_dp.c | 6 ++ .../drm/i915/display/intel_privacy_screen.c | 72 +++++++++++++++++++ .../drm/i915/display/intel_privacy_screen.h | 25 +++++++ 8 files changed, 169 insertions(+), 4 deletions(-) create mode 100644 drivers/gpu/drm/i915/display/intel_privacy_screen.c create mode 100644 drivers/gpu/drm/i915/display/intel_privacy_screen.h diff --git a/drivers/gpu/drm/i915/Makefile b/drivers/gpu/drm/i915/Makefile index 90dcf09f52cc..f7067c8f0407 100644 --- a/drivers/gpu/drm/i915/Makefile +++ b/drivers/gpu/drm/i915/Makefile @@ -197,7 +197,8 @@ i915-y +=3D \ display/intel_vga.o i915-$(CONFIG_ACPI) +=3D \ display/intel_acpi.o \ - display/intel_opregion.o + display/intel_opregion.o \ + display/intel_privacy_screen.o i915-$(CONFIG_DRM_FBDEV_EMULATION) +=3D \ display/intel_fbdev.o =20 diff --git a/drivers/gpu/drm/i915/display/intel_atomic.c b/drivers/gpu/drm/= i915/display/intel_atomic.c index c2875b10adf9..c73b81c4c3f6 100644 --- a/drivers/gpu/drm/i915/display/intel_atomic.c +++ b/drivers/gpu/drm/i915/display/intel_atomic.c @@ -37,6 +37,7 @@ #include "intel_atomic.h" #include "intel_display_types.h" #include "intel_hdcp.h" +#include "intel_privacy_screen.h" #include "intel_sprite.h" =20 /** @@ -57,11 +58,14 @@ int intel_digital_connector_atomic_get_property(struct = drm_connector *connector, struct drm_i915_private *dev_priv =3D to_i915(dev); struct intel_digital_connector_state *intel_conn_state =3D to_intel_digital_connector_state(state); + struct intel_connector *intel_connector =3D to_intel_connector(connector)= ; =20 if (property =3D=3D dev_priv->force_audio_property) *val =3D intel_conn_state->force_audio; else if (property =3D=3D dev_priv->broadcast_rgb_property) *val =3D intel_conn_state->broadcast_rgb; + else if (property =3D=3D intel_connector->privacy_screen_property) + *val =3D intel_conn_state->privacy_screen_status; else { DRM_DEBUG_ATOMIC("Unknown property [PROP:%d:%s]\n", property->base.id, property->name); @@ -89,15 +93,18 @@ int intel_digital_connector_atomic_set_property(struct = drm_connector *connector, struct drm_i915_private *dev_priv =3D to_i915(dev); struct intel_digital_connector_state *intel_conn_state =3D to_intel_digital_connector_state(state); + struct intel_connector *intel_connector =3D to_intel_connector(connector)= ; =20 if (property =3D=3D dev_priv->force_audio_property) { intel_conn_state->force_audio =3D val; return 0; - } - - if (property =3D=3D dev_priv->broadcast_rgb_property) { + } else if (property =3D=3D dev_priv->broadcast_rgb_property) { intel_conn_state->broadcast_rgb =3D val; return 0; + } else if (property =3D=3D intel_connector->privacy_screen_property) { + intel_privacy_screen_set_val(intel_connector, val); + intel_conn_state->privacy_screen_status =3D val; + return 0; } =20 DRM_DEBUG_ATOMIC("Unknown property [PROP:%d:%s]\n", diff --git a/drivers/gpu/drm/i915/display/intel_connector.c b/drivers/gpu/d= rm/i915/display/intel_connector.c index 1133c4e97bb4..f3e041c737de 100644 --- a/drivers/gpu/drm/i915/display/intel_connector.c +++ b/drivers/gpu/drm/i915/display/intel_connector.c @@ -296,3 +296,38 @@ intel_attach_colorspace_property(struct drm_connector = *connector) drm_object_attach_property(&connector->base, connector->colorspace_property, 0); } + +static const struct drm_prop_enum_list privacy_screen_enum[] =3D { + { PRIVACY_SCREEN_DISABLED, "Disabled" }, + { PRIVACY_SCREEN_ENABLED, "Enabled" }, +}; + +/** + * intel_attach_privacy_screen_property - + * create and attach the connecter's privacy-screen property. * + * @connector: connector for which to init the privacy-screen property + * + * This function creates and attaches the "privacy-screen" property to the + * connector. Initial state of privacy-screen is set to disabled. + */ +void +intel_attach_privacy_screen_property(struct drm_connector *connector) +{ + struct intel_connector *intel_connector =3D to_intel_connector(connector)= ; + struct drm_property *prop; + + if (!intel_connector->privacy_screen_property) { + prop =3D drm_property_create_enum(connector->dev, + DRM_MODE_PROP_ENUM, + "privacy-screen", + privacy_screen_enum, + ARRAY_SIZE(privacy_screen_enum)); + if (!prop) + return; + + intel_connector->privacy_screen_property =3D prop; + } + + drm_object_attach_property(&connector->base, prop, + PRIVACY_SCREEN_DISABLED); +} diff --git a/drivers/gpu/drm/i915/display/intel_connector.h b/drivers/gpu/d= rm/i915/display/intel_connector.h index 93a7375c8196..61005f37a338 100644 --- a/drivers/gpu/drm/i915/display/intel_connector.h +++ b/drivers/gpu/drm/i915/display/intel_connector.h @@ -31,5 +31,6 @@ void intel_attach_force_audio_property(struct drm_connect= or *connector); void intel_attach_broadcast_rgb_property(struct drm_connector *connector); void intel_attach_aspect_ratio_property(struct drm_connector *connector); void intel_attach_colorspace_property(struct drm_connector *connector); +void intel_attach_privacy_screen_property(struct drm_connector *connector)= ; =20 #endif /* __INTEL_CONNECTOR_H__ */ diff --git a/drivers/gpu/drm/i915/display/intel_display_types.h b/drivers/g= pu/drm/i915/display/intel_display_types.h index 0a4a04116091..a0addd2c5376 100644 --- a/drivers/gpu/drm/i915/display/intel_display_types.h +++ b/drivers/gpu/drm/i915/display/intel_display_types.h @@ -433,6 +433,23 @@ struct intel_connector { struct work_struct modeset_retry_work; =20 struct intel_hdcp hdcp; + + /* Optional "privacy-screen" property for the connector panel */ + struct drm_property *privacy_screen_property; +}; + +/** + * enum intel_privacy_screen_status - privacy_screen status + * + * This enum is used to track and control the state of the integrated priv= acy + * screen present on some display panels, via the "privacy-screen" propert= y. + * + * @PRIVACY_SCREEN_DISABLED: The privacy-screen on the panel is disabled + * @PRIVACY_SCREEN_ENABLED: The privacy-screen on the panel is enabled + **/ +enum intel_privacy_screen_status { + PRIVACY_SCREEN_DISABLED =3D 0, + PRIVACY_SCREEN_ENABLED =3D 1, }; =20 struct intel_digital_connector_state { @@ -440,6 +457,7 @@ struct intel_digital_connector_state { =20 enum hdmi_force_audio force_audio; int broadcast_rgb; + enum intel_privacy_screen_status privacy_screen_status; }; =20 #define to_intel_digital_connector_state(x) container_of(x, struct intel_d= igital_connector_state, base) diff --git a/drivers/gpu/drm/i915/display/intel_dp.c b/drivers/gpu/drm/i915= /display/intel_dp.c index 6b209193cbbb..79e5a4dade1d 100644 --- a/drivers/gpu/drm/i915/display/intel_dp.c +++ b/drivers/gpu/drm/i915/display/intel_dp.c @@ -62,6 +62,7 @@ #include "intel_lspcon.h" #include "intel_lvds.h" #include "intel_panel.h" +#include "intel_privacy_screen.h" #include "intel_psr.h" #include "intel_sideband.h" #include "intel_tc.h" @@ -6601,6 +6602,7 @@ intel_dp_add_properties(struct intel_dp *intel_dp, st= ruct drm_connector *connect { struct drm_i915_private *dev_priv =3D to_i915(connector->dev); enum port port =3D dp_to_dig_port(intel_dp)->base.port; + struct intel_connector *intel_connector =3D to_intel_connector(connector)= ; =20 if (!IS_G4X(dev_priv) && port !=3D PORT_A) intel_attach_force_audio_property(connector); @@ -6631,6 +6633,10 @@ intel_dp_add_properties(struct intel_dp *intel_dp, s= truct drm_connector *connect =20 /* Lookup the ACPI node corresponding to the connector */ intel_acpi_device_id_update(dev_priv); + + /* Check for integrated Privacy screen support */ + if (intel_privacy_screen_present(intel_connector)) + intel_attach_privacy_screen_property(connector); } } =20 diff --git a/drivers/gpu/drm/i915/display/intel_privacy_screen.c b/drivers/= gpu/drm/i915/display/intel_privacy_screen.c new file mode 100644 index 000000000000..c8a5b64f94fb --- /dev/null +++ b/drivers/gpu/drm/i915/display/intel_privacy_screen.c @@ -0,0 +1,72 @@ +// SPDX-License-Identifier: GPL-2.0 OR MIT +/* + * Intel ACPI privacy screen code + * + * Copyright =C2=A9 2019 Google Inc. + */ + +#include + +#include "intel_privacy_screen.h" + +#define CONNECTOR_DSM_REVID 1 + +#define CONNECTOR_DSM_FN_PRIVACY_ENABLE 2 +#define CONNECTOR_DSM_FN_PRIVACY_DISABLE 3 + +static const guid_t drm_conn_dsm_guid =3D + GUID_INIT(0xC7033113, 0x8720, 0x4CEB, + 0x90, 0x90, 0x9D, 0x52, 0xB3, 0xE5, 0x2D, 0x73); + +/* Makes _DSM call to set privacy screen status */ +static void acpi_privacy_screen_call_dsm(acpi_handle conn_handle, u64 func= ) +{ + union acpi_object *obj; + + obj =3D acpi_evaluate_dsm(conn_handle, &drm_conn_dsm_guid, + CONNECTOR_DSM_REVID, func, NULL); + if (!obj) { + DRM_DEBUG_DRIVER("failed to evaluate _DSM for fn %llx\n", func); + return; + } + + ACPI_FREE(obj); +} + +void intel_privacy_screen_set_val(struct intel_connector *connector, + enum intel_privacy_screen_status val) +{ + acpi_handle acpi_handle =3D connector->acpi_handle; + + if (!acpi_handle) + return; + + if (val =3D=3D PRIVACY_SCREEN_DISABLED) + acpi_privacy_screen_call_dsm(acpi_handle, + CONNECTOR_DSM_FN_PRIVACY_DISABLE); + else if (val =3D=3D PRIVACY_SCREEN_ENABLED) + acpi_privacy_screen_call_dsm(acpi_handle, + CONNECTOR_DSM_FN_PRIVACY_ENABLE); + else + DRM_WARN("%s: Cannot set privacy screen to invalid val %u\n", + dev_name(connector->base.dev->dev), val); +} + +bool intel_privacy_screen_present(struct intel_connector *connector) +{ + acpi_handle handle =3D connector->acpi_handle; + + if (!handle) + return false; + + if (!acpi_check_dsm(handle, &drm_conn_dsm_guid, + CONNECTOR_DSM_REVID, + 1 << CONNECTOR_DSM_FN_PRIVACY_ENABLE | + 1 << CONNECTOR_DSM_FN_PRIVACY_DISABLE)) { + DRM_WARN("%s: Odd, connector ACPI node but no privacy scrn?\n", + dev_name(connector->base.dev->dev)); + return false; + } + DRM_DEV_INFO(connector->base.dev->dev, "supports privacy screen\n"); + return true; +} diff --git a/drivers/gpu/drm/i915/display/intel_privacy_screen.h b/drivers/= gpu/drm/i915/display/intel_privacy_screen.h new file mode 100644 index 000000000000..6ada234a2161 --- /dev/null +++ b/drivers/gpu/drm/i915/display/intel_privacy_screen.h @@ -0,0 +1,25 @@ +/* SPDX-License-Identifier: GPL-2.0 OR MIT */ +/* + * Copyright =C2=A9 2019 Google Inc. + */ + +#ifndef __DRM_PRIVACY_SCREEN_H__ +#define __DRM_PRIVACY_SCREEN_H__ + +#include "intel_display_types.h" + +#ifdef CONFIG_ACPI +bool intel_privacy_screen_present(struct intel_connector *connector); +void intel_privacy_screen_set_val(struct intel_connector *connector, + enum intel_privacy_screen_status val); +#else +bool intel_privacy_screen_present(struct intel_connector *connector); +{ + return false; +} +void intel_privacy_screen_set_val(struct intel_connector *connector, + enum intel_privacy_screen_status val) +{ } +#endif /* CONFIG_ACPI */ + +#endif /* __DRM_PRIVACY_SCREEN_H__ */ --=20 2.24.0.393.g34dc348eaf-goog