From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from gabe.freedesktop.org (gabe.freedesktop.org [131.252.210.177]) (using TLSv1.2 with cipher ECDHE-RSA-AES256-GCM-SHA384 (256/256 bits)) (No client certificate requested) by smtp.lore.kernel.org (Postfix) with ESMTPS id E1A07C00140 for ; Mon, 8 Aug 2022 15:18:37 +0000 (UTC) Received: from gabe.freedesktop.org (localhost [127.0.0.1]) by gabe.freedesktop.org (Postfix) with ESMTP id B68B1112595; Mon, 8 Aug 2022 15:18:03 +0000 (UTC) Received: from mail-wr1-x432.google.com (mail-wr1-x432.google.com [IPv6:2a00:1450:4864:20::432]) by gabe.freedesktop.org (Postfix) with ESMTPS id 8C3032B225 for ; Mon, 8 Aug 2022 15:15:33 +0000 (UTC) Received: by mail-wr1-x432.google.com with SMTP id l22so11271832wrz.7 for ; Mon, 08 Aug 2022 08:15:33 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=gmail.com; s=20210112; h=mime-version:references:in-reply-to:message-id:cc:to:subject:from :date:from:to:cc; bh=aX5eN81HWESmJRv88W0AI5OJIWxgo2+CRoZwYXMKtCU=; b=Vr8BTyxqf7qQirOBiaFgxwMwXWYUu6kCtfYbMorBmgoIgZNEa/5G3nIby8Tpd+q8Vs 4ywVnUUiQRlSTa19nLHwf+zzuIYnlSVyCL7lD0napvmtuwxhqSLkJX9rg7+vG5o7I9al 8LNsTBc4nc8q/LGEOeqWBN42XpS9aE8ZkroOE7h5V81CSmR0IK2Alt9hcVZ8r+n1Tm/M CoQ7pY7Fz83+gaOdq2GXxtwpYksNK7/mwwaOVX/VQtKuuKbhB2MO4pr2o0uPdaxjUdwC bxiIT3a9mzNYs8ZojcPW2TMlO/w37khvO6LmMNmo1/ccsE15gjIy88jIb5KpjTMNUbeW KGMg== X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20210112; h=mime-version:references:in-reply-to:message-id:cc:to:subject:from :date:x-gm-message-state:from:to:cc; bh=aX5eN81HWESmJRv88W0AI5OJIWxgo2+CRoZwYXMKtCU=; b=oHBIasTPgXnmm1M7Mj1MSazZxDpttT6cwYJ2ZCA8eXv9Y5UOgnI9ugX/YqniZ2qDmX 2idxt27B9S1Vj2jzgn9PBE3HZK4eeKZV8JFzq5OMTcD+6sflJq9yv4J5m+PtjjgFUYUA VMq2L1Cc1Si+qvSWLKtm2QJyM1vuqXYQ88AlfBWBd4H7Gh9aGPLMoRKFJ3vtnNazYWto jSDriti0m+aRicnKmhxiNNVKDSG5NL8bjfq0lD8m8qs9upVLJ9K6tdnBCKGPxNpNPGF1 2lOgMP70ecr4kMiaLs78ImSgi+AlSneHASXr1X8Scj8mgk73RX9q0jarsxXLlkxGT6LK tE5Q== X-Gm-Message-State: ACgBeo3ZR9ubFPgBYJX8OnpJZLeJ9lchldem285xaI9ETVDEKQaZI3Ki vVmOygVLkWP0/txnMHaZOWo= X-Google-Smtp-Source: AA6agR6koSiS4NDnWWMI/JyRAsMcpmCzaZWXxVzQjc9ZAPjykYm54Z/3WZDM7o2o+zcmpDEaIS+pUA== X-Received: by 2002:adf:e111:0:b0:21d:665e:2fa5 with SMTP id t17-20020adfe111000000b0021d665e2fa5mr12226602wrz.652.1659971731902; Mon, 08 Aug 2022 08:15:31 -0700 (PDT) Received: from [192.168.0.181] (82-64-129-2.subs.proxad.net. [82.64.129.2]) by smtp.gmail.com with ESMTPSA id u7-20020a05600c210700b003a3561d4f3fsm2810035wml.43.2022.08.08.08.15.31 (version=TLS1_3 cipher=TLS_AES_256_GCM_SHA384 bits=256/256); Mon, 08 Aug 2022 08:15:31 -0700 (PDT) Date: Mon, 08 Aug 2022 17:15:25 +0200 From: Matthieu CHARETTE Subject: Re: [PATCH] drm: Fix EDID firmware load on resume To: Jani Nikula Message-Id: In-Reply-To: <87wnbqen2f.fsf@intel.com> References: <202207172035.mtErdlaw-lkp@intel.com> <20220727074152.43059-1-matthieu.charette@gmail.com> <87wnbqen2f.fsf@intel.com> X-Mailer: geary/40.0 MIME-Version: 1.0 Content-Type: multipart/alternative; boundary="=-bxxlzlNWaqsntO0fg8Im" X-BeenThere: dri-devel@lists.freedesktop.org X-Mailman-Version: 2.1.29 Precedence: list List-Id: Direct Rendering Infrastructure - Development List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , Cc: kbuild-all@lists.01.org, lkp@intel.com, airlied@linux.ie, linux-kernel@vger.kernel.org, dri-devel@lists.freedesktop.org, tzimmermann@suse.de, andrealmeid@igalia.com Errors-To: dri-devel-bounces@lists.freedesktop.org Sender: "dri-devel" --=-bxxlzlNWaqsntO0fg8Im Content-Type: text/plain; charset=us-ascii; format=flowed Sorry, What do you mean? Matthieu On Tue, Aug 2 2022 at 05:29:12 PM +0300, Jani Nikula wrote: > On Wed, 27 Jul 2022, Matthieu CHARETTE > wrote: >> Loading an EDID using drm.edid_firmware parameter makes resume to >> fail >> after firmware cache is being cleaned. This is because edid_load() >> use a >> temporary device to request the firmware. This cause the EDID >> firmware >> not to be cached from suspend. And, requesting the EDID firmware >> return >> an error during resume. >> So the request_firmware() call should use a permanent device for >> each >> connector. Also, we should cache the EDID even if no monitor is >> connected, in case it's plugged while suspended. > > AFAICT this breaks changing drm.edid_firmware runtime. > > BR, > Jani. > >> >> Link: >> Signed-off-by: Matthieu CHARETTE > > >> --- >> drivers/gpu/drm/drm_connector.c | 9 ++++ >> drivers/gpu/drm/drm_edid_load.c | 81 >> ++++++++++++++++++++++++++++----- >> include/drm/drm_connector.h | 12 +++++ >> include/drm/drm_edid.h | 3 ++ >> 4 files changed, 94 insertions(+), 11 deletions(-) >> >> diff --git a/drivers/gpu/drm/drm_connector.c >> b/drivers/gpu/drm/drm_connector.c >> index 1c48d162c77e..e8819ebf1c4b 100644 >> --- a/drivers/gpu/drm/drm_connector.c >> +++ b/drivers/gpu/drm/drm_connector.c >> @@ -31,6 +31,7 @@ >> #include >> #include >> >> +#include >> #include >> >> #include "drm_crtc_internal.h" >> @@ -289,6 +290,9 @@ int drm_connector_init(struct drm_device *dev, >> >> drm_connector_get_cmdline_mode(connector); >> >> + connector->edid_load_pdev = NULL; >> + drm_cache_edid_firmware(connector); >> + >> /* We should add connectors at the end to avoid upsetting the >> connector >> * index too much. >> */ >> @@ -473,6 +477,11 @@ void drm_connector_cleanup(struct >> drm_connector *connector) >> connector->tile_group = NULL; >> } >> >> + if (connector->edid_load_pdev) { >> + platform_device_unregister(connector->edid_load_pdev); >> + connector->edid_load_pdev = NULL; >> + } >> + >> list_for_each_entry_safe(mode, t, &connector->probed_modes, head) >> drm_mode_remove(connector, mode); >> >> diff --git a/drivers/gpu/drm/drm_edid_load.c >> b/drivers/gpu/drm/drm_edid_load.c >> index 37d8ba3ddb46..5a82be9917ec 100644 >> --- a/drivers/gpu/drm/drm_edid_load.c >> +++ b/drivers/gpu/drm/drm_edid_load.c >> @@ -167,6 +167,19 @@ static int edid_size(const u8 *edid, int >> data_size) >> return (edid[0x7e] + 1) * EDID_LENGTH; >> } >> >> +static struct platform_device *edid_pdev(const char >> *connector_name) >> +{ >> + struct platform_device *pdev = >> platform_device_register_simple(connector_name, -1, NULL, 0); >> + >> + if (IS_ERR(pdev)) { >> + DRM_ERROR("Failed to register EDID firmware platform device " >> + "for connector \"%s\"\n", connector_name); >> + return ERR_CAST(pdev); >> + } >> + >> + return pdev; >> +} >> + >> static void *edid_load(struct drm_connector *connector, const char >> *name, >> const char *connector_name) >> { >> @@ -182,18 +195,17 @@ static void *edid_load(struct drm_connector >> *connector, const char *name, >> fwdata = generic_edid[builtin]; >> fwsize = sizeof(generic_edid[builtin]); >> } else { >> - struct platform_device *pdev; >> + struct platform_device *pdev = connector->edid_load_pdev; >> int err; >> >> - pdev = platform_device_register_simple(connector_name, -1, NULL, >> 0); >> - if (IS_ERR(pdev)) { >> - DRM_ERROR("Failed to register EDID firmware platform device " >> - "for connector \"%s\"\n", connector_name); >> - return ERR_CAST(pdev); >> + if (WARN_ON(!pdev)) { >> + pdev = edid_pdev(connector_name); >> + if (IS_ERR(pdev)) >> + return ERR_CAST(pdev); >> + connector->edid_load_pdev = pdev; >> } >> >> err = request_firmware(&fw, name, &pdev->dev); >> - platform_device_unregister(pdev); >> if (err) { >> DRM_ERROR("Requesting EDID firmware \"%s\" failed (err=%d)\n", >> name, err); >> @@ -263,11 +275,9 @@ static void *edid_load(struct drm_connector >> *connector, const char *name, >> return edid; >> } >> >> -struct edid *drm_load_edid_firmware(struct drm_connector >> *connector) >> +static char *edid_name(const char *connector_name) >> { >> - const char *connector_name = connector->name; >> char *edidname, *last, *colon, *fwstr, *edidstr, *fallback = NULL; >> - struct edid *edid; >> >> if (edid_firmware[0] == '\0') >> return ERR_PTR(-ENOENT); >> @@ -310,8 +320,57 @@ struct edid *drm_load_edid_firmware(struct >> drm_connector *connector) >> if (*last == '\n') >> *last = '\0'; >> >> - edid = edid_load(connector, edidname, connector_name); >> + edidname = kstrdup(edidname, GFP_KERNEL); >> + if (!edidname) { >> + kfree(fwstr); >> + return ERR_PTR(-ENOMEM); >> + } >> + >> kfree(fwstr); >> + return edidname; >> +} >> + >> +void drm_cache_edid_firmware(struct drm_connector *connector) >> +{ >> + const char *connector_name = connector->name; >> + const char *edidname = edid_name(connector_name); >> + struct platform_device *pdev; >> + int err; >> + >> + if (IS_ERR(edidname)) >> + return; >> + >> + if (match_string(generic_edid_name, GENERIC_EDIDS, edidname) >= >> 0) { >> + kfree(edidname); >> + return; >> + } >> + >> + pdev = edid_pdev(connector_name); >> + if (IS_ERR(pdev)) { >> + kfree(edidname); >> + return; >> + } >> + connector->edid_load_pdev = pdev; >> + >> + err = firmware_request_cache(&pdev->dev, edidname); >> + if (err) >> + DRM_ERROR("Requesting EDID firmware cache \"%s\" failed >> (err=%d)\n", >> + edidname, err); >> + >> + kfree(edidname); >> +} >> + >> +struct edid *drm_load_edid_firmware(struct drm_connector >> *connector) >> +{ >> + const char *connector_name = connector->name; >> + const char *edidname = edid_name(connector_name); >> + struct edid *edid; >> + >> + if (IS_ERR(edidname)) >> + return ERR_CAST(edidname); >> + >> + edid = edid_load(connector, edidname, connector_name); >> + kfree(edidname); >> >> return edid; >> } >> diff --git a/include/drm/drm_connector.h >> b/include/drm/drm_connector.h >> index 3ac4bf87f257..47c84741517e 100644 >> --- a/include/drm/drm_connector.h >> +++ b/include/drm/drm_connector.h >> @@ -1573,6 +1573,18 @@ struct drm_connector { >> */ >> struct i2c_adapter *ddc; >> >> + /** >> + * @edid_load_pdev: Platform device for loading EDID via firmware. >> + * >> + * The platform device is registered in drm_connector_init() in >> case a >> + * custom EDID firmware is used with `edid_firmware` parameter. >> Otherwise, >> + * it is set to NULL. >> + * >> + * Platform device is unregistered in drm_connector_cleanup() if >> it >> + * is not NULL. >> + */ >> + struct platform_device *edid_load_pdev; >> + >> /** >> * @null_edid_counter: track sinks that give us all zeros for the >> EDID. >> * Needed to workaround some HW bugs where we get all 0s >> diff --git a/include/drm/drm_edid.h b/include/drm/drm_edid.h >> index b2756753370b..e907c928a35d 100644 >> --- a/include/drm/drm_edid.h >> +++ b/include/drm/drm_edid.h >> @@ -378,10 +378,13 @@ int drm_av_sync_delay(struct drm_connector >> *connector, >> const struct drm_display_mode *mode); >> >> #ifdef CONFIG_DRM_LOAD_EDID_FIRMWARE >> +void drm_cache_edid_firmware(struct drm_connector *connector); >> struct edid *drm_load_edid_firmware(struct drm_connector >> *connector); >> int __drm_set_edid_firmware_path(const char *path); >> int __drm_get_edid_firmware_path(char *buf, size_t bufsize); >> #else >> +inline void >> +drm_cache_edid_firmware(struct drm_connector *connector); >> static inline struct edid * >> drm_load_edid_firmware(struct drm_connector *connector) >> { > > -- > Jani Nikula, Intel Open Source Graphics Center --=-bxxlzlNWaqsntO0fg8Im Content-Type: text/html; charset=us-ascii
Sorry, What do you mean?

