2017-03-08 15:12:42 +01:00
/*
* Copyright 1999 Precision Insight , Inc . , Cedar Park , Texas .
* Copyright 2000 VA Linux Systems , Inc . , Sunnyvale , California .
* Copyright ( c ) 2009 - 2010 , Code Aurora Forum .
* All rights reserved .
*
* Author : Rickard E . ( Rik ) Faith < faith @ valinux . com >
* Author : Gareth Hughes < gareth @ valinux . com >
*
* Permission is hereby granted , free of charge , to any person obtaining a
* copy of this software and associated documentation files ( the " Software " ) ,
* to deal in the Software without restriction , including without limitation
* the rights to use , copy , modify , merge , publish , distribute , sublicense ,
* and / or sell copies of the Software , and to permit persons to whom the
* Software is furnished to do so , subject to the following conditions :
*
* The above copyright notice and this permission notice ( including the next
* paragraph ) shall be included in all copies or substantial portions of the
* Software .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS OR
* IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY ,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT . IN NO EVENT SHALL
* VA LINUX SYSTEMS AND / OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM , DAMAGES OR
* OTHER LIABILITY , WHETHER IN AN ACTION OF CONTRACT , TORT OR OTHERWISE ,
* ARISING FROM , OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE .
*/
# ifndef _DRM_FILE_H_
# define _DRM_FILE_H_
# include <linux/types.h>
# include <linux/completion.h>
2018-12-27 14:56:38 +02:00
# include <linux/idr.h>
2017-03-08 15:12:42 +01:00
# include <uapi/drm/drm.h>
# include <drm/drm_prime.h>
struct dma_fence ;
struct drm_file ;
struct drm_device ;
2017-05-31 11:21:46 +02:00
struct device ;
2019-11-07 18:05:58 +00:00
struct file ;
2017-03-08 15:12:42 +01:00
/*
* FIXME : Not sure we want to have drm_minor here in the end , but to avoid
* header include loops we need it here for now .
*/
2017-03-08 15:12:44 +01:00
2018-05-08 17:14:25 -07:00
/* Note that the order of this enum is ABI (it determines
* / dev / dri / renderD * numbers ) .
*/
2017-03-08 15:12:42 +01:00
enum drm_minor_type {
DRM_MINOR_PRIMARY ,
2018-05-08 17:14:25 -07:00
DRM_MINOR_CONTROL ,
2017-03-08 15:12:42 +01:00
DRM_MINOR_RENDER ,
} ;
/**
2017-03-08 15:12:44 +01:00
* struct drm_minor - DRM device minor structure
*
* This structure represents a DRM minor number for device nodes in / dev .
* Entirely opaque to drivers and should never be inspected directly by drivers .
* Drivers instead should only interact with & struct drm_file and of course
* & struct drm_device , which is also where driver - private data and resources can
* be attached to .
2017-03-08 15:12:42 +01:00
*/
struct drm_minor {
2017-03-08 15:12:44 +01:00
/* private: */
int index ; /* Minor device number */
int type ; /* Control or render */
struct device * kdev ; /* Linux device */
2017-03-08 15:12:42 +01:00
struct drm_device * dev ;
struct dentry * debugfs_root ;
struct list_head debugfs_list ;
struct mutex debugfs_lock ; /* Protects debugfs_list. */
} ;
2017-03-08 15:12:44 +01:00
/**
* struct drm_pending_event - Event queued up for userspace to read
*
* This represents a DRM event . Drivers can use this as a generic completion
* mechanism , which supports kernel - internal & struct completion , & struct dma_fence
* and also the DRM - specific & struct drm_event delivery mechanism .
*/
2017-03-08 15:12:42 +01:00
struct drm_pending_event {
2017-03-08 15:12:44 +01:00
/**
* @ completion :
*
* Optional pointer to a kernel internal completion signalled when
* drm_send_event ( ) is called , useful to internally synchronize with
* nonblocking operations .
*/
2017-03-08 15:12:42 +01:00
struct completion * completion ;
2017-03-08 15:12:44 +01:00
/**
* @ completion_release :
*
* Optional callback currently only used by the atomic modeset helpers
* to clean up the reference count for the structure @ completion is
* stored in .
*/
2017-03-08 15:12:42 +01:00
void ( * completion_release ) ( struct completion * completion ) ;
2017-03-08 15:12:44 +01:00
/**
* @ event :
*
* Pointer to the actual event that should be sent to userspace to be
* read using drm_read ( ) . Can be optional , since nowadays events are
* also used to signal kernel internal threads with @ completion or DMA
* transactions using @ fence .
*/
2017-03-08 15:12:42 +01:00
struct drm_event * event ;
2017-03-08 15:12:44 +01:00
/**
* @ fence :
*
* Optional DMA fence to unblock other hardware transactions which
* depend upon the nonblocking DRM operation this event represents .
*/
2017-03-08 15:12:42 +01:00
struct dma_fence * fence ;
2017-03-08 15:12:44 +01:00
/**
* @ file_priv :
*
* & struct drm_file where @ event should be delivered to . Only set when
* @ event is set .
*/
struct drm_file * file_priv ;
/**
* @ link :
*
* Double - linked list to keep track of this event . Can be used by the
* driver up to the point when it calls drm_send_event ( ) , after that
* this list entry is owned by the core for its own book - keeping .
*/
2017-03-08 15:12:42 +01:00
struct list_head link ;
2017-03-08 15:12:44 +01:00
/**
* @ pending_link :
*
* Entry on & drm_file . pending_event_list , to keep track of all pending
* events for @ file_priv , to allow correct unwinding of them when
* userspace closes the file before the event is delivered .
*/
2017-03-08 15:12:42 +01:00
struct list_head pending_link ;
} ;
2017-03-08 15:12:44 +01:00
/**
* struct drm_file - DRM file private data
*
* This structure tracks DRM state per open file descriptor .
*/
2017-03-08 15:12:42 +01:00
struct drm_file {
2017-03-08 15:12:44 +01:00
/**
* @ authenticated :
*
* Whether the client is allowed to submit rendering , which for legacy
* nodes means it must be authenticated .
*
* See also the : ref : ` section on primary nodes and authentication
* < drm_primary_node > ` .
*/
2018-11-02 14:25:42 +01:00
bool authenticated ;
2017-03-08 15:12:44 +01:00
/**
* @ stereo_allowed :
*
* True when the client has asked us to expose stereo 3 D mode flags .
*/
2018-11-02 14:25:42 +01:00
bool stereo_allowed ;
2017-03-08 15:12:44 +01:00
/**
* @ universal_planes :
*
* True if client understands CRTC primary planes and cursor planes
* in the plane list . Automatically set when @ atomic is set .
2017-03-08 15:12:42 +01:00
*/
2018-11-02 14:25:42 +01:00
bool universal_planes ;
2017-03-08 15:12:44 +01:00
/** @atomic: True if client understands atomic properties. */
2018-11-02 14:25:42 +01:00
bool atomic ;
2017-03-08 15:12:44 +01:00
2018-05-08 16:39:41 +05:30
/**
* @ aspect_ratio_allowed :
*
* True , if client can handle picture aspect ratios , and has requested
* to pass this information along with the mode .
*/
2018-11-02 14:25:42 +01:00
bool aspect_ratio_allowed ;
2018-05-08 16:39:41 +05:30
2018-02-28 14:11:23 +00:00
/**
* @ writeback_connectors :
*
* True if client understands writeback connectors
*/
2018-11-02 14:25:42 +01:00
bool writeback_connectors ;
2018-02-28 14:11:23 +00:00
2020-03-19 17:29:29 +00:00
/**
* @ was_master :
*
* This client has or had , master capability . Protected by struct
* & drm_device . master_mutex .
*
* This is used to ensure that CAP_SYS_ADMIN is not enforced , if the
* client is or was master in the past .
*/
bool was_master ;
2017-03-08 15:12:44 +01:00
/**
* @ is_master :
*
* This client is the creator of @ master . Protected by struct
* & drm_device . master_mutex .
*
* See also the : ref : ` section on primary nodes and authentication
* < drm_primary_node > ` .
2017-03-08 15:12:42 +01:00
*/
2018-11-02 14:25:42 +01:00
bool is_master ;
2017-03-08 15:12:42 +01:00
2017-03-08 15:12:44 +01:00
/**
* @ master :
*
drm: serialize drm_file.master with a new spinlock
Currently, drm_file.master pointers should be protected by
drm_device.master_mutex when being dereferenced. This is because
drm_file.master is not invariant for the lifetime of drm_file. If
drm_file is not the creator of master, then drm_file.is_master is
false, and a call to drm_setmaster_ioctl will invoke
drm_new_set_master, which then allocates a new master for drm_file and
puts the old master.
Thus, without holding drm_device.master_mutex, the old value of
drm_file.master could be freed while it is being used by another
concurrent process.
However, it is not always possible to lock drm_device.master_mutex to
dereference drm_file.master. Through the fbdev emulation code, this
might occur in a deep nest of other locks. But drm_device.master_mutex
is also the outermost lock in the nesting hierarchy, so this leads to
potential deadlocks.
To address this, we introduce a new spin lock at the bottom of the
lock hierarchy that only serializes drm_file.master. With this change,
the value of drm_file.master changes only when both
drm_device.master_mutex and drm_file.master_lookup_lock are
held. Hence, any process holding either of those locks can ensure that
the value of drm_file.master will not change concurrently.
Since no lock depends on the new drm_file.master_lookup_lock, when
drm_file.master is dereferenced, but drm_device.master_mutex cannot be
held, we can safely protect the master pointer with
drm_file.master_lookup_lock.
Reported-by: Daniel Vetter <daniel.vetter@ffwll.ch>
Signed-off-by: Desmond Cheong Zhi Xi <desmondcheongzx@gmail.com>
Signed-off-by: Daniel Vetter <daniel.vetter@ffwll.ch>
Link: https://patchwork.freedesktop.org/patch/msgid/20210712043508.11584-5-desmondcheongzx@gmail.com
2021-07-12 12:35:07 +08:00
* Master this node is currently associated with . Protected by struct
* & drm_device . master_mutex , and serialized by @ master_lookup_lock .
*
* Only relevant if drm_is_primary_client ( ) returns true . Note that
* this only matches & drm_device . master if the master is the currently
* active one .
2017-03-08 15:12:44 +01:00
*
2021-08-02 18:59:57 +08:00
* To update @ master , both & drm_device . master_mutex and
* @ master_lookup_lock need to be held , therefore holding either of
* them is safe and enough for the read side .
*
2021-07-12 12:35:08 +08:00
* When dereferencing this pointer , either hold struct
* & drm_device . master_mutex for the duration of the pointer ' s use , or
* use drm_file_get_master ( ) if struct & drm_device . master_mutex is not
* currently held and there is no other need to hold it . This prevents
* @ master from being freed during use .
*
2017-03-08 15:12:44 +01:00
* See also @ authentication and @ is_master and the : ref : ` section on
* primary nodes and authentication < drm_primary_node > ` .
*/
struct drm_master * master ;
drm: serialize drm_file.master with a new spinlock
Currently, drm_file.master pointers should be protected by
drm_device.master_mutex when being dereferenced. This is because
drm_file.master is not invariant for the lifetime of drm_file. If
drm_file is not the creator of master, then drm_file.is_master is
false, and a call to drm_setmaster_ioctl will invoke
drm_new_set_master, which then allocates a new master for drm_file and
puts the old master.
Thus, without holding drm_device.master_mutex, the old value of
drm_file.master could be freed while it is being used by another
concurrent process.
However, it is not always possible to lock drm_device.master_mutex to
dereference drm_file.master. Through the fbdev emulation code, this
might occur in a deep nest of other locks. But drm_device.master_mutex
is also the outermost lock in the nesting hierarchy, so this leads to
potential deadlocks.
To address this, we introduce a new spin lock at the bottom of the
lock hierarchy that only serializes drm_file.master. With this change,
the value of drm_file.master changes only when both
drm_device.master_mutex and drm_file.master_lookup_lock are
held. Hence, any process holding either of those locks can ensure that
the value of drm_file.master will not change concurrently.
Since no lock depends on the new drm_file.master_lookup_lock, when
drm_file.master is dereferenced, but drm_device.master_mutex cannot be
held, we can safely protect the master pointer with
drm_file.master_lookup_lock.
Reported-by: Daniel Vetter <daniel.vetter@ffwll.ch>
Signed-off-by: Desmond Cheong Zhi Xi <desmondcheongzx@gmail.com>
Signed-off-by: Daniel Vetter <daniel.vetter@ffwll.ch>
Link: https://patchwork.freedesktop.org/patch/msgid/20210712043508.11584-5-desmondcheongzx@gmail.com
2021-07-12 12:35:07 +08:00
/** @master_lock: Serializes @master. */
spinlock_t master_lookup_lock ;
2017-03-08 15:12:44 +01:00
/** @pid: Process that opened this file. */
2017-03-08 15:12:42 +01:00
struct pid * pid ;
2017-03-08 15:12:44 +01:00
/** @magic: Authentication magic, see @authenticated. */
2017-03-08 15:12:42 +01:00
drm_magic_t magic ;
2017-03-08 15:12:44 +01:00
/**
* @ lhead :
*
* List of all open files of a DRM device , linked into
* & drm_device . filelist . Protected by & drm_device . filelist_mutex .
*/
2017-03-08 15:12:42 +01:00
struct list_head lhead ;
2017-03-08 15:12:44 +01:00
/** @minor: &struct drm_minor for this file. */
2017-03-08 15:12:42 +01:00
struct drm_minor * minor ;
2017-03-08 15:12:44 +01:00
/**
* @ object_idr :
*
* Mapping of mm object handles to object pointers . Used by the GEM
* subsystem . Protected by @ table_lock .
*/
2017-03-08 15:12:42 +01:00
struct idr object_idr ;
2017-03-08 15:12:44 +01:00
/** @table_lock: Protects @object_idr. */
2017-03-08 15:12:42 +01:00
spinlock_t table_lock ;
2017-04-04 13:26:24 +10:00
/** @syncobj_idr: Mapping of sync object handles to object pointers. */
struct idr syncobj_idr ;
/** @syncobj_table_lock: Protects @syncobj_idr. */
spinlock_t syncobj_table_lock ;
2017-03-08 15:12:44 +01:00
/** @filp: Pointer to the core file structure. */
2017-03-08 15:12:42 +01:00
struct file * filp ;
2017-03-08 15:12:44 +01:00
/**
* @ driver_priv :
*
* Optional pointer for driver private data . Can be allocated in
* & drm_driver . open and should be freed in & drm_driver . postclose .
*/
2017-03-08 15:12:42 +01:00
void * driver_priv ;
/**
2017-03-08 15:12:44 +01:00
* @ fbs :
*
* List of & struct drm_framebuffer associated with this file , using the
* & drm_framebuffer . filp_head entry .
2017-03-08 15:12:42 +01:00
*
2017-03-08 15:12:44 +01:00
* Protected by @ fbs_lock . Note that the @ fbs list holds a reference on
* the framebuffer object to prevent it from untimely disappearing .
2017-03-08 15:12:42 +01:00
*/
struct list_head fbs ;
2017-03-08 15:12:44 +01:00
/** @fbs_lock: Protects @fbs. */
2017-03-08 15:12:42 +01:00
struct mutex fbs_lock ;
2017-03-08 15:12:44 +01:00
/**
* @ blobs :
*
* User - created blob properties ; this retains a reference on the
* property .
*
* Protected by @ drm_mode_config . blob_lock ;
*/
2017-03-08 15:12:42 +01:00
struct list_head blobs ;
2017-03-08 15:12:44 +01:00
/** @event_wait: Waitqueue for new events added to @event_list. */
2017-03-08 15:12:42 +01:00
wait_queue_head_t event_wait ;
2017-03-08 15:12:44 +01:00
/**
* @ pending_event_list :
*
* List of pending & struct drm_pending_event , used to clean up pending
* events in case this file gets closed before the event is signalled .
* Uses the & drm_pending_event . pending_link entry .
*
* Protect by & drm_device . event_lock .
*/
2017-03-08 15:12:42 +01:00
struct list_head pending_event_list ;
2017-03-08 15:12:44 +01:00
/**
* @ event_list :
*
* List of & struct drm_pending_event , ready for delivery to userspace
* through drm_read ( ) . Uses the & drm_pending_event . link entry .
*
* Protect by & drm_device . event_lock .
*/
2017-03-08 15:12:42 +01:00
struct list_head event_list ;
2017-03-08 15:12:44 +01:00
/**
* @ event_space :
*
* Available event space to prevent userspace from
* exhausting kernel memory . Currently limited to the fairly arbitrary
* value of 4 KB .
*/
2017-03-08 15:12:42 +01:00
int event_space ;
2017-03-08 15:12:44 +01:00
/** @event_read_lock: Serializes drm_read(). */
2017-03-08 15:12:42 +01:00
struct mutex event_read_lock ;
2017-03-08 15:12:44 +01:00
/**
* @ prime :
*
* Per - file buffer caches used by the PRIME buffer sharing code .
*/
2017-03-08 15:12:42 +01:00
struct drm_prime_file_private prime ;
2017-03-08 15:12:44 +01:00
/* private: */
2019-04-23 10:01:50 +10:00
# if IS_ENABLED(CONFIG_DRM_LEGACY)
2017-03-08 15:12:44 +01:00
unsigned long lock_count ; /* DRI1 legacy lock count */
2019-04-23 10:01:50 +10:00
# endif
2017-03-08 15:12:42 +01:00
} ;
2017-03-08 15:12:44 +01:00
/**
* drm_is_primary_client - is this an open file of the primary node
* @ file_priv : DRM file
*
* Returns true if this is an open file of the primary node , i . e .
* & drm_file . minor of @ file_priv is a primary minor .
*
* See also the : ref : ` section on primary nodes and authentication
* < drm_primary_node > ` .
*/
static inline bool drm_is_primary_client ( const struct drm_file * file_priv )
{
return file_priv - > minor - > type = = DRM_MINOR_PRIMARY ;
}
/**
* drm_is_render_client - is this an open file of the render node
* @ file_priv : DRM file
*
* Returns true if this is an open file of the render node , i . e .
* & drm_file . minor of @ file_priv is a render minor .
*
* See also the : ref : ` section on render nodes < drm_render_node > ` .
*/
2017-03-08 15:12:42 +01:00
static inline bool drm_is_render_client ( const struct drm_file * file_priv )
{
return file_priv - > minor - > type = = DRM_MINOR_RENDER ;
}
int drm_open ( struct inode * inode , struct file * filp ) ;
ssize_t drm_read ( struct file * filp , char __user * buffer ,
size_t count , loff_t * offset ) ;
int drm_release ( struct inode * inode , struct file * filp ) ;
2020-01-24 12:56:26 +00:00
int drm_release_noglobal ( struct inode * inode , struct file * filp ) ;
2017-07-03 06:39:46 -04:00
__poll_t drm_poll ( struct file * filp , struct poll_table_struct * wait ) ;
2017-03-08 15:12:42 +01:00
int drm_event_reserve_init_locked ( struct drm_device * dev ,
struct drm_file * file_priv ,
struct drm_pending_event * p ,
struct drm_event * e ) ;
int drm_event_reserve_init ( struct drm_device * dev ,
struct drm_file * file_priv ,
struct drm_pending_event * p ,
struct drm_event * e ) ;
void drm_event_cancel_free ( struct drm_device * dev ,
struct drm_pending_event * p ) ;
void drm_send_event_locked ( struct drm_device * dev , struct drm_pending_event * e ) ;
void drm_send_event ( struct drm_device * dev , struct drm_pending_event * e ) ;
2021-01-15 16:31:47 -08:00
void drm_send_event_timestamp_locked ( struct drm_device * dev ,
struct drm_pending_event * e ,
ktime_t timestamp ) ;
2017-03-08 15:12:42 +01:00
2019-11-07 18:05:58 +00:00
struct file * mock_drm_getfile ( struct drm_minor * minor , unsigned int flags ) ;
2020-03-24 18:49:26 +01:00
# ifdef CONFIG_MMU
struct drm_vma_offset_manager ;
unsigned long drm_get_unmapped_area ( struct file * file ,
unsigned long uaddr , unsigned long len ,
unsigned long pgoff , unsigned long flags ,
struct drm_vma_offset_manager * mgr ) ;
# endif /* CONFIG_MMU */
2017-03-08 15:12:42 +01:00
# endif /* _DRM_FILE_H_ */