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"
2016-03-15 16:40:02 +02:00
static const u16 lfsr_converts [ ] = {
2013-08-27 23:40:56 +03:00
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
/* Get DSI clock from pixel clock */
2016-03-16 12:21:40 +02:00
static u32 dsi_clk_from_pclk ( u32 pclk , enum mipi_dsi_pixel_format fmt ,
int lane_count )
2013-08-27 23:40:56 +03:00
{
2013-12-10 12:14:56 +05:30
u32 dsi_clk_khz ;
2016-03-16 12:21:40 +02:00
u32 bpp = mipi_dsi_pixel_format_to_bpp ( fmt ) ;
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
}
2015-07-01 15:58:52 +03:00
static int dsi_calc_mnp ( struct drm_i915_private * dev_priv ,
2016-04-12 22:14:35 +03:00
struct intel_crtc_state * config ,
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 ] ;
2016-04-12 22:14:35 +03:00
config - > dsi_pll . ctrl = 1 < < ( DSI_PLL_P1_POST_DIV_SHIFT + calc_p - 2 ) ;
config - > 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 .
*/
2016-04-12 22:14:35 +03:00
static int vlv_compute_dsi_pll ( struct intel_encoder * encoder ,
struct intel_crtc_state * config )
2013-08-27 23:40:56 +03:00
{
struct drm_i915_private * dev_priv = encoder - > base . dev - > dev_private ;
struct intel_dsi * intel_dsi = enc_to_intel_dsi ( & encoder - > base ) ;
int ret ;
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
2016-04-12 22:14:35 +03:00
ret = dsi_calc_mnp ( dev_priv , config , dsi_clk ) ;
2013-08-27 23:40:56 +03:00
if ( ret ) {
DRM_DEBUG_KMS ( " dsi_calc_mnp failed \n " ) ;
2016-04-12 22:14:35 +03:00
return ret ;
2013-08-27 23:40:56 +03:00
}
2014-12-09 10:57:00 +05:30
if ( intel_dsi - > ports & ( 1 < < PORT_A ) )
2016-04-12 22:14:35 +03:00
config - > 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 ) )
2016-04-12 22:14:35 +03:00
config - > dsi_pll . ctrl | = DSI_PLL_CLK_GATE_DSI1_DSIPLL ;
config - > dsi_pll . ctrl | = DSI_PLL_VCO_EN ;
2014-12-04 10:58:52 +05:30
2013-08-27 23:40:56 +03:00
DRM_DEBUG_KMS ( " dsi pll div %08x, ctrl %08x \n " ,
2016-04-12 22:14:35 +03:00
config - > dsi_pll . div , config - > dsi_pll . ctrl ) ;
return 0 ;
}
static void vlv_enable_dsi_pll ( struct intel_encoder * encoder ,
const struct intel_crtc_state * config )
2013-08-27 23:40:56 +03:00
{
2016-04-12 22:14:35 +03:00
struct drm_i915_private * dev_priv = to_i915 ( encoder - > base . dev ) ;
2013-08-27 23:40:56 +03:00
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
2016-04-12 22:14:36 +03:00
vlv_cck_write ( dev_priv , CCK_REG_DSI_PLL_CONTROL , 0 ) ;
vlv_cck_write ( dev_priv , CCK_REG_DSI_PLL_DIVIDER , config - > dsi_pll . div ) ;
vlv_cck_write ( dev_priv , CCK_REG_DSI_PLL_CONTROL ,
config - > dsi_pll . ctrl & ~ DSI_PLL_VCO_EN ) ;
2013-08-27 23:40:56 +03:00
/* wait at least 0.5 us after ungating before enabling VCO */
usleep_range ( 1 , 10 ) ;
2016-04-12 22:14:35 +03:00
vlv_cck_write ( dev_priv , CCK_REG_DSI_PLL_CONTROL , config - > dsi_pll . ctrl ) ;
2013-08-27 23:40:56 +03:00
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 " ) ;
}
2015-09-01 19:41:39 +05:30
static void vlv_disable_dsi_pll ( struct intel_encoder * encoder )
2013-08-27 23:40:56 +03:00
{
2016-04-12 22:14:35 +03:00
struct drm_i915_private * dev_priv = to_i915 ( encoder - > base . dev ) ;
2013-08-27 23:40:56 +03:00
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
2016-03-24 12:41:40 +02:00
static bool bxt_dsi_pll_is_enabled ( struct drm_i915_private * dev_priv )
{
bool enabled ;
u32 val ;
u32 mask ;
mask = BXT_DSI_PLL_DO_ENABLE | BXT_DSI_PLL_LOCKED ;
val = I915_READ ( BXT_DSI_PLL_ENABLE ) ;
enabled = ( val & mask ) = = mask ;
if ( ! enabled )
return false ;
/*
* Both dividers must be programmed with valid values even if only one
* of the PLL is used , see BSpec / Broxton Clocks . Check this here for
* paranoia , since BIOS is known to misconfigure PLLs in this way at
* times , and since accessing DSI registers with invalid dividers
* causes a system hang .
*/
val = I915_READ ( BXT_DSI_PLL_CTL ) ;
if ( ! ( val & BXT_DSIA_16X_MASK ) | | ! ( val & BXT_DSIC_16X_MASK ) ) {
DRM_DEBUG_DRIVER ( " PLL is enabled with invalid divider settings (%08x) \n " ,
val ) ;
enabled = false ;
}
return enabled ;
}
2015-09-01 19:41:39 +05:30
static void bxt_disable_dsi_pll ( struct intel_encoder * encoder )
{
2016-04-12 22:14:35 +03:00
struct drm_i915_private * dev_priv = to_i915 ( encoder - > base . dev ) ;
2015-09-01 19:41:39 +05:30
u32 val ;
DRM_DEBUG_KMS ( " \n " ) ;
val = I915_READ ( BXT_DSI_PLL_ENABLE ) ;
val & = ~ BXT_DSI_PLL_DO_ENABLE ;
I915_WRITE ( BXT_DSI_PLL_ENABLE , val ) ;
/*
* PLL lock should deassert within 200u s .
* Wait up to 1 ms before timing out .
*/
if ( wait_for ( ( I915_READ ( BXT_DSI_PLL_ENABLE )
& BXT_DSI_PLL_LOCKED ) = = 0 , 1 ) )
DRM_ERROR ( " Timeout waiting for PLL lock deassertion \n " ) ;
}
2016-03-16 12:21:40 +02:00
static void assert_bpp_mismatch ( enum mipi_dsi_pixel_format fmt , int pipe_bpp )
2014-07-30 20:32:37 +05:30
{
2016-03-16 12:21:40 +02:00
int bpp = mipi_dsi_pixel_format_to_bpp ( fmt ) ;
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
}
2016-04-12 22:14:35 +03:00
static u32 vlv_dsi_get_pclk ( struct intel_encoder * encoder , int pipe_bpp ,
struct intel_crtc_state * config )
2014-07-30 20:32:37 +05:30
{
2016-04-12 22:14:35 +03:00
struct drm_i915_private * dev_priv = to_i915 ( encoder - > base . dev ) ;
2014-07-30 20:32:37 +05:30
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 ;
2016-03-15 16:40:05 +02:00
int refclk = IS_CHERRYVIEW ( dev_priv ) ? 100000 : 25000 ;
2014-07-30 20:32:37 +05:30
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
2016-04-12 22:14:35 +03:00
config - > dsi_pll . ctrl = pll_ctl & ~ DSI_PLL_LOCK ;
config - > dsi_pll . div = pll_div ;
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
2016-04-12 22:14:35 +03:00
static u32 bxt_dsi_get_pclk ( struct intel_encoder * encoder , int pipe_bpp ,
struct intel_crtc_state * config )
2015-09-01 19:41:46 +05:30
{
u32 pclk ;
u32 dsi_clk ;
u32 dsi_ratio ;
struct intel_dsi * intel_dsi = enc_to_intel_dsi ( & encoder - > base ) ;
struct drm_i915_private * dev_priv = encoder - > base . dev - > dev_private ;
/* Divide by zero */
if ( ! pipe_bpp ) {
DRM_ERROR ( " Invalid BPP(0) \n " ) ;
return 0 ;
}
2016-04-12 22:14:35 +03:00
config - > dsi_pll . ctrl = I915_READ ( BXT_DSI_PLL_CTL ) ;
2015-09-01 19:41:46 +05:30
2016-04-12 22:14:35 +03:00
dsi_ratio = config - > dsi_pll . ctrl & BXT_DSI_PLL_RATIO_MASK ;
2015-09-01 19:41:46 +05:30
dsi_clk = ( dsi_ratio * BXT_REF_CLOCK_KHZ ) / 2 ;
/* pixel_format and pipe_bpp should agree */
assert_bpp_mismatch ( intel_dsi - > pixel_format , pipe_bpp ) ;
pclk = DIV_ROUND_CLOSEST ( dsi_clk * intel_dsi - > lane_count , pipe_bpp ) ;
DRM_DEBUG_DRIVER ( " Calculated pclk=%u \n " , pclk ) ;
return pclk ;
}
2016-04-12 22:14:35 +03:00
u32 intel_dsi_get_pclk ( struct intel_encoder * encoder , int pipe_bpp ,
struct intel_crtc_state * config )
2016-01-08 12:45:39 +02:00
{
if ( IS_BROXTON ( encoder - > base . dev ) )
2016-04-12 22:14:35 +03:00
return bxt_dsi_get_pclk ( encoder , pipe_bpp , config ) ;
2016-01-08 12:45:39 +02:00
else
2016-04-12 22:14:35 +03:00
return vlv_dsi_get_pclk ( encoder , pipe_bpp , config ) ;
2016-01-08 12:45:39 +02:00
}
2015-10-05 16:21:11 +08:00
static void vlv_dsi_reset_clocks ( struct intel_encoder * encoder , enum port port )
2015-09-01 19:41:44 +05:30
{
u32 temp ;
struct drm_i915_private * dev_priv = encoder - > base . dev - > dev_private ;
struct intel_dsi * intel_dsi = enc_to_intel_dsi ( & encoder - > base ) ;
temp = I915_READ ( MIPI_CTRL ( port ) ) ;
temp & = ~ ESCAPE_CLOCK_DIVIDER_MASK ;
I915_WRITE ( MIPI_CTRL ( port ) , temp |
intel_dsi - > escape_clk_div < <
ESCAPE_CLOCK_DIVIDER_SHIFT ) ;
}
2015-09-23 23:27:17 +05:30
/* Program BXT Mipi clocks and dividers */
2016-04-12 22:14:35 +03:00
static void bxt_dsi_program_clocks ( struct drm_device * dev , enum port port ,
const struct intel_crtc_state * config )
2015-09-23 23:27:17 +05:30
{
struct drm_i915_private * dev_priv = dev - > dev_private ;
2016-02-15 22:43:57 +05:30
u32 tmp ;
u32 dsi_rate = 0 ;
u32 pll_ratio = 0 ;
u32 rx_div ;
u32 tx_div ;
u32 rx_div_upper ;
u32 rx_div_lower ;
u32 mipi_8by3_divider ;
2015-09-23 23:27:17 +05:30
/* Clear old configurations */
tmp = I915_READ ( BXT_MIPI_CLOCK_CTL ) ;
tmp & = ~ ( BXT_MIPI_TX_ESCLK_FIXDIV_MASK ( port ) ) ;
2016-02-15 22:43:57 +05:30
tmp & = ~ ( BXT_MIPI_RX_ESCLK_UPPER_FIXDIV_MASK ( port ) ) ;
tmp & = ~ ( BXT_MIPI_8X_BY3_DIVIDER_MASK ( port ) ) ;
tmp & = ~ ( BXT_MIPI_RX_ESCLK_LOWER_FIXDIV_MASK ( port ) ) ;
2015-09-23 23:27:17 +05:30
/* Get the current DSI rate(actual) */
2016-04-12 22:14:35 +03:00
pll_ratio = config - > dsi_pll . ctrl & BXT_DSI_PLL_RATIO_MASK ;
2015-09-23 23:27:17 +05:30
dsi_rate = ( BXT_REF_CLOCK_KHZ * pll_ratio ) / 2 ;
2016-02-15 22:43:57 +05:30
/*
* tx clock should be < = 20 MHz and the div value must be
* subtracted by 1 as per bspec
*/
tx_div = DIV_ROUND_UP ( dsi_rate , 20000 ) - 1 ;
/*
* rx clock should be < = 150 MHz and the div value must be
* subtracted by 1 as per bspec
*/
rx_div = DIV_ROUND_UP ( dsi_rate , 150000 ) - 1 ;
2015-09-23 23:27:17 +05:30
/*
2016-02-15 22:43:57 +05:30
* rx divider value needs to be updated in the
* two differnt bit fields in the register hence splitting the
* rx divider value accordingly
2015-09-23 23:27:17 +05:30
*/
2016-02-15 22:43:57 +05:30
rx_div_lower = rx_div & RX_DIVIDER_BIT_1_2 ;
rx_div_upper = ( rx_div & RX_DIVIDER_BIT_3_4 ) > > 2 ;
/* As per bpsec program the 8/3X clock divider to the below value */
if ( dev_priv - > vbt . dsi . config - > is_cmd_mode )
mipi_8by3_divider = 0x2 ;
else
mipi_8by3_divider = 0x3 ;
2015-09-23 23:27:17 +05:30
2016-02-15 22:43:57 +05:30
tmp | = BXT_MIPI_8X_BY3_DIVIDER ( port , mipi_8by3_divider ) ;
tmp | = BXT_MIPI_TX_ESCLK_DIVIDER ( port , tx_div ) ;
tmp | = BXT_MIPI_RX_ESCLK_LOWER_DIVIDER ( port , rx_div_lower ) ;
tmp | = BXT_MIPI_RX_ESCLK_UPPER_DIVIDER ( port , rx_div_upper ) ;
2015-09-23 23:27:17 +05:30
I915_WRITE ( BXT_MIPI_CLOCK_CTL , tmp ) ;
}
2016-04-12 22:14:35 +03:00
static int bxt_compute_dsi_pll ( struct intel_encoder * encoder ,
struct intel_crtc_state * config )
2015-09-01 19:41:38 +05:30
{
struct intel_dsi * intel_dsi = enc_to_intel_dsi ( & encoder - > base ) ;
u8 dsi_ratio ;
u32 dsi_clk ;
dsi_clk = dsi_clk_from_pclk ( intel_dsi - > pclk , intel_dsi - > pixel_format ,
2016-04-12 22:14:35 +03:00
intel_dsi - > lane_count ) ;
2015-09-01 19:41:38 +05:30
/*
* 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 | |
2016-04-12 22:14:35 +03:00
dsi_ratio > BXT_DSI_PLL_RATIO_MAX ) {
2015-09-01 19:41:38 +05:30
DRM_ERROR ( " Cant get a suitable ratio from DSI PLL ratios \n " ) ;
2016-04-12 22:14:35 +03:00
return - ECHRNG ;
2015-09-01 19:41:38 +05:30
}
/*
* 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
*/
2016-04-12 22:14:35 +03:00
config - > dsi_pll . ctrl = dsi_ratio | BXT_DSIA_16X_BY2 | BXT_DSIC_16X_BY2 ;
2015-09-01 19:41:38 +05:30
/* As per recommendation from hardware team,
* Prog PVD ratio = 1 if dsi ratio < = 50
*/
2016-04-12 22:14:35 +03:00
if ( dsi_ratio < = 50 )
config - > dsi_pll . ctrl | = BXT_DSI_PLL_PVD_RATIO_1 ;
2015-09-01 19:41:38 +05:30
2016-04-12 22:14:35 +03:00
return 0 ;
}
2015-09-01 19:41:38 +05:30
2016-04-12 22:14:35 +03:00
static void bxt_enable_dsi_pll ( struct intel_encoder * encoder ,
const struct intel_crtc_state * config )
2015-09-01 19:41:38 +05:30
{
struct drm_i915_private * dev_priv = encoder - > base . dev - > dev_private ;
2015-09-23 23:27:17 +05:30
struct intel_dsi * intel_dsi = enc_to_intel_dsi ( & encoder - > base ) ;
enum port port ;
2015-09-01 19:41:38 +05:30
u32 val ;
DRM_DEBUG_KMS ( " \n " ) ;
/* Configure PLL vales */
2016-04-12 22:14:36 +03:00
I915_WRITE ( BXT_DSI_PLL_CTL , config - > dsi_pll . ctrl ) ;
POSTING_READ ( BXT_DSI_PLL_CTL ) ;
2015-09-01 19:41:38 +05:30
2015-09-23 23:27:17 +05:30
/* Program TX, RX, Dphy clocks */
for_each_dsi_port ( port , intel_dsi - > ports )
2016-04-12 22:14:35 +03:00
bxt_dsi_program_clocks ( encoder - > base . dev , port , config ) ;
2015-09-23 23:27:17 +05:30
2015-09-01 19:41:38 +05:30
/* 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 " ) ;
}
2016-03-24 12:41:40 +02:00
bool intel_dsi_pll_is_enabled ( struct drm_i915_private * dev_priv )
{
if ( IS_BROXTON ( dev_priv ) )
return bxt_dsi_pll_is_enabled ( dev_priv ) ;
MISSING_CASE ( INTEL_DEVID ( dev_priv ) ) ;
return false ;
}
2016-04-12 22:14:35 +03:00
int intel_compute_dsi_pll ( struct intel_encoder * encoder ,
struct intel_crtc_state * config )
{
struct drm_device * dev = encoder - > base . dev ;
if ( IS_VALLEYVIEW ( dev ) | | IS_CHERRYVIEW ( dev ) )
return vlv_compute_dsi_pll ( encoder , config ) ;
else if ( IS_BROXTON ( dev ) )
return bxt_compute_dsi_pll ( encoder , config ) ;
return - ENODEV ;
}
void intel_enable_dsi_pll ( struct intel_encoder * encoder ,
const struct intel_crtc_state * config )
2015-09-01 19:41:38 +05:30
{
struct drm_device * dev = encoder - > base . dev ;
2015-12-09 12:29:35 -08:00
if ( IS_VALLEYVIEW ( dev ) | | IS_CHERRYVIEW ( dev ) )
2016-04-12 22:14:35 +03:00
vlv_enable_dsi_pll ( encoder , config ) ;
2015-09-01 19:41:38 +05:30
else if ( IS_BROXTON ( dev ) )
2016-04-12 22:14:35 +03:00
bxt_enable_dsi_pll ( encoder , config ) ;
2015-09-01 19:41:38 +05:30
}
2015-09-01 19:41:39 +05:30
void intel_disable_dsi_pll ( struct intel_encoder * encoder )
{
struct drm_device * dev = encoder - > base . dev ;
2015-12-09 12:29:35 -08:00
if ( IS_VALLEYVIEW ( dev ) | | IS_CHERRYVIEW ( dev ) )
2015-09-01 19:41:39 +05:30
vlv_disable_dsi_pll ( encoder ) ;
else if ( IS_BROXTON ( dev ) )
bxt_disable_dsi_pll ( encoder ) ;
}
2015-09-01 19:41:44 +05:30
2015-10-05 16:21:11 +08:00
static void bxt_dsi_reset_clocks ( struct intel_encoder * encoder , enum port port )
2015-09-01 19:41:44 +05:30
{
u32 tmp ;
struct drm_device * dev = encoder - > base . dev ;
struct drm_i915_private * dev_priv = dev - > dev_private ;
/* Clear old configurations */
tmp = I915_READ ( BXT_MIPI_CLOCK_CTL ) ;
tmp & = ~ ( BXT_MIPI_TX_ESCLK_FIXDIV_MASK ( port ) ) ;
2016-02-15 22:43:57 +05:30
tmp & = ~ ( BXT_MIPI_RX_ESCLK_UPPER_FIXDIV_MASK ( port ) ) ;
tmp & = ~ ( BXT_MIPI_8X_BY3_DIVIDER_MASK ( port ) ) ;
tmp & = ~ ( BXT_MIPI_RX_ESCLK_LOWER_FIXDIV_MASK ( port ) ) ;
2015-09-01 19:41:44 +05:30
I915_WRITE ( BXT_MIPI_CLOCK_CTL , tmp ) ;
I915_WRITE ( MIPI_EOT_DISABLE ( port ) , CLOCKSTOP ) ;
}
void intel_dsi_reset_clocks ( struct intel_encoder * encoder , enum port port )
{
struct drm_device * dev = encoder - > base . dev ;
if ( IS_BROXTON ( dev ) )
bxt_dsi_reset_clocks ( encoder , port ) ;
2015-12-09 12:29:35 -08:00
else if ( IS_VALLEYVIEW ( dev ) | | IS_CHERRYVIEW ( dev ) )
2015-09-01 19:41:44 +05:30
vlv_dsi_reset_clocks ( encoder , port ) ;
}