2011-10-04 19:19:01 +09:00
/*
* Copyright ( c ) 2011 Samsung Electronics Co . , Ltd .
* Authors :
* Inki Dae < inki . dae @ samsung . com >
* Joonyoung Shim < jy0922 . shim @ samsung . com >
* Seung - Woo Kim < sw0312 . kim @ samsung . 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 .
*/
# include "drmP.h"
# include "drm.h"
2011-10-18 16:58:05 +09:00
# include "drm_crtc_helper.h"
2011-10-04 19:19:01 +09:00
# include <drm/exynos_drm.h>
# include "exynos_drm_drv.h"
# include "exynos_drm_crtc.h"
2012-02-15 11:25:19 +09:00
# include "exynos_drm_encoder.h"
2011-10-04 19:19:01 +09:00
# include "exynos_drm_fbdev.h"
# include "exynos_drm_fb.h"
# include "exynos_drm_gem.h"
2011-12-08 17:54:07 +09:00
# include "exynos_drm_plane.h"
2012-03-21 10:55:26 +09:00
# include "exynos_drm_vidi.h"
2011-10-04 19:19:01 +09:00
2011-12-15 17:31:24 +09:00
# define DRIVER_NAME "exynos"
2011-10-04 19:19:01 +09:00
# define DRIVER_DESC "Samsung SoC DRM"
# define DRIVER_DATE "20110530"
# define DRIVER_MAJOR 1
# define DRIVER_MINOR 0
2011-12-16 21:31:12 +09:00
# define VBLANK_OFF_DELAY 50000
2011-10-04 19:19:01 +09:00
static int exynos_drm_load ( struct drm_device * dev , unsigned long flags )
{
struct exynos_drm_private * private ;
int ret ;
int nr ;
DRM_DEBUG_DRIVER ( " %s \n " , __FILE__ ) ;
private = kzalloc ( sizeof ( struct exynos_drm_private ) , GFP_KERNEL ) ;
if ( ! private ) {
DRM_ERROR ( " failed to allocate private \n " ) ;
return - ENOMEM ;
}
INIT_LIST_HEAD ( & private - > pageflip_event_list ) ;
dev - > dev_private = ( void * ) private ;
drm_mode_config_init ( dev ) ;
2011-10-18 16:58:05 +09:00
/* init kms poll for handling hpd */
drm_kms_helper_poll_init ( dev ) ;
2011-10-04 19:19:01 +09:00
exynos_drm_mode_config_init ( dev ) ;
/*
* EXYNOS4 is enough to have two CRTCs and each crtc would be used
* without dependency of hardware .
*/
for ( nr = 0 ; nr < MAX_CRTC ; nr + + ) {
ret = exynos_drm_crtc_create ( dev , nr ) ;
if ( ret )
goto err_crtc ;
}
2011-12-08 17:54:07 +09:00
for ( nr = 0 ; nr < MAX_PLANE ; nr + + ) {
ret = exynos_plane_init ( dev , nr ) ;
if ( ret )
goto err_crtc ;
}
2011-10-04 19:19:01 +09:00
ret = drm_vblank_init ( dev , MAX_CRTC ) ;
if ( ret )
goto err_crtc ;
/*
* probe sub drivers such as display controller and hdmi driver ,
* that were registered at probe ( ) of platform driver
* to the sub driver and create encoder and connector for them .
*/
ret = exynos_drm_device_register ( dev ) ;
if ( ret )
goto err_vblank ;
2012-02-15 11:25:19 +09:00
/* setup possible_clones. */
exynos_drm_encoder_setup ( dev ) ;
2011-10-04 19:19:01 +09:00
/*
* create and configure fb helper and also exynos specific
* fbdev object .
*/
ret = exynos_drm_fbdev_init ( dev ) ;
if ( ret ) {
DRM_ERROR ( " failed to initialize drm fbdev \n " ) ;
goto err_drm_device ;
}
2011-12-16 21:31:12 +09:00
drm_vblank_offdelay = VBLANK_OFF_DELAY ;
2011-10-04 19:19:01 +09:00
return 0 ;
err_drm_device :
exynos_drm_device_unregister ( dev ) ;
err_vblank :
drm_vblank_cleanup ( dev ) ;
err_crtc :
drm_mode_config_cleanup ( dev ) ;
kfree ( private ) ;
return ret ;
}
static int exynos_drm_unload ( struct drm_device * dev )
{
DRM_DEBUG_DRIVER ( " %s \n " , __FILE__ ) ;
exynos_drm_fbdev_fini ( dev ) ;
exynos_drm_device_unregister ( dev ) ;
drm_vblank_cleanup ( dev ) ;
2011-10-18 16:58:05 +09:00
drm_kms_helper_poll_fini ( dev ) ;
2011-10-04 19:19:01 +09:00
drm_mode_config_cleanup ( dev ) ;
kfree ( dev - > dev_private ) ;
dev - > dev_private = NULL ;
return 0 ;
}
2012-03-16 18:47:09 +09:00
static int exynos_drm_open ( struct drm_device * dev , struct drm_file * file )
{
DRM_DEBUG_DRIVER ( " %s \n " , __FILE__ ) ;
return exynos_drm_subdrv_open ( dev , file ) ;
}
2011-10-14 13:29:51 +09:00
static void exynos_drm_preclose ( struct drm_device * dev ,
2012-02-15 11:25:18 +09:00
struct drm_file * file )
2011-10-14 13:29:51 +09:00
{
2012-03-16 18:47:07 +09:00
struct exynos_drm_private * private = dev - > dev_private ;
struct drm_pending_vblank_event * e , * t ;
unsigned long flags ;
2012-02-15 11:25:18 +09:00
DRM_DEBUG_DRIVER ( " %s \n " , __FILE__ ) ;
2011-10-14 13:29:51 +09:00
2012-03-16 18:47:07 +09:00
/* release events of current file */
spin_lock_irqsave ( & dev - > event_lock , flags ) ;
list_for_each_entry_safe ( e , t , & private - > pageflip_event_list ,
base . link ) {
if ( e - > base . file_priv = = file ) {
list_del ( & e - > base . link ) ;
e - > base . destroy ( & e - > base ) ;
}
}
spin_unlock_irqrestore ( & dev - > event_lock , flags ) ;
2012-03-16 18:47:09 +09:00
exynos_drm_subdrv_close ( dev , file ) ;
2011-10-14 13:29:51 +09:00
}
2012-02-15 11:25:22 +09:00
static void exynos_drm_postclose ( struct drm_device * dev , struct drm_file * file )
{
DRM_DEBUG_DRIVER ( " %s \n " , __FILE__ ) ;
if ( ! file - > driver_priv )
return ;
kfree ( file - > driver_priv ) ;
file - > driver_priv = NULL ;
}
2011-10-04 19:19:01 +09:00
static void exynos_drm_lastclose ( struct drm_device * dev )
{
DRM_DEBUG_DRIVER ( " %s \n " , __FILE__ ) ;
exynos_drm_fbdev_restore_mode ( dev ) ;
}
static struct vm_operations_struct exynos_drm_gem_vm_ops = {
. fault = exynos_drm_gem_fault ,
. open = drm_gem_vm_open ,
. close = drm_gem_vm_close ,
} ;
static struct drm_ioctl_desc exynos_ioctls [ ] = {
DRM_IOCTL_DEF_DRV ( EXYNOS_GEM_CREATE , exynos_drm_gem_create_ioctl ,
DRM_UNLOCKED | DRM_AUTH ) ,
DRM_IOCTL_DEF_DRV ( EXYNOS_GEM_MAP_OFFSET ,
exynos_drm_gem_map_offset_ioctl , DRM_UNLOCKED |
DRM_AUTH ) ,
DRM_IOCTL_DEF_DRV ( EXYNOS_GEM_MMAP ,
exynos_drm_gem_mmap_ioctl , DRM_UNLOCKED | DRM_AUTH ) ,
2011-12-08 17:54:07 +09:00
DRM_IOCTL_DEF_DRV ( EXYNOS_PLANE_SET_ZPOS , exynos_plane_set_zpos_ioctl ,
DRM_UNLOCKED | DRM_AUTH ) ,
2012-03-21 10:55:26 +09:00
DRM_IOCTL_DEF_DRV ( EXYNOS_VIDI_CONNECTION ,
vidi_connection_ioctl , DRM_UNLOCKED | DRM_AUTH ) ,
2011-10-04 19:19:01 +09:00
} ;
2011-12-08 15:00:20 +09:00
static const struct file_operations exynos_drm_driver_fops = {
. owner = THIS_MODULE ,
. open = drm_open ,
. mmap = exynos_drm_gem_mmap ,
. poll = drm_poll ,
. read = drm_read ,
. unlocked_ioctl = drm_ioctl ,
. release = drm_release ,
} ;
2011-10-04 19:19:01 +09:00
static struct drm_driver exynos_drm_driver = {
. driver_features = DRIVER_HAVE_IRQ | DRIVER_BUS_PLATFORM |
DRIVER_MODESET | DRIVER_GEM ,
. load = exynos_drm_load ,
. unload = exynos_drm_unload ,
2012-03-16 18:47:09 +09:00
. open = exynos_drm_open ,
2011-10-14 13:29:51 +09:00
. preclose = exynos_drm_preclose ,
2011-10-04 19:19:01 +09:00
. lastclose = exynos_drm_lastclose ,
2012-02-15 11:25:22 +09:00
. postclose = exynos_drm_postclose ,
2011-10-04 19:19:01 +09:00
. get_vblank_counter = drm_vblank_count ,
. enable_vblank = exynos_drm_crtc_enable_vblank ,
. disable_vblank = exynos_drm_crtc_disable_vblank ,
. gem_init_object = exynos_drm_gem_init_object ,
. gem_free_object = exynos_drm_gem_free_object ,
. gem_vm_ops = & exynos_drm_gem_vm_ops ,
. dumb_create = exynos_drm_gem_dumb_create ,
. dumb_map_offset = exynos_drm_gem_dumb_map_offset ,
. dumb_destroy = exynos_drm_gem_dumb_destroy ,
. ioctls = exynos_ioctls ,
2011-12-08 15:00:20 +09:00
. fops = & exynos_drm_driver_fops ,
2011-10-04 19:19:01 +09:00
. name = DRIVER_NAME ,
. desc = DRIVER_DESC ,
. date = DRIVER_DATE ,
. major = DRIVER_MAJOR ,
. minor = DRIVER_MINOR ,
} ;
static int exynos_drm_platform_probe ( struct platform_device * pdev )
{
DRM_DEBUG_DRIVER ( " %s \n " , __FILE__ ) ;
exynos_drm_driver . num_ioctls = DRM_ARRAY_SIZE ( exynos_ioctls ) ;
return drm_platform_init ( & exynos_drm_driver , pdev ) ;
}
static int exynos_drm_platform_remove ( struct platform_device * pdev )
{
DRM_DEBUG_DRIVER ( " %s \n " , __FILE__ ) ;
drm_platform_exit ( & exynos_drm_driver , pdev ) ;
return 0 ;
}
static struct platform_driver exynos_drm_platform_driver = {
. probe = exynos_drm_platform_probe ,
. remove = __devexit_p ( exynos_drm_platform_remove ) ,
. driver = {
. owner = THIS_MODULE ,
2012-03-05 12:02:30 +01:00
. name = " exynos-drm " ,
2011-10-04 19:19:01 +09:00
} ,
} ;
static int __init exynos_drm_init ( void )
{
2012-03-16 18:47:08 +09:00
int ret ;
2011-10-04 19:19:01 +09:00
DRM_DEBUG_DRIVER ( " %s \n " , __FILE__ ) ;
2012-03-16 18:47:08 +09:00
# ifdef CONFIG_DRM_EXYNOS_FIMD
ret = platform_driver_register ( & fimd_driver ) ;
if ( ret < 0 )
goto out_fimd ;
# endif
# ifdef CONFIG_DRM_EXYNOS_HDMI
ret = platform_driver_register ( & hdmi_driver ) ;
if ( ret < 0 )
goto out_hdmi ;
ret = platform_driver_register ( & mixer_driver ) ;
if ( ret < 0 )
goto out_mixer ;
ret = platform_driver_register ( & exynos_drm_common_hdmi_driver ) ;
if ( ret < 0 )
goto out_common_hdmi ;
# endif
2012-03-21 10:55:26 +09:00
# ifdef CONFIG_DRM_EXYNOS_VIDI
ret = platform_driver_register ( & vidi_driver ) ;
if ( ret < 0 )
goto out_vidi ;
# endif
2012-03-16 18:47:08 +09:00
ret = platform_driver_register ( & exynos_drm_platform_driver ) ;
if ( ret < 0 )
goto out ;
return 0 ;
out :
2012-03-21 10:55:26 +09:00
# ifdef CONFIG_DRM_EXYNOS_VIDI
out_vidi :
platform_driver_unregister ( & vidi_driver ) ;
# endif
2012-03-16 18:47:08 +09:00
# ifdef CONFIG_DRM_EXYNOS_HDMI
platform_driver_unregister ( & exynos_drm_common_hdmi_driver ) ;
out_common_hdmi :
platform_driver_unregister ( & mixer_driver ) ;
out_mixer :
platform_driver_unregister ( & hdmi_driver ) ;
out_hdmi :
# endif
# ifdef CONFIG_DRM_EXYNOS_FIMD
platform_driver_unregister ( & fimd_driver ) ;
out_fimd :
# endif
return ret ;
2011-10-04 19:19:01 +09:00
}
static void __exit exynos_drm_exit ( void )
{
DRM_DEBUG_DRIVER ( " %s \n " , __FILE__ ) ;
platform_driver_unregister ( & exynos_drm_platform_driver ) ;
2012-03-16 18:47:08 +09:00
# ifdef CONFIG_DRM_EXYNOS_HDMI
platform_driver_unregister ( & exynos_drm_common_hdmi_driver ) ;
platform_driver_unregister ( & mixer_driver ) ;
platform_driver_unregister ( & hdmi_driver ) ;
# endif
2012-03-21 10:55:26 +09:00
# ifdef CONFIG_DRM_EXYNOS_VIDI
platform_driver_unregister ( & vidi_driver ) ;
# endif
2012-03-16 18:47:08 +09:00
# ifdef CONFIG_DRM_EXYNOS_FIMD
platform_driver_unregister ( & fimd_driver ) ;
# endif
2011-10-04 19:19:01 +09:00
}
module_init ( exynos_drm_init ) ;
module_exit ( exynos_drm_exit ) ;
MODULE_AUTHOR ( " Inki Dae <inki.dae@samsung.com> " ) ;
MODULE_AUTHOR ( " Joonyoung Shim <jy0922.shim@samsung.com> " ) ;
MODULE_AUTHOR ( " Seung-Woo Kim <sw0312.kim@samsung.com> " ) ;
MODULE_DESCRIPTION ( " Samsung SoC DRM Driver " ) ;
MODULE_LICENSE ( " GPL " ) ;