Re: [PATCH for drm-misc-fixes v5 2/4] drm/hisilicon/hibmc: fix no showing when no connectors connected
From: Yongbang Shi
Date: Tue Apr 28 2026 - 09:06:37 EST
> Hi
>
> Am 28.04.26 um 05:58 schrieb Yongbang Shi:
> [...]
>>> There's a problem with the overall logic here: if the physical status is 'connected' and the helper could not retrieve
>>> any modes, the helper should return 0 here. Then the DRM helpers do the right thing with setting up a few modes or an
>>> EDID override as a fallback. See [2]. For example, on my broken test system, I'd be able to provide my display's EDID
>>> and get the correct output.
>>>
>>> [2] https://elixir.bootlin.com/linux/v7.0.1/source/drivers/gpu/drm/drm_probe_helper.c#L436
>>>
>>> Any code below is BMC setup and should run in an else branch, just like in ast.
>>>
>> The `drm_edid_override_connector_update` interface is used to retrieve an overridden EDID from debugfs or from firmware:
>> * Debugfs: Requires the user to manually perform file operations to configure it;
>> * Firmware: Requires the distribution’s operating system or the user to place the EDID binary file in the
>> `/lib/firmware/` directory, and the `drm.edid_firmware` parameter must be specified in the GRUB boot
>> parameters;
>>
>> The modification method you provided allows for display even when the EDID cannot be retrieved. But both of these
>> methods require cooperation from the user or the distribution's operating system, making implementation relatively
>> difficult.
>
> Yes, it's the established way for users to override the EDID.
>
>>
>> Our use case requires that the KVM's display functionality remain intact even when no VGA or DisplayPort monitor is
>> connected. I believe setting `drm_set_preferred_mode` (1024x768) as the default resolution when no monitor is present
>> would be a more universal solution. Similarly, on your test system, this approach would ensure basic display
>> functionality.
>
> If no modes could be detected on the VGA and no EDID has been provided by the user, DRM helpers will install a set of
> default modes that are suitable for graphics cards. See [1]. The modes installed for the KVM are not all compatible with
> VGA. If you rely on DRM helpers, you'll always get correct modes for VGA and KVM.
>
> Generally speaking, the decision of VGA-vs-KVM should be in ->detect. Having phys_state/phys_status does this very well.
> The ->get_modes function should then use whatever has been detected.
>
> [1] https://elixir.bootlin.com/linux/v7.0.1/source/drivers/gpu/drm/drm_probe_helper.c#L653
>
Yes, I understand what you mean. But there's another issue here; we've also tested the scenario where we directly return
0. However, we found that only three resolutions remain supported: 1024*768, 800*600 and 640*480. This is because the
resolution added via `drm_add_modes_noedid` in the helper has a maximum of 1024*768. This doesn't look good—there are
too few resolutions available for users to choose from in KVM.
With the current implementation, when `count == 0`, we set the resolution to the highest resolution supported by our
CRTC, allowing users more options.
Of course, this implementation has its drawbacks; it doesn't allow users to provide a custom resolution list via debugfs
using the override method when `count == 0`. But this method isn't very important for how our product is used.
Thanks,
Yongbang.
> Best regards
> Thomas
>
>>
>>>> + drm_edid_connector_update(connector, NULL);
>>>> count = drm_add_modes_noedid(connector,
>>>> connector->dev->mode_config.max_width,
>>>> connector->dev->mode_config.max_height);
>>>> drm_set_preferred_mode(connector, 1024, 768);
>>>> -out:
>>>> - drm_edid_free(drm_edid);
>>>> -
>>>> return count;
>>>> }
>>>> @@ -57,10 +50,32 @@ static void hibmc_connector_destroy(struct drm_connector *connector)
>>>> drm_connector_cleanup(connector);
>>>> }
>>>> +static int hibmc_vdac_detect(struct drm_connector *connector,
>>>> + struct drm_modeset_acquire_ctx *ctx,
>>>> + bool force)
>>>> +{
>>>> + struct hibmc_drm_private *priv = to_hibmc_drm_private(connector->dev);
>>>> + int state = drm_connector_helper_detect_from_ddc(connector, ctx,
>>>> + force);
>>> 'state' -> 'status'
>>>
>> Okay.
>>
>>>> + struct hibmc_vdac *vdac = to_hibmc_vdac(connector);
>>>> +
>>>> + if (priv->dp.phys_state == connector_status_connected)
>>>> + return vdac->phys_state = state;
>>> Please only one statement per line. First assign, then return.
>>>
>> Yes, Sorry about that. According to the Linux kernel coding guidelines, this line of code should be split.
>>
>>
>> Thanks,
>> Yongbang.
>>
>>>> +
>>>> + if (state != vdac->phys_state)
>>>> + ++connector->epoch_counter;
>>>> + vdac->phys_state = state;
>>>> +
>>>> + /* If both the DP and VDAC physical states are disconnected,
>>>> + * the "connected" status is returned to support KVM display.
>>>> + */
>>>> + return connector_status_connected;
>>> I haven't tried, but I think this should also resolve the problems on my test systems. Great, thanks a lot! I might just
>>> get default resolutions for now, but that's OK.
>>>
>>> Best regards
>>> Thomas
>>>
>>>> +}
>>>> +
>>>> static const struct drm_connector_helper_funcs
>>>> hibmc_connector_helper_funcs = {
>>>> .get_modes = hibmc_connector_get_modes,
>>>> - .detect_ctx = drm_connector_helper_detect_from_ddc,
>>>> + .detect_ctx = hibmc_vdac_detect,
>>>> };
>>>> static const struct drm_connector_funcs hibmc_connector_funcs = {
>>>> @@ -130,6 +145,8 @@ int hibmc_vdac_init(struct hibmc_drm_private *priv)
>>>> connector->polled = DRM_CONNECTOR_POLL_CONNECT | DRM_CONNECTOR_POLL_DISCONNECT;
>>>> + vdac->phys_state = connector_status_connected;
>>>> +
>>>> return 0;
>>>> err:
>