2012-03-21 10:55:26 +09:00
/* exynos_drm_vidi.c
*
* Copyright ( C ) 2012 Samsung Electronics Co . Ltd
* Authors :
* Inki Dae < inki . dae @ samsung . com >
*
* 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 .
*
*/
2012-10-02 18:01:07 +01:00
# include <drm/drmP.h>
2012-03-21 10:55:26 +09:00
# include <linux/kernel.h>
# include <linux/platform_device.h>
2014-11-24 14:55:41 +09:00
# include <linux/component.h>
2016-09-23 10:15:23 +02:00
# include <linux/timer.h>
2012-03-21 10:55:26 +09:00
# include <drm/exynos_drm.h>
2012-10-02 18:01:07 +01:00
# include <drm/drm_edid.h>
# include <drm/drm_crtc_helper.h>
2015-06-01 12:04:44 -03:00
# include <drm/drm_atomic_helper.h>
2012-03-21 10:55:26 +09:00
# include "exynos_drm_drv.h"
# include "exynos_drm_crtc.h"
2015-11-30 14:53:21 +01:00
# include "exynos_drm_fb.h"
2015-04-03 21:03:40 +09:00
# include "exynos_drm_plane.h"
2013-08-13 00:46:40 +01:00
# include "exynos_drm_vidi.h"
2012-03-21 10:55:26 +09:00
2016-09-23 10:15:23 +02:00
/* VIDI uses fixed refresh rate of 50Hz */
# define VIDI_REFRESH_TIME (1000 / 50)
2012-03-21 10:55:26 +09:00
/* vidi has totally three virtual windows. */
# define WINDOWS_NR 3
2014-01-30 16:38:07 -05:00
# define ctx_from_connector(c) container_of(c, struct vidi_context, \
connector )
2012-03-21 10:55:26 +09:00
struct vidi_context {
2015-08-15 12:14:08 -03:00
struct drm_encoder encoder ;
2014-11-24 14:55:41 +09:00
struct platform_device * pdev ;
2014-02-19 21:02:55 +09:00
struct drm_device * drm_dev ;
2015-01-18 18:16:23 +09:00
struct exynos_drm_crtc * crtc ;
2014-01-30 16:38:07 -05:00
struct drm_connector connector ;
2015-04-03 21:03:40 +09:00
struct exynos_drm_plane planes [ WINDOWS_NR ] ;
2012-03-21 10:55:26 +09:00
struct edid * raw_edid ;
unsigned int clkdiv ;
unsigned int connected ;
bool suspended ;
2016-09-23 10:15:23 +02:00
struct timer_list timer ;
2012-03-21 10:55:26 +09:00
struct mutex lock ;
} ;
2015-08-15 12:14:08 -03:00
static inline struct vidi_context * encoder_to_vidi ( struct drm_encoder * e )
2014-11-17 09:54:23 +01:00
{
2015-08-11 17:38:06 +09:00
return container_of ( e , struct vidi_context , encoder ) ;
2014-11-17 09:54:23 +01:00
}
2012-03-21 10:55:26 +09:00
static const char fake_edid_info [ ] = {
0x00 , 0xff , 0xff , 0xff , 0xff , 0xff , 0xff , 0x00 , 0x4c , 0x2d , 0x05 , 0x05 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x30 , 0x12 , 0x01 , 0x03 , 0x80 , 0x10 , 0x09 , 0x78 ,
0x0a , 0xee , 0x91 , 0xa3 , 0x54 , 0x4c , 0x99 , 0x26 , 0x0f , 0x50 , 0x54 , 0xbd ,
0xee , 0x00 , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 ,
0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x01 , 0x66 , 0x21 , 0x50 , 0xb0 , 0x51 , 0x00 ,
0x1b , 0x30 , 0x40 , 0x70 , 0x36 , 0x00 , 0xa0 , 0x5a , 0x00 , 0x00 , 0x00 , 0x1e ,
0x01 , 0x1d , 0x00 , 0x72 , 0x51 , 0xd0 , 0x1e , 0x20 , 0x6e , 0x28 , 0x55 , 0x00 ,
0xa0 , 0x5a , 0x00 , 0x00 , 0x00 , 0x1e , 0x00 , 0x00 , 0x00 , 0xfd , 0x00 , 0x18 ,
0x4b , 0x1a , 0x44 , 0x17 , 0x00 , 0x0a , 0x20 , 0x20 , 0x20 , 0x20 , 0x20 , 0x20 ,
0x00 , 0x00 , 0x00 , 0xfc , 0x00 , 0x53 , 0x41 , 0x4d , 0x53 , 0x55 , 0x4e , 0x47 ,
0x0a , 0x20 , 0x20 , 0x20 , 0x20 , 0x20 , 0x01 , 0xbc , 0x02 , 0x03 , 0x1e , 0xf1 ,
0x46 , 0x84 , 0x05 , 0x03 , 0x10 , 0x20 , 0x22 , 0x23 , 0x09 , 0x07 , 0x07 , 0x83 ,
0x01 , 0x00 , 0x00 , 0xe2 , 0x00 , 0x0f , 0x67 , 0x03 , 0x0c , 0x00 , 0x10 , 0x00 ,
0xb8 , 0x2d , 0x01 , 0x1d , 0x80 , 0x18 , 0x71 , 0x1c , 0x16 , 0x20 , 0x58 , 0x2c ,
0x25 , 0x00 , 0xa0 , 0x5a , 0x00 , 0x00 , 0x00 , 0x9e , 0x8c , 0x0a , 0xd0 , 0x8a ,
0x20 , 0xe0 , 0x2d , 0x10 , 0x10 , 0x3e , 0x96 , 0x00 , 0xa0 , 0x5a , 0x00 , 0x00 ,
0x00 , 0x18 , 0x02 , 0x3a , 0x80 , 0x18 , 0x71 , 0x38 , 0x2d , 0x40 , 0x58 , 0x2c ,
0x45 , 0x00 , 0xa0 , 0x5a , 0x00 , 0x00 , 0x00 , 0x1e , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 , 0x00 ,
0x00 , 0x00 , 0x00 , 0x06
} ;
2015-08-31 00:53:57 +09:00
static const uint32_t formats [ ] = {
DRM_FORMAT_XRGB8888 ,
DRM_FORMAT_ARGB8888 ,
DRM_FORMAT_NV12 ,
} ;
2015-11-30 14:53:25 +01:00
static const enum drm_plane_type vidi_win_types [ WINDOWS_NR ] = {
DRM_PLANE_TYPE_PRIMARY ,
DRM_PLANE_TYPE_OVERLAY ,
DRM_PLANE_TYPE_CURSOR ,
} ;
2015-01-18 18:16:23 +09:00
static int vidi_enable_vblank ( struct exynos_drm_crtc * crtc )
2012-03-21 10:55:26 +09:00
{
2015-01-18 18:16:23 +09:00
struct vidi_context * ctx = crtc - > ctx ;
2012-03-21 10:55:26 +09:00
if ( ctx - > suspended )
return - EPERM ;
2016-09-23 10:15:23 +02:00
mod_timer ( & ctx - > timer ,
jiffies + msecs_to_jiffies ( VIDI_REFRESH_TIME ) - 1 ) ;
2012-09-19 11:02:43 +09:00
2012-03-21 10:55:26 +09:00
return 0 ;
}
2015-01-18 18:16:23 +09:00
static void vidi_disable_vblank ( struct exynos_drm_crtc * crtc )
2012-03-21 10:55:26 +09:00
{
}
2015-08-03 14:39:36 +09:00
static void vidi_update_plane ( struct exynos_drm_crtc * crtc ,
struct exynos_drm_plane * plane )
2012-03-21 10:55:26 +09:00
{
2015-11-30 14:53:21 +01:00
struct drm_plane_state * state = plane - > base . state ;
2015-01-18 18:16:23 +09:00
struct vidi_context * ctx = crtc - > ctx ;
2015-11-30 14:53:21 +01:00
dma_addr_t addr ;
2012-03-21 10:55:26 +09:00
if ( ctx - > suspended )
return ;
2015-11-30 14:53:21 +01:00
addr = exynos_drm_fb_dma_addr ( state - > fb , 0 ) ;
DRM_DEBUG_KMS ( " dma_addr = %pad \n " , & addr ) ;
2012-03-21 10:55:26 +09:00
}
2015-06-01 12:04:55 -03:00
static void vidi_enable ( struct exynos_drm_crtc * crtc )
2014-01-30 16:19:27 -05:00
{
2015-06-01 12:04:55 -03:00
struct vidi_context * ctx = crtc - > ctx ;
2015-04-03 21:03:40 +09:00
2015-06-01 12:04:55 -03:00
mutex_lock ( & ctx - > lock ) ;
2014-01-30 16:19:27 -05:00
2015-06-01 12:04:55 -03:00
ctx - > suspended = false ;
2014-01-30 16:19:27 -05:00
2015-06-01 12:04:55 -03:00
mutex_unlock ( & ctx - > lock ) ;
2016-09-23 10:15:23 +02:00
drm_crtc_vblank_on ( & crtc - > base ) ;
2014-01-30 16:19:27 -05:00
}
2015-06-01 12:04:55 -03:00
static void vidi_disable ( struct exynos_drm_crtc * crtc )
2014-01-30 16:19:27 -05:00
{
2015-01-18 18:16:23 +09:00
struct vidi_context * ctx = crtc - > ctx ;
2014-01-30 16:19:27 -05:00
2016-09-23 10:15:23 +02:00
drm_crtc_vblank_off ( & crtc - > base ) ;
2014-01-30 16:19:27 -05:00
mutex_lock ( & ctx - > lock ) ;
2015-06-01 12:04:55 -03:00
ctx - > suspended = true ;
2014-01-30 16:19:27 -05:00
mutex_unlock ( & ctx - > lock ) ;
}
2015-05-07 09:04:45 +09:00
static const struct exynos_drm_crtc_ops vidi_crtc_ops = {
2015-06-01 12:04:55 -03:00
. enable = vidi_enable ,
. disable = vidi_disable ,
2014-01-30 16:19:02 -05:00
. enable_vblank = vidi_enable_vblank ,
. disable_vblank = vidi_disable_vblank ,
2015-08-03 14:38:05 +09:00
. update_plane = vidi_update_plane ,
2017-03-14 09:27:56 +01:00
. atomic_flush = exynos_crtc_handle_event ,
2012-03-21 10:55:26 +09:00
} ;
2016-09-23 10:15:23 +02:00
static void vidi_fake_vblank_timer ( unsigned long arg )
2012-03-21 10:55:26 +09:00
{
2016-09-23 10:15:23 +02:00
struct vidi_context * ctx = ( void * ) arg ;
2012-03-21 10:55:26 +09:00
2016-09-23 10:15:23 +02:00
if ( drm_crtc_handle_vblank ( & ctx - > crtc - > base ) )
mod_timer ( & ctx - > timer ,
jiffies + msecs_to_jiffies ( VIDI_REFRESH_TIME ) - 1 ) ;
2012-03-21 10:55:26 +09:00
}
2016-02-03 13:42:47 +01:00
static ssize_t vidi_show_connection ( struct device * dev ,
2012-03-21 10:55:26 +09:00
struct device_attribute * attr , char * buf )
{
2014-11-17 09:54:16 +01:00
struct vidi_context * ctx = dev_get_drvdata ( dev ) ;
2012-03-21 10:55:26 +09:00
int rc ;
mutex_lock ( & ctx - > lock ) ;
rc = sprintf ( buf , " %d \n " , ctx - > connected ) ;
mutex_unlock ( & ctx - > lock ) ;
return rc ;
}
2016-02-03 13:42:47 +01:00
static ssize_t vidi_store_connection ( struct device * dev ,
2012-03-21 10:55:26 +09:00
struct device_attribute * attr ,
const char * buf , size_t len )
{
2014-11-17 09:54:16 +01:00
struct vidi_context * ctx = dev_get_drvdata ( dev ) ;
2012-03-21 10:55:26 +09:00
int ret ;
ret = kstrtoint ( buf , 0 , & ctx - > connected ) ;
if ( ret )
return ret ;
if ( ctx - > connected > 1 )
return - EINVAL ;
2012-06-27 16:00:56 +09:00
/* use fake edid data for test. */
if ( ! ctx - > raw_edid )
ctx - > raw_edid = ( struct edid * ) fake_edid_info ;
2012-06-27 16:16:26 +09:00
/* if raw_edid isn't same as fake data then it can't be tested. */
if ( ctx - > raw_edid ! = ( struct edid * ) fake_edid_info ) {
DRM_DEBUG_KMS ( " edid data is not fake data. \n " ) ;
return - EINVAL ;
}
2012-03-21 10:55:26 +09:00
DRM_DEBUG_KMS ( " requested connection. \n " ) ;
2014-02-19 21:02:55 +09:00
drm_helper_hpd_irq_event ( ctx - > drm_dev ) ;
2012-03-21 10:55:26 +09:00
return len ;
}
static DEVICE_ATTR ( connection , 0644 , vidi_show_connection ,
vidi_store_connection ) ;
int vidi_connection_ioctl ( struct drm_device * drm_dev , void * data ,
struct drm_file * file_priv )
{
2015-08-11 17:38:06 +09:00
struct vidi_context * ctx = dev_get_drvdata ( drm_dev - > dev ) ;
2012-03-21 10:55:26 +09:00
struct drm_exynos_vidi_connection * vidi = data ;
if ( ! vidi ) {
DRM_DEBUG_KMS ( " user data for vidi is null. \n " ) ;
return - EINVAL ;
}
if ( vidi - > connection > 1 ) {
DRM_DEBUG_KMS ( " connection should be 0 or 1. \n " ) ;
return - EINVAL ;
}
if ( ctx - > connected = = vidi - > connection ) {
DRM_DEBUG_KMS ( " same connection request. \n " ) ;
return - EINVAL ;
}
2012-06-27 16:36:12 +09:00
if ( vidi - > connection ) {
2016-02-03 13:42:47 +01:00
struct edid * raw_edid ;
raw_edid = ( struct edid * ) ( unsigned long ) vidi - > edid ;
2013-01-10 19:35:06 +09:00
if ( ! drm_edid_is_valid ( raw_edid ) ) {
DRM_DEBUG_KMS ( " edid data is invalid. \n " ) ;
2012-06-27 16:36:12 +09:00
return - EINVAL ;
}
2013-09-27 15:08:29 +03:00
ctx - > raw_edid = drm_edid_duplicate ( raw_edid ) ;
2012-06-27 16:36:12 +09:00
if ( ! ctx - > raw_edid ) {
DRM_DEBUG_KMS ( " failed to allocate raw_edid. \n " ) ;
return - ENOMEM ;
}
} else {
/*
* with connection = 0 , free raw_edid
* only if raw edid data isn ' t same as fake data .
*/
if ( ctx - > raw_edid & & ctx - > raw_edid ! =
( struct edid * ) fake_edid_info ) {
kfree ( ctx - > raw_edid ) ;
ctx - > raw_edid = NULL ;
}
}
2012-03-21 10:55:26 +09:00
ctx - > connected = vidi - > connection ;
2014-02-19 21:02:55 +09:00
drm_helper_hpd_irq_event ( ctx - > drm_dev ) ;
2012-03-21 10:55:26 +09:00
return 0 ;
}
2014-01-30 16:38:07 -05:00
static enum drm_connector_status vidi_detect ( struct drm_connector * connector ,
bool force )
{
struct vidi_context * ctx = ctx_from_connector ( connector ) ;
/*
* connection request would come from user side
* to do hotplug through specific ioctl .
*/
return ctx - > connected ? connector_status_connected :
connector_status_disconnected ;
}
static void vidi_connector_destroy ( struct drm_connector * connector )
{
}
2015-12-15 12:21:06 +01:00
static const struct drm_connector_funcs vidi_connector_funcs = {
2014-01-30 16:38:07 -05:00
. fill_modes = drm_helper_probe_single_connector_modes ,
. detect = vidi_detect ,
. destroy = vidi_connector_destroy ,
2015-06-01 12:04:44 -03:00
. reset = drm_atomic_helper_connector_reset ,
. atomic_duplicate_state = drm_atomic_helper_connector_duplicate_state ,
. atomic_destroy_state = drm_atomic_helper_connector_destroy_state ,
2014-01-30 16:38:07 -05:00
} ;
static int vidi_get_modes ( struct drm_connector * connector )
{
struct vidi_context * ctx = ctx_from_connector ( connector ) ;
struct edid * edid ;
int edid_len ;
/*
* the edid data comes from user side and it would be set
* to ctx - > raw_edid through specific ioctl .
*/
if ( ! ctx - > raw_edid ) {
DRM_DEBUG_KMS ( " raw_edid is null. \n " ) ;
return - EFAULT ;
}
edid_len = ( 1 + ctx - > raw_edid - > extensions ) * EDID_LENGTH ;
edid = kmemdup ( ctx - > raw_edid , edid_len , GFP_KERNEL ) ;
if ( ! edid ) {
DRM_DEBUG_KMS ( " failed to allocate edid \n " ) ;
return - ENOMEM ;
}
drm_mode_connector_update_edid_property ( connector , edid ) ;
return drm_add_edid_modes ( connector , edid ) ;
}
2015-12-15 12:21:06 +01:00
static const struct drm_connector_helper_funcs vidi_connector_helper_funcs = {
2014-01-30 16:38:07 -05:00
. get_modes = vidi_get_modes ,
} ;
2015-08-15 12:14:08 -03:00
static int vidi_create_connector ( struct drm_encoder * encoder )
2014-01-30 16:38:07 -05:00
{
2015-08-15 12:14:08 -03:00
struct vidi_context * ctx = encoder_to_vidi ( encoder ) ;
2014-01-30 16:38:07 -05:00
struct drm_connector * connector = & ctx - > connector ;
int ret ;
connector - > polled = DRM_CONNECTOR_POLL_HPD ;
ret = drm_connector_init ( ctx - > drm_dev , connector ,
& vidi_connector_funcs , DRM_MODE_CONNECTOR_VIRTUAL ) ;
if ( ret ) {
DRM_ERROR ( " Failed to initialize connector with drm \n " ) ;
return ret ;
}
drm_connector_helper_add ( connector , & vidi_connector_helper_funcs ) ;
drm_mode_connector_attach_encoder ( connector , encoder ) ;
return 0 ;
}
2015-08-15 12:14:08 -03:00
static void exynos_vidi_mode_set ( struct drm_encoder * encoder ,
struct drm_display_mode * mode ,
struct drm_display_mode * adjusted_mode )
{
}
static void exynos_vidi_enable ( struct drm_encoder * encoder )
{
}
static void exynos_vidi_disable ( struct drm_encoder * encoder )
{
}
2015-12-15 12:21:06 +01:00
static const struct drm_encoder_helper_funcs exynos_vidi_encoder_helper_funcs = {
2015-08-15 12:14:08 -03:00
. mode_set = exynos_vidi_mode_set ,
. enable = exynos_vidi_enable ,
. disable = exynos_vidi_disable ,
} ;
2015-12-15 12:21:06 +01:00
static const struct drm_encoder_funcs exynos_vidi_encoder_funcs = {
2015-08-15 12:14:08 -03:00
. destroy = drm_encoder_cleanup ,
} ;
2014-11-24 14:55:41 +09:00
static int vidi_bind ( struct device * dev , struct device * master , void * data )
2014-05-09 14:25:20 +09:00
{
2014-11-17 09:54:16 +01:00
struct vidi_context * ctx = dev_get_drvdata ( dev ) ;
2014-11-24 14:55:41 +09:00
struct drm_device * drm_dev = data ;
2015-08-15 12:14:08 -03:00
struct drm_encoder * encoder = & ctx - > encoder ;
2015-04-03 21:03:40 +09:00
struct exynos_drm_plane * exynos_plane ;
2015-11-30 14:53:25 +01:00
struct exynos_drm_plane_config plane_config = { 0 } ;
unsigned int i ;
2015-08-15 12:14:08 -03:00
int pipe , ret ;
2014-05-09 14:25:20 +09:00
2017-03-15 15:41:06 +01:00
ctx - > drm_dev = drm_dev ;
2015-01-30 16:43:01 +09:00
2015-11-30 14:53:25 +01:00
plane_config . pixel_formats = formats ;
plane_config . num_pixel_formats = ARRAY_SIZE ( formats ) ;
for ( i = 0 ; i < WINDOWS_NR ; i + + ) {
plane_config . zpos = i ;
plane_config . type = vidi_win_types [ i ] ;
2015-12-16 13:21:42 +01:00
ret = exynos_plane_init ( drm_dev , & ctx - > planes [ i ] , i ,
2017-03-15 15:41:05 +01:00
& plane_config ) ;
2015-04-03 21:03:40 +09:00
if ( ret )
return ret ;
}
2015-10-12 22:07:48 +09:00
exynos_plane = & ctx - > planes [ DEFAULT_WIN ] ;
2015-04-03 21:03:40 +09:00
ctx - > crtc = exynos_drm_crtc_create ( drm_dev , & exynos_plane - > base ,
2017-05-29 10:05:25 +09:00
EXYNOS_DISPLAY_TYPE_VIDI , & vidi_crtc_ops , ctx ) ;
2015-01-18 18:16:23 +09:00
if ( IS_ERR ( ctx - > crtc ) ) {
2014-05-09 14:25:20 +09:00
DRM_ERROR ( " failed to create crtc. \n " ) ;
2015-01-18 18:16:23 +09:00
return PTR_ERR ( ctx - > crtc ) ;
2014-05-09 14:25:20 +09:00
}
2015-08-15 12:14:08 -03:00
pipe = exynos_drm_crtc_get_pipe_from_type ( drm_dev ,
EXYNOS_DISPLAY_TYPE_VIDI ) ;
if ( pipe < 0 )
return pipe ;
encoder - > possible_crtcs = 1 < < pipe ;
DRM_DEBUG_KMS ( " possible_crtcs = 0x%x \n " , encoder - > possible_crtcs ) ;
drm_encoder_init ( drm_dev , encoder , & exynos_vidi_encoder_funcs ,
drm: Pass 'name' to drm_encoder_init()
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.
@@
identifier dev, encoder, funcs;
@@
int drm_encoder_init(struct drm_device *dev,
struct drm_encoder *encoder,
const struct drm_encoder_funcs *funcs,
int encoder_type
+ ,const char *name, int DOTDOTDOT
)
{ ... }
@@
identifier dev, encoder, funcs;
@@
int drm_encoder_init(struct drm_device *dev,
struct drm_encoder *encoder,
const struct drm_encoder_funcs *funcs,
int encoder_type
+ ,const char *name, int DOTDOTDOT
);
@@
expression E1, E2, E3, E4;
@@
drm_encoder_init(E1, E2, E3, E4
+ ,NULL
)
v2: 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/1449670818-2966-1-git-send-email-ville.syrjala@linux.intel.com
2015-12-09 16:20:18 +02:00
DRM_MODE_ENCODER_TMDS , NULL ) ;
2015-08-15 12:14:08 -03:00
drm_encoder_helper_add ( encoder , & exynos_vidi_encoder_helper_funcs ) ;
2015-08-05 20:24:20 -03:00
2015-08-15 12:14:08 -03:00
ret = vidi_create_connector ( encoder ) ;
2014-05-09 14:25:20 +09:00
if ( ret ) {
2015-08-05 20:24:20 -03:00
DRM_ERROR ( " failed to create connector ret = %d \n " , ret ) ;
2015-08-15 12:14:08 -03:00
drm_encoder_cleanup ( encoder ) ;
2014-05-09 14:25:20 +09:00
return ret ;
}
return 0 ;
}
2014-11-24 14:55:41 +09:00
static void vidi_unbind ( struct device * dev , struct device * master , void * data )
{
2016-09-23 10:15:23 +02:00
struct vidi_context * ctx = dev_get_drvdata ( dev ) ;
del_timer_sync ( & ctx - > timer ) ;
2014-11-24 14:55:41 +09:00
}
static const struct component_ops vidi_component_ops = {
. bind = vidi_bind ,
. unbind = vidi_unbind ,
} ;
2012-12-21 15:09:25 -08:00
static int vidi_probe ( struct platform_device * pdev )
2012-03-21 10:55:26 +09:00
{
struct vidi_context * ctx ;
int ret ;
2014-05-09 14:25:20 +09:00
ctx = devm_kzalloc ( & pdev - > dev , sizeof ( * ctx ) , GFP_KERNEL ) ;
2012-03-21 10:55:26 +09:00
if ( ! ctx )
return - ENOMEM ;
2014-11-24 14:55:41 +09:00
ctx - > pdev = pdev ;
2016-09-23 10:15:23 +02:00
setup_timer ( & ctx - > timer , vidi_fake_vblank_timer , ( unsigned long ) ctx ) ;
2012-03-21 10:55:26 +09:00
mutex_init ( & ctx - > lock ) ;
2014-11-17 09:54:16 +01:00
platform_set_drvdata ( pdev , ctx ) ;
2012-03-21 10:55:26 +09:00
2014-05-09 14:25:20 +09:00
ret = device_create_file ( & pdev - > dev , & dev_attr_connection ) ;
if ( ret < 0 ) {
2014-11-24 14:55:41 +09:00
DRM_ERROR ( " failed to create connection sysfs. \n " ) ;
2015-06-11 23:23:37 +09:00
return ret ;
2014-05-09 14:25:20 +09:00
}
2012-03-21 10:55:26 +09:00
2014-11-24 14:55:41 +09:00
ret = component_add ( & pdev - > dev , & vidi_component_ops ) ;
if ( ret )
goto err_remove_file ;
return ret ;
err_remove_file :
device_remove_file ( & pdev - > dev , & dev_attr_connection ) ;
return ret ;
2012-03-21 10:55:26 +09:00
}
2012-12-21 15:09:25 -08:00
static int vidi_remove ( struct platform_device * pdev )
2012-03-21 10:55:26 +09:00
{
2014-11-17 09:54:16 +01:00
struct vidi_context * ctx = platform_get_drvdata ( pdev ) ;
2012-03-21 10:55:26 +09:00
2012-06-27 16:36:12 +09:00
if ( ctx - > raw_edid ! = ( struct edid * ) fake_edid_info ) {
kfree ( ctx - > raw_edid ) ;
ctx - > raw_edid = NULL ;
2014-05-09 14:25:20 +09:00
return - EINVAL ;
2012-06-27 16:36:12 +09:00
}
2014-11-24 14:55:41 +09:00
component_del ( & pdev - > dev , & vidi_component_ops ) ;
2012-03-21 10:55:26 +09:00
return 0 ;
}
struct platform_driver vidi_driver = {
. probe = vidi_probe ,
2012-12-21 15:09:25 -08:00
. remove = vidi_remove ,
2012-03-21 10:55:26 +09:00
. driver = {
. name = " exynos-drm-vidi " ,
. owner = THIS_MODULE ,
} ,
} ;