Matthieu

On Tue, Aug 2 2022 at 05:29:12 PM +0300, Jani Nikula <jani.nikula@linux.intel.com> wrote:
On Wed, 27 Jul 2022, Matthieu CHARETTE <matthieu.charette@gmail.com> wrote:
Loading an EDID using drm.edid_firmware parameter makes resume to fail after firmware cache is being cleaned. This is because edid_load() use a temporary device to request the firmware. This cause the EDID firmware not to be cached from suspend. And, requesting the EDID firmware return an error during resume. So the request_firmware() call should use a permanent device for each connector. Also, we should cache the EDID even if no monitor is connected, in case it's plugged while suspended.
AFAICT this breaks changing drm.edid_firmware runtime. BR, Jani.
Link: https://gitlab.freedesktop.org/drm/amd/-/issues/2061 Signed-off-by: Matthieu CHARETTE <matthieu.charette@gmail.com> --- drivers/gpu/drm/drm_connector.c | 9 ++++ drivers/gpu/drm/drm_edid_load.c | 81 ++++++++++++++++++++++++++++----- include/drm/drm_connector.h | 12 +++++ include/drm/drm_edid.h | 3 ++ 4 files changed, 94 insertions(+), 11 deletions(-) diff --git a/drivers/gpu/drm/drm_connector.c b/drivers/gpu/drm/drm_connector.c index 1c48d162c77e..e8819ebf1c4b 100644 --- a/drivers/gpu/drm/drm_connector.c +++ b/drivers/gpu/drm/drm_connector.c @@ -31,6 +31,7 @@ #include <drm/drm_privacy_screen_consumer.h> #include <drm/drm_sysfs.h> +#include <linux/platform_device.h> #include <linux/uaccess.h> #include "drm_crtc_internal.h" @@ -289,6 +290,9 @@ int drm_connector_init(struct drm_device *dev, drm_connector_get_cmdline_mode(connector); + connector->edid_load_pdev = NULL; + drm_cache_edid_firmware(connector); + /* We should add connectors at the end to avoid upsetting the connector * index too much. */ @@ -473,6 +477,11 @@ void drm_connector_cleanup(struct drm_connector *connector) connector->tile_group = NULL; } + if (connector->edid_load_pdev) { + platform_device_unregister(connector->edid_load_pdev); + connector->edid_load_pdev = NULL; + } + list_for_each_entry_safe(mode, t, &connector->probed_modes, head) drm_mode_remove(connector, mode); diff --git a/drivers/gpu/drm/drm_edid_load.c b/drivers/gpu/drm/drm_edid_load.c index 37d8ba3ddb46..5a82be9917ec 100644 --- a/drivers/gpu/drm/drm_edid_load.c +++ b/drivers/gpu/drm/drm_edid_load.c @@ -167,6 +167,19 @@ static int edid_size(const u8 *edid, int data_size) return (edid[0x7e] + 1) * EDID_LENGTH; } +static struct platform_device *edid_pdev(const char *connector_name) +{ + struct platform_device *pdev = platform_device_register_simple(connector_name, -1, NULL, 0); + + if (IS_ERR(pdev)) { + DRM_ERROR("Failed to register EDID firmware platform device " + "for connector \"%s\"\n", connector_name); + return ERR_CAST(pdev); + } + + return pdev; +} + static void *edid_load(struct drm_connector *connector, const char *name, const char *connector_name) { @@ -182,18 +195,17 @@ static void *edid_load(struct drm_connector *connector, const char *name, fwdata = generic_edid[builtin]; fwsize = sizeof(generic_edid[builtin]); } else { - struct platform_device *pdev; + struct platform_device *pdev = connector->edid_load_pdev; int err; - pdev = platform_device_register_simple(connector_name, -1, NULL, 0); - if (IS_ERR(pdev)) { - DRM_ERROR("Failed to register EDID firmware platform device " - "for connector \"%s\"\n", connector_name); - return ERR_CAST(pdev); + if (WARN_ON(!pdev)) { + pdev = edid_pdev(connector_name); + if (IS_ERR(pdev)) + return ERR_CAST(pdev); + connector->edid_load_pdev = pdev; } err = request_firmware(&fw, name, &pdev->dev); - platform_device_unregister(pdev); if (err) { DRM_ERROR("Requesting EDID firmware \"%s\" failed (err=%d)\n", name, err); @@ -263,11 +275,9 @@ static void *edid_load(struct drm_connector *connector, const char *name, return edid; } -struct edid *drm_load_edid_firmware(struct drm_connector *connector) +static char *edid_name(const char *connector_name) { - const char *connector_name = connector->name; char *edidname, *last, *colon, *fwstr, *edidstr, *fallback = NULL; - struct edid *edid; if (edid_firmware[0] == '\0') return ERR_PTR(-ENOENT); @@ -310,8 +320,57 @@ struct edid *drm_load_edid_firmware(struct drm_connector *connector) if (*last == '\n') *last = '\0'; - edid = edid_load(connector, edidname, connector_name); + edidname = kstrdup(edidname, GFP_KERNEL); + if (!edidname) { + kfree(fwstr); + return ERR_PTR(-ENOMEM); + } + kfree(fwstr); + return edidname; +} + +void drm_cache_edid_firmware(struct drm_connector *connector) +{ + const char *connector_name = connector->name; + const char *edidname = edid_name(connector_name); + struct platform_device *pdev; + int err; + + if (IS_ERR(edidname)) + return; + + if (match_string(generic_edid_name, GENERIC_EDIDS, edidname) >= 0) { + kfree(edidname); + return; + } + + pdev = edid_pdev(connector_name); + if (IS_ERR(pdev)) { + kfree(edidname); + return; + } + connector->edid_load_pdev = pdev; + + err = firmware_request_cache(&pdev->dev, edidname); + if (err) + DRM_ERROR("Requesting EDID firmware cache \"%s\" failed (err=%d)\n", + edidname, err); + + kfree(edidname); +} + +struct edid *drm_load_edid_firmware(struct drm_connector *connector) +{ + const char *connector_name = connector->name; + const char *edidname = edid_name(connector_name); + struct edid *edid; + + if (IS_ERR(edidname)) + return ERR_CAST(edidname); + + edid = edid_load(connector, edidname, connector_name); + kfree(edidname); return edid; } diff --git a/include/drm/drm_connector.h b/include/drm/drm_connector.h index 3ac4bf87f257..47c84741517e 100644 --- a/include/drm/drm_connector.h +++ b/include/drm/drm_connector.h @@ -1573,6 +1573,18 @@ struct drm_connector { */ struct i2c_adapter *ddc; + /** + * @edid_load_pdev: Platform device for loading EDID via firmware. + * + * The platform device is registered in drm_connector_init() in case a + * custom EDID firmware is used with `edid_firmware` parameter. Otherwise, + * it is set to NULL. + * + * Platform device is unregistered in drm_connector_cleanup() if it + * is not NULL. + */ + struct platform_device *edid_load_pdev; + /** * @null_edid_counter: track sinks that give us all zeros for the EDID. * Needed to workaround some HW bugs where we get all 0s diff --git a/include/drm/drm_edid.h b/include/drm/drm_edid.h index b2756753370b..e907c928a35d 100644 --- a/include/drm/drm_edid.h +++ b/include/drm/drm_edid.h @@ -378,10 +378,13 @@ int drm_av_sync_delay(struct drm_connector *connector, const struct drm_display_mode *mode); #ifdef CONFIG_DRM_LOAD_EDID_FIRMWARE +void drm_cache_edid_firmware(struct drm_connector *connector); struct edid *drm_load_edid_firmware(struct drm_connector *connector); int __drm_set_edid_firmware_path(const char *path); int __drm_get_edid_firmware_path(char *buf, size_t bufsize); #else +inline void +drm_cache_edid_firmware(struct drm_connector *connector); static inline struct edid * drm_load_edid_firmware(struct drm_connector *connector) {
--
Jani Nikula, Intel Open Source Graphics Center
--=-bxxlzlNWaqsntO0fg8Im-- From mboxrd@z Thu Jan 1 00:00:00 1970 Content-Type: multipart/mixed; boundary="===============7393658385179695874==" MIME-Version: 1.0 From: Matthieu CHARETTE To: kbuild-all@lists.01.org Subject: Re: [PATCH] drm: Fix EDID firmware load on resume Date: Mon, 08 Aug 2022 17:15:25 +0200 Message-ID: In-Reply-To: <87wnbqen2f.fsf@intel.com> List-Id: --===============7393658385179695874== Content-Type: text/plain; charset="utf-8" MIME-Version: 1.0 Content-Transfer-Encoding: quoted-printable Sorry, What do you mean? Matthieu On Tue, Aug 2 2022 at 05:29:12 PM +0300, Jani Nikula = wrote: > On Wed, 27 Jul 2022, Matthieu CHARETTE > wrote: >> Loading an EDID using drm.edid_firmware parameter makes resume to = >> fail >> after firmware cache is being cleaned. This is because edid_load() = >> use a >> temporary device to request the firmware. This cause the EDID = >> firmware >> not to be cached from suspend. And, requesting the EDID firmware = >> return >> an error during resume. >> So the request_firmware() call should use a permanent device for = >> each >> connector. Also, we should cache the EDID even if no monitor is >> connected, in case it's plugged while suspended. > = > AFAICT this breaks changing drm.edid_firmware runtime. > = > BR, > Jani. > = >> = >> Link: >> Signed-off-by: Matthieu CHARETTE > > >> --- >> drivers/gpu/drm/drm_connector.c | 9 ++++ >> drivers/gpu/drm/drm_edid_load.c | 81 = >> ++++++++++++++++++++++++++++----- >> include/drm/drm_connector.h | 12 +++++ >> include/drm/drm_edid.h | 3 ++ >> 4 files changed, 94 insertions(+), 11 deletions(-) >> = >> diff --git a/drivers/gpu/drm/drm_connector.c = >> b/drivers/gpu/drm/drm_connector.c >> index 1c48d162c77e..e8819ebf1c4b 100644 >> --- a/drivers/gpu/drm/drm_connector.c >> +++ b/drivers/gpu/drm/drm_connector.c >> @@ -31,6 +31,7 @@ >> #include >> #include >> = >> +#include >> #include >> = >> #include "drm_crtc_internal.h" >> @@ -289,6 +290,9 @@ int drm_connector_init(struct drm_device *dev, >> = >> drm_connector_get_cmdline_mode(connector); >> = >> + connector->edid_load_pdev =3D NULL; >> + drm_cache_edid_firmware(connector); >> + >> /* We should add connectors at the end to avoid upsetting the = >> connector >> * index too much. >> */ >> @@ -473,6 +477,11 @@ void drm_connector_cleanup(struct = >> drm_connector *connector) >> connector->tile_group =3D NULL; >> } >> = >> + if (connector->edid_load_pdev) { >> + platform_device_unregister(connector->edid_load_pdev); >> + connector->edid_load_pdev =3D NULL; >> + } >> + >> list_for_each_entry_safe(mode, t, &connector->probed_modes, head) >> drm_mode_remove(connector, mode); >> = >> diff --git a/drivers/gpu/drm/drm_edid_load.c = >> b/drivers/gpu/drm/drm_edid_load.c >> index 37d8ba3ddb46..5a82be9917ec 100644 >> --- a/drivers/gpu/drm/drm_edid_load.c >> +++ b/drivers/gpu/drm/drm_edid_load.c >> @@ -167,6 +167,19 @@ static int edid_size(const u8 *edid, int = >> data_size) >> return (edid[0x7e] + 1) * EDID_LENGTH; >> } >> = >> +static struct platform_device *edid_pdev(const char = >> *connector_name) >> +{ >> + struct platform_device *pdev =3D = >> platform_device_register_simple(connector_name, -1, NULL, 0); >> + >> + if (IS_ERR(pdev)) { >> + DRM_ERROR("Failed to register EDID firmware platform device " >> + "for connector \"%s\"\n", connector_name); >> + return ERR_CAST(pdev); >> + } >> + >> + return pdev; >> +} >> + >> static void *edid_load(struct drm_connector *connector, const char = >> *name, >> const char *connector_name) >> { >> @@ -182,18 +195,17 @@ static void *edid_load(struct drm_connector = >> *connector, const char *name, >> fwdata =3D generic_edid[builtin]; >> fwsize =3D sizeof(generic_edid[builtin]); >> } else { >> - struct platform_device *pdev; >> + struct platform_device *pdev =3D connector->edid_load_pdev; >> int err; >> = >> - pdev =3D platform_device_register_simple(connector_name, -1, NULL, = >> 0); >> - if (IS_ERR(pdev)) { >> - DRM_ERROR("Failed to register EDID firmware platform device " >> - "for connector \"%s\"\n", connector_name); >> - return ERR_CAST(pdev); >> + if (WARN_ON(!pdev)) { >> + pdev =3D edid_pdev(connector_name); >> + if (IS_ERR(pdev)) >> + return ERR_CAST(pdev); >> + connector->edid_load_pdev =3D pdev; >> } >> = >> err =3D request_firmware(&fw, name, &pdev->dev); >> - platform_device_unregister(pdev); >> if (err) { >> DRM_ERROR("Requesting EDID firmware \"%s\" failed (err=3D%d)\n", >> name, err); >> @@ -263,11 +275,9 @@ static void *edid_load(struct drm_connector = >> *connector, const char *name, >> return edid; >> } >> = >> -struct edid *drm_load_edid_firmware(struct drm_connector = >> *connector) >> +static char *edid_name(const char *connector_name) >> { >> - const char *connector_name =3D connector->name; >> char *edidname, *last, *colon, *fwstr, *edidstr, *fallback =3D NULL; >> - struct edid *edid; >> = >> if (edid_firmware[0] =3D=3D '\0') >> return ERR_PTR(-ENOENT); >> @@ -310,8 +320,57 @@ struct edid *drm_load_edid_firmware(struct = >> drm_connector *connector) >> if (*last =3D=3D '\n') >> *last =3D '\0'; >> = >> - edid =3D edid_load(connector, edidname, connector_name); >> + edidname =3D kstrdup(edidname, GFP_KERNEL); >> + if (!edidname) { >> + kfree(fwstr); >> + return ERR_PTR(-ENOMEM); >> + } >> + >> kfree(fwstr); >> + return edidname; >> +} >> + >> +void drm_cache_edid_firmware(struct drm_connector *connector) >> +{ >> + const char *connector_name =3D connector->name; >> + const char *edidname =3D edid_name(connector_name); >> + struct platform_device *pdev; >> + int err; >> + >> + if (IS_ERR(edidname)) >> + return; >> + >> + if (match_string(generic_edid_name, GENERIC_EDIDS, edidname) >=3D = >> 0) { >> + kfree(edidname); >> + return; >> + } >> + >> + pdev =3D edid_pdev(connector_name); >> + if (IS_ERR(pdev)) { >> + kfree(edidname); >> + return; >> + } >> + connector->edid_load_pdev =3D pdev; >> + >> + err =3D firmware_request_cache(&pdev->dev, edidname); >> + if (err) >> + DRM_ERROR("Requesting EDID firmware cache \"%s\" failed = >> (err=3D%d)\n", >> + edidname, err); >> + >> + kfree(edidname); >> +} >> + >> +struct edid *drm_load_edid_firmware(struct drm_connector = >> *connector) >> +{ >> + const char *connector_name =3D connector->name; >> + const char *edidname =3D edid_name(connector_name); >> + struct edid *edid; >> + >> + if (IS_ERR(edidname)) >> + return ERR_CAST(edidname); >> + >> + edid =3D edid_load(connector, edidname, connector_name); >> + kfree(edidname); >> = >> return edid; >> } >> diff --git a/include/drm/drm_connector.h = >> b/include/drm/drm_connector.h >> index 3ac4bf87f257..47c84741517e 100644 >> --- a/include/drm/drm_connector.h >> +++ b/include/drm/drm_connector.h >> @@ -1573,6 +1573,18 @@ struct drm_connector { >> */ >> struct i2c_adapter *ddc; >> = >> + /** >> + * @edid_load_pdev: Platform device for loading EDID via firmware. >> + * >> + * The platform device is registered in drm_connector_init() in = >> case a >> + * custom EDID firmware is used with `edid_firmware` parameter. = >> Otherwise, >> + * it is set to NULL. >> + * >> + * Platform device is unregistered in drm_connector_cleanup() if = >> it >> + * is not NULL. >> + */ >> + struct platform_device *edid_load_pdev; >> + >> /** >> * @null_edid_counter: track sinks that give us all zeros for the = >> EDID. >> * Needed to workaround some HW bugs where we get all 0s >> diff --git a/include/drm/drm_edid.h b/include/drm/drm_edid.h >> index b2756753370b..e907c928a35d 100644 >> --- a/include/drm/drm_edid.h >> +++ b/include/drm/drm_edid.h >> @@ -378,10 +378,13 @@ int drm_av_sync_delay(struct drm_connector = >> *connector, >> const struct drm_display_mode *mode); >> = >> #ifdef CONFIG_DRM_LOAD_EDID_FIRMWARE >> +void drm_cache_edid_firmware(struct drm_connector *connector); >> struct edid *drm_load_edid_firmware(struct drm_connector = >> *connector); >> int __drm_set_edid_firmware_path(const char *path); >> int __drm_get_edid_firmware_path(char *buf, size_t bufsize); >> #else >> +inline void >> +drm_cache_edid_firmware(struct drm_connector *connector); >> static inline struct edid * >> drm_load_edid_firmware(struct drm_connector *connector) >> { > = > -- > Jani Nikula, Intel Open Source Graphics Center --===============7393658385179695874== Content-Type: text/html MIME-Version: 1.0 Content-Transfer-Encoding: base64 Content-Disposition: attachment; filename="attachment.htm" PGRpdiBpZD0iZ2VhcnktYm9keSIgZGlyPSJhdXRvIj48ZGl2PlNvcnJ5LCBXaGF0IGRvIHlvdSBt ZWFuPzwvZGl2PjxkaXY+PGJyPjwvZGl2PjxkaXY+TWF0dGhpZXU8L2Rpdj48L2Rpdj48ZGl2IGlk PSJnZWFyeS1xdW90ZSIgZGlyPSJhdXRvIj48YnI+T24gVHVlLCBBdWcgMiAyMDIyIGF0IDA1OjI5 OjEyIFBNICswMzAwLCBKYW5pIE5pa3VsYSAmbHQ7amFuaS5uaWt1bGFAbGludXguaW50ZWwuY29t Jmd0OyB3cm90ZTo8YnI+PGJsb2NrcXVvdGUgdHlwZT0iY2l0ZSI+PGRpdiBjbGFzcz0icGxhaW50 ZXh0IiBzdHlsZT0id2hpdGUtc3BhY2U6IGJyZWFrLXNwYWNlczsiPk9uIFdlZCwgMjcgSnVsIDIw MjIsIE1hdHRoaWV1IENIQVJFVFRFICZsdDs8YSBocmVmPSJtYWlsdG86bWF0dGhpZXUuY2hhcmV0 dGVAZ21haWwuY29tIj5tYXR0aGlldS5jaGFyZXR0ZUBnbWFpbC5jb208L2E+Jmd0OyB3cm90ZToK PGJsb2NrcXVvdGU+IExvYWRpbmcgYW4gRURJRCB1c2luZyBkcm0uZWRpZF9maXJtd2FyZSBwYXJh bWV0ZXIgbWFrZXMgcmVzdW1lIHRvIGZhaWwKIGFmdGVyIGZpcm13YXJlIGNhY2hlIGlzIGJlaW5n IGNsZWFuZWQuIFRoaXMgaXMgYmVjYXVzZSBlZGlkX2xvYWQoKSB1c2UgYQogdGVtcG9yYXJ5IGRl dmljZSB0byByZXF1ZXN0IHRoZSBmaXJtd2FyZS4gVGhpcyBjYXVzZSB0aGUgRURJRCBmaXJtd2Fy ZQogbm90IHRvIGJlIGNhY2hlZCBmcm9tIHN1c3BlbmQuIEFuZCwgcmVxdWVzdGluZyB0aGUgRURJ RCBmaXJtd2FyZSByZXR1cm4KIGFuIGVycm9yIGR1cmluZyByZXN1bWUuCiBTbyB0aGUgcmVxdWVz dF9maXJtd2FyZSgpIGNhbGwgc2hvdWxkIHVzZSBhIHBlcm1hbmVudCBkZXZpY2UgZm9yIGVhY2gK IGNvbm5lY3Rvci4gQWxzbywgd2Ugc2hvdWxkIGNhY2hlIHRoZSBFRElEIGV2ZW4gaWYgbm8gbW9u aXRvciBpcwogY29ubmVjdGVkLCBpbiBjYXNlIGl0J3MgcGx1Z2dlZCB3aGlsZSBzdXNwZW5kZWQu CjwvYmxvY2txdW90ZT4KQUZBSUNUIHRoaXMgYnJlYWtzIGNoYW5naW5nIGRybS5lZGlkX2Zpcm13 YXJlIHJ1bnRpbWUuCgpCUiwKSmFuaS4KCjxibG9ja3F1b3RlPgogTGluazogPGEgaHJlZj0iaHR0 cHM6Ly9naXRsYWIuZnJlZWRlc2t0b3Aub3JnL2RybS9hbWQvLS9pc3N1ZXMvMjA2MSI+aHR0cHM6 Ly9naXRsYWIuZnJlZWRlc2t0b3Aub3JnL2RybS9hbWQvLS9pc3N1ZXMvMjA2MTwvYT4KIFNpZ25l ZC1vZmYtYnk6IE1hdHRoaWV1IENIQVJFVFRFICZsdDs8YSBocmVmPSJtYWlsdG86bWF0dGhpZXUu Y2hhcmV0dGVAZ21haWwuY29tIj5tYXR0aGlldS5jaGFyZXR0ZUBnbWFpbC5jb208L2E+Jmd0Owog LS0tCiAgZHJpdmVycy9ncHUvZHJtL2RybV9jb25uZWN0b3IuYyB8ICA5ICsrKysKICBkcml2ZXJz L2dwdS9kcm0vZHJtX2VkaWRfbG9hZC5jIHwgODEgKysrKysrKysrKysrKysrKysrKysrKysrKysr Ky0tLS0tCiAgaW5jbHVkZS9kcm0vZHJtX2Nvbm5lY3Rvci5oICAgICB8IDEyICsrKysrCiAgaW5j bHVkZS9kcm0vZHJtX2VkaWQuaCAgICAgICAgICB8ICAzICsrCiAgNCBmaWxlcyBjaGFuZ2VkLCA5 NCBpbnNlcnRpb25zKCspLCAxMSBkZWxldGlvbnMoLSkKCiBkaWZmIC0tZ2l0IGEvZHJpdmVycy9n cHUvZHJtL2RybV9jb25uZWN0b3IuYyBiL2RyaXZlcnMvZ3B1L2RybS9kcm1fY29ubmVjdG9yLmMK IGluZGV4IDFjNDhkMTYyYzc3ZS4uZTg4MTllYmYxYzRiIDEwMDY0NAogLS0tIGEvZHJpdmVycy9n cHUvZHJtL2RybV9jb25uZWN0b3IuYwogKysrIGIvZHJpdmVycy9ncHUvZHJtL2RybV9jb25uZWN0 b3IuYwogQEAgLTMxLDYgKzMxLDcgQEAKICAjaW5jbHVkZSAmbHQ7ZHJtL2RybV9wcml2YWN5X3Nj cmVlbl9jb25zdW1lci5oJmd0OwogICNpbmNsdWRlICZsdDtkcm0vZHJtX3N5c2ZzLmgmZ3Q7CiAg CiArI2luY2x1ZGUgJmx0O2xpbnV4L3BsYXRmb3JtX2RldmljZS5oJmd0OwogICNpbmNsdWRlICZs dDtsaW51eC91YWNjZXNzLmgmZ3Q7CiAgCiAgI2luY2x1ZGUgImRybV9jcnRjX2ludGVybmFsLmgi CiBAQCAtMjg5LDYgKzI5MCw5IEBAIGludCBkcm1fY29ubmVjdG9yX2luaXQoc3RydWN0IGRybV9k ZXZpY2UgKmRldiwKICAKICAJZHJtX2Nvbm5lY3Rvcl9nZXRfY21kbGluZV9tb2RlKGNvbm5lY3Rv cik7CiAgCiArCWNvbm5lY3Rvci0mZ3Q7ZWRpZF9sb2FkX3BkZXYgPSBOVUxMOwogKwlkcm1fY2Fj aGVfZWRpZF9maXJtd2FyZShjb25uZWN0b3IpOwogKwogIAkvKiBXZSBzaG91bGQgYWRkIGNvbm5l Y3RvcnMgYXQgdGhlIGVuZCB0byBhdm9pZCB1cHNldHRpbmcgdGhlIGNvbm5lY3RvcgogIAkgKiBp bmRleCB0b28gbXVjaC4KICAJICovCiBAQCAtNDczLDYgKzQ3NywxMSBAQCB2b2lkIGRybV9jb25u ZWN0b3JfY2xlYW51cChzdHJ1Y3QgZHJtX2Nvbm5lY3RvciAqY29ubmVjdG9yKQogIAkJY29ubmVj dG9yLSZndDt0aWxlX2dyb3VwID0gTlVMTDsKICAJfQogIAogKwlpZiAoY29ubmVjdG9yLSZndDtl ZGlkX2xvYWRfcGRldikgewogKwkJcGxhdGZvcm1fZGV2aWNlX3VucmVnaXN0ZXIoY29ubmVjdG9y LSZndDtlZGlkX2xvYWRfcGRldik7CiArCQljb25uZWN0b3ItJmd0O2VkaWRfbG9hZF9wZGV2ID0g TlVMTDsKICsJfQogKwogIAlsaXN0X2Zvcl9lYWNoX2VudHJ5X3NhZmUobW9kZSwgdCwgJmFtcDtj b25uZWN0b3ItJmd0O3Byb2JlZF9tb2RlcywgaGVhZCkKICAJCWRybV9tb2RlX3JlbW92ZShjb25u ZWN0b3IsIG1vZGUpOwogIAogZGlmZiAtLWdpdCBhL2RyaXZlcnMvZ3B1L2RybS9kcm1fZWRpZF9s b2FkLmMgYi9kcml2ZXJzL2dwdS9kcm0vZHJtX2VkaWRfbG9hZC5jCiBpbmRleCAzN2Q4YmEzZGRi NDYuLjVhODJiZTk5MTdlYyAxMDA2NDQKIC0tLSBhL2RyaXZlcnMvZ3B1L2RybS9kcm1fZWRpZF9s b2FkLmMKICsrKyBiL2RyaXZlcnMvZ3B1L2RybS9kcm1fZWRpZF9sb2FkLmMKIEBAIC0xNjcsNiAr MTY3LDE5IEBAIHN0YXRpYyBpbnQgZWRpZF9zaXplKGNvbnN0IHU4ICplZGlkLCBpbnQgZGF0YV9z aXplKQogIAlyZXR1cm4gKGVkaWRbMHg3ZV0gKyAxKSAqIEVESURfTEVOR1RIOwogIH0KICAKICtz dGF0aWMgc3RydWN0IHBsYXRmb3JtX2RldmljZSAqZWRpZF9wZGV2KGNvbnN0IGNoYXIgKmNvbm5l Y3Rvcl9uYW1lKQogK3sKICsJc3RydWN0IHBsYXRmb3JtX2RldmljZSAqcGRldiA9IHBsYXRmb3Jt X2RldmljZV9yZWdpc3Rlcl9zaW1wbGUoY29ubmVjdG9yX25hbWUsIC0xLCBOVUxMLCAwKTsKICsK ICsJaWYgKElTX0VSUihwZGV2KSkgewogKwkJRFJNX0VSUk9SKCJGYWlsZWQgdG8gcmVnaXN0ZXIg RURJRCBmaXJtd2FyZSBwbGF0Zm9ybSBkZXZpY2UgIgogKwkJCSJmb3IgY29ubmVjdG9yIFwiJXNc IlxuIiwgY29ubmVjdG9yX25hbWUpOwogKwkJcmV0dXJuIEVSUl9DQVNUKHBkZXYpOwogKwl9CiAr CiArCXJldHVybiBwZGV2OwogK30KICsKICBzdGF0aWMgdm9pZCAqZWRpZF9sb2FkKHN0cnVjdCBk cm1fY29ubmVjdG9yICpjb25uZWN0b3IsIGNvbnN0IGNoYXIgKm5hbWUsCiAgCQkJY29uc3QgY2hh ciAqY29ubmVjdG9yX25hbWUpCiAgewogQEAgLTE4MiwxOCArMTk1LDE3IEBAIHN0YXRpYyB2b2lk ICplZGlkX2xvYWQoc3RydWN0IGRybV9jb25uZWN0b3IgKmNvbm5lY3RvciwgY29uc3QgY2hhciAq bmFtZSwKICAJCWZ3ZGF0YSA9IGdlbmVyaWNfZWRpZFtidWlsdGluXTsKICAJCWZ3c2l6ZSA9IHNp emVvZihnZW5lcmljX2VkaWRbYnVpbHRpbl0pOwogIAl9IGVsc2UgewogLQkJc3RydWN0IHBsYXRm b3JtX2RldmljZSAqcGRldjsKICsJCXN0cnVjdCBwbGF0Zm9ybV9kZXZpY2UgKnBkZXYgPSBjb25u ZWN0b3ItJmd0O2VkaWRfbG9hZF9wZGV2OwogIAkJaW50IGVycjsKICAKIC0JCXBkZXYgPSBwbGF0 Zm9ybV9kZXZpY2VfcmVnaXN0ZXJfc2ltcGxlKGNvbm5lY3Rvcl9uYW1lLCAtMSwgTlVMTCwgMCk7 CiAtCQlpZiAoSVNfRVJSKHBkZXYpKSB7CiAtCQkJRFJNX0VSUk9SKCJGYWlsZWQgdG8gcmVnaXN0 ZXIgRURJRCBmaXJtd2FyZSBwbGF0Zm9ybSBkZXZpY2UgIgogLQkJCQkgICJmb3IgY29ubmVjdG9y IFwiJXNcIlxuIiwgY29ubmVjdG9yX25hbWUpOwogLQkJCXJldHVybiBFUlJfQ0FTVChwZGV2KTsK ICsJCWlmIChXQVJOX09OKCFwZGV2KSkgewogKwkJCXBkZXYgPSBlZGlkX3BkZXYoY29ubmVjdG9y X25hbWUpOwogKwkJCWlmIChJU19FUlIocGRldikpCiArCQkJCXJldHVybiBFUlJfQ0FTVChwZGV2 KTsKICsJCQljb25uZWN0b3ItJmd0O2VkaWRfbG9hZF9wZGV2ID0gcGRldjsKICAJCX0KICAKICAJ CWVyciA9IHJlcXVlc3RfZmlybXdhcmUoJmFtcDtmdywgbmFtZSwgJmFtcDtwZGV2LSZndDtkZXYp OwogLQkJcGxhdGZvcm1fZGV2aWNlX3VucmVnaXN0ZXIocGRldik7CiAgCQlpZiAoZXJyKSB7CiAg CQkJRFJNX0VSUk9SKCJSZXF1ZXN0aW5nIEVESUQgZmlybXdhcmUgXCIlc1wiIGZhaWxlZCAoZXJy PSVkKVxuIiwKICAJCQkJICBuYW1lLCBlcnIpOwogQEAgLTI2MywxMSArMjc1LDkgQEAgc3RhdGlj IHZvaWQgKmVkaWRfbG9hZChzdHJ1Y3QgZHJtX2Nvbm5lY3RvciAqY29ubmVjdG9yLCBjb25zdCBj aGFyICpuYW1lLAogIAlyZXR1cm4gZWRpZDsKICB9CiAgCiAtc3RydWN0IGVkaWQgKmRybV9sb2Fk X2VkaWRfZmlybXdhcmUoc3RydWN0IGRybV9jb25uZWN0b3IgKmNvbm5lY3RvcikKICtzdGF0aWMg Y2hhciAqZWRpZF9uYW1lKGNvbnN0IGNoYXIgKmNvbm5lY3Rvcl9uYW1lKQogIHsKIC0JY29uc3Qg Y2hhciAqY29ubmVjdG9yX25hbWUgPSBjb25uZWN0b3ItJmd0O25hbWU7CiAgCWNoYXIgKmVkaWRu YW1lLCAqbGFzdCwgKmNvbG9uLCAqZndzdHIsICplZGlkc3RyLCAqZmFsbGJhY2sgPSBOVUxMOwog LQlzdHJ1Y3QgZWRpZCAqZWRpZDsKICAKICAJaWYgKGVkaWRfZmlybXdhcmVbMF0gPT0gJ1wwJykK ICAJCXJldHVybiBFUlJfUFRSKC1FTk9FTlQpOwogQEAgLTMxMCw4ICszMjAsNTcgQEAgc3RydWN0 IGVkaWQgKmRybV9sb2FkX2VkaWRfZmlybXdhcmUoc3RydWN0IGRybV9jb25uZWN0b3IgKmNvbm5l Y3RvcikKICAJaWYgKCpsYXN0ID09ICdcbicpCiAgCQkqbGFzdCA9ICdcMCc7CiAgCiAtCWVkaWQg PSBlZGlkX2xvYWQoY29ubmVjdG9yLCBlZGlkbmFtZSwgY29ubmVjdG9yX25hbWUpOwogKwllZGlk bmFtZSA9IGtzdHJkdXAoZWRpZG5hbWUsIEdGUF9LRVJORUwpOwogKwlpZiAoIWVkaWRuYW1lKSB7 CiArCQlrZnJlZShmd3N0cik7CiArCQlyZXR1cm4gRVJSX1BUUigtRU5PTUVNKTsKICsJfQogKwog IAlrZnJlZShmd3N0cik7CiArCXJldHVybiBlZGlkbmFtZTsKICt9CiArCiArdm9pZCBkcm1fY2Fj aGVfZWRpZF9maXJtd2FyZShzdHJ1Y3QgZHJtX2Nvbm5lY3RvciAqY29ubmVjdG9yKQogK3sKICsJ Y29uc3QgY2hhciAqY29ubmVjdG9yX25hbWUgPSBjb25uZWN0b3ItJmd0O25hbWU7CiArCWNvbnN0 IGNoYXIgKmVkaWRuYW1lID0gZWRpZF9uYW1lKGNvbm5lY3Rvcl9uYW1lKTsKICsJc3RydWN0IHBs YXRmb3JtX2RldmljZSAqcGRldjsKICsJaW50IGVycjsKICsKICsJaWYgKElTX0VSUihlZGlkbmFt ZSkpCiArCQlyZXR1cm47CiArCiArCWlmIChtYXRjaF9zdHJpbmcoZ2VuZXJpY19lZGlkX25hbWUs IEdFTkVSSUNfRURJRFMsIGVkaWRuYW1lKSAmZ3Q7PSAwKSB7CiArCQlrZnJlZShlZGlkbmFtZSk7 CiArCQlyZXR1cm47CiArCX0KICsKICsJcGRldiA9IGVkaWRfcGRldihjb25uZWN0b3JfbmFtZSk7 CiArCWlmIChJU19FUlIocGRldikpIHsKICsJCWtmcmVlKGVkaWRuYW1lKTsKICsJCXJldHVybjsK ICsJfQogKwljb25uZWN0b3ItJmd0O2VkaWRfbG9hZF9wZGV2ID0gcGRldjsKICsKICsJZXJyID0g ZmlybXdhcmVfcmVxdWVzdF9jYWNoZSgmYW1wO3BkZXYtJmd0O2RldiwgZWRpZG5hbWUpOwogKwlp ZiAoZXJyKQogKwkJRFJNX0VSUk9SKCJSZXF1ZXN0aW5nIEVESUQgZmlybXdhcmUgY2FjaGUgXCIl c1wiIGZhaWxlZCAoZXJyPSVkKVxuIiwKICsJCQllZGlkbmFtZSwgZXJyKTsKICsKICsJa2ZyZWUo ZWRpZG5hbWUpOwogK30KICsKICtzdHJ1Y3QgZWRpZCAqZHJtX2xvYWRfZWRpZF9maXJtd2FyZShz dHJ1Y3QgZHJtX2Nvbm5lY3RvciAqY29ubmVjdG9yKQogK3sKICsJY29uc3QgY2hhciAqY29ubmVj dG9yX25hbWUgPSBjb25uZWN0b3ItJmd0O25hbWU7CiArCWNvbnN0IGNoYXIgKmVkaWRuYW1lID0g ZWRpZF9uYW1lKGNvbm5lY3Rvcl9uYW1lKTsKICsJc3RydWN0IGVkaWQgKmVkaWQ7CiArCiArCWlm IChJU19FUlIoZWRpZG5hbWUpKQogKwkJcmV0dXJuIEVSUl9DQVNUKGVkaWRuYW1lKTsKICsKICsJ ZWRpZCA9IGVkaWRfbG9hZChjb25uZWN0b3IsIGVkaWRuYW1lLCBjb25uZWN0b3JfbmFtZSk7CiAr CWtmcmVlKGVkaWRuYW1lKTsKICAKICAJcmV0dXJuIGVkaWQ7CiAgfQogZGlmZiAtLWdpdCBhL2lu Y2x1ZGUvZHJtL2RybV9jb25uZWN0b3IuaCBiL2luY2x1ZGUvZHJtL2RybV9jb25uZWN0b3IuaAog aW5kZXggM2FjNGJmODdmMjU3Li40N2M4NDc0MTUxN2UgMTAwNjQ0CiAtLS0gYS9pbmNsdWRlL2Ry bS9kcm1fY29ubmVjdG9yLmgKICsrKyBiL2luY2x1ZGUvZHJtL2RybV9jb25uZWN0b3IuaAogQEAg LTE1NzMsNiArMTU3MywxOCBAQCBzdHJ1Y3QgZHJtX2Nvbm5lY3RvciB7CiAgCSAqLwogIAlzdHJ1 Y3QgaTJjX2FkYXB0ZXIgKmRkYzsKICAKICsJLyoqCiArCSAqIEBlZGlkX2xvYWRfcGRldjogUGxh dGZvcm0gZGV2aWNlIGZvciBsb2FkaW5nIEVESUQgdmlhIGZpcm13YXJlLgogKwkgKgogKwkgKiBU aGUgcGxhdGZvcm0gZGV2aWNlIGlzIHJlZ2lzdGVyZWQgaW4gZHJtX2Nvbm5lY3Rvcl9pbml0KCkg aW4gY2FzZSBhCiArCSAqIGN1c3RvbSBFRElEIGZpcm13YXJlIGlzIHVzZWQgd2l0aCBgZWRpZF9m aXJtd2FyZWAgcGFyYW1ldGVyLiBPdGhlcndpc2UsCiArCSAqIGl0IGlzIHNldCB0byBOVUxMLgog KwkgKgogKwkgKiBQbGF0Zm9ybSBkZXZpY2UgaXMgdW5yZWdpc3RlcmVkIGluIGRybV9jb25uZWN0 b3JfY2xlYW51cCgpIGlmIGl0CiArCSAqIGlzIG5vdCBOVUxMLgogKwkgKi8KICsJc3RydWN0IHBs YXRmb3JtX2RldmljZSAqZWRpZF9sb2FkX3BkZXY7CiArCiAgCS8qKgogIAkgKiBAbnVsbF9lZGlk X2NvdW50ZXI6IHRyYWNrIHNpbmtzIHRoYXQgZ2l2ZSB1cyBhbGwgemVyb3MgZm9yIHRoZSBFRElE LgogIAkgKiBOZWVkZWQgdG8gd29ya2Fyb3VuZCBzb21lIEhXIGJ1Z3Mgd2hlcmUgd2UgZ2V0IGFs bCAwcwogZGlmZiAtLWdpdCBhL2luY2x1ZGUvZHJtL2RybV9lZGlkLmggYi9pbmNsdWRlL2RybS9k cm1fZWRpZC5oCiBpbmRleCBiMjc1Njc1MzM3MGIuLmU5MDdjOTI4YTM1ZCAxMDA2NDQKIC0tLSBh L2luY2x1ZGUvZHJtL2RybV9lZGlkLmgKICsrKyBiL2luY2x1ZGUvZHJtL2RybV9lZGlkLmgKIEBA IC0zNzgsMTAgKzM3OCwxMyBAQCBpbnQgZHJtX2F2X3N5bmNfZGVsYXkoc3RydWN0IGRybV9jb25u ZWN0b3IgKmNvbm5lY3RvciwKICAJCSAgICAgIGNvbnN0IHN0cnVjdCBkcm1fZGlzcGxheV9tb2Rl ICptb2RlKTsKICAKICAjaWZkZWYgQ09ORklHX0RSTV9MT0FEX0VESURfRklSTVdBUkUKICt2b2lk IGRybV9jYWNoZV9lZGlkX2Zpcm13YXJlKHN0cnVjdCBkcm1fY29ubmVjdG9yICpjb25uZWN0b3Ip OwogIHN0cnVjdCBlZGlkICpkcm1fbG9hZF9lZGlkX2Zpcm13YXJlKHN0cnVjdCBkcm1fY29ubmVj dG9yICpjb25uZWN0b3IpOwogIGludCBfX2RybV9zZXRfZWRpZF9maXJtd2FyZV9wYXRoKGNvbnN0 IGNoYXIgKnBhdGgpOwogIGludCBfX2RybV9nZXRfZWRpZF9maXJtd2FyZV9wYXRoKGNoYXIgKmJ1 Ziwgc2l6ZV90IGJ1ZnNpemUpOwogICNlbHNlCiAraW5saW5lIHZvaWQKICtkcm1fY2FjaGVfZWRp ZF9maXJtd2FyZShzdHJ1Y3QgZHJtX2Nvbm5lY3RvciAqY29ubmVjdG9yKTsKICBzdGF0aWMgaW5s aW5lIHN0cnVjdCBlZGlkICoKICBkcm1fbG9hZF9lZGlkX2Zpcm13YXJlKHN0cnVjdCBkcm1fY29u bmVjdG9yICpjb25uZWN0b3IpCiAgewo8L2Jsb2NrcXVvdGU+CjxkaXY+LS0gCjwvZGl2Pkphbmkg TmlrdWxhLCBJbnRlbCBPcGVuIFNvdXJjZSBHcmFwaGljcyBDZW50ZXIKPC9kaXY+PC9ibG9ja3F1 b3RlPjwvZGl2Pg== --===============7393658385179695874==--