blob: aff0b0d8195c5987f98f1e36cfa47507495b4145 [file] [log] [blame]
/*
* arch/arm/cpu/armv8/gxtvbb/hdmitx20/enc_clk_config.c
*
* Copyright (C) 2015 Amlogic, Inc. 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 as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* 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.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include <common.h>
#include <amlogic/enc_clk_config.h>
#include <asm/arch/io.h>
#include <asm/io.h>
#include "hw_enc_clk_config.h"
#include "mach_reg.h"
#include <amlogic/hdmi.h>
#define msleep(i) udelay(i*1000)
#define check_clk_config(para)\
if (para == -1)\
return;
#define check_div() \
if (div == -1)\
return ;\
switch (div) {\
case 1:\
div = 0; break;\
case 2:\
div = 1; break;\
case 4:\
div = 2; break;\
case 6:\
div = 3; break;\
case 12:\
div = 4; break;\
default:\
break;\
}
#define WAIT_FOR_PLL_LOCKED(reg) \
do { \
unsigned int st = 0, cnt = 10; \
while (cnt--) { \
msleep(5); \
st = !!(hd_read_reg(reg) & (1 << 31)); \
if (st) \
break; \
else { /* reset hpll */ \
hd_set_reg_bits(reg, 1, 28, 1); \
hd_set_reg_bits(reg, 0, 28, 1); \
} \
} \
if (cnt < 9) \
printk("pll[0x%x] reset %d times\n", reg, 9 - cnt);\
} while (0)
// viu_channel_sel: 1 or 2
// viu_type_sel: 0: 0=ENCL, 1=ENCI, 2=ENCP, 3=ENCT.
int set_viu_path(unsigned viu_channel_sel, enum viu_type viu_type_sel)
{
if ((viu_channel_sel > 2) || (viu_channel_sel == 0))
return -1;
if (viu_channel_sel == 1)
hd_set_reg_bits(P_VPU_VIU_VENC_MUX_CTRL, viu_type_sel, 0, 2);
else
//viu_channel_sel ==2
hd_set_reg_bits(P_VPU_VIU_VENC_MUX_CTRL, viu_type_sel, 2, 2);
return 0;
}
static void set_hdmitx_sys_clk(void)
{
hd_set_reg_bits(P_HHI_HDMI_CLK_CNTL, 0, 9, 3);
hd_set_reg_bits(P_HHI_HDMI_CLK_CNTL, 0, 0, 7);
hd_set_reg_bits(P_HHI_HDMI_CLK_CNTL, 1, 8, 1);
}
static void set_hpll_clk_out(unsigned clk)
{
check_clk_config(clk);
printk("config HPLL = %d\n", clk);
switch (clk) {
case 5940:
hd_write_reg(P_HHI_HDMI_PLL_CNTL, 0x5800027b);
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 0x4300, 0, 16); // div_frac
hd_write_reg(P_HHI_HDMI_PLL_CNTL3, 0x12dc5081);
hd_write_reg(P_HHI_HDMI_PLL_CNTL4, 0x801da72c);
hd_write_reg(P_HHI_HDMI_PLL_CNTL5, 0x71486980);
hd_write_reg(P_HHI_HDMI_PLL_CNTL6, 0x00002e55);
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL, 0x0, 28, 1);
WAIT_FOR_PLL_LOCKED(P_HHI_HDMI_PLL_CNTL);
printk("HPLL: 0x%lx\n", hd_read_reg(P_HHI_HDMI_PLL_CNTL));
break;
case 3712:
hd_write_reg(P_HHI_HDMI_PLL_CNTL, 0x5800024d);
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 0x4160, 0, 16);
hd_write_reg(P_HHI_HDMI_PLL_CNTL3, 0x0d5c5091);
hd_write_reg(P_HHI_HDMI_PLL_CNTL4, 0x801da72c);
hd_write_reg(P_HHI_HDMI_PLL_CNTL5, 0x71486980);
hd_write_reg(P_HHI_HDMI_PLL_CNTL6, 0x00002e55);
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL, 0x4, 28, 3);
WAIT_FOR_PLL_LOCKED(P_HHI_HDMI_PLL_CNTL);
printk("HPLL: 0x%lx\n", hd_read_reg(P_HHI_HDMI_PLL_CNTL));
break;
case 2970:
hd_write_reg(P_HHI_HDMI_PLL_CNTL, 0x5800023d);
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 0x4380, 0, 16); // div_frac
hd_write_reg(P_HHI_HDMI_PLL_CNTL3, 0x0d5c5091);
hd_write_reg(P_HHI_HDMI_PLL_CNTL4, 0x801da72c);
hd_write_reg(P_HHI_HDMI_PLL_CNTL5, 0x71486980);
hd_write_reg(P_HHI_HDMI_PLL_CNTL6, 0x00002e55);
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL, 0x4, 28, 3);
WAIT_FOR_PLL_LOCKED(P_HHI_HDMI_PLL_CNTL);
printk("HPLL: 0x%lx\n", hd_read_reg(P_HHI_HDMI_PLL_CNTL));
break;
case 4320:
hd_write_reg(P_HHI_HDMI_PLL_CNTL, 0x5800025a);
hd_write_reg(P_HHI_HDMI_PLL_CNTL2, 0x00000000);
hd_write_reg(P_HHI_HDMI_PLL_CNTL3, 0x0d5c5091);
hd_write_reg(P_HHI_HDMI_PLL_CNTL4, 0x801da72c);
hd_write_reg(P_HHI_HDMI_PLL_CNTL5, 0x71486980);
hd_write_reg(P_HHI_HDMI_PLL_CNTL6, 0x00002e55);
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL, 0x0, 28, 1);
WAIT_FOR_PLL_LOCKED(P_HHI_HDMI_PLL_CNTL);
printk("HPLL: 0x%lx\n", hd_read_reg(P_HHI_HDMI_PLL_CNTL));
break;
default:
printk("error hpll clk: %d\n", clk);
break;
}
printk("config HPLL done\n");
}
static void set_hpll_od1(unsigned div)
{
switch (div) {
case 1:
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 0, 16, 2);
break;
case 2:
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 1, 16, 2);
break;
case 4:
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 2, 16, 2);
break;
case 8:
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 3, 16, 2);
break;
default:
break;
}
}
static void set_hpll_od2(unsigned div)
{
switch (div) {
case 1:
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 0, 22, 2);
break;
case 2:
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 1, 22, 2);
break;
case 4:
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 2, 22, 2);
break;
case 8:
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 3, 22, 2);
break;
default:
break;
}
}
static void set_hpll_od3(unsigned div)
{
switch (div) {
case 1:
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 0, 18, 2);
break;
case 2:
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 1, 18, 2);
break;
case 4:
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 2, 18, 2);
break;
case 8:
hd_set_reg_bits(P_HHI_HDMI_PLL_CNTL2, 3, 18, 2);
break;
default:
break;
}
}
// --------------------------------------------------
// clocks_set_vid_clk_div
// --------------------------------------------------
// wire clk_final_en = control[19];
// wire clk_div1 = control[18];
// wire [1:0] clk_sel = control[17:16];
// wire set_preset = control[15];
// wire [14:0] shift_preset = control[14:0];
static void set_hpll_od3_clk_div(int div_sel)
{
int shift_val = 0;
int shift_sel = 0;
/* When div 6.25, need to reset vid_pll_div */
if (div_sel == CLK_UTIL_VID_PLL_DIV_6p25) {
msleep(1);
hd_set_reg_bits(P_RESET0_REGISTER, 1, 7, 1);
}
// Disable the output clock
hd_set_reg_bits(P_HHI_VID_PLL_CLK_DIV, 0, 19, 1);
hd_set_reg_bits(P_HHI_VID_PLL_CLK_DIV, 0, 15, 1);
switch (div_sel) {
case CLK_UTIL_VID_PLL_DIV_1: shift_val = 0xFFFF; shift_sel = 0; break;
case CLK_UTIL_VID_PLL_DIV_2: shift_val = 0x0aaa; shift_sel = 0; break;
case CLK_UTIL_VID_PLL_DIV_3: shift_val = 0x0db6; shift_sel = 0; break;
case CLK_UTIL_VID_PLL_DIV_3p5: shift_val = 0x36cc; shift_sel = 1; break;
case CLK_UTIL_VID_PLL_DIV_3p75: shift_val = 0x6666; shift_sel = 2; break;
case CLK_UTIL_VID_PLL_DIV_4: shift_val = 0x0ccc; shift_sel = 0; break;
case CLK_UTIL_VID_PLL_DIV_5: shift_val = 0x739c; shift_sel = 2; break;
case CLK_UTIL_VID_PLL_DIV_6: shift_val = 0x0e38; shift_sel = 0; break;
case CLK_UTIL_VID_PLL_DIV_6p25: shift_val = 0x0000; shift_sel = 3; break;
case CLK_UTIL_VID_PLL_DIV_7: shift_val = 0x3c78; shift_sel = 1; break;
case CLK_UTIL_VID_PLL_DIV_7p5: shift_val = 0x78f0; shift_sel = 2; break;
case CLK_UTIL_VID_PLL_DIV_12: shift_val = 0x0fc0; shift_sel = 0; break;
case CLK_UTIL_VID_PLL_DIV_14: shift_val = 0x3f80; shift_sel = 1; break;
case CLK_UTIL_VID_PLL_DIV_15: shift_val = 0x7f80; shift_sel = 2; break;
case CLK_UTIL_VID_PLL_DIV_2p5: shift_val = 0x5294; shift_sel = 2; break;
default:
printk("Error: clocks_set_vid_clk_div: Invalid parameter\n");
break;
}
if (shift_val == 0xffff ) { // if divide by 1
hd_set_reg_bits(P_HHI_VID_PLL_CLK_DIV, 1, 18, 1);
} else {
hd_set_reg_bits(P_HHI_VID_PLL_CLK_DIV, 0, 18, 1);
hd_set_reg_bits(P_HHI_VID_PLL_CLK_DIV, 0, 16, 2);
hd_set_reg_bits(P_HHI_VID_PLL_CLK_DIV, 0, 15, 1);
hd_set_reg_bits(P_HHI_VID_PLL_CLK_DIV, 0, 0, 14);
hd_set_reg_bits(P_HHI_VID_PLL_CLK_DIV, shift_sel, 16, 2);
hd_set_reg_bits(P_HHI_VID_PLL_CLK_DIV, 1, 15, 1);
hd_set_reg_bits(P_HHI_VID_PLL_CLK_DIV, shift_val, 0, 14);
hd_set_reg_bits(P_HHI_VID_PLL_CLK_DIV, 0, 15, 1);
}
// Enable the final output clock
hd_set_reg_bits(P_HHI_VID_PLL_CLK_DIV, 1, 19, 1);
}
static void set_vid_clk_div(unsigned div)
{
check_clk_config(div);
if (div == 0)
div = 1;
hd_set_reg_bits(P_HHI_VID_CLK_CNTL, 0, 16, 3); // select vid_pll_clk
hd_set_reg_bits(P_HHI_VID_CLK_DIV, div-1, 0, 8);
hd_set_reg_bits(P_HHI_VID_CLK_CNTL, 7, 0, 3);
}
static void set_hdmi_tx_pixel_div(unsigned div)
{
check_div();
hd_set_reg_bits(P_HHI_HDMI_CLK_CNTL, div, 16, 4);
hd_set_reg_bits(P_HHI_VID_CLK_CNTL2, 1, 5, 1); //enable gate
}
static void set_encp_div(unsigned div)
{
check_div();
hd_set_reg_bits(P_HHI_VID_CLK_DIV, div, 24, 4);
hd_set_reg_bits(P_HHI_VID_CLK_CNTL2, 1, 2, 1); //enable gate
hd_set_reg_bits(P_HHI_VID_CLK_CNTL, 1, 19, 1);
}
static void set_enci_div(unsigned div)
{
check_div();
hd_set_reg_bits(P_HHI_VID_CLK_DIV, div, 28, 4);
hd_set_reg_bits(P_HHI_VID_CLK_CNTL2, 1, 0, 1); //enable gate
hd_set_reg_bits(P_HHI_VID_CLK_CNTL, 1, 19, 1);
}
// mode viu_path viu_type hpll_clk_out od1 od2 od3
// vid_pll_div vid_clk_div hdmi_tx_pixel_div encp_div enci_div
static struct hw_enc_clk_val_group setting_enc_clk_val[] = {
{{HDMI_720x480i60_16x9, HDMI_720x576i50_16x9, GROUP_END},
1, VIU_ENCI, 4320, 4, 4, 1, CLK_UTIL_VID_PLL_DIV_5, 1, 2, -1, 2},
{{HDMI_720x576p50_16x9, HDMI_720x480p60_16x9, GROUP_END},
1, VIU_ENCP, 4320, 4, 4, 1, CLK_UTIL_VID_PLL_DIV_5, 1, 2, 1, -1},
{{HDMI_1280x720p50_16x9, HDMI_1280x720p60_16x9, GROUP_END},
1, VIU_ENCP, 2970, 4, 1, 1, CLK_UTIL_VID_PLL_DIV_5, 1, 2, 1, -1},
{{HDMI_1920x1080i60_16x9, HDMI_1920x1080i50_16x9, GROUP_END},
1, VIU_ENCP, 2970, 4, 1, 1, CLK_UTIL_VID_PLL_DIV_5, 1, 2, 1, -1},
{{HDMI_1920x1080p60_16x9, HDMI_1920x1080p50_16x9, GROUP_END},
1, VIU_ENCP, 2970, 1, 2, 2, CLK_UTIL_VID_PLL_DIV_5, 1, 1, 1, -1},
{{HDMI_1920x1080p30_16x9, HDMI_1920x1080p24_16x9, HDMI_1920x1080p25_16x9, GROUP_END},
1, VIU_ENCP, 2970, 2, 2, 2, CLK_UTIL_VID_PLL_DIV_5, 1, 1, 1, -1},
{{HDMI_3840x2160p30_16x9, HDMI_3840x2160p25_16x9, HDMI_3840x2160p24_16x9,
HDMI_4096x2160p24_256x135, HDMI_4096x2160p25_256x135, HDMI_4096x2160p30_256x135, GROUP_END},
1, VIU_ENCP, 2970, 1, 1, 1, CLK_UTIL_VID_PLL_DIV_5, 2, 1, 1, -1},
{{HDMI_3840x2160p60_16x9, HDMI_3840x2160p50_16x9, HDMI_4096x2160p60_256x135,
HDMI_4096x2160p50_256x135, GROUP_END},
1, VIU_ENCP, 5940, 1, 1, 2, CLK_UTIL_VID_PLL_DIV_5, 1, 1, 1, -1},
};
/* mode hpll_clk_out od1 od2(PHY) od3
* vid_pll_div vid_clk_div hdmi_tx_pixel_div encp_div enci_div
*/
static struct hw_enc_clk_val_group setting_enc_clk_val_30[] = {
{{HDMI_1920x1080p60_16x9, HDMI_1920x1080p50_16x9,
GROUP_END},
1, VIU_ENCP, 3712, 1, 2, 2, CLK_UTIL_VID_PLL_DIV_6p25, 1, 1, 1, -1},
{{HDMI_4096x2160p60_256x135, HDMI_4096x2160p50_256x135, HDMI_3840x2160p60_16x9, HDMI_3840x2160p50_16x9,
GROUP_END},
1, VIU_ENCP, 3712, 1, 1, 1, CLK_UTIL_VID_PLL_DIV_6p25, 1, 2, 1, -1},
{{HDMI_3840x2160p24_16x9, HDMI_3840x2160p24_64x27, HDMI_3840x2160p25_16x9, HDMI_3840x2160p25_64x27,
HDMI_3840x2160p30_16x9, HDMI_3840x2160p30_64x27,
GROUP_END},
1, VIU_ENCP, 3712, 1, 1, 1, CLK_UTIL_VID_PLL_DIV_6p25, 1, 2, 2, -1},
};
void set_hdmitx_clk(enum hdmi_vic vic)
{
int i = 0;
int j = 0;
struct hw_enc_clk_val_group *p_enc =NULL;
p_enc = &setting_enc_clk_val[0];
for (j = 0; j < ARRAY_SIZE(setting_enc_clk_val); j++) {
for (i = 0; ((i < GROUP_MAX) && (p_enc[j].group[i] != GROUP_END)); i ++) {
if (vic == p_enc[j].group[i])
goto next;
}
}
if (j == ARRAY_SIZE(setting_enc_clk_val)) {
printf("Not find VIC = %d for hpll setting\n", vic);
return;
}
next:
set_viu_path(p_enc[j].viu_path, p_enc[j].viu_type);
set_hdmitx_sys_clk();
set_hpll_clk_out(p_enc[j].hpll_clk_out);
set_hpll_od1(p_enc[j].od1);
set_hpll_od2(p_enc[j].od2);
set_hpll_od3(p_enc[j].od3);
set_hpll_od3_clk_div(p_enc[j].vid_pll_div);
printk("j = %d vid_clk_div = %d\n", j, p_enc[j].vid_clk_div);
set_vid_clk_div(p_enc[j].vid_clk_div);
set_hdmi_tx_pixel_div(p_enc[j].hdmi_tx_pixel_div);
set_encp_div(p_enc[j].encp_div);
set_enci_div(p_enc[j].enci_div);
}
void hdmitx_set_clk_30b(enum hdmi_vic vic)
{
int i = 0;
int j = 0;
struct hw_enc_clk_val_group *p_enc = NULL;
p_enc = &setting_enc_clk_val_30[0];
for (j = 0; j < sizeof(setting_enc_clk_val_30)
/ sizeof(struct hw_enc_clk_val_group); j++) {
for (i = 0; ((i < GROUP_MAX) && (p_enc[j].group[i]
!= GROUP_END)); i++) {
if (vic == p_enc[j].group[i])
goto next;
}
}
if (j == sizeof(setting_enc_clk_val_30) /
sizeof(struct hw_enc_clk_val_group)) {
printk("Not find VIC = %d for hpll setting\n", vic);
return;
}
next:
set_hdmitx_sys_clk();
set_hpll_clk_out(p_enc[j].hpll_clk_out);
set_hpll_od1(p_enc[j].od1);
set_hpll_od2(p_enc[j].od2);
set_hpll_od3(p_enc[j].od3);
set_hpll_od3_clk_div(p_enc[j].vid_pll_div);
set_vid_clk_div(p_enc[j].vid_clk_div);
set_hdmi_tx_pixel_div(p_enc[j].hdmi_tx_pixel_div);
set_encp_div(p_enc[j].encp_div);
set_enci_div(p_enc[j].enci_div);
}