[Intel-gfx] [RFC v2 6/8] drm: Handle fbdev emulation in core

Noralf Trønnes noralf at tronnes.org
Thu Jan 11 14:09:14 UTC 2018


Den 11.01.2018 08.45, skrev Daniel Vetter:
> On Wed, Jan 10, 2018 at 06:02:38PM +0100, Noralf Trønnes wrote:
>> Den 09.01.2018 11.38, skrev Daniel Vetter:
>>> On Wed, Jan 03, 2018 at 11:21:08PM +0100, Noralf Trønnes wrote:
>>>> Prepare for generic fbdev emulation by letting DRM core work directly
>>>> with the fbdev compatibility layer. This is done by adding new fbdev
>>>> helper vtable callbacks for restore, hotplug_event, unregister and
>>>> release.
>>>>
>>>> Signed-off-by: Noralf Trønnes <noralf at tronnes.org>
>>> No clue whether my idea is any good, but inspired by the bootsplash
>>> discussion, and the prospect that we might get other in-kernel drm/kms
>>> clients I'm wondering whether we should make this a bit more generic and
>>> tie it to drm_file?
>>>
>>> The idea would be to have a list of internal drm clients by putting all
>>> the internal drm_files onto a list (same way we do with the userspace
>>> ones). Then we'd just walk that list and call ->hotplug, ->unregister and
>>> ->release at the right places.
>>>
>>> ->restore would be a bit different, I guess for that we'd only call the
>>> ->restore callback of the very first kernel-internal client.
>>>
>>> With that we could then add/remove kernel-internal drm clients like normal
>>> userspace clients, which should make integration of emergency consoles,
>>> boot splashes and whatever else real easy. And I think it would be a lot
>>> cleaner than leaking fb_helper knowledge into the drm core, which feels a
>>> rather backwards.
>>>
>>> Otoh that feels a bit overengineered (but at least it shouldn't be a lot
>>> more code really). If the list is too much work we could start with 1
>>> drm_file * pointer for internal stuff (but would still need locking, so
>>> list_head is probably easier).
>>>
>>> Thoughts?
>> I prefer to have a proper in-kernel client API from the get go, instead
>> of fixing it up later. The reason I skipped spending time on it in this
>> RFC, was that I didn't know if I was on the right track at the right time
>> to get the necessary attention to make this dumb_buffer idea happen.
>>
>> With a drm_file centric approach, are you thinking something like this:
>>
>>   struct drm_device {
>> +    struct list_head filelist_internal;
>>   };
>>
>> +struct drm_file_funcs {
>> +    int (*restore)(struct drm_file *file);
>> +    void (*hotplug)(struct drm_file *file);
>> +    void (*unregister)(struct drm_file *file);
> I guess we still need a cleanup callback here? For the usual two-stage
> driver unload sequence of 1. unregister 2. cleanup.

There's no need for a cleanup callback in this scenario.
The client holds a ref on drm_device through drm_internal_open() and if
it can't release the drm_file on .unregister, there won't be any cleanup.
When the client is in a position to release drm_file (fb_close), it will
do so and the ref is dropped.

If for some reason we can't take a ref, then we need to use
drm_device.open_count to avoid drm_device going away in drm_dev_unplug().

Noralf.

>
>> +};
>>
>>   struct drm_file {
>> +    struct drm_device *dev;
>> +    const struct drm_file_funcs *funcs;
>>   };
>>
>>   struct drm_file *drm_file_alloc(struct drm_minor *minor)
>>   {
>> +    file->dev = dev;
>>   }
>>
>> struct drm_file* drm_internal_open(struct drm_device *dev,
>>                     const struct drm_file_funcs *funcs)
>>      struct drm_file *file;
>>      int ret;
>>
>>      if (!try_module_get(dev->driver->fops->owner))
>>          return ERR_PTR(-ENODEV);
>>
>>      drm_dev_ref(dev);
>>
>>      file = drm_file_alloc(dev);
>>      if (IS_ERR(file)) {
>>          drm_dev_unref(dev);
>>          module_put(dev->driver->fops->owner);
>>          return file;
>>      }
>>
>>      file->funcs = funcs;
>>
>>      mutex_lock(&dev->filelist_mutex);
>>      list_add(&file->lhead, &dev->filelist_internal);
>>      mutex_unlock(&dev->filelist_mutex);
>>
>>      return file;
>> }
>>
>> void drm_internal_release(struct drm_file *file)
>> {
>>      struct drm_device *dev = file->dev;
>>
>>      mutex_lock(&dev->filelist_mutex);
>>      list_del(&file->lhead);
>>      mutex_unlock(&dev->filelist_mutex);
>>
>>      drm_file_free(file);
>>      drm_dev_unref(dev);
>>      module_put(dev->driver->fops->owner);
>> }
>>
>>   void drm_lastclose(struct drm_device *dev)
>>   {
>>
>> +    mutex_lock(&dev->filelist_mutex);
>> +    list_for_each_entry(file, &dev->filelist_internal, lhead) {
>> +        if (file->funcs && file->funcs->restore &&
>> +            !file->funcs->restore(file))
>> +                break;
>> +    mutex_unlock(&dev->filelist_mutex);
>>   }
>>
>>   void drm_kms_helper_hotplug_event(struct drm_device *dev)
>>   {
>>
>> +    mutex_lock(&dev->filelist_mutex);
>> +    list_for_each_entry(file, &dev->filelist_internal, lhead) {
>> +        if (file->funcs && file->funcs->hotplug)
>> +            file->funcs->hotplug(file);
>> +    mutex_unlock(&dev->filelist_mutex);
>>   }
>>
>> How is locking done when .unregister will call into drm_internal_release()?
>>
>>   void drm_dev_unregister(struct drm_device *dev)
>>   {
>>
>> +    list_for_each_entry(file, &dev->filelist_internal, lhead) {
>> +        if (file->funcs && file->funcs->unregister)
>> +            file->funcs->unregister(file);
>>   }
>>
>> There is also the case where .unregister can't release the drm_file
>> because userspace has mmap'ed the buffer (fbdev). The client holds a ref
>> on drm_device so cleanup is stalled but that requires a driver that can
>> handle that (ref the tinydrm unplug series which is awaiting a timeslot).
> Yeah we need the usual split into 2 things, with unregister only taking
> away the uapi (but keeping references still to the drm_device/module).
> Once fb_close has been called and the drm_device can be freed, then we
> need a separate cleanup callback.
>
> This also means that locking is no issue for ->unregister. It's only an
> issue for ->cleanup:
> - list_for_each_entry_safe, since the ->cleanup callback will self-remove
> - a comment like we already have for the fb list that since cleanup is
>    called only from 1 thread, at a time where we know nothing else can get
>    at the drm_device anymore (the last reference was dropped) it's safe to
>    iterate without locking.
>
>> amdgpu iterates dev->filelist in amdgpu_gem_force_release() and
>> amdgpu_sched_process_priority_override(). I don't know if it needs to
>> handle the internal ones as well.
> That's for the scheduler, not for kms. As long as the internal clients are
> kms-only (I don't expect that to ever change, gpu command submission needs
> non-generic userspace) this shouldn't be a problem.
>
>> What's the rational for having separate lists for internal and userspace?
> Hm I guess we could unify them. But probably more work since we need to
> make sure that all the existing filelist walkers won't be confused by the
> internal stuff. I figured it's easier to just split them. If we end up
> having lots of duplicated loops we can reconsider I think.
>
>> There is one thing missing and that is notification of new drm_devices.
>> The client iterates the available drm_devices on module init, but it
>> needs a way to know about any later drm_device registrations.
> Oh, I didn't notice that part. That's the exact same nasty issue fbcon
> with trying to keep up to date fbdev drivers. fbcon uses a notifier, and
> the resulting locking loops are terrible.
>
> But I think if we do a notifier _only_ for new drm_device registrations
> (i.e. we call the notifier from drm_dev_register), and let the
> unregister/cleanup all get handled through the new drm_file callbacks, it
> should be all fine. The only tricky bit will be closing the race when
> loading the fbdev emulation code (or anything else really).
>
> The real downside of this is that it forces distros to manually load the
> fbdev emulation module (right now the explicit driver call is forcing
> that), which is annoying.
>
> But over the past years we've cleaned up the separation between the fbdev
> emulation and the other helpers, and we could simply move the fbdev
> emulation into the core drm.ko module. Then you'd simply put a direct call
> to register the fbdev instance into drm_dev_register, done. No locking
> headache, no races, not manual module loading needed.
>
> Right now I'm leaning towards making the fbdev stuff built-into drm.ko,
> but need to ponder this some more first.
> -Daniel
>
>> Noralf.
>>
>>> -Daniel
>>>
>>>> ---
>>>>    drivers/gpu/drm/drm_file.c         | 12 +++++++++++-
>>>>    drivers/gpu/drm/drm_mode_config.c  | 10 ++++++++++
>>>>    drivers/gpu/drm/drm_probe_helper.c |  4 ++++
>>>>    include/drm/drm_fb_helper.h        | 33 +++++++++++++++++++++++++++++++++
>>>>    4 files changed, 58 insertions(+), 1 deletion(-)
>>>>
>>>> diff --git a/drivers/gpu/drm/drm_file.c b/drivers/gpu/drm/drm_file.c
>>>> index 400d44437e93..7ec09fb83135 100644
>>>> --- a/drivers/gpu/drm/drm_file.c
>>>> +++ b/drivers/gpu/drm/drm_file.c
>>>> @@ -35,6 +35,7 @@
>>>>    #include <linux/slab.h>
>>>>    #include <linux/module.h>
>>>> +#include <drm/drm_fb_helper.h>
>>>>    #include <drm/drm_file.h>
>>>>    #include <drm/drmP.h>
>>>> @@ -441,10 +442,19 @@ static void drm_legacy_dev_reinit(struct drm_device *dev)
>>>>    void drm_lastclose(struct drm_device * dev)
>>>>    {
>>>> +	struct drm_fb_helper *fb_helper = dev->fb_helper;
>>>> +	int ret;
>>>> +
>>>>    	DRM_DEBUG("\n");
>>>> -	if (dev->driver->lastclose)
>>>> +	if (dev->driver->lastclose) {
>>>>    		dev->driver->lastclose(dev);
>>>> +	} else if (fb_helper && fb_helper->funcs && fb_helper->funcs->restore) {
>>>> +		ret = fb_helper->funcs->restore(fb_helper);
>>>> +		if (ret)
>>>> +			DRM_ERROR("Failed to restore fbdev: %d\n", ret);
>>>> +	}
>>>> +
>>>>    	DRM_DEBUG("driver lastclose completed\n");
>>>>    	if (drm_core_check_feature(dev, DRIVER_LEGACY))
>>>> diff --git a/drivers/gpu/drm/drm_mode_config.c b/drivers/gpu/drm/drm_mode_config.c
>>>> index bc5c46306b3d..260eb1730244 100644
>>>> --- a/drivers/gpu/drm/drm_mode_config.c
>>>> +++ b/drivers/gpu/drm/drm_mode_config.c
>>>> @@ -21,6 +21,7 @@
>>>>     */
>>>>    #include <drm/drm_encoder.h>
>>>> +#include <drm/drm_fb_helper.h>
>>>>    #include <drm/drm_mode_config.h>
>>>>    #include <drm/drmP.h>
>>>> @@ -61,6 +62,11 @@ int drm_modeset_register_all(struct drm_device *dev)
>>>>    void drm_modeset_unregister_all(struct drm_device *dev)
>>>>    {
>>>> +	struct drm_fb_helper *fb_helper = dev->fb_helper;
>>>> +
>>>> +	if (fb_helper && fb_helper->funcs && fb_helper->funcs->unregister)
>>>> +		fb_helper->funcs->unregister(fb_helper);
>>>> +
>>>>    	drm_connector_unregister_all(dev);
>>>>    	drm_encoder_unregister_all(dev);
>>>>    	drm_crtc_unregister_all(dev);
>>>> @@ -408,6 +414,7 @@ EXPORT_SYMBOL(drm_mode_config_init);
>>>>     */
>>>>    void drm_mode_config_cleanup(struct drm_device *dev)
>>>>    {
>>>> +	struct drm_fb_helper *fb_helper = dev->fb_helper;
>>>>    	struct drm_connector *connector;
>>>>    	struct drm_connector_list_iter conn_iter;
>>>>    	struct drm_crtc *crtc, *ct;
>>>> @@ -417,6 +424,9 @@ void drm_mode_config_cleanup(struct drm_device *dev)
>>>>    	struct drm_property_blob *blob, *bt;
>>>>    	struct drm_plane *plane, *plt;
>>>> +	if (fb_helper && fb_helper->funcs && fb_helper->funcs->release)
>>>> +		fb_helper->funcs->release(fb_helper);
>>>> +
>>>>    	list_for_each_entry_safe(encoder, enct, &dev->mode_config.encoder_list,
>>>>    				 head) {
>>>>    		encoder->funcs->destroy(encoder);
>>>> diff --git a/drivers/gpu/drm/drm_probe_helper.c b/drivers/gpu/drm/drm_probe_helper.c
>>>> index 555fbe54d6e2..9d8b0ba54173 100644
>>>> --- a/drivers/gpu/drm/drm_probe_helper.c
>>>> +++ b/drivers/gpu/drm/drm_probe_helper.c
>>>> @@ -559,10 +559,14 @@ EXPORT_SYMBOL(drm_helper_probe_single_connector_modes);
>>>>     */
>>>>    void drm_kms_helper_hotplug_event(struct drm_device *dev)
>>>>    {
>>>> +	struct drm_fb_helper *fb_helper = dev->fb_helper;
>>>> +
>>>>    	/* send a uevent + call fbdev */
>>>>    	drm_sysfs_hotplug_event(dev);
>>>>    	if (dev->mode_config.funcs->output_poll_changed)
>>>>    		dev->mode_config.funcs->output_poll_changed(dev);
>>>> +	else if (fb_helper && fb_helper->funcs && fb_helper->funcs->hotplug_event)
>>>> +		fb_helper->funcs->hotplug_event(fb_helper);
>>>>    }
>>>>    EXPORT_SYMBOL(drm_kms_helper_hotplug_event);
>>>> diff --git a/include/drm/drm_fb_helper.h b/include/drm/drm_fb_helper.h
>>>> index 16d8773b60e3..385f967c3552 100644
>>>> --- a/include/drm/drm_fb_helper.h
>>>> +++ b/include/drm/drm_fb_helper.h
>>>> @@ -125,6 +125,39 @@ struct drm_fb_helper_funcs {
>>>>    			       struct drm_display_mode **modes,
>>>>    			       struct drm_fb_offset *offsets,
>>>>    			       bool *enabled, int width, int height);
>>>> +
>>>> +	/**
>>>> +	 * @restore:
>>>> +	 *
>>>> +	 * Optional callback for restoring fbdev emulation.
>>>> +	 * Called by drm_lastclose() if &drm_driver->lastclose is not set.
>>>> +	 */
>>>> +	int (*restore)(struct drm_fb_helper *fb_helper);
>>>> +
>>>> +	/**
>>>> +	 * @hotplug_event:
>>>> +	 *
>>>> +	 * Optional callback for hotplug events.
>>>> +	 * Called by drm_kms_helper_hotplug_event() if
>>>> +	 * &drm_mode_config_funcs->output_poll_changed  is not set.
>>>> +	 */
>>>> +	int (*hotplug_event)(struct drm_fb_helper *fb_helper);
>>>> +
>>>> +	/**
>>>> +	 * @unregister:
>>>> +	 *
>>>> +	 * Optional callback for unregistrering fbdev emulation.
>>>> +	 * Called by drm_dev_unregister().
>>>> +	 */
>>>> +	void (*unregister)(struct drm_fb_helper *fb_helper);
>>>> +
>>>> +	/**
>>>> +	 * @release:
>>>> +	 *
>>>> +	 * Optional callback for releasing fbdev emulation resources.
>>>> +	 * Called by drm_mode_config_cleanup().
>>>> +	 */
>>>> +	void (*release)(struct drm_fb_helper *fb_helper);
>>>>    };
>>>>    struct drm_fb_helper_connector {
>>>> -- 
>>>> 2.14.2
>>>>



More information about the Intel-gfx mailing list