2013-08-27 23:40:56 +03:00
/*
* Copyright © 2013 Intel Corporation
*
* 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
* THE AUTHORS OR COPYRIGHT HOLDERS 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 .
*
* Authors :
* Shobhit Kumar < shobhit . kumar @ intel . com >
* Yogesh Mohan Marimuthu < yogesh . mohan . marimuthu @ intel . com >
*/
# include <linux/kernel.h>
# include "intel_drv.h"
# include "i915_drv.h"
# include "intel_dsi.h"
# define DSI_HSS_PACKET_SIZE 4
# define DSI_HSE_PACKET_SIZE 4
# define DSI_HSA_PACKET_EXTRA_SIZE 6
# define DSI_HBP_PACKET_EXTRA_SIZE 6
# define DSI_HACTIVE_PACKET_EXTRA_SIZE 6
# define DSI_HFP_PACKET_EXTRA_SIZE 6
# define DSI_EOTP_PACKET_SIZE 4
2015-07-01 15:58:50 +03:00
static int dsi_pixel_format_bpp ( int pixel_format )
{
int bpp ;
switch ( pixel_format ) {
default :
case VID_MODE_FORMAT_RGB888 :
case VID_MODE_FORMAT_RGB666_LOOSE :
bpp = 24 ;
break ;
case VID_MODE_FORMAT_RGB666 :
bpp = 18 ;
break ;
case VID_MODE_FORMAT_RGB565 :
bpp = 16 ;
break ;
}
return bpp ;
}
2013-08-27 23:40:56 +03:00
struct dsi_mnp {
u32 dsi_pll_ctrl ;
u32 dsi_pll_div ;
} ;
static const u32 lfsr_converts [ ] = {
426 , 469 , 234 , 373 , 442 , 221 , 110 , 311 , 411 , /* 62 - 70 */
461 , 486 , 243 , 377 , 188 , 350 , 175 , 343 , 427 , 213 , /* 71 - 80 */
2015-07-01 15:58:51 +03:00
106 , 53 , 282 , 397 , 454 , 227 , 113 , 56 , 284 , 142 , /* 81 - 90 */
71 , 35 , 273 , 136 , 324 , 418 , 465 , 488 , 500 , 506 /* 91 - 100 */
2013-08-27 23:40:56 +03:00
} ;
2013-12-10 12:14:56 +05:30
# ifdef DSI_CLK_FROM_RR
2013-09-04 18:25:27 +03:00
static u32 dsi_rr_formula ( const struct drm_display_mode * mode ,
2013-08-27 23:40:56 +03:00
int pixel_format , int video_mode_format ,
int lane_count , bool eotp )
{
u32 bpp ;
u32 hactive , vactive , hfp , hsync , hbp , vfp , vsync , vbp ;
u32 hsync_bytes , hbp_bytes , hactive_bytes , hfp_bytes ;
u32 bytes_per_line , bytes_per_frame ;
u32 num_frames ;
u32 bytes_per_x_frames , bytes_per_x_frames_x_lanes ;
u32 dsi_bit_clock_hz ;
u32 dsi_clk ;
2015-07-01 15:58:50 +03:00
bpp = dsi_pixel_format_bpp ( pixel_format ) ;
2013-08-27 23:40:56 +03:00
hactive = mode - > hdisplay ;
vactive = mode - > vdisplay ;
hfp = mode - > hsync_start - mode - > hdisplay ;
hsync = mode - > hsync_end - mode - > hsync_start ;
hbp = mode - > htotal - mode - > hsync_end ;
vfp = mode - > vsync_start - mode - > vdisplay ;
vsync = mode - > vsync_end - mode - > vsync_start ;
vbp = mode - > vtotal - mode - > vsync_end ;
hsync_bytes = DIV_ROUND_UP ( hsync * bpp , 8 ) ;
hbp_bytes = DIV_ROUND_UP ( hbp * bpp , 8 ) ;
hactive_bytes = DIV_ROUND_UP ( hactive * bpp , 8 ) ;
hfp_bytes = DIV_ROUND_UP ( hfp * bpp , 8 ) ;
bytes_per_line = DSI_HSS_PACKET_SIZE + hsync_bytes +
DSI_HSA_PACKET_EXTRA_SIZE + DSI_HSE_PACKET_SIZE +
hbp_bytes + DSI_HBP_PACKET_EXTRA_SIZE +
hactive_bytes + DSI_HACTIVE_PACKET_EXTRA_SIZE +
hfp_bytes + DSI_HFP_PACKET_EXTRA_SIZE ;
/*
* XXX : Need to accurately calculate LP to HS transition timeout and add
* it to bytes_per_line / bytes_per_frame .
*/
if ( eotp & & video_mode_format = = VIDEO_MODE_BURST )
bytes_per_line + = DSI_EOTP_PACKET_SIZE ;
bytes_per_frame = vsync * bytes_per_line + vbp * bytes_per_line +
vactive * bytes_per_line + vfp * bytes_per_line ;
if ( eotp & &
( video_mode_format = = VIDEO_MODE_NON_BURST_WITH_SYNC_PULSE | |
video_mode_format = = VIDEO_MODE_NON_BURST_WITH_SYNC_EVENTS ) )
bytes_per_frame + = DSI_EOTP_PACKET_SIZE ;
num_frames = drm_mode_vrefresh ( mode ) ;
bytes_per_x_frames = num_frames * bytes_per_frame ;
bytes_per_x_frames_x_lanes = bytes_per_x_frames / lane_count ;
/* the dsi clock is divided by 2 in the hardware to get dsi ddr clock */
dsi_bit_clock_hz = bytes_per_x_frames_x_lanes * 8 ;
2013-12-10 12:14:56 +05:30
dsi_clk = dsi_bit_clock_hz / 1000 ;
2013-08-27 23:40:56 +03:00
if ( eotp & & video_mode_format = = VIDEO_MODE_BURST )
dsi_clk * = 2 ;
return dsi_clk ;
}
2013-12-10 12:14:56 +05:30
# else
2013-08-27 23:40:56 +03:00
2013-12-10 12:14:56 +05:30
/* Get DSI clock from pixel clock */
2014-07-30 20:34:57 +05:30
static u32 dsi_clk_from_pclk ( u32 pclk , int pixel_format , int lane_count )
2013-08-27 23:40:56 +03:00
{
2013-12-10 12:14:56 +05:30
u32 dsi_clk_khz ;
2015-07-01 15:58:50 +03:00
u32 bpp = dsi_pixel_format_bpp ( pixel_format ) ;
2013-08-27 23:40:56 +03:00
2013-12-10 12:14:56 +05:30
/* DSI data rate = pixel clock * bits per pixel / lane count
pixel clock is converted from KHz to Hz */
2014-07-30 20:34:57 +05:30
dsi_clk_khz = DIV_ROUND_CLOSEST ( pclk * bpp , lane_count ) ;
2013-08-27 23:40:56 +03:00
2013-12-10 12:14:56 +05:30
return dsi_clk_khz ;
2013-08-27 23:40:56 +03:00
}
2013-12-10 12:14:56 +05:30
# endif
2013-08-27 23:40:56 +03:00
2015-07-01 15:58:52 +03:00
static int dsi_calc_mnp ( struct drm_i915_private * dev_priv ,
struct dsi_mnp * dsi_mnp , int target_dsi_clk )
2013-08-27 23:40:56 +03:00
{
2015-05-12 15:23:09 +03:00
unsigned int calc_m = 0 , calc_p = 0 ;
2015-07-01 15:58:52 +03:00
unsigned int m_min , m_max , p_min = 2 , p_max = 6 ;
unsigned int m , n , p ;
int ref_clk ;
2015-05-12 15:23:09 +03:00
int delta = target_dsi_clk ;
2013-08-27 23:40:56 +03:00
u32 m_seed ;
2015-05-12 15:23:09 +03:00
/* target_dsi_clk is expected in kHz */
if ( target_dsi_clk < 300000 | | target_dsi_clk > 1150000 ) {
2013-08-27 23:40:56 +03:00
DRM_ERROR ( " DSI CLK Out of Range \n " ) ;
return - ECHRNG ;
}
2015-07-01 15:58:52 +03:00
if ( IS_CHERRYVIEW ( dev_priv ) ) {
ref_clk = 100000 ;
n = 4 ;
m_min = 70 ;
m_max = 96 ;
} else {
ref_clk = 25000 ;
n = 1 ;
m_min = 62 ;
m_max = 92 ;
}
for ( m = m_min ; m < = m_max & & delta ; m + + ) {
for ( p = p_min ; p < = p_max & & delta ; p + + ) {
2015-05-12 15:23:09 +03:00
/*
* Find the optimal m and p divisors with minimal delta
* + / - the required clock
*/
2015-05-13 10:35:25 +03:00
int calc_dsi_clk = ( m * ref_clk ) / ( p * n ) ;
2015-05-12 15:23:09 +03:00
int d = abs ( target_dsi_clk - calc_dsi_clk ) ;
if ( d < delta ) {
delta = d ;
2013-12-10 12:14:57 +05:30
calc_m = m ;
calc_p = p ;
2013-08-27 23:40:56 +03:00
}
}
}
2015-05-13 10:35:25 +03:00
/* register has log2(N1), this works fine for powers of two */
n = ffs ( n ) - 1 ;
2013-08-27 23:40:56 +03:00
m_seed = lfsr_converts [ calc_m - 62 ] ;
dsi_mnp - > dsi_pll_ctrl = 1 < < ( DSI_PLL_P1_POST_DIV_SHIFT + calc_p - 2 ) ;
2015-05-13 10:35:25 +03:00
dsi_mnp - > dsi_pll_div = n < < DSI_PLL_N1_DIV_SHIFT |
2013-08-27 23:40:56 +03:00
m_seed < < DSI_PLL_M1_DIV_SHIFT ;
return 0 ;
}
/*
* XXX : The muxing and gating is hard coded for now . Need to add support for
* sharing PLLs with two DSI outputs .
*/
static void vlv_configure_dsi_pll ( struct intel_encoder * encoder )
{
struct drm_i915_private * dev_priv = encoder - > base . dev - > dev_private ;
struct intel_dsi * intel_dsi = enc_to_intel_dsi ( & encoder - > base ) ;
int ret ;
struct dsi_mnp dsi_mnp ;
u32 dsi_clk ;
2014-07-30 20:34:57 +05:30
dsi_clk = dsi_clk_from_pclk ( intel_dsi - > pclk , intel_dsi - > pixel_format ,
2014-07-30 22:34:27 +02:00
intel_dsi - > lane_count ) ;
2013-08-27 23:40:56 +03:00
2015-07-01 15:58:52 +03:00
ret = dsi_calc_mnp ( dev_priv , & dsi_mnp , dsi_clk ) ;
2013-08-27 23:40:56 +03:00
if ( ret ) {
DRM_DEBUG_KMS ( " dsi_calc_mnp failed \n " ) ;
return ;
}
2014-12-09 10:57:00 +05:30
if ( intel_dsi - > ports & ( 1 < < PORT_A ) )
dsi_mnp . dsi_pll_ctrl | = DSI_PLL_CLK_GATE_DSI0_DSIPLL ;
2013-08-27 23:40:56 +03:00
2014-12-09 10:57:00 +05:30
if ( intel_dsi - > ports & ( 1 < < PORT_C ) )
2014-12-04 10:58:52 +05:30
dsi_mnp . dsi_pll_ctrl | = DSI_PLL_CLK_GATE_DSI1_DSIPLL ;
2013-08-27 23:40:56 +03:00
DRM_DEBUG_KMS ( " dsi pll div %08x, ctrl %08x \n " ,
dsi_mnp . dsi_pll_div , dsi_mnp . dsi_pll_ctrl ) ;
vlv_cck_write ( dev_priv , CCK_REG_DSI_PLL_CONTROL , 0 ) ;
vlv_cck_write ( dev_priv , CCK_REG_DSI_PLL_DIVIDER , dsi_mnp . dsi_pll_div ) ;
vlv_cck_write ( dev_priv , CCK_REG_DSI_PLL_CONTROL , dsi_mnp . dsi_pll_ctrl ) ;
}
2015-09-01 19:41:38 +05:30
static void vlv_enable_dsi_pll ( struct intel_encoder * encoder )
2013-08-27 23:40:56 +03:00
{
struct drm_i915_private * dev_priv = encoder - > base . dev - > dev_private ;
u32 tmp ;
DRM_DEBUG_KMS ( " \n " ) ;
2015-05-26 20:42:30 +03:00
mutex_lock ( & dev_priv - > sb_lock ) ;
2013-08-27 23:40:56 +03:00
vlv_configure_dsi_pll ( encoder ) ;
/* wait at least 0.5 us after ungating before enabling VCO */
usleep_range ( 1 , 10 ) ;
tmp = vlv_cck_read ( dev_priv , CCK_REG_DSI_PLL_CONTROL ) ;
tmp | = DSI_PLL_VCO_EN ;
vlv_cck_write ( dev_priv , CCK_REG_DSI_PLL_CONTROL , tmp ) ;
2014-12-05 14:16:58 +05:30
if ( wait_for ( vlv_cck_read ( dev_priv , CCK_REG_DSI_PLL_CONTROL ) &
DSI_PLL_LOCK , 20 ) ) {
2013-08-27 23:40:56 +03:00
2015-05-26 20:42:30 +03:00
mutex_unlock ( & dev_priv - > sb_lock ) ;
2013-08-27 23:40:56 +03:00
DRM_ERROR ( " DSI PLL lock failed \n " ) ;
return ;
}
2015-05-26 20:42:30 +03:00
mutex_unlock ( & dev_priv - > sb_lock ) ;
2013-08-27 23:40:56 +03:00
DRM_DEBUG_KMS ( " DSI PLL locked \n " ) ;
}
void vlv_disable_dsi_pll ( struct intel_encoder * encoder )
{
struct drm_i915_private * dev_priv = encoder - > base . dev - > dev_private ;
u32 tmp ;
DRM_DEBUG_KMS ( " \n " ) ;
2015-05-26 20:42:30 +03:00
mutex_lock ( & dev_priv - > sb_lock ) ;
2013-08-27 23:40:56 +03:00
tmp = vlv_cck_read ( dev_priv , CCK_REG_DSI_PLL_CONTROL ) ;
tmp & = ~ DSI_PLL_VCO_EN ;
tmp | = DSI_PLL_LDO_GATE ;
vlv_cck_write ( dev_priv , CCK_REG_DSI_PLL_CONTROL , tmp ) ;
2015-05-26 20:42:30 +03:00
mutex_unlock ( & dev_priv - > sb_lock ) ;
2013-08-27 23:40:56 +03:00
}
2014-07-30 20:32:37 +05:30
static void assert_bpp_mismatch ( int pixel_format , int pipe_bpp )
{
2015-07-01 15:58:50 +03:00
int bpp = dsi_pixel_format_bpp ( pixel_format ) ;
2014-07-30 20:32:37 +05:30
WARN ( bpp ! = pipe_bpp ,
2014-07-30 22:34:27 +02:00
" bpp match assertion failure (expected %d, current %d) \n " ,
bpp , pipe_bpp ) ;
2014-07-30 20:32:37 +05:30
}
u32 vlv_get_dsi_pclk ( struct intel_encoder * encoder , int pipe_bpp )
{
struct drm_i915_private * dev_priv = encoder - > base . dev - > dev_private ;
struct intel_dsi * intel_dsi = enc_to_intel_dsi ( & encoder - > base ) ;
u32 dsi_clock , pclk ;
u32 pll_ctl , pll_div ;
2015-05-13 10:35:25 +03:00
u32 m = 0 , p = 0 , n ;
2014-07-30 20:32:37 +05:30
int refclk = 25000 ;
int i ;
DRM_DEBUG_KMS ( " \n " ) ;
2015-05-26 20:42:30 +03:00
mutex_lock ( & dev_priv - > sb_lock ) ;
2014-07-30 20:32:37 +05:30
pll_ctl = vlv_cck_read ( dev_priv , CCK_REG_DSI_PLL_CONTROL ) ;
pll_div = vlv_cck_read ( dev_priv , CCK_REG_DSI_PLL_DIVIDER ) ;
2015-05-26 20:42:30 +03:00
mutex_unlock ( & dev_priv - > sb_lock ) ;
2014-07-30 20:32:37 +05:30
/* mask out other bits and extract the P1 divisor */
pll_ctl & = DSI_PLL_P1_POST_DIV_MASK ;
pll_ctl = pll_ctl > > ( DSI_PLL_P1_POST_DIV_SHIFT - 2 ) ;
2015-05-13 10:35:25 +03:00
/* N1 divisor */
n = ( pll_div & DSI_PLL_N1_DIV_MASK ) > > DSI_PLL_N1_DIV_SHIFT ;
n = 1 < < n ; /* register has log2(N1) */
2014-07-30 20:32:37 +05:30
/* mask out the other bits and extract the M1 divisor */
pll_div & = DSI_PLL_M1_DIV_MASK ;
pll_div = pll_div > > DSI_PLL_M1_DIV_SHIFT ;
while ( pll_ctl ) {
pll_ctl = pll_ctl > > 1 ;
p + + ;
}
p - - ;
if ( ! p ) {
DRM_ERROR ( " wrong P1 divisor \n " ) ;
return 0 ;
}
for ( i = 0 ; i < ARRAY_SIZE ( lfsr_converts ) ; i + + ) {
if ( lfsr_converts [ i ] = = pll_div )
break ;
}
if ( i = = ARRAY_SIZE ( lfsr_converts ) ) {
DRM_ERROR ( " wrong m_seed programmed \n " ) ;
return 0 ;
}
m = i + 62 ;
2015-05-13 10:35:25 +03:00
dsi_clock = ( m * refclk ) / ( p * n ) ;
2014-07-30 20:32:37 +05:30
/* pixel_format and pipe_bpp should agree */
assert_bpp_mismatch ( intel_dsi - > pixel_format , pipe_bpp ) ;
pclk = DIV_ROUND_CLOSEST ( dsi_clock * intel_dsi - > lane_count , pipe_bpp ) ;
return pclk ;
}
2015-09-01 19:41:38 +05:30
static bool bxt_configure_dsi_pll ( struct intel_encoder * encoder )
{
struct drm_i915_private * dev_priv = encoder - > base . dev - > dev_private ;
struct intel_dsi * intel_dsi = enc_to_intel_dsi ( & encoder - > base ) ;
u8 dsi_ratio ;
u32 dsi_clk ;
u32 val ;
dsi_clk = dsi_clk_from_pclk ( intel_dsi - > pclk , intel_dsi - > pixel_format ,
intel_dsi - > lane_count ) ;
/*
* From clock diagram , to get PLL ratio divider , divide double of DSI
* link rate ( i . e . , 2 * 8 x = 16 x frequency value ) by ref clock . Make sure to
* round ' up ' the result
*/
dsi_ratio = DIV_ROUND_UP ( dsi_clk * 2 , BXT_REF_CLOCK_KHZ ) ;
if ( dsi_ratio < BXT_DSI_PLL_RATIO_MIN | |
dsi_ratio > BXT_DSI_PLL_RATIO_MAX ) {
DRM_ERROR ( " Cant get a suitable ratio from DSI PLL ratios \n " ) ;
return false ;
}
/*
* Program DSI ratio and Select MIPIC and MIPIA PLL output as 8 x
* Spec says both have to be programmed , even if one is not getting
* used . Configure MIPI_CLOCK_CTL dividers in modeset
*/
val = I915_READ ( BXT_DSI_PLL_CTL ) ;
val & = ~ BXT_DSI_PLL_PVD_RATIO_MASK ;
val & = ~ BXT_DSI_FREQ_SEL_MASK ;
val & = ~ BXT_DSI_PLL_RATIO_MASK ;
val | = ( dsi_ratio | BXT_DSIA_16X_BY2 | BXT_DSIC_16X_BY2 ) ;
/* As per recommendation from hardware team,
* Prog PVD ratio = 1 if dsi ratio < = 50
*/
if ( dsi_ratio < = 50 ) {
val & = ~ BXT_DSI_PLL_PVD_RATIO_MASK ;
val | = BXT_DSI_PLL_PVD_RATIO_1 ;
}
I915_WRITE ( BXT_DSI_PLL_CTL , val ) ;
POSTING_READ ( BXT_DSI_PLL_CTL ) ;
return true ;
}
static void bxt_enable_dsi_pll ( struct intel_encoder * encoder )
{
struct drm_i915_private * dev_priv = encoder - > base . dev - > dev_private ;
u32 val ;
DRM_DEBUG_KMS ( " \n " ) ;
val = I915_READ ( BXT_DSI_PLL_ENABLE ) ;
if ( val & BXT_DSI_PLL_DO_ENABLE ) {
WARN ( 1 , " DSI PLL already enabled. Disabling it. \n " ) ;
val & = ~ BXT_DSI_PLL_DO_ENABLE ;
I915_WRITE ( BXT_DSI_PLL_ENABLE , val ) ;
}
/* Configure PLL vales */
if ( ! bxt_configure_dsi_pll ( encoder ) ) {
DRM_ERROR ( " Configure DSI PLL failed, abort PLL enable \n " ) ;
return ;
}
/* Enable DSI PLL */
val = I915_READ ( BXT_DSI_PLL_ENABLE ) ;
val | = BXT_DSI_PLL_DO_ENABLE ;
I915_WRITE ( BXT_DSI_PLL_ENABLE , val ) ;
/* Timeout and fail if PLL not locked */
if ( wait_for ( I915_READ ( BXT_DSI_PLL_ENABLE ) & BXT_DSI_PLL_LOCKED , 1 ) ) {
DRM_ERROR ( " Timed out waiting for DSI PLL to lock \n " ) ;
return ;
}
DRM_DEBUG_KMS ( " DSI PLL locked \n " ) ;
}
void intel_enable_dsi_pll ( struct intel_encoder * encoder )
{
struct drm_device * dev = encoder - > base . dev ;
if ( IS_VALLEYVIEW ( dev ) )
vlv_enable_dsi_pll ( encoder ) ;
else if ( IS_BROXTON ( dev ) )
bxt_enable_dsi_pll ( encoder ) ;
}