[PATCH] drm: Fix EDID firmware load on resume
Matthieu CHARETTE
matthieu.charette at gmail.com
Mon Sep 12 16:25:14 UTC 2022
Hi,
Sorry for late reply.
How do you propose to go then?
Can we still use a persistent platform device to load the firmware and
cache it? But, in this case the system will still crash in case the
user change drm.edid_firmware then, without replugging the device, he
suspends and resumes the machine.
Or you may have a better idea?
Thanks,
Matthieu
On Mon, Aug 8 2022 at 08:19:01 PM +0300, Jani Nikula
<jani.nikula at linux.intel.com> wrote:
> On Mon, 08 Aug 2022, Matthieu CHARETTE <matthieu.charette at gmail.com>
> wrote:
>> Sorry, What do you mean?
>
> You cache with one name at connector init time, but the name specified
> using drm.edid_firmware may be changed whenever, to cause the next
> EDID
> read to use a different EDID firmware.
>
> BR,
> Jani.
>
>
>>
>> Matthieu
>>
>> On Tue, Aug 2 2022 at 05:29:12 PM +0300, Jani Nikula
>> <jani.nikula at linux.intel.com> wrote:
>>> On Wed, 27 Jul 2022, Matthieu CHARETTE <matthieu.charette at gmail.com
>>> <mailto:matthieu.charette at 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 at gmail.com
>>>> <mailto:matthieu.charette at 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
>>
>
> --
> Jani Nikula, Intel Open Source Graphics Center
More information about the dri-devel
mailing list