2019-05-27 08:55:01 +02:00
// SPDX-License-Identifier: GPL-2.0-or-later
2016-03-29 09:57:30 +08:00
/*
* Rockchip SoC DP ( Display Port ) interface driver .
*
* Copyright ( C ) Fuzhou Rockchip Electronics Co . , Ltd .
* Author : Andy Yan < andy . yan @ rock - chips . com >
* Yakir Yang < ykk @ rock - chips . com >
* Jeff Chen < jeff . chen @ rock - chips . com >
*/
# include <linux/component.h>
# include <linux/mfd/syscon.h>
2016-06-29 17:15:01 +08:00
# include <linux/of_device.h>
2016-03-29 09:57:30 +08:00
# include <linux/of_graph.h>
# include <linux/regmap.h>
# include <linux/reset.h>
# include <linux/clk.h>
# include <video/of_videomode.h>
# include <video/videomode.h>
2019-06-11 12:08:22 -04:00
# include <drm/drm_atomic.h>
# include <drm/drm_atomic_helper.h>
2016-03-29 09:57:30 +08:00
# include <drm/bridge/analogix_dp.h>
2019-07-16 08:42:19 +02:00
# include <drm/drm_dp_helper.h>
# include <drm/drm_of.h>
# include <drm/drm_panel.h>
# include <drm/drm_probe_helper.h>
2016-03-29 09:57:30 +08:00
# include "rockchip_drm_drv.h"
# include "rockchip_drm_vop.h"
2016-06-29 17:15:01 +08:00
# define RK3288_GRF_SOC_CON6 0x25c
# define RK3288_EDP_LCDC_SEL BIT(5)
2016-06-29 17:15:26 +08:00
# define RK3399_GRF_SOC_CON20 0x6250
# define RK3399_EDP_LCDC_SEL BIT(5)
2016-06-29 17:15:01 +08:00
# define HIWORD_UPDATE(val, mask) (val | (mask) << 16)
2016-07-24 14:57:52 +08:00
# define PSR_WAIT_LINE_FLAG_TIMEOUT_MS 100
2016-03-29 09:57:30 +08:00
# define to_dp(nm) container_of(nm, struct rockchip_dp_device, nm)
2016-06-29 17:15:01 +08:00
/**
* struct rockchip_dp_chip_data - splite the grf setting of kind of chips
* @ lcdsel_grf_reg : grf register offset of lcdc select
* @ lcdsel_big : reg value of selecting vop big for eDP
* @ lcdsel_lit : reg value of selecting vop little for eDP
* @ chip_type : specific chip type
*/
struct rockchip_dp_chip_data {
u32 lcdsel_grf_reg ;
u32 lcdsel_big ;
u32 lcdsel_lit ;
u32 chip_type ;
} ;
2016-03-29 09:57:30 +08:00
struct rockchip_dp_device {
struct drm_device * drm_dev ;
struct device * dev ;
struct drm_encoder encoder ;
struct drm_display_mode mode ;
struct clk * pclk ;
2016-06-29 17:16:05 +08:00
struct clk * grfclk ;
2016-03-29 09:57:30 +08:00
struct regmap * grf ;
struct reset_control * rst ;
2016-06-29 17:15:01 +08:00
const struct rockchip_dp_chip_data * data ;
2018-01-10 17:23:41 +01:00
struct analogix_dp_device * adp ;
2016-03-29 09:57:30 +08:00
struct analogix_dp_plat_data plat_data ;
} ;
static int rockchip_dp_pre_init ( struct rockchip_dp_device * dp )
{
reset_control_assert ( dp - > rst ) ;
usleep_range ( 10 , 20 ) ;
reset_control_deassert ( dp - > rst ) ;
return 0 ;
}
2018-04-23 12:49:58 +02:00
static int rockchip_dp_poweron_start ( struct analogix_dp_plat_data * plat_data )
2016-03-29 09:57:30 +08:00
{
struct rockchip_dp_device * dp = to_dp ( plat_data ) ;
int ret ;
ret = clk_prepare_enable ( dp - > pclk ) ;
if ( ret < 0 ) {
2017-09-15 02:36:03 -06:00
DRM_DEV_ERROR ( dp - > dev , " failed to enable pclk %d \n " , ret ) ;
2016-03-29 09:57:30 +08:00
return ret ;
}
ret = rockchip_dp_pre_init ( dp ) ;
if ( ret < 0 ) {
2017-09-15 02:36:03 -06:00
DRM_DEV_ERROR ( dp - > dev , " failed to dp pre init %d \n " , ret ) ;
2016-07-19 11:32:43 +00:00
clk_disable_unprepare ( dp - > pclk ) ;
2016-03-29 09:57:30 +08:00
return ret ;
}
2018-04-23 12:49:58 +02:00
return ret ;
}
2016-03-29 09:57:30 +08:00
static int rockchip_dp_powerdown ( struct analogix_dp_plat_data * plat_data )
{
struct rockchip_dp_device * dp = to_dp ( plat_data ) ;
clk_disable_unprepare ( dp - > pclk ) ;
return 0 ;
}
2016-06-29 17:15:39 +08:00
static int rockchip_dp_get_modes ( struct analogix_dp_plat_data * plat_data ,
struct drm_connector * connector )
{
struct drm_display_info * di = & connector - > display_info ;
/* VOP couldn't output YUV video format for eDP rightly */
u32 mask = DRM_COLOR_FORMAT_YCRCB444 | DRM_COLOR_FORMAT_YCRCB422 ;
if ( ( di - > color_formats & mask ) ) {
DRM_DEBUG_KMS ( " Swapping display color format from YUV to RGB \n " ) ;
di - > color_formats & = ~ mask ;
di - > color_formats | = DRM_COLOR_FORMAT_RGB444 ;
di - > bpc = 8 ;
}
return 0 ;
}
2016-03-29 09:57:30 +08:00
static bool
rockchip_dp_drm_encoder_mode_fixup ( struct drm_encoder * encoder ,
const struct drm_display_mode * mode ,
struct drm_display_mode * adjusted_mode )
{
/* do nothing */
return true ;
}
static void rockchip_dp_drm_encoder_mode_set ( struct drm_encoder * encoder ,
struct drm_display_mode * mode ,
struct drm_display_mode * adjusted )
{
/* do nothing */
}
2019-06-11 12:08:22 -04:00
static
struct drm_crtc * rockchip_dp_drm_get_new_crtc ( struct drm_encoder * encoder ,
struct drm_atomic_state * state )
{
struct drm_connector * connector ;
struct drm_connector_state * conn_state ;
connector = drm_atomic_get_new_connector_for_encoder ( state , encoder ) ;
if ( ! connector )
return NULL ;
conn_state = drm_atomic_get_new_connector_state ( state , connector ) ;
if ( ! conn_state )
return NULL ;
return conn_state - > crtc ;
}
static void rockchip_dp_drm_encoder_enable ( struct drm_encoder * encoder ,
struct drm_atomic_state * state )
2016-03-29 09:57:30 +08:00
{
struct rockchip_dp_device * dp = to_dp ( encoder ) ;
2019-06-11 12:08:22 -04:00
struct drm_crtc * crtc ;
struct drm_crtc_state * old_crtc_state ;
2016-03-29 09:57:30 +08:00
int ret ;
u32 val ;
2019-06-11 12:08:22 -04:00
crtc = rockchip_dp_drm_get_new_crtc ( encoder , state ) ;
if ( ! crtc )
return ;
old_crtc_state = drm_atomic_get_old_crtc_state ( state , crtc ) ;
/* Coming back from self refresh, nothing to do */
if ( old_crtc_state & & old_crtc_state - > self_refresh_active )
return ;
2016-03-29 09:57:30 +08:00
ret = drm_of_encoder_active_endpoint_id ( dp - > dev - > of_node , encoder ) ;
if ( ret < 0 )
return ;
if ( ret )
2016-06-29 17:15:01 +08:00
val = dp - > data - > lcdsel_lit ;
2016-03-29 09:57:30 +08:00
else
2016-06-29 17:15:01 +08:00
val = dp - > data - > lcdsel_big ;
2016-03-29 09:57:30 +08:00
2017-09-15 02:36:03 -06:00
DRM_DEV_DEBUG ( dp - > dev , " vop %s output to dp \n " , ( ret ) ? " LIT " : " BIG " ) ;
2016-03-29 09:57:30 +08:00
2016-06-29 17:16:05 +08:00
ret = clk_prepare_enable ( dp - > grfclk ) ;
if ( ret < 0 ) {
2017-09-15 02:36:03 -06:00
DRM_DEV_ERROR ( dp - > dev , " failed to enable grfclk %d \n " , ret ) ;
2016-03-29 09:57:30 +08:00
return ;
}
2016-06-29 17:16:05 +08:00
ret = regmap_write ( dp - > grf , dp - > data - > lcdsel_grf_reg , val ) ;
if ( ret ! = 0 )
2017-09-15 02:36:03 -06:00
DRM_DEV_ERROR ( dp - > dev , " Could not write to GRF: %d \n " , ret ) ;
2016-06-29 17:16:05 +08:00
clk_disable_unprepare ( dp - > grfclk ) ;
2016-03-29 09:57:30 +08:00
}
2019-06-11 12:08:22 -04:00
static void rockchip_dp_drm_encoder_disable ( struct drm_encoder * encoder ,
struct drm_atomic_state * state )
2016-03-29 09:57:30 +08:00
{
2019-06-11 12:08:22 -04:00
struct rockchip_dp_device * dp = to_dp ( encoder ) ;
struct drm_crtc * crtc ;
struct drm_crtc_state * new_crtc_state = NULL ;
int ret ;
crtc = rockchip_dp_drm_get_new_crtc ( encoder , state ) ;
/* No crtc means we're doing a full shutdown */
if ( ! crtc )
return ;
new_crtc_state = drm_atomic_get_new_crtc_state ( state , crtc ) ;
/* If we're not entering self-refresh, no need to wait for vact */
if ( ! new_crtc_state | | ! new_crtc_state - > self_refresh_active )
return ;
ret = rockchip_drm_wait_vact_end ( crtc , PSR_WAIT_LINE_FLAG_TIMEOUT_MS ) ;
if ( ret )
DRM_DEV_ERROR ( dp - > dev , " line flag irq timed out \n " ) ;
2016-03-29 09:57:30 +08:00
}
2016-04-20 10:41:42 +08:00
static int
rockchip_dp_drm_encoder_atomic_check ( struct drm_encoder * encoder ,
struct drm_crtc_state * crtc_state ,
struct drm_connector_state * conn_state )
{
struct rockchip_crtc_state * s = to_rockchip_crtc_state ( crtc_state ) ;
2018-04-23 12:49:57 +02:00
struct drm_display_info * di = & conn_state - > connector - > display_info ;
2016-04-20 10:41:42 +08:00
/*
2016-06-29 17:15:44 +08:00
* The hardware IC designed that VOP must output the RGB10 video
* format to eDP controller , and if eDP panel only support RGB8 ,
* then eDP controller should cut down the video data , not via VOP
* controller , that ' s why we need to hardcode the VOP output mode
* to RGA10 here .
2016-04-20 10:41:42 +08:00
*/
2016-06-29 17:15:26 +08:00
2016-04-20 10:41:42 +08:00
s - > output_mode = ROCKCHIP_OUT_MODE_AAAA ;
s - > output_type = DRM_MODE_CONNECTOR_eDP ;
2018-04-23 12:49:57 +02:00
s - > output_bpc = di - > bpc ;
2016-04-20 10:41:42 +08:00
return 0 ;
}
2016-03-29 09:57:30 +08:00
static struct drm_encoder_helper_funcs rockchip_dp_encoder_helper_funcs = {
. mode_fixup = rockchip_dp_drm_encoder_mode_fixup ,
. mode_set = rockchip_dp_drm_encoder_mode_set ,
2019-06-11 12:08:22 -04:00
. atomic_enable = rockchip_dp_drm_encoder_enable ,
. atomic_disable = rockchip_dp_drm_encoder_disable ,
2016-04-20 10:41:42 +08:00
. atomic_check = rockchip_dp_drm_encoder_atomic_check ,
2016-03-29 09:57:30 +08:00
} ;
static struct drm_encoder_funcs rockchip_dp_encoder_funcs = {
2018-01-10 17:23:42 +01:00
. destroy = drm_encoder_cleanup ,
2016-03-29 09:57:30 +08:00
} ;
2017-10-19 11:48:04 +08:00
static int rockchip_dp_of_probe ( struct rockchip_dp_device * dp )
2016-03-29 09:57:30 +08:00
{
struct device * dev = dp - > dev ;
struct device_node * np = dev - > of_node ;
dp - > grf = syscon_regmap_lookup_by_phandle ( np , " rockchip,grf " ) ;
if ( IS_ERR ( dp - > grf ) ) {
2017-09-15 02:36:03 -06:00
DRM_DEV_ERROR ( dev , " failed to get rockchip,grf property \n " ) ;
2016-03-29 09:57:30 +08:00
return PTR_ERR ( dp - > grf ) ;
}
2016-06-29 17:16:05 +08:00
dp - > grfclk = devm_clk_get ( dev , " grf " ) ;
if ( PTR_ERR ( dp - > grfclk ) = = - ENOENT ) {
dp - > grfclk = NULL ;
} else if ( PTR_ERR ( dp - > grfclk ) = = - EPROBE_DEFER ) {
return - EPROBE_DEFER ;
} else if ( IS_ERR ( dp - > grfclk ) ) {
2017-09-15 02:36:03 -06:00
DRM_DEV_ERROR ( dev , " failed to get grf clock \n " ) ;
2016-06-29 17:16:05 +08:00
return PTR_ERR ( dp - > grfclk ) ;
}
2016-03-29 09:57:30 +08:00
dp - > pclk = devm_clk_get ( dev , " pclk " ) ;
if ( IS_ERR ( dp - > pclk ) ) {
2017-09-15 02:36:03 -06:00
DRM_DEV_ERROR ( dev , " failed to get pclk property \n " ) ;
2016-03-29 09:57:30 +08:00
return PTR_ERR ( dp - > pclk ) ;
}
dp - > rst = devm_reset_control_get ( dev , " dp " ) ;
if ( IS_ERR ( dp - > rst ) ) {
2017-09-15 02:36:03 -06:00
DRM_DEV_ERROR ( dev , " failed to get dp reset control \n " ) ;
2016-03-29 09:57:30 +08:00
return PTR_ERR ( dp - > rst ) ;
}
return 0 ;
}
static int rockchip_dp_drm_create_encoder ( struct rockchip_dp_device * dp )
{
struct drm_encoder * encoder = & dp - > encoder ;
struct drm_device * drm_dev = dp - > drm_dev ;
struct device * dev = dp - > dev ;
int ret ;
encoder - > possible_crtcs = drm_of_find_possible_crtcs ( drm_dev ,
dev - > of_node ) ;
DRM_DEBUG_KMS ( " possible_crtcs = 0x%x \n " , encoder - > possible_crtcs ) ;
ret = drm_encoder_init ( drm_dev , encoder , & rockchip_dp_encoder_funcs ,
DRM_MODE_ENCODER_TMDS , NULL ) ;
if ( ret ) {
DRM_ERROR ( " failed to initialize encoder with drm \n " ) ;
return ret ;
}
drm_encoder_helper_add ( encoder , & rockchip_dp_encoder_helper_funcs ) ;
return 0 ;
}
static int rockchip_dp_bind ( struct device * dev , struct device * master ,
void * data )
{
struct rockchip_dp_device * dp = dev_get_drvdata ( dev ) ;
2016-06-29 17:15:01 +08:00
const struct rockchip_dp_chip_data * dp_data ;
2016-03-29 09:57:30 +08:00
struct drm_device * drm_dev = data ;
int ret ;
2016-06-29 17:15:01 +08:00
dp_data = of_device_get_match_data ( dev ) ;
if ( ! dp_data )
return - ENODEV ;
dp - > data = dp_data ;
2016-03-29 09:57:30 +08:00
dp - > drm_dev = drm_dev ;
ret = rockchip_dp_drm_create_encoder ( dp ) ;
if ( ret ) {
DRM_ERROR ( " failed to create drm encoder \n " ) ;
return ret ;
}
dp - > plat_data . encoder = & dp - > encoder ;
2016-06-29 17:15:01 +08:00
dp - > plat_data . dev_type = dp - > data - > chip_type ;
2018-04-23 12:49:58 +02:00
dp - > plat_data . power_on_start = rockchip_dp_poweron_start ;
2016-03-29 09:57:30 +08:00
dp - > plat_data . power_off = rockchip_dp_powerdown ;
2016-06-29 17:15:39 +08:00
dp - > plat_data . get_modes = rockchip_dp_get_modes ;
2016-03-29 09:57:30 +08:00
2018-01-10 17:23:41 +01:00
dp - > adp = analogix_dp_bind ( dev , dp - > drm_dev , & dp - > plat_data ) ;
2018-01-10 17:23:42 +01:00
if ( IS_ERR ( dp - > adp ) ) {
2018-01-10 17:23:43 +01:00
ret = PTR_ERR ( dp - > adp ) ;
2019-06-11 12:08:22 -04:00
goto err_cleanup_encoder ;
2018-01-10 17:23:42 +01:00
}
2018-01-10 17:23:41 +01:00
return 0 ;
2018-01-10 17:23:43 +01:00
err_cleanup_encoder :
dp - > encoder . funcs - > destroy ( & dp - > encoder ) ;
return ret ;
2016-03-29 09:57:30 +08:00
}
static void rockchip_dp_unbind ( struct device * dev , struct device * master ,
void * data )
{
2016-07-24 14:57:52 +08:00
struct rockchip_dp_device * dp = dev_get_drvdata ( dev ) ;
2018-01-10 17:23:41 +01:00
analogix_dp_unbind ( dp - > adp ) ;
2018-01-10 17:23:42 +01:00
dp - > encoder . funcs - > destroy ( & dp - > encoder ) ;
2018-04-23 12:49:59 +02:00
dp - > adp = ERR_PTR ( - ENODEV ) ;
2016-03-29 09:57:30 +08:00
}
static const struct component_ops rockchip_dp_component_ops = {
. bind = rockchip_dp_bind ,
. unbind = rockchip_dp_unbind ,
} ;
static int rockchip_dp_probe ( struct platform_device * pdev )
{
struct device * dev = & pdev - > dev ;
2016-06-29 17:15:30 +08:00
struct drm_panel * panel = NULL ;
2016-03-29 09:57:30 +08:00
struct rockchip_dp_device * dp ;
2017-03-29 13:55:46 -05:00
int ret ;
2016-03-29 09:57:30 +08:00
2017-03-29 13:55:46 -05:00
ret = drm_of_find_panel_or_bridge ( dev - > of_node , 1 , 0 , & panel , NULL ) ;
2017-10-19 11:48:04 +08:00
if ( ret < 0 )
2017-03-29 13:55:46 -05:00
return ret ;
2016-03-29 09:57:30 +08:00
dp = devm_kzalloc ( dev , sizeof ( * dp ) , GFP_KERNEL ) ;
if ( ! dp )
return - ENOMEM ;
dp - > dev = dev ;
2018-04-23 12:49:59 +02:00
dp - > adp = ERR_PTR ( - ENODEV ) ;
2016-03-29 09:57:30 +08:00
dp - > plat_data . panel = panel ;
2017-10-19 11:48:04 +08:00
ret = rockchip_dp_of_probe ( dp ) ;
if ( ret < 0 )
return ret ;
2016-03-29 09:57:30 +08:00
platform_set_drvdata ( pdev , dp ) ;
return component_add ( dev , & rockchip_dp_component_ops ) ;
}
static int rockchip_dp_remove ( struct platform_device * pdev )
{
component_del ( & pdev - > dev , & rockchip_dp_component_ops ) ;
return 0 ;
}
2018-01-10 17:23:41 +01:00
# ifdef CONFIG_PM_SLEEP
static int rockchip_dp_suspend ( struct device * dev )
{
struct rockchip_dp_device * dp = dev_get_drvdata ( dev ) ;
2018-04-23 12:49:59 +02:00
if ( IS_ERR ( dp - > adp ) )
return 0 ;
2018-01-10 17:23:41 +01:00
return analogix_dp_suspend ( dp - > adp ) ;
}
static int rockchip_dp_resume ( struct device * dev )
{
struct rockchip_dp_device * dp = dev_get_drvdata ( dev ) ;
2018-04-23 12:49:59 +02:00
if ( IS_ERR ( dp - > adp ) )
return 0 ;
2018-01-10 17:23:41 +01:00
return analogix_dp_resume ( dp - > adp ) ;
}
# endif
2016-06-06 16:53:33 +02:00
static const struct dev_pm_ops rockchip_dp_pm_ops = {
2016-03-29 09:57:30 +08:00
# ifdef CONFIG_PM_SLEEP
2019-08-02 11:46:16 -07:00
. suspend_late = rockchip_dp_suspend ,
2018-01-10 17:23:41 +01:00
. resume_early = rockchip_dp_resume ,
2016-03-29 09:57:30 +08:00
# endif
} ;
2016-06-29 17:15:26 +08:00
static const struct rockchip_dp_chip_data rk3399_edp = {
. lcdsel_grf_reg = RK3399_GRF_SOC_CON20 ,
. lcdsel_big = HIWORD_UPDATE ( 0 , RK3399_EDP_LCDC_SEL ) ,
. lcdsel_lit = HIWORD_UPDATE ( RK3399_EDP_LCDC_SEL , RK3399_EDP_LCDC_SEL ) ,
. chip_type = RK3399_EDP ,
} ;
2016-06-29 17:15:01 +08:00
static const struct rockchip_dp_chip_data rk3288_dp = {
. lcdsel_grf_reg = RK3288_GRF_SOC_CON6 ,
. lcdsel_big = HIWORD_UPDATE ( 0 , RK3288_EDP_LCDC_SEL ) ,
. lcdsel_lit = HIWORD_UPDATE ( RK3288_EDP_LCDC_SEL , RK3288_EDP_LCDC_SEL ) ,
. chip_type = RK3288_DP ,
} ;
2016-03-29 09:57:30 +08:00
static const struct of_device_id rockchip_dp_dt_ids [ ] = {
2016-06-29 17:15:01 +08:00
{ . compatible = " rockchip,rk3288-dp " , . data = & rk3288_dp } ,
2016-06-29 17:15:26 +08:00
{ . compatible = " rockchip,rk3399-edp " , . data = & rk3399_edp } ,
2016-03-29 09:57:30 +08:00
{ }
} ;
MODULE_DEVICE_TABLE ( of , rockchip_dp_dt_ids ) ;
2017-03-22 11:21:20 +08:00
struct platform_driver rockchip_dp_driver = {
2016-03-29 09:57:30 +08:00
. probe = rockchip_dp_probe ,
. remove = rockchip_dp_remove ,
. driver = {
. name = " rockchip-dp " ,
. pm = & rockchip_dp_pm_ops ,
. of_match_table = of_match_ptr ( rockchip_dp_dt_ids ) ,
} ,
} ;