You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
kernel_samsung_sm7125/drivers/clk/qcom/mdss/mdss-dsi-20nm-pll-util.c

1011 lines
30 KiB

/* Copyright (c) 2014-2017, The Linux Foundation. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2 and
* only version 2 as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*/
#define pr_fmt(fmt) "%s: " fmt, __func__
#include <linux/kernel.h>
#include <linux/err.h>
#include <linux/iopoll.h>
#include <linux/delay.h>
#include <linux/clk/msm-clock-generic.h>
#include "mdss-pll.h"
#include "mdss-dsi-pll.h"
#define MMSS_DSI_PHY_PLL_SYS_CLK_CTRL 0x0000
#define MMSS_DSI_PHY_PLL_PLL_VCOTAIL_EN 0x0004
#define MMSS_DSI_PHY_PLL_CMN_MODE 0x0008
#define MMSS_DSI_PHY_PLL_IE_TRIM 0x000C
#define MMSS_DSI_PHY_PLL_IP_TRIM 0x0010
#define MMSS_DSI_PHY_PLL_PLL_PHSEL_CONTROL 0x0018
#define MMSS_DSI_PHY_PLL_IPTAT_TRIM_VCCA_TX_SEL 0x001C
#define MMSS_DSI_PHY_PLL_PLL_PHSEL_DC 0x0020
#define MMSS_DSI_PHY_PLL_PLL_IP_SETI 0x0024
#define MMSS_DSI_PHY_PLL_CORE_CLK_IN_SYNC_SEL 0x0028
#define MMSS_DSI_PHY_PLL_BIAS_EN_CLKBUFLR_EN 0x0030
#define MMSS_DSI_PHY_PLL_PLL_CP_SETI 0x0034
#define MMSS_DSI_PHY_PLL_PLL_IP_SETP 0x0038
#define MMSS_DSI_PHY_PLL_PLL_CP_SETP 0x003C
#define MMSS_DSI_PHY_PLL_ATB_SEL1 0x0040
#define MMSS_DSI_PHY_PLL_ATB_SEL2 0x0044
#define MMSS_DSI_PHY_PLL_SYSCLK_EN_SEL_TXBAND 0x0048
#define MMSS_DSI_PHY_PLL_RESETSM_CNTRL 0x004C
#define MMSS_DSI_PHY_PLL_RESETSM_CNTRL2 0x0050
#define MMSS_DSI_PHY_PLL_RESETSM_CNTRL3 0x0054
#define MMSS_DSI_PHY_PLL_RESETSM_PLL_CAL_COUNT1 0x0058
#define MMSS_DSI_PHY_PLL_RESETSM_PLL_CAL_COUNT2 0x005C
#define MMSS_DSI_PHY_PLL_DIV_REF1 0x0060
#define MMSS_DSI_PHY_PLL_DIV_REF2 0x0064
#define MMSS_DSI_PHY_PLL_KVCO_COUNT1 0x0068
#define MMSS_DSI_PHY_PLL_KVCO_COUNT2 0x006C
#define MMSS_DSI_PHY_PLL_KVCO_CAL_CNTRL 0x0070
#define MMSS_DSI_PHY_PLL_KVCO_CODE 0x0074
#define MMSS_DSI_PHY_PLL_VREF_CFG1 0x0078
#define MMSS_DSI_PHY_PLL_VREF_CFG2 0x007C
#define MMSS_DSI_PHY_PLL_VREF_CFG3 0x0080
#define MMSS_DSI_PHY_PLL_VREF_CFG4 0x0084
#define MMSS_DSI_PHY_PLL_VREF_CFG5 0x0088
#define MMSS_DSI_PHY_PLL_VREF_CFG6 0x008C
#define MMSS_DSI_PHY_PLL_PLLLOCK_CMP1 0x0090
#define MMSS_DSI_PHY_PLL_PLLLOCK_CMP2 0x0094
#define MMSS_DSI_PHY_PLL_PLLLOCK_CMP3 0x0098
#define MMSS_DSI_PHY_PLL_BGTC 0x00A0
#define MMSS_DSI_PHY_PLL_PLL_TEST_UPDN 0x00A4
#define MMSS_DSI_PHY_PLL_PLL_VCO_TUNE 0x00A8
#define MMSS_DSI_PHY_PLL_DEC_START1 0x00AC
#define MMSS_DSI_PHY_PLL_PLL_AMP_OS 0x00B0
#define MMSS_DSI_PHY_PLL_SSC_EN_CENTER 0x00B4
#define MMSS_DSI_PHY_PLL_SSC_ADJ_PER1 0x00B8
#define MMSS_DSI_PHY_PLL_SSC_ADJ_PER2 0x00BC
#define MMSS_DSI_PHY_PLL_SSC_PER1 0x00C0
#define MMSS_DSI_PHY_PLL_SSC_PER2 0x00C4
#define MMSS_DSI_PHY_PLL_SSC_STEP_SIZE1 0x00C8
#define MMSS_DSI_PHY_PLL_SSC_STEP_SIZE2 0x00CC
#define MMSS_DSI_PHY_PLL_RES_CODE_UP 0x00D0
#define MMSS_DSI_PHY_PLL_RES_CODE_DN 0x00D4
#define MMSS_DSI_PHY_PLL_RES_CODE_UP_OFFSET 0x00D8
#define MMSS_DSI_PHY_PLL_RES_CODE_DN_OFFSET 0x00DC
#define MMSS_DSI_PHY_PLL_RES_CODE_START_SEG1 0x00E0
#define MMSS_DSI_PHY_PLL_RES_CODE_START_SEG2 0x00E4
#define MMSS_DSI_PHY_PLL_RES_CODE_CAL_CSR 0x00E8
#define MMSS_DSI_PHY_PLL_RES_CODE 0x00EC
#define MMSS_DSI_PHY_PLL_RES_TRIM_CONTROL 0x00F0
#define MMSS_DSI_PHY_PLL_RES_TRIM_CONTROL2 0x00F4
#define MMSS_DSI_PHY_PLL_RES_TRIM_EN_VCOCALDONE 0x00F8
#define MMSS_DSI_PHY_PLL_FAUX_EN 0x00FC
#define MMSS_DSI_PHY_PLL_DIV_FRAC_START1 0x0100
#define MMSS_DSI_PHY_PLL_DIV_FRAC_START2 0x0104
#define MMSS_DSI_PHY_PLL_DIV_FRAC_START3 0x0108
#define MMSS_DSI_PHY_PLL_DEC_START2 0x010C
#define MMSS_DSI_PHY_PLL_PLL_RXTXEPCLK_EN 0x0110
#define MMSS_DSI_PHY_PLL_PLL_CRCTRL 0x0114
#define MMSS_DSI_PHY_PLL_LOW_POWER_RO_CONTROL 0x013C
#define MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL 0x0140
#define MMSS_DSI_PHY_PLL_HR_OCLK2_DIVIDER 0x0144
#define MMSS_DSI_PHY_PLL_HR_OCLK3_DIVIDER 0x0148
#define MMSS_DSI_PHY_PLL_PLL_VCO_HIGH 0x014C
#define MMSS_DSI_PHY_PLL_RESET_SM 0x0150
#define MMSS_DSI_PHY_PLL_MUXVAL 0x0154
#define MMSS_DSI_PHY_PLL_CORE_RES_CODE_DN 0x0158
#define MMSS_DSI_PHY_PLL_CORE_RES_CODE_UP 0x015C
#define MMSS_DSI_PHY_PLL_CORE_VCO_TUNE 0x0160
#define MMSS_DSI_PHY_PLL_CORE_VCO_TAIL 0x0164
#define MMSS_DSI_PHY_PLL_CORE_KVCO_CODE 0x0168
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL0 0x014
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL1 0x018
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL2 0x01C
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL3 0x020
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL4 0x024
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL5 0x028
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL6 0x02C
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL7 0x030
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL8 0x034
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL9 0x038
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL10 0x03C
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL11 0x040
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL12 0x044
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL13 0x048
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL14 0x04C
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL15 0x050
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL16 0x054
#define MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL17 0x058
#define DSI_PLL_POLL_DELAY_US 1000
#define DSI_PLL_POLL_TIMEOUT_US 15000
int set_mdss_byte_mux_sel(struct mux_clk *clk, int sel)
{
return 0;
}
int get_mdss_byte_mux_sel(struct mux_clk *clk)
{
return 0;
}
int set_mdss_pixel_mux_sel(struct mux_clk *clk, int sel)
{
return 0;
}
int get_mdss_pixel_mux_sel(struct mux_clk *clk)
{
return 0;
}
static void pll_20nm_cache_trim_codes(struct mdss_pll_resources *dsi_pll_res)
{
int rc;
if (dsi_pll_res->reg_upd)
return;
rc = mdss_pll_resource_enable(dsi_pll_res, true);
if (rc) {
pr_err("Failed to enable mdss dsi pll resources\n");
return;
}
dsi_pll_res->cache_pll_trim_codes[0] =
MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_CORE_KVCO_CODE);
dsi_pll_res->cache_pll_trim_codes[1] =
MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_CORE_VCO_TUNE);
pr_debug("core_kvco_code=0x%x core_vco_turn=0x%x\n",
dsi_pll_res->cache_pll_trim_codes[0],
dsi_pll_res->cache_pll_trim_codes[1]);
mdss_pll_resource_enable(dsi_pll_res, false);
dsi_pll_res->reg_upd = true;
}
static void pll_20nm_override_trim_codes(struct mdss_pll_resources *dsi_pll_res)
{
u32 reg_data;
void __iomem *pll_base = dsi_pll_res->pll_base;
/*
* Override mux config for all cached trim codes from
* saved config except for VCO Tune
*/
reg_data = (dsi_pll_res->cache_pll_trim_codes[0] & 0x3f) | BIT(5);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_KVCO_CODE, reg_data);
reg_data = (dsi_pll_res->cache_pll_trim_codes[1] & 0x7f) | BIT(7);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_VCO_TUNE, reg_data);
}
int set_bypass_lp_div_mux_sel(struct mux_clk *clk, int sel)
{
struct mdss_pll_resources *dsi_pll_res = clk->priv;
int reg_data;
pr_debug("bypass_lp_div mux set to %s mode\n",
sel ? "indirect" : "direct");
pr_debug("POST_DIVIDER_CONTROL = 0x%x\n",
MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL));
reg_data = MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL);
reg_data |= BIT(7);
MDSS_PLL_REG_W(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL,
reg_data | (sel << 5));
pr_debug("POST_DIVIDER_CONTROL = 0x%x\n",
MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL));
return 0;
}
int set_shadow_bypass_lp_div_mux_sel(struct mux_clk *clk, int sel)
{
struct mdss_pll_resources *dsi_pll_res = clk->priv;
int reg_data, rem;
if (!dsi_pll_res->resource_enable) {
pr_err("PLL resources disabled. Dynamic fps invalid\n");
return -EINVAL;
}
reg_data = MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL);
reg_data |= BIT(7);
pr_debug("%d: reg_data = %x\n", __LINE__, reg_data);
/* Repeat POST DIVIDER 2 times (4 writes)*/
for (rem = 0; rem < 2; rem++)
MDSS_DYN_PLL_REG_W(dsi_pll_res->dyn_pll_base,
MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL16 + (4 * rem),
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL,
(reg_data | (sel << 5)), (reg_data | (sel << 5)));
return 0;
}
int get_bypass_lp_div_mux_sel(struct mux_clk *clk)
{
int mux_mode, rc;
struct mdss_pll_resources *dsi_pll_res = clk->priv;
if (is_gdsc_disabled(dsi_pll_res))
return 0;
rc = mdss_pll_resource_enable(dsi_pll_res, true);
if (rc) {
pr_err("Failed to enable mdss dsi pll resources\n");
return rc;
}
mux_mode = MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL) & BIT(5);
pr_debug("bypass_lp_div mux mode = %s",
mux_mode ? "indirect" : "direct");
mdss_pll_resource_enable(dsi_pll_res, false);
return !!mux_mode;
}
int ndiv_set_div(struct div_clk *clk, int div)
{
int rc, reg_data;
struct mdss_pll_resources *dsi_pll_res = clk->priv;
rc = mdss_pll_resource_enable(dsi_pll_res, true);
if (rc) {
pr_err("Failed to enable mdss dsi pll resources\n");
return rc;
}
reg_data = MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL);
MDSS_PLL_REG_W(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL,
reg_data | div);
pr_debug("POST_DIVIDER_CONTROL = 0x%x\n",
MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL));
mdss_pll_resource_enable(dsi_pll_res, false);
return rc;
}
int shadow_ndiv_set_div(struct div_clk *clk, int div)
{
struct mdss_pll_resources *dsi_pll_res = clk->priv;
if (!dsi_pll_res->resource_enable) {
pr_err("PLL resources disabled. Dynamic fps invalid\n");
return -EINVAL;
}
pr_debug("%d div=%i\n", __LINE__, div);
MDSS_DYN_PLL_REG_W(dsi_pll_res->dyn_pll_base,
MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL14,
MMSS_DSI_PHY_PLL_RESETSM_CNTRL3,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL,
0x07, (0xB | div));
return 0;
}
int ndiv_get_div(struct div_clk *clk)
{
int div = 0, rc;
struct mdss_pll_resources *dsi_pll_res = clk->priv;
if (is_gdsc_disabled(dsi_pll_res))
return 0;
rc = mdss_pll_resource_enable(clk->priv, true);
if (rc) {
pr_err("Failed to enable mdss dsi pll resources\n");
return rc;
}
div = MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL) & 0x0F;
mdss_pll_resource_enable(dsi_pll_res, false);
return div;
}
int fixed_hr_oclk2_set_div(struct div_clk *clk, int div)
{
int rc;
struct mdss_pll_resources *dsi_pll_res = clk->priv;
rc = mdss_pll_resource_enable(dsi_pll_res, true);
if (rc) {
pr_err("Failed to enable mdss dsi pll resources\n");
return rc;
}
MDSS_PLL_REG_W(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_HR_OCLK2_DIVIDER,
(div - 1));
mdss_pll_resource_enable(dsi_pll_res, false);
return rc;
}
int shadow_fixed_hr_oclk2_set_div(struct div_clk *clk, int div)
{
struct mdss_pll_resources *dsi_pll_res = clk->priv;
if (!dsi_pll_res->resource_enable) {
pr_err("PLL resources disabled. Dynamic fps invalid\n");
return -EINVAL;
}
pr_debug("%d div = %d\n", __LINE__, div);
MDSS_DYN_PLL_REG_W(dsi_pll_res->dyn_pll_base,
MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL5,
MMSS_DSI_PHY_PLL_HR_OCLK2_DIVIDER,
MMSS_DSI_PHY_PLL_HR_OCLK2_DIVIDER,
(div - 1), (div - 1));
return 0;
}
int fixed_hr_oclk2_get_div(struct div_clk *clk)
{
int div = 0, rc;
struct mdss_pll_resources *dsi_pll_res = clk->priv;
if (is_gdsc_disabled(dsi_pll_res))
return 0;
rc = mdss_pll_resource_enable(dsi_pll_res, true);
if (rc) {
pr_err("Failed to enable mdss dsi pll resources\n");
return rc;
}
div = MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_HR_OCLK2_DIVIDER);
mdss_pll_resource_enable(dsi_pll_res, false);
return div + 1;
}
int hr_oclk3_set_div(struct div_clk *clk, int div)
{
int rc;
struct mdss_pll_resources *dsi_pll_res = clk->priv;
rc = mdss_pll_resource_enable(dsi_pll_res, true);
if (rc) {
pr_err("Failed to enable mdss dsi pll resources\n");
return rc;
}
pr_debug("%d div = %d\n", __LINE__, div);
MDSS_PLL_REG_W(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_HR_OCLK3_DIVIDER,
(div - 1));
pr_debug("%s: HR_OCLK3_DIVIDER = 0x%x\n", __func__,
MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_HR_OCLK3_DIVIDER));
mdss_pll_resource_enable(dsi_pll_res, false);
return rc;
}
int shadow_hr_oclk3_set_div(struct div_clk *clk, int div)
{
struct mdss_pll_resources *dsi_pll_res = clk->priv;
if (!dsi_pll_res->resource_enable) {
pr_err("PLL resources disabled. Dynamic fps invalid\n");
return -EINVAL;
}
pr_debug("%d div = %d\n", __LINE__, div);
MDSS_DYN_PLL_REG_W(dsi_pll_res->dyn_pll_base,
MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL6,
MMSS_DSI_PHY_PLL_HR_OCLK3_DIVIDER,
MMSS_DSI_PHY_PLL_HR_OCLK3_DIVIDER,
(div - 1), (div - 1));
return 0;
}
int hr_oclk3_get_div(struct div_clk *clk)
{
int div = 0, rc;
struct mdss_pll_resources *dsi_pll_res = clk->priv;
if (is_gdsc_disabled(dsi_pll_res))
return 0;
rc = mdss_pll_resource_enable(dsi_pll_res, true);
if (rc) {
pr_err("Failed to enable mdss dsi pll resources\n");
return rc;
}
div = MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_HR_OCLK3_DIVIDER);
mdss_pll_resource_enable(dsi_pll_res, false);
return div + 1;
}
static bool pll_20nm_is_pll_locked(struct mdss_pll_resources *dsi_pll_res)
{
u32 status;
bool pll_locked;
/* poll for PLL ready status */
if (readl_poll_timeout_atomic((dsi_pll_res->pll_base +
MMSS_DSI_PHY_PLL_RESET_SM),
status,
((status & BIT(5)) > 0),
DSI_PLL_POLL_DELAY_US,
DSI_PLL_POLL_TIMEOUT_US)) {
pr_debug("DSI PLL status=%x failed to Lock\n", status);
pll_locked = false;
} else if (readl_poll_timeout_atomic((dsi_pll_res->pll_base +
MMSS_DSI_PHY_PLL_RESET_SM),
status,
((status & BIT(6)) > 0),
DSI_PLL_POLL_DELAY_US,
DSI_PLL_POLL_TIMEOUT_US)) {
pr_debug("DSI PLL status=%x PLl not ready\n", status);
pll_locked = false;
} else {
pll_locked = true;
}
return pll_locked;
}
void __dsi_pll_disable(void __iomem *pll_base)
{
if (!pll_base) {
pr_err("Invalid pll base\n");
return;
}
pr_debug("Disabling PHY PLL for PLL_BASE=%p\n", pll_base);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_VCOTAIL_EN, 0x02);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_RESETSM_CNTRL3, 0x06);
}
static void pll_20nm_config_powerdown(void __iomem *pll_base)
{
if (!pll_base) {
pr_err("Invalid pll base.\n");
return;
}
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_SYS_CLK_CTRL, 0x00);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_CMN_MODE, 0x01);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_VCOTAIL_EN, 0x82);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_BIAS_EN_CLKBUFLR_EN, 0x02);
}
static int dsi_pll_enable(struct clk *c)
{
int i, rc;
struct dsi_pll_vco_clk *vco = to_vco_clk(c);
struct mdss_pll_resources *dsi_pll_res = vco->priv;
rc = mdss_pll_resource_enable(dsi_pll_res, true);
if (rc) {
pr_err("Failed to enable mdss dsi pll resources\n");
return rc;
}
/* Try all enable sequences until one succeeds */
for (i = 0; i < vco->pll_en_seq_cnt; i++) {
rc = vco->pll_enable_seqs[i](dsi_pll_res);
pr_debug("DSI PLL %s after sequence #%d\n",
rc ? "unlocked" : "locked", i + 1);
if (!rc)
break;
}
/* Disable PLL1 to avoid current leakage while toggling MDSS GDSC */
if (dsi_pll_res->pll_1_base)
pll_20nm_config_powerdown(dsi_pll_res->pll_1_base);
if (rc) {
mdss_pll_resource_enable(dsi_pll_res, false);
pr_err("DSI PLL failed to lock\n");
}
dsi_pll_res->pll_on = true;
return rc;
}
static void dsi_pll_disable(struct clk *c)
{
struct dsi_pll_vco_clk *vco = to_vco_clk(c);
struct mdss_pll_resources *dsi_pll_res = vco->priv;
if (!dsi_pll_res->pll_on &&
mdss_pll_resource_enable(dsi_pll_res, true)) {
pr_err("Failed to enable mdss dsi pll resources\n");
return;
}
dsi_pll_res->handoff_resources = false;
__dsi_pll_disable(dsi_pll_res->pll_base);
/* Disable PLL1 to avoid current leakage while toggling MDSS GDSC */
if (dsi_pll_res->pll_1_base)
pll_20nm_config_powerdown(dsi_pll_res->pll_1_base);
pll_20nm_config_powerdown(dsi_pll_res->pll_base);
mdss_pll_resource_enable(dsi_pll_res, false);
dsi_pll_res->pll_on = false;
pr_debug("DSI PLL Disabled\n");
}
static void pll_20nm_config_common_block_1(void __iomem *pll_base)
{
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_VCOTAIL_EN, 0x82);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_BIAS_EN_CLKBUFLR_EN, 0x2a);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_BIAS_EN_CLKBUFLR_EN, 0x2b);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_RESETSM_CNTRL3, 0x02);
}
static void pll_20nm_config_common_block_2(void __iomem *pll_base)
{
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_SYS_CLK_CTRL, 0x40);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_IE_TRIM, 0x0F);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_IP_TRIM, 0x0F);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_PHSEL_CONTROL, 0x08);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_IPTAT_TRIM_VCCA_TX_SEL, 0x0E);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_BKG_KVCO_CAL_EN, 0x08);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_SYSCLK_EN_SEL_TXBAND, 0x4A);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_DIV_REF1, 0x00);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_DIV_REF2, 0x01);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_CNTRL, 0x07);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_KVCO_CAL_CNTRL, 0x1f);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_KVCO_COUNT1, 0x8A);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_VREF_CFG3, 0x10);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_SSC_EN_CENTER, 0x00);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_FAUX_EN, 0x0C);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_RXTXEPCLK_EN, 0x0a);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_LOW_POWER_RO_CONTROL, 0x0f);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_CMN_MODE, 0x00);
}
static void pll_20nm_config_loop_bw(void __iomem *pll_base)
{
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_IP_SETI, 0x03);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_CP_SETI, 0x3F);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_IP_SETP, 0x03);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_CP_SETP, 0x1F);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_CRCTRL, 0x77);
}
static void pll_20nm_vco_rate_calc(struct mdss_pll_vco_calc *vco_calc,
s64 vco_clk_rate, s64 ref_clk_rate)
{
s64 multiplier = (1 << 20);
s64 duration = 1024, pll_comp_val;
s64 dec_start_multiple, dec_start;
s32 div_frac_start;
s64 dec_start1, dec_start2;
s32 div_frac_start1, div_frac_start2, div_frac_start3;
s64 pll_plllock_cmp1, pll_plllock_cmp2, pll_plllock_cmp3;
memset(vco_calc, 0, sizeof(*vco_calc));
pr_debug("vco_clk_rate=%lld ref_clk_rate=%lld\n", vco_clk_rate,
ref_clk_rate);
dec_start_multiple = div_s64(vco_clk_rate * multiplier,
2 * ref_clk_rate);
div_s64_rem(dec_start_multiple,
multiplier, &div_frac_start);
dec_start = div_s64(dec_start_multiple, multiplier);
dec_start1 = (dec_start & 0x7f) | BIT(7);
dec_start2 = ((dec_start & 0x80) >> 7) | BIT(1);
div_frac_start1 = (div_frac_start & 0x7f) | BIT(7);
div_frac_start2 = ((div_frac_start >> 7) & 0x7f) | BIT(7);
div_frac_start3 = ((div_frac_start >> 14) & 0x3f) | BIT(6);
pll_comp_val = (div_s64(dec_start_multiple * 2 * duration,
10 * multiplier)) - 1;
pll_plllock_cmp1 = pll_comp_val & 0xff;
pll_plllock_cmp2 = (pll_comp_val >> 8) & 0xff;
pll_plllock_cmp3 = (pll_comp_val >> 16) & 0xff;
pr_debug("dec_start_multiple = 0x%llx\n", dec_start_multiple);
pr_debug("dec_start = 0x%llx, div_frac_start = 0x%x\n",
dec_start, div_frac_start);
pr_debug("dec_start1 = 0x%llx, dec_start2 = 0x%llx\n",
dec_start1, dec_start2);
pr_debug("div_frac_start1 = 0x%x, div_frac_start2 = 0x%x\n",
div_frac_start1, div_frac_start2);
pr_debug("div_frac_start3 = 0x%x\n", div_frac_start3);
pr_debug("pll_comp_val = 0x%llx\n", pll_comp_val);
pr_debug("pll_plllock_cmp1 = 0x%llx, pll_plllock_cmp2 =%llx\n",
pll_plllock_cmp1, pll_plllock_cmp2);
pr_debug("pll_plllock_cmp3 = 0x%llx\n", pll_plllock_cmp3);
/* Assign to vco struct */
vco_calc->div_frac_start1 = div_frac_start1;
vco_calc->div_frac_start2 = div_frac_start2;
vco_calc->div_frac_start3 = div_frac_start3;
vco_calc->dec_start1 = dec_start1;
vco_calc->dec_start2 = dec_start2;
vco_calc->pll_plllock_cmp1 = pll_plllock_cmp1;
vco_calc->pll_plllock_cmp2 = pll_plllock_cmp2;
vco_calc->pll_plllock_cmp3 = pll_plllock_cmp3;
}
static void pll_20nm_config_vco_rate(void __iomem *pll_base,
struct mdss_pll_vco_calc *vco_calc)
{
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_DIV_FRAC_START1,
vco_calc->div_frac_start1);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_DIV_FRAC_START2,
vco_calc->div_frac_start2);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_DIV_FRAC_START3,
vco_calc->div_frac_start3);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_DEC_START1,
vco_calc->dec_start1);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_DEC_START2,
vco_calc->dec_start2);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLLLOCK_CMP1,
vco_calc->pll_plllock_cmp1);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLLLOCK_CMP2,
vco_calc->pll_plllock_cmp2);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLLLOCK_CMP3,
vco_calc->pll_plllock_cmp3);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLLLOCK_CMP_EN, 0x01);
}
int pll_20nm_vco_set_rate(struct dsi_pll_vco_clk *vco, unsigned long rate)
{
struct mdss_pll_resources *dsi_pll_res = vco->priv;
dsi_pll_res->vco_current_rate = rate;
dsi_pll_res->vco_ref_clk_rate = vco->ref_clk_rate;
return 0;
}
int shadow_pll_20nm_vco_set_rate(struct dsi_pll_vco_clk *vco,
unsigned long rate)
{
struct mdss_pll_resources *dsi_pll_res = vco->priv;
struct mdss_pll_vco_calc vco_calc;
s64 vco_clk_rate = rate;
u32 rem;
if (!dsi_pll_res->resource_enable) {
pr_err("PLL resources disabled. Dynamic fps invalid\n");
return -EINVAL;
}
pr_debug("req vco set rate: %lld\n", vco_clk_rate);
pll_20nm_override_trim_codes(dsi_pll_res);
/* div fraction, start and comp calculations */
pll_20nm_vco_rate_calc(&vco_calc, vco_clk_rate,
dsi_pll_res->vco_ref_clk_rate);
MDSS_DYN_PLL_REG_W(dsi_pll_res->dyn_pll_base,
MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL0,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL,
MMSS_DSI_PHY_PLL_PLLLOCK_CMP_EN,
0xB1, 0);
MDSS_DYN_PLL_REG_W(dsi_pll_res->dyn_pll_base,
MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL1,
MMSS_DSI_PHY_PLL_PLLLOCK_CMP1,
MMSS_DSI_PHY_PLL_PLLLOCK_CMP2,
vco_calc.pll_plllock_cmp1, vco_calc.pll_plllock_cmp2);
MDSS_DYN_PLL_REG_W(dsi_pll_res->dyn_pll_base,
MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL2,
MMSS_DSI_PHY_PLL_PLLLOCK_CMP3,
MMSS_DSI_PHY_PLL_DEC_START1,
vco_calc.pll_plllock_cmp3, vco_calc.dec_start1);
MDSS_DYN_PLL_REG_W(dsi_pll_res->dyn_pll_base,
MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL3,
MMSS_DSI_PHY_PLL_DEC_START2,
MMSS_DSI_PHY_PLL_DIV_FRAC_START1,
vco_calc.dec_start2, vco_calc.div_frac_start1);
MDSS_DYN_PLL_REG_W(dsi_pll_res->dyn_pll_base,
MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL4,
MMSS_DSI_PHY_PLL_DIV_FRAC_START2,
MMSS_DSI_PHY_PLL_DIV_FRAC_START3,
vco_calc.div_frac_start2, vco_calc.div_frac_start3);
/* Method 2 - Auto PLL calibration */
MDSS_DYN_PLL_REG_W(dsi_pll_res->dyn_pll_base,
MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL7,
MMSS_DSI_PHY_PLL_PLL_VCO_TUNE,
MMSS_DSI_PHY_PLL_PLLLOCK_CMP_EN,
0, 0x0D);
MDSS_DYN_PLL_REG_W(dsi_pll_res->dyn_pll_base,
MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL8,
MMSS_DSI_PHY_PLL_POST_DIVIDER_CONTROL,
MMSS_DSI_PHY_PLL_RESETSM_CNTRL3,
0xF0, 0x07);
/*
* RESETSM_CTRL3 has to be set for 12 times (6 reg writes),
* Each register setting write 2 times, running in loop for 5
* times (5 reg writes) and other two iterations are taken
* care (one above and other in shadow_bypass
*/
for (rem = 0; rem < 5; rem++) {
MDSS_DYN_PLL_REG_W(dsi_pll_res->dyn_pll_base,
MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL9 + (4 * rem),
MMSS_DSI_PHY_PLL_RESETSM_CNTRL3,
MMSS_DSI_PHY_PLL_RESETSM_CNTRL3,
0x07, 0x07);
}
MDSS_DYN_PLL_REG_W(dsi_pll_res->dyn_pll_base,
MMSS_DSI_DYNAMIC_REFRESH_PLL_CTRL15,
MMSS_DSI_PHY_PLL_RESETSM_CNTRL3,
MMSS_DSI_PHY_PLL_RESETSM_CNTRL3,
0x03, 0x03);
/* memory barrier */
wmb();
return 0;
}
unsigned long pll_20nm_vco_get_rate(struct clk *c)
{
u64 vco_rate, multiplier = (1 << 20);
s32 div_frac_start;
u32 dec_start;
struct dsi_pll_vco_clk *vco = to_vco_clk(c);
u64 ref_clk = vco->ref_clk_rate;
int rc;
struct mdss_pll_resources *dsi_pll_res = vco->priv;
if (is_gdsc_disabled(dsi_pll_res))
return 0;
rc = mdss_pll_resource_enable(dsi_pll_res, true);
if (rc) {
pr_err("Failed to enable mdss dsi pll resources\n");
return rc;
}
dec_start = (MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_DEC_START2) & BIT(0)) << 7;
dec_start |= (MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_DEC_START1) & 0x7f);
pr_debug("dec_start = 0x%x\n", dec_start);
div_frac_start = (MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_DIV_FRAC_START3) & 0x3f) << 14;
div_frac_start |= (MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_DIV_FRAC_START2) & 0x7f) << 7;
div_frac_start |= MDSS_PLL_REG_R(dsi_pll_res->pll_base,
MMSS_DSI_PHY_PLL_DIV_FRAC_START1) & 0x7f;
pr_debug("div_frac_start = 0x%x\n", div_frac_start);
vco_rate = ref_clk * 2 * dec_start;
vco_rate += ((ref_clk * 2 * div_frac_start) / multiplier);
pr_debug("returning vco rate = %lu\n", (unsigned long)vco_rate);
mdss_pll_resource_enable(dsi_pll_res, false);
return (unsigned long)vco_rate;
}
long pll_20nm_vco_round_rate(struct clk *c, unsigned long rate)
{
unsigned long rrate = rate;
struct dsi_pll_vco_clk *vco = to_vco_clk(c);
if (rate < vco->min_rate)
rrate = vco->min_rate;
if (rate > vco->max_rate)
rrate = vco->max_rate;
return rrate;
}
enum handoff pll_20nm_vco_handoff(struct clk *c)
{
int rc;
enum handoff ret = HANDOFF_DISABLED_CLK;
struct dsi_pll_vco_clk *vco = to_vco_clk(c);
struct mdss_pll_resources *dsi_pll_res = vco->priv;
if (is_gdsc_disabled(dsi_pll_res))
return HANDOFF_DISABLED_CLK;
rc = mdss_pll_resource_enable(dsi_pll_res, true);
if (rc) {
pr_err("Failed to enable mdss dsi pll resources\n");
return ret;
}
if (pll_20nm_is_pll_locked(dsi_pll_res)) {
dsi_pll_res->handoff_resources = true;
dsi_pll_res->pll_on = true;
c->rate = pll_20nm_vco_get_rate(c);
ret = HANDOFF_ENABLED_CLK;
dsi_pll_res->vco_locking_rate = c->rate;
dsi_pll_res->is_init_locked = true;
pll_20nm_cache_trim_codes(dsi_pll_res);
pr_debug("handoff vco_locking_rate=%llu\n",
dsi_pll_res->vco_locking_rate);
} else {
mdss_pll_resource_enable(dsi_pll_res, false);
dsi_pll_res->vco_locking_rate = 0;
dsi_pll_res->is_init_locked = false;
}
return ret;
}
int pll_20nm_vco_prepare(struct clk *c)
{
int rc = 0;
struct dsi_pll_vco_clk *vco = to_vco_clk(c);
struct mdss_pll_resources *dsi_pll_res = vco->priv;
if (!dsi_pll_res) {
pr_err("Dsi pll resources are not available\n");
return -EINVAL;
}
if ((dsi_pll_res->vco_cached_rate != 0)
&& (dsi_pll_res->vco_cached_rate == c->rate)) {
rc = c->ops->set_rate(c, dsi_pll_res->vco_cached_rate);
if (rc) {
pr_err("vco_set_rate failed. rc=%d\n", rc);
goto error;
}
}
rc = dsi_pll_enable(c);
error:
return rc;
}
void pll_20nm_vco_unprepare(struct clk *c)
{
struct dsi_pll_vco_clk *vco = to_vco_clk(c);
struct mdss_pll_resources *dsi_pll_res = vco->priv;
if (!dsi_pll_res) {
pr_err("Dsi pll resources are not available\n");
return;
}
dsi_pll_res->vco_cached_rate = c->rate;
dsi_pll_disable(c);
}
static void pll_20nm_config_resetsm(void __iomem *pll_base)
{
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_RESETSM_CNTRL, 0x24);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_RESETSM_CNTRL2, 0x07);
}
static void pll_20nm_config_vco_start(void __iomem *pll_base)
{
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_VCOTAIL_EN, 0x03);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_RESETSM_CNTRL3, 0x02);
udelay(10);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_RESETSM_CNTRL3, 0x03);
}
static void pll_20nm_config_bypass_cal(void __iomem *pll_base)
{
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_RESETSM_CNTRL, 0xac);
MDSS_PLL_REG_W(pll_base, MMSS_DSI_PHY_PLL_PLL_BKG_KVCO_CAL_EN, 0x28);
}
static int pll_20nm_vco_relock(struct mdss_pll_resources *dsi_pll_res)
{
int rc = 0;
pll_20nm_override_trim_codes(dsi_pll_res);
pll_20nm_config_bypass_cal(dsi_pll_res->pll_base);
pll_20nm_config_vco_start(dsi_pll_res->pll_base);
if (!pll_20nm_is_pll_locked(dsi_pll_res)) {
pr_err("DSI PLL re-lock failed\n");
rc = -EINVAL;
}
return rc;
}
static int pll_20nm_vco_init_lock(struct mdss_pll_resources *dsi_pll_res)
{
int rc = 0;
pll_20nm_config_resetsm(dsi_pll_res->pll_base);
pll_20nm_config_vco_start(dsi_pll_res->pll_base);
if (!pll_20nm_is_pll_locked(dsi_pll_res)) {
pr_err("DSI PLL init lock failed\n");
rc = -EINVAL;
goto init_lock_err;
}
pll_20nm_cache_trim_codes(dsi_pll_res);
init_lock_err:
return rc;
}
int pll_20nm_vco_enable_seq(struct mdss_pll_resources *dsi_pll_res)
{
int rc = 0;
struct mdss_pll_vco_calc vco_calc;
if (!dsi_pll_res) {
pr_err("Invalid PLL resources\n");
return -EINVAL;
}
pll_20nm_config_common_block_1(dsi_pll_res->pll_1_base);
pll_20nm_config_common_block_1(dsi_pll_res->pll_base);
pll_20nm_config_common_block_2(dsi_pll_res->pll_base);
pll_20nm_config_loop_bw(dsi_pll_res->pll_base);
pll_20nm_vco_rate_calc(&vco_calc, dsi_pll_res->vco_current_rate,
dsi_pll_res->vco_ref_clk_rate);
pll_20nm_config_vco_rate(dsi_pll_res->pll_base, &vco_calc);
pr_debug("init lock=%d prev vco_rate=%llu, new vco_rate=%llu\n",
dsi_pll_res->is_init_locked, dsi_pll_res->vco_locking_rate,
dsi_pll_res->vco_current_rate);
/*
* Run auto-lock sequence if it is either bootup initial
* locking or when the vco rate is changed. Otherwise, just
* use stored codes and bypass caliberation.
*/
if (!dsi_pll_res->is_init_locked || (dsi_pll_res->vco_locking_rate !=
dsi_pll_res->vco_current_rate)) {
rc = pll_20nm_vco_init_lock(dsi_pll_res);
dsi_pll_res->is_init_locked = (rc) ? false : true;
} else {
rc = pll_20nm_vco_relock(dsi_pll_res);
}
dsi_pll_res->vco_locking_rate = (rc) ? 0 :
dsi_pll_res->vco_current_rate;
return rc;
}