2012-09-21 10:07:47 +02:00
/*
* Freescale i . MX drm driver
*
* Copyright ( C ) 2011 Sascha Hauer , Pengutronix
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version 2
* of the License , or ( at your option ) any later version .
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
*/
2013-11-03 11:23:34 +00:00
# include <linux/component.h>
2012-09-21 10:07:47 +02:00
# include <linux/device.h>
2016-07-08 17:40:59 +08:00
# include <linux/dma-buf.h>
2013-11-03 15:38:09 +00:00
# include <linux/module.h>
2012-09-21 10:07:47 +02:00
# include <linux/platform_device.h>
2016-07-08 17:40:59 +08:00
# include <linux/reservation.h>
2012-09-21 10:07:47 +02:00
# include <drm/drmP.h>
2016-07-08 17:40:59 +08:00
# include <drm/drm_atomic.h>
# include <drm/drm_atomic_helper.h>
2012-09-21 10:07:47 +02:00
# include <drm/drm_fb_helper.h>
# include <drm/drm_crtc_helper.h>
# include <drm/drm_gem_cma_helper.h>
# include <drm/drm_fb_cma_helper.h>
2014-10-29 10:03:57 +01:00
# include <drm/drm_plane_helper.h>
2014-07-03 17:52:57 +01:00
# include <drm/drm_of.h>
2016-05-12 15:00:44 +02:00
# include <video/imx-ipu-v3.h>
2012-09-21 10:07:47 +02:00
# include "imx-drm.h"
# define MAX_CRTC 4
2014-03-05 10:20:52 +01:00
struct imx_drm_component {
struct device_node * of_node ;
struct list_head list ;
} ;
2012-09-21 10:07:47 +02:00
struct imx_drm_device {
struct drm_device * drm ;
2013-11-24 13:23:17 +00:00
struct imx_drm_crtc * crtc [ MAX_CRTC ] ;
2015-12-15 12:20:51 +01:00
unsigned int pipes ;
2012-09-21 10:07:47 +02:00
struct drm_fbdev_cma * fbhelper ;
2016-07-08 17:40:59 +08:00
struct drm_atomic_state * state ;
2012-09-21 10:07:47 +02:00
} ;
struct imx_drm_crtc {
struct drm_crtc * crtc ;
struct imx_drm_crtc_helper_funcs imx_drm_helper_funcs ;
} ;
2015-10-27 13:40:57 +05:30
# if IS_ENABLED(CONFIG_DRM_FBDEV_EMULATION)
2013-11-03 12:13:47 +00:00
static int legacyfb_depth = 16 ;
module_param ( legacyfb_depth , int , 0444 ) ;
2015-10-27 13:40:57 +05:30
# endif
2013-11-03 12:13:47 +00:00
2012-09-21 10:07:47 +02:00
static void imx_drm_driver_lastclose ( struct drm_device * drm )
{
struct imx_drm_device * imxdrm = drm - > dev_private ;
2015-07-05 22:45:23 +02:00
drm_fbdev_cma_restore_mode ( imxdrm - > fbhelper ) ;
2012-09-21 10:07:47 +02:00
}
static int imx_drm_driver_unload ( struct drm_device * drm )
{
struct imx_drm_device * imxdrm = drm - > dev_private ;
2013-11-03 22:18:40 +00:00
drm_kms_helper_poll_fini ( drm ) ;
2012-09-21 10:07:47 +02:00
2013-11-03 12:13:47 +00:00
if ( imxdrm - > fbhelper )
drm_fbdev_cma_fini ( imxdrm - > fbhelper ) ;
2013-11-03 11:23:34 +00:00
component_unbind_all ( drm - > dev , drm ) ;
2013-12-17 19:10:47 +00:00
drm_vblank_cleanup ( drm ) ;
drm_mode_config_cleanup ( drm ) ;
2012-09-21 10:07:47 +02:00
2014-09-10 22:43:51 +08:00
platform_set_drvdata ( drm - > platformdev , NULL ) ;
2012-09-21 10:07:47 +02:00
return 0 ;
}
2015-09-24 18:35:31 +02:00
static int imx_drm_enable_vblank ( struct drm_device * drm , unsigned int pipe )
2012-09-21 10:07:47 +02:00
{
struct imx_drm_device * imxdrm = drm - > dev_private ;
2015-09-24 18:35:31 +02:00
struct imx_drm_crtc * imx_drm_crtc = imxdrm - > crtc [ pipe ] ;
2012-09-21 10:07:47 +02:00
int ret ;
if ( ! imx_drm_crtc )
return - EINVAL ;
if ( ! imx_drm_crtc - > imx_drm_helper_funcs . enable_vblank )
return - ENOSYS ;
ret = imx_drm_crtc - > imx_drm_helper_funcs . enable_vblank (
imx_drm_crtc - > crtc ) ;
return ret ;
}
2015-09-24 18:35:31 +02:00
static void imx_drm_disable_vblank ( struct drm_device * drm , unsigned int pipe )
2012-09-21 10:07:47 +02:00
{
struct imx_drm_device * imxdrm = drm - > dev_private ;
2015-09-24 18:35:31 +02:00
struct imx_drm_crtc * imx_drm_crtc = imxdrm - > crtc [ pipe ] ;
2012-09-21 10:07:47 +02:00
if ( ! imx_drm_crtc )
return ;
if ( ! imx_drm_crtc - > imx_drm_helper_funcs . disable_vblank )
return ;
imx_drm_crtc - > imx_drm_helper_funcs . disable_vblank ( imx_drm_crtc - > crtc ) ;
}
static const struct file_operations imx_drm_driver_fops = {
. owner = THIS_MODULE ,
. open = drm_open ,
. release = drm_release ,
. unlocked_ioctl = drm_ioctl ,
. mmap = drm_gem_cma_mmap ,
. poll = drm_poll ,
. read = drm_read ,
. llseek = noop_llseek ,
} ;
2013-11-03 13:28:24 +00:00
void imx_drm_connector_destroy ( struct drm_connector * connector )
{
2014-05-29 16:57:41 +01:00
drm_connector_unregister ( connector ) ;
2013-11-03 13:28:24 +00:00
drm_connector_cleanup ( connector ) ;
}
EXPORT_SYMBOL_GPL ( imx_drm_connector_destroy ) ;
void imx_drm_encoder_destroy ( struct drm_encoder * encoder )
{
drm_encoder_cleanup ( encoder ) ;
}
EXPORT_SYMBOL_GPL ( imx_drm_encoder_destroy ) ;
2013-11-03 22:18:40 +00:00
static void imx_drm_output_poll_changed ( struct drm_device * drm )
{
struct imx_drm_device * imxdrm = drm - > dev_private ;
drm_fbdev_cma_hotplug_event ( imxdrm - > fbhelper ) ;
}
2015-12-15 12:21:09 +01:00
static const struct drm_mode_config_funcs imx_drm_mode_config_funcs = {
2013-11-03 16:04:48 +00:00
. fb_create = drm_fb_cma_create ,
2013-11-03 22:18:40 +00:00
. output_poll_changed = imx_drm_output_poll_changed ,
2016-07-08 17:40:59 +08:00
. atomic_check = drm_atomic_helper_check ,
. atomic_commit = drm_atomic_helper_commit ,
} ;
static void imx_drm_atomic_commit_tail ( struct drm_atomic_state * state )
{
struct drm_device * dev = state - > dev ;
struct drm_crtc * crtc ;
struct drm_crtc_state * crtc_state ;
struct drm_plane_state * plane_state ;
struct drm_gem_cma_object * cma_obj ;
struct fence * excl ;
unsigned shared_count ;
struct fence * * shared ;
unsigned int i , j ;
int ret ;
/* Wait for fences. */
for_each_crtc_in_state ( state , crtc , crtc_state , i ) {
plane_state = crtc - > primary - > state ;
if ( plane_state - > fb ) {
cma_obj = drm_fb_cma_get_gem_obj ( plane_state - > fb , 0 ) ;
if ( cma_obj - > base . dma_buf ) {
ret = reservation_object_get_fences_rcu (
cma_obj - > base . dma_buf - > resv , & excl ,
& shared_count , & shared ) ;
if ( unlikely ( ret ) )
DRM_ERROR ( " failed to get fences "
" for buffer \n " ) ;
if ( excl ) {
fence_wait ( excl , false ) ;
fence_put ( excl ) ;
}
for ( j = 0 ; j < shared_count ; i + + ) {
fence_wait ( shared [ j ] , false ) ;
fence_put ( shared [ j ] ) ;
}
}
}
}
drm_atomic_helper_commit_modeset_disables ( dev , state ) ;
drm_atomic_helper_commit_planes ( dev , state , true ) ;
drm_atomic_helper_commit_modeset_enables ( dev , state ) ;
drm_atomic_helper_commit_hw_done ( state ) ;
drm_atomic_helper_wait_for_vblanks ( dev , state ) ;
drm_atomic_helper_cleanup_planes ( dev , state ) ;
}
static struct drm_mode_config_helper_funcs imx_drm_mode_config_helpers = {
. atomic_commit_tail = imx_drm_atomic_commit_tail ,
2013-11-03 16:04:48 +00:00
} ;
2012-09-21 10:07:47 +02:00
/*
2013-11-03 11:23:34 +00:00
* Main DRM initialisation . This binds , initialises and registers
* with DRM the subcomponents of the driver .
2012-09-21 10:07:47 +02:00
*/
static int imx_drm_driver_load ( struct drm_device * drm , unsigned long flags )
{
2013-11-03 15:31:22 +00:00
struct imx_drm_device * imxdrm ;
2013-11-03 18:39:29 +00:00
struct drm_connector * connector ;
2012-09-21 10:07:47 +02:00
int ret ;
2013-11-03 15:31:22 +00:00
imxdrm = devm_kzalloc ( drm - > dev , sizeof ( * imxdrm ) , GFP_KERNEL ) ;
if ( ! imxdrm )
return - ENOMEM ;
2012-09-21 10:07:47 +02:00
imxdrm - > drm = drm ;
drm - > dev_private = imxdrm ;
/*
* enable drm irq mode .
2013-10-04 14:53:37 +03:00
* - with irq_enabled = true , we can use the vblank feature .
2012-09-21 10:07:47 +02:00
*
* P . S . note that we wouldn ' t use drm irq handler but
* just specific driver own one instead because
* drm framework supports only one irq handler and
* drivers can well take care of their interrupts
*/
2013-10-04 14:53:37 +03:00
drm - > irq_enabled = true ;
2012-09-21 10:07:47 +02:00
2013-11-03 16:04:48 +00:00
/*
* set max width and height as default value ( 4096 x4096 ) .
* this value would be used to check framebuffer size limitation
* at drm_mode_addfb ( ) .
*/
drm - > mode_config . min_width = 64 ;
drm - > mode_config . min_height = 64 ;
drm - > mode_config . max_width = 4096 ;
drm - > mode_config . max_height = 4096 ;
drm - > mode_config . funcs = & imx_drm_mode_config_funcs ;
2016-07-08 17:40:59 +08:00
drm - > mode_config . helper_private = & imx_drm_mode_config_helpers ;
2013-11-03 16:04:48 +00:00
2012-09-21 10:07:47 +02:00
drm_mode_config_init ( drm ) ;
2013-12-17 19:10:47 +00:00
ret = drm_vblank_init ( drm , MAX_CRTC ) ;
2012-09-21 10:07:47 +02:00
if ( ret )
2013-12-16 11:33:44 +00:00
goto err_kms ;
2012-09-21 10:07:47 +02:00
2013-11-03 14:31:09 +01:00
platform_set_drvdata ( drm - > platformdev , drm ) ;
2013-11-03 13:55:34 +00:00
2013-11-03 11:23:34 +00:00
/* Now try and bind all our sub-components */
ret = component_bind_all ( drm - > dev , drm ) ;
if ( ret )
2013-11-03 15:20:18 +00:00
goto err_vblank ;
2013-11-03 18:39:29 +00:00
/*
* All components are now added , we can publish the connector sysfs
* entries to userspace . This will generate hotplug events and so
* userspace will expect to be able to access DRM at this point .
*/
list_for_each_entry ( connector , & drm - > mode_config . connector_list , head ) {
2014-05-29 16:57:41 +01:00
ret = drm_connector_register ( connector ) ;
2013-11-03 18:39:29 +00:00
if ( ret ) {
dev_err ( drm - > dev ,
2014-05-29 16:57:41 +01:00
" [CONNECTOR:%d:%s] drm_connector_register failed: %d \n " ,
2013-11-03 18:39:29 +00:00
connector - > base . id ,
2014-06-03 14:56:16 +03:00
connector - > name , ret ) ;
2013-11-03 18:39:29 +00:00
goto err_unbind ;
}
}
2016-07-08 17:40:56 +08:00
drm_mode_config_reset ( drm ) ;
2013-11-03 12:13:47 +00:00
/*
* All components are now initialised , so setup the fb helper .
* The fb helper takes copies of key hardware information , so the
* crtcs / connectors / encoders must not change after this point .
*/
2015-10-27 13:40:57 +05:30
# if IS_ENABLED(CONFIG_DRM_FBDEV_EMULATION)
2013-11-03 12:13:47 +00:00
if ( legacyfb_depth ! = 16 & & legacyfb_depth ! = 32 ) {
dev_warn ( drm - > dev , " Invalid legacyfb_depth. Defaulting to 16bpp \n " ) ;
legacyfb_depth = 16 ;
}
imxdrm - > fbhelper = drm_fbdev_cma_init ( drm , legacyfb_depth ,
drm - > mode_config . num_crtc , MAX_CRTC ) ;
if ( IS_ERR ( imxdrm - > fbhelper ) ) {
ret = PTR_ERR ( imxdrm - > fbhelper ) ;
imxdrm - > fbhelper = NULL ;
goto err_unbind ;
}
# endif
2013-11-03 22:18:40 +00:00
drm_kms_helper_poll_init ( drm ) ;
2013-12-16 11:33:44 +00:00
return 0 ;
2013-11-03 14:31:09 +01:00
2013-11-03 18:39:29 +00:00
err_unbind :
component_unbind_all ( drm - > dev , drm ) ;
2013-11-03 15:20:18 +00:00
err_vblank :
2013-12-16 11:33:44 +00:00
drm_vblank_cleanup ( drm ) ;
err_kms :
drm_mode_config_cleanup ( drm ) ;
2012-09-21 10:07:47 +02:00
return ret ;
}
/*
* imx_drm_add_crtc - add a new crtc
*/
2013-11-03 12:26:23 +00:00
int imx_drm_add_crtc ( struct drm_device * drm , struct drm_crtc * crtc ,
2015-11-06 11:08:02 +01:00
struct imx_drm_crtc * * new_crtc , struct drm_plane * primary_plane ,
2012-09-21 10:07:47 +02:00
const struct imx_drm_crtc_helper_funcs * imx_drm_helper_funcs ,
2014-03-05 10:20:52 +01:00
struct device_node * port )
2012-09-21 10:07:47 +02:00
{
2013-11-03 12:26:23 +00:00
struct imx_drm_device * imxdrm = drm - > dev_private ;
2012-09-21 10:07:47 +02:00
struct imx_drm_crtc * imx_drm_crtc ;
2013-12-16 12:39:31 +00:00
/*
* The vblank arrays are dimensioned by MAX_CRTC - we can ' t
* pass IDs greater than this to those functions .
*/
2013-11-03 15:38:09 +00:00
if ( imxdrm - > pipes > = MAX_CRTC )
return - EINVAL ;
2013-12-16 12:39:31 +00:00
2013-11-03 15:38:09 +00:00
if ( imxdrm - > drm - > open_count )
return - EBUSY ;
2012-09-21 10:07:47 +02:00
imx_drm_crtc = kzalloc ( sizeof ( * imx_drm_crtc ) , GFP_KERNEL ) ;
2013-11-03 15:38:09 +00:00
if ( ! imx_drm_crtc )
return - ENOMEM ;
2012-09-21 10:07:47 +02:00
imx_drm_crtc - > imx_drm_helper_funcs = * imx_drm_helper_funcs ;
imx_drm_crtc - > crtc = crtc ;
2014-07-03 17:52:57 +01:00
crtc - > port = port ;
2015-12-15 12:20:52 +01:00
imxdrm - > crtc [ imxdrm - > pipes + + ] = imx_drm_crtc ;
2012-09-21 10:07:47 +02:00
* new_crtc = imx_drm_crtc ;
2013-12-17 19:11:07 +00:00
drm_crtc_helper_add ( crtc ,
imx_drm_crtc - > imx_drm_helper_funcs . crtc_helper_funcs ) ;
2015-11-06 11:08:02 +01:00
drm_crtc_init_with_planes ( drm , crtc , primary_plane , NULL ,
drm: Pass 'name' to drm_crtc_init_with_planes()
Done with coccinelle for the most part. However, it thinks '...' is
part of the semantic patch, so I put an 'int DOTDOTDOT' placeholder
in its place and got rid of it with sed afterwards.
I didn't convert drm_crtc_init() since passing the varargs through
would mean either cpp macros or va_list, and I figured we don't
care about these legacy functions enough to warrant the extra pain.
@@
identifier dev, crtc, primary, cursor, funcs;
@@
int drm_crtc_init_with_planes(struct drm_device *dev,
struct drm_crtc *crtc,
struct drm_plane *primary, struct drm_plane *cursor,
const struct drm_crtc_funcs *funcs
+ ,const char *name, int DOTDOTDOT
)
{ ... }
@@
identifier dev, crtc, primary, cursor, funcs;
@@
int drm_crtc_init_with_planes(struct drm_device *dev,
struct drm_crtc *crtc,
struct drm_plane *primary, struct drm_plane *cursor,
const struct drm_crtc_funcs *funcs
+ ,const char *name, int DOTDOTDOT
);
@@
expression E1, E2, E3, E4, E5;
@@
drm_crtc_init_with_planes(E1, E2, E3, E4, E5
+ ,NULL
)
v2: Split crtc and plane changes apart
Pass NULL for no-name instead of ""
Leave drm_crtc_init() alone
v3: Add ', or NULL...' to @name kernel doc (Jani)
Annotate the function with __printf() attribute (Jani)
Signed-off-by: Ville Syrjälä <ville.syrjala@linux.intel.com>
Signed-off-by: Daniel Vetter <daniel.vetter@ffwll.ch>
Link: http://patchwork.freedesktop.org/patch/msgid/1449670771-2751-1-git-send-email-ville.syrjala@linux.intel.com
2015-12-09 16:19:31 +02:00
imx_drm_crtc - > imx_drm_helper_funcs . crtc_funcs , NULL ) ;
2013-12-17 19:11:07 +00:00
2012-09-21 10:07:47 +02:00
return 0 ;
}
EXPORT_SYMBOL_GPL ( imx_drm_add_crtc ) ;
/*
* imx_drm_remove_crtc - remove a crtc
*/
int imx_drm_remove_crtc ( struct imx_drm_crtc * imx_drm_crtc )
{
2013-11-03 15:20:18 +00:00
struct imx_drm_device * imxdrm = imx_drm_crtc - > crtc - > dev - > dev_private ;
2015-12-15 12:20:52 +01:00
unsigned int pipe = drm_crtc_index ( imx_drm_crtc - > crtc ) ;
2012-09-21 10:07:47 +02:00
drm_crtc_cleanup ( imx_drm_crtc - > crtc ) ;
2015-12-15 12:20:52 +01:00
imxdrm - > crtc [ pipe ] = NULL ;
2012-09-21 10:07:47 +02:00
kfree ( imx_drm_crtc ) ;
return 0 ;
}
EXPORT_SYMBOL_GPL ( imx_drm_remove_crtc ) ;
2014-07-03 17:52:57 +01:00
int imx_drm_encoder_parse_of ( struct drm_device * drm ,
struct drm_encoder * encoder , struct device_node * np )
2013-11-03 14:04:47 +00:00
{
2014-07-03 17:52:57 +01:00
uint32_t crtc_mask = drm_of_find_possible_crtcs ( drm , np ) ;
2013-11-03 14:04:47 +00:00
2014-07-03 17:52:57 +01:00
/*
* If we failed to find the CRTC ( s ) which this encoder is
* supposed to be connected to , it ' s because the CRTC has
* not been registered yet . Defer probing , and hope that
* the required CRTC is added later .
*/
if ( crtc_mask = = 0 )
return - EPROBE_DEFER ;
2014-03-05 10:20:52 +01:00
2014-07-03 17:52:57 +01:00
encoder - > possible_crtcs = crtc_mask ;
2014-08-04 21:07:07 +02:00
2014-07-03 17:52:57 +01:00
/* FIXME: this is the mask of outputs which can clone this output. */
encoder - > possible_clones = ~ 0 ;
2013-11-03 14:04:47 +00:00
return 0 ;
}
2014-07-03 17:52:57 +01:00
EXPORT_SYMBOL_GPL ( imx_drm_encoder_parse_of ) ;
2013-11-03 14:04:47 +00:00
2013-08-02 13:27:49 -04:00
static const struct drm_ioctl_desc imx_drm_ioctls [ ] = {
2012-09-21 10:07:47 +02:00
/* none so far */
} ;
static struct drm_driver imx_drm_driver = {
2016-07-08 17:41:02 +08:00
. driver_features = DRIVER_MODESET | DRIVER_GEM | DRIVER_PRIME |
DRIVER_ATOMIC ,
2012-09-21 10:07:47 +02:00
. load = imx_drm_driver_load ,
. unload = imx_drm_driver_unload ,
. lastclose = imx_drm_driver_lastclose ,
2016-04-26 19:29:52 +02:00
. gem_free_object_unlocked = drm_gem_cma_free_object ,
2012-09-21 10:07:47 +02:00
. gem_vm_ops = & drm_gem_cma_vm_ops ,
. dumb_create = drm_gem_cma_dumb_create ,
. dumb_map_offset = drm_gem_cma_dumb_map_offset ,
2013-07-16 09:12:04 +02:00
. dumb_destroy = drm_gem_dumb_destroy ,
2012-09-21 10:07:47 +02:00
2013-10-10 16:18:46 +02:00
. prime_handle_to_fd = drm_gem_prime_handle_to_fd ,
. prime_fd_to_handle = drm_gem_prime_fd_to_handle ,
. gem_prime_import = drm_gem_prime_import ,
. gem_prime_export = drm_gem_prime_export ,
. gem_prime_get_sg_table = drm_gem_cma_prime_get_sg_table ,
. gem_prime_import_sg_table = drm_gem_cma_prime_import_sg_table ,
. gem_prime_vmap = drm_gem_cma_prime_vmap ,
. gem_prime_vunmap = drm_gem_cma_prime_vunmap ,
. gem_prime_mmap = drm_gem_cma_prime_mmap ,
2015-09-30 16:46:48 +03:00
. get_vblank_counter = drm_vblank_no_hw_counter ,
2012-09-21 10:07:47 +02:00
. enable_vblank = imx_drm_enable_vblank ,
. disable_vblank = imx_drm_disable_vblank ,
. ioctls = imx_drm_ioctls ,
. num_ioctls = ARRAY_SIZE ( imx_drm_ioctls ) ,
. fops = & imx_drm_driver_fops ,
. name = " imx-drm " ,
. desc = " i.MX DRM graphics " ,
. date = " 20120507 " ,
. major = 1 ,
. minor = 0 ,
. patchlevel = 0 ,
} ;
2013-11-03 11:23:34 +00:00
static int compare_of ( struct device * dev , void * data )
{
2014-03-05 10:20:52 +01:00
struct device_node * np = data ;
2013-11-03 11:23:34 +00:00
2016-05-12 15:00:44 +02:00
/* Special case for DI, dev->of_node may not be set yet */
if ( strcmp ( dev - > driver - > name , " imx-ipuv3-crtc " ) = = 0 ) {
struct ipu_client_platformdata * pdata = dev - > platform_data ;
return pdata - > of_node = = np ;
}
2014-03-05 10:20:52 +01:00
/* Special case for LDB, one device for two channels */
if ( of_node_cmp ( np - > name , " lvds-channel " ) = = 0 ) {
np = of_get_parent ( np ) ;
of_node_put ( np ) ;
2013-11-03 11:23:34 +00:00
}
2014-03-05 10:20:52 +01:00
return dev - > of_node = = np ;
}
2013-11-03 11:23:34 +00:00
static int imx_drm_bind ( struct device * dev )
{
return drm_platform_init ( & imx_drm_driver , to_platform_device ( dev ) ) ;
}
static void imx_drm_unbind ( struct device * dev )
{
drm_put_dev ( dev_get_drvdata ( dev ) ) ;
}
static const struct component_master_ops imx_drm_ops = {
. bind = imx_drm_bind ,
. unbind = imx_drm_unbind ,
} ;
2012-09-21 10:07:47 +02:00
static int imx_drm_platform_probe ( struct platform_device * pdev )
{
2015-10-20 10:23:13 +01:00
int ret = drm_of_component_probe ( & pdev - > dev , compare_of , & imx_drm_ops ) ;
2014-03-05 10:20:52 +01:00
2015-10-20 10:23:13 +01:00
if ( ! ret )
ret = dma_set_coherent_mask ( & pdev - > dev , DMA_BIT_MASK ( 32 ) ) ;
2014-03-05 10:20:52 +01:00
2015-10-20 10:23:13 +01:00
return ret ;
2012-09-21 10:07:47 +02:00
}
static int imx_drm_platform_remove ( struct platform_device * pdev )
{
2013-11-03 11:23:34 +00:00
component_master_del ( & pdev - > dev , & imx_drm_ops ) ;
2012-09-21 10:07:47 +02:00
return 0 ;
}
2014-09-10 22:43:51 +08:00
# ifdef CONFIG_PM_SLEEP
static int imx_drm_suspend ( struct device * dev )
{
struct drm_device * drm_dev = dev_get_drvdata ( dev ) ;
2016-07-08 17:40:59 +08:00
struct imx_drm_device * imxdrm ;
2014-09-10 22:43:51 +08:00
/* The drm_dev is NULL before .load hook is called */
if ( drm_dev = = NULL )
return 0 ;
drm_kms_helper_poll_disable ( drm_dev ) ;
2016-07-08 17:40:59 +08:00
imxdrm = drm_dev - > dev_private ;
imxdrm - > state = drm_atomic_helper_suspend ( drm_dev ) ;
if ( IS_ERR ( imxdrm - > state ) ) {
drm_kms_helper_poll_enable ( drm_dev ) ;
return PTR_ERR ( imxdrm - > state ) ;
}
2014-09-10 22:43:51 +08:00
return 0 ;
}
static int imx_drm_resume ( struct device * dev )
{
struct drm_device * drm_dev = dev_get_drvdata ( dev ) ;
2016-07-08 17:40:59 +08:00
struct imx_drm_device * imx_drm ;
2014-09-10 22:43:51 +08:00
if ( drm_dev = = NULL )
return 0 ;
2016-07-08 17:40:59 +08:00
imx_drm = drm_dev - > dev_private ;
drm_atomic_helper_resume ( drm_dev , imx_drm - > state ) ;
2014-09-10 22:43:51 +08:00
drm_kms_helper_poll_enable ( drm_dev ) ;
return 0 ;
}
# endif
static SIMPLE_DEV_PM_OPS ( imx_drm_pm_ops , imx_drm_suspend , imx_drm_resume ) ;
2013-11-03 11:23:34 +00:00
static const struct of_device_id imx_drm_dt_ids [ ] = {
2014-03-05 10:20:52 +01:00
{ . compatible = " fsl,imx-display-subsystem " , } ,
2013-11-03 11:23:34 +00:00
{ /* sentinel */ } ,
} ;
MODULE_DEVICE_TABLE ( of , imx_drm_dt_ids ) ;
2012-09-21 10:07:47 +02:00
static struct platform_driver imx_drm_pdrv = {
. probe = imx_drm_platform_probe ,
2012-11-19 13:20:51 -05:00
. remove = imx_drm_platform_remove ,
2012-09-21 10:07:47 +02:00
. driver = {
. name = " imx-drm " ,
2014-09-10 22:43:51 +08:00
. pm = & imx_drm_pm_ops ,
2013-11-03 11:23:34 +00:00
. of_match_table = imx_drm_dt_ids ,
2012-09-21 10:07:47 +02:00
} ,
} ;
2013-11-03 15:31:22 +00:00
module_platform_driver ( imx_drm_pdrv ) ;
2012-09-21 10:07:47 +02:00
MODULE_AUTHOR ( " Sascha Hauer <s.hauer@pengutronix.de> " ) ;
MODULE_DESCRIPTION ( " i.MX drm driver core " ) ;
MODULE_LICENSE ( " GPL " ) ;