mirror of
https://github.com/hardkernel/linux.git
synced 2026-06-08 03:40:35 +09:00
clk: sm1: fix cpu clock hangup [1/1]
PD#SWPL-10033 Problem: it will hangup when play video Solution: using the origin clk-mux ops to register cpu clock Verify: test passed on sm1 Change-Id: I1a21eb28d8af3d68a70c89c37fff857fb416c9a8 Signed-off-by: Jian Hu <jian.hu@amlogic.com>
This commit is contained in:
@@ -21,6 +21,7 @@
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/delay.h>
|
||||
#include <dt-bindings/clock/amlogic,g12a-clkc.h>
|
||||
|
||||
#include <linux/amlogic/cpu_version.h>
|
||||
@@ -380,7 +381,7 @@ static struct meson_clk_mpll g12a_mpll3 = {
|
||||
};
|
||||
|
||||
static u32 mux_table_cpu_p[] = { 0, 1, 2 };
|
||||
static u32 mux_table_cpu_px[] = { 0, 1 };
|
||||
/*static u32 mux_table_cpu_px[] = { 0, 1 };*/
|
||||
static struct meson_cpu_mux_divider g12a_cpu_fclk_p = {
|
||||
.reg = (void *)HHI_SYS_CPU_CLK_CNTL0,
|
||||
.cpu_fclk_p00 = {
|
||||
@@ -430,42 +431,74 @@ static struct meson_cpu_mux_divider g12a_cpu_fclk_p = {
|
||||
},
|
||||
};
|
||||
|
||||
static struct meson_clk_cpu g12a_cpu_clk = {
|
||||
.reg_off = HHI_SYS_CPU_CLK_CNTL0,
|
||||
.clk_nb.notifier_call = meson_clk_cpu_notifier_cb,
|
||||
.mux.reg = (void *)HHI_SYS_CPU_CLK_CNTL0,
|
||||
.mux.shift = 11,
|
||||
.mux.mask = 0x1,
|
||||
.mux.lock = &clk_lock,
|
||||
.mux.table = mux_table_cpu_px,
|
||||
.mux.hw.init = &(struct clk_init_data){
|
||||
.name = "cpu_clk",
|
||||
.ops = &meson_clk_cpu_ops,
|
||||
.parent_names = (const char *[]){ "cpu_fixedpll_p", "sys_pll"},
|
||||
.num_parents = 2,
|
||||
.flags = CLK_GET_RATE_NOCACHE,
|
||||
},
|
||||
/*
|
||||
*static struct meson_clk_cpu g12a_cpu_clk = {
|
||||
* .reg_off = HHI_SYS_CPU_CLK_CNTL0,
|
||||
* .clk_nb.notifier_call = meson_clk_cpu_notifier_cb,
|
||||
* .mux.reg = (void *)HHI_SYS_CPU_CLK_CNTL0,
|
||||
* .mux.shift = 11,
|
||||
* .mux.mask = 0x1,
|
||||
* .mux.lock = &clk_lock,
|
||||
* .mux.table = mux_table_cpu_px,
|
||||
* .mux.hw.init = &(struct clk_init_data){
|
||||
* .name = "cpu_clk",
|
||||
* .ops = &meson_clk_cpu_ops,
|
||||
* .parent_names = (const char *[]){ "cpu_fixedpll_p", "sys_pll"},
|
||||
* .num_parents = 2,
|
||||
* .flags = CLK_GET_RATE_NOCACHE,
|
||||
* },
|
||||
*};
|
||||
*/
|
||||
|
||||
/*
|
||||
* static struct meson_clk_cpu g12b_cpu_clk1 = {
|
||||
* .reg_off = HHI_SYS_CPU_CLK_CNTL0,
|
||||
* .clk_nb.notifier_call = meson_clk_cpu_notifier_cb,
|
||||
* .mux.reg = (void *)HHI_SYS_CPU_CLK_CNTL0,
|
||||
* .mux.shift = 11,
|
||||
* .mux.mask = 0x1,
|
||||
* .mux.lock = &clk_lock,
|
||||
* .mux.table = mux_table_cpu_px,
|
||||
* .mux.hw.init = &(struct clk_init_data){
|
||||
* .name = "cpu_clk",
|
||||
* .ops = &meson_clk_cpu_ops,
|
||||
* .parent_names = (const char *[]){ "cpu_fixedpll_p", "sys1_pll"},
|
||||
* .num_parents = 2,
|
||||
* .flags = CLK_GET_RATE_NOCACHE,
|
||||
* },
|
||||
*};
|
||||
*/
|
||||
|
||||
static struct clk_mux g12a_cpu_clk = {
|
||||
.reg = (void *)HHI_SYS_CPU_CLK_CNTL0,
|
||||
.mask = 0x1,
|
||||
.shift = 11,
|
||||
.lock = &clk_lock,
|
||||
.hw.init = &(struct clk_init_data){
|
||||
.name = "cpu_clk",
|
||||
.ops = &clk_mux_ops,
|
||||
.parent_names = (const char *[]){ "cpu_fixedpll_p",
|
||||
"sys_pll" },
|
||||
.num_parents = 2,
|
||||
.flags = CLK_GET_RATE_NOCACHE,
|
||||
},
|
||||
};
|
||||
|
||||
static struct meson_clk_cpu g12b_cpu_clk1 = {
|
||||
.reg_off = HHI_SYS_CPU_CLK_CNTL0,
|
||||
.clk_nb.notifier_call = meson_clk_cpu_notifier_cb,
|
||||
.mux.reg = (void *)HHI_SYS_CPU_CLK_CNTL0,
|
||||
.mux.shift = 11,
|
||||
.mux.mask = 0x1,
|
||||
.mux.lock = &clk_lock,
|
||||
.mux.table = mux_table_cpu_px,
|
||||
.mux.hw.init = &(struct clk_init_data){
|
||||
.name = "cpu_clk",
|
||||
.ops = &meson_clk_cpu_ops,
|
||||
.parent_names = (const char *[]){ "cpu_fixedpll_p", "sys1_pll"},
|
||||
.num_parents = 2,
|
||||
.flags = CLK_GET_RATE_NOCACHE,
|
||||
},
|
||||
static struct clk_mux g12b_cpu_clk1 = {
|
||||
.reg = (void *)HHI_SYS_CPU_CLK_CNTL0,
|
||||
.mask = 0x1,
|
||||
.shift = 11,
|
||||
.lock = &clk_lock,
|
||||
.hw.init = &(struct clk_init_data){
|
||||
.name = "cpu_clk",
|
||||
.ops = &clk_mux_ops,
|
||||
.parent_names = (const char *[]){ "cpu_fixedpll_p",
|
||||
"sys1_pll" },
|
||||
.num_parents = 2,
|
||||
.flags = CLK_GET_RATE_NOCACHE,
|
||||
},
|
||||
};
|
||||
|
||||
|
||||
|
||||
static u32 mux_table_clk81[] = { 6, 5, 7 };
|
||||
|
||||
static struct clk_mux g12a_mpeg_clk_sel = {
|
||||
@@ -795,7 +828,7 @@ static struct clk_hw *g12a_clk_hws[] = {
|
||||
[CLKID_EFUSE] = &g12a_efuse.hw,
|
||||
|
||||
[CLKID_CPU_FCLK_P] = &g12a_cpu_fclk_p.hw,
|
||||
[CLKID_CPU_CLK] = &g12a_cpu_clk.mux.hw,
|
||||
[CLKID_CPU_CLK] = &g12a_cpu_clk.hw,
|
||||
|
||||
[CLKID_PCIE_PLL] = &g12a_pcie_pll.hw,
|
||||
[CLKID_24M] = &g12a_24m.hw,
|
||||
@@ -894,12 +927,84 @@ static struct clk_gate *g12a_clk_gates[] = {
|
||||
&g12a_gen_clk,
|
||||
};
|
||||
|
||||
static int g12a_cpu_clk_notifier_cb(struct notifier_block *nb,
|
||||
unsigned long event, void *data)
|
||||
{
|
||||
struct clk_hw **hws = g12a_clk_hws;
|
||||
struct clk_hw *cpu_clk_hw, *parent_clk_hw;
|
||||
struct clk *cpu_clk, *parent_clk;
|
||||
int ret;
|
||||
|
||||
switch (event) {
|
||||
case PRE_RATE_CHANGE:
|
||||
parent_clk_hw = hws[CLKID_CPU_FCLK_P];
|
||||
break;
|
||||
case POST_RATE_CHANGE:
|
||||
parent_clk_hw = &g12a_sys_pll.hw;
|
||||
break;
|
||||
default:
|
||||
return NOTIFY_DONE;
|
||||
}
|
||||
|
||||
cpu_clk_hw = hws[CLKID_CPU_CLK];
|
||||
cpu_clk = __clk_lookup(clk_hw_get_name(cpu_clk_hw));
|
||||
parent_clk = __clk_lookup(clk_hw_get_name(parent_clk_hw));
|
||||
|
||||
ret = clk_set_parent(cpu_clk, parent_clk);
|
||||
if (ret)
|
||||
return notifier_from_errno(ret);
|
||||
|
||||
udelay(80);
|
||||
|
||||
return NOTIFY_OK;
|
||||
}
|
||||
|
||||
static struct notifier_block g12a_cpu_nb_data = {
|
||||
.notifier_call = g12a_cpu_clk_notifier_cb,
|
||||
};
|
||||
|
||||
static int g12b_cpu_clk_notifier_cb(struct notifier_block *nb,
|
||||
unsigned long event, void *data)
|
||||
{
|
||||
struct clk_hw **hws = g12a_clk_hws;
|
||||
struct clk_hw *cpu_clk_hw, *parent_clk_hw;
|
||||
struct clk *cpu_clk, *parent_clk;
|
||||
int ret;
|
||||
|
||||
switch (event) {
|
||||
case PRE_RATE_CHANGE:
|
||||
parent_clk_hw = hws[CLKID_CPU_FCLK_P];
|
||||
break;
|
||||
case POST_RATE_CHANGE:
|
||||
parent_clk_hw = &g12b_sys1_pll.hw;
|
||||
break;
|
||||
default:
|
||||
return NOTIFY_DONE;
|
||||
}
|
||||
|
||||
cpu_clk_hw = hws[CLKID_CPU_CLK];
|
||||
cpu_clk = __clk_lookup(clk_hw_get_name(cpu_clk_hw));
|
||||
parent_clk = __clk_lookup(clk_hw_get_name(parent_clk_hw));
|
||||
|
||||
ret = clk_set_parent(cpu_clk, parent_clk);
|
||||
if (ret)
|
||||
return notifier_from_errno(ret);
|
||||
|
||||
udelay(80);
|
||||
|
||||
return NOTIFY_OK;
|
||||
}
|
||||
|
||||
static struct notifier_block g12b_cpu_nb_data = {
|
||||
.notifier_call = g12b_cpu_clk_notifier_cb,
|
||||
};
|
||||
|
||||
static void __init g12a_clkc_init(struct device_node *np)
|
||||
{
|
||||
int ret = 0, clkid, i;
|
||||
struct clk_hw *parent_hw;
|
||||
struct clk *parent_clk;
|
||||
|
||||
/*struct clk_hw *parent_hw;
|
||||
*struct clk *parent_clk;
|
||||
*/
|
||||
meson_cpu_version_init();
|
||||
|
||||
/* Generic clocks and PLLs */
|
||||
@@ -925,15 +1030,12 @@ static void __init g12a_clkc_init(struct device_node *np)
|
||||
|
||||
/* Populate the base address for CPU clk */
|
||||
if (is_meson_g12b_cpu()) {
|
||||
g12a_clk_hws[CLKID_CPU_CLK] = &g12b_cpu_clk1.mux.hw;
|
||||
g12b_cpu_clk1.base = clk_base;
|
||||
g12b_cpu_clk1.mux.reg = clk_base
|
||||
+ (unsigned long)g12b_cpu_clk1.mux.reg;
|
||||
} else {
|
||||
g12a_cpu_clk.base = clk_base;
|
||||
g12a_cpu_clk.mux.reg = clk_base
|
||||
+ (unsigned long)g12a_cpu_clk.mux.reg;
|
||||
}
|
||||
g12a_clk_hws[CLKID_CPU_CLK] = &g12b_cpu_clk1.hw;
|
||||
g12b_cpu_clk1.reg = clk_base
|
||||
+ (unsigned long)g12b_cpu_clk1.reg;
|
||||
} else
|
||||
g12a_cpu_clk.reg = clk_base
|
||||
+ (unsigned long)g12a_cpu_clk.reg;
|
||||
|
||||
g12a_cpu_fclk_p.reg = clk_base
|
||||
+ (unsigned long)g12a_cpu_fclk_p.reg;
|
||||
@@ -1013,13 +1115,21 @@ static void __init g12a_clkc_init(struct device_node *np)
|
||||
* feature before that time solves the problem :-)
|
||||
*/
|
||||
if (is_meson_g12b_cpu()) {
|
||||
parent_hw = clk_hw_get_parent(&g12b_cpu_clk1.mux.hw);
|
||||
parent_clk = parent_hw->clk;
|
||||
ret = clk_notifier_register(parent_clk, &g12b_cpu_clk1.clk_nb);
|
||||
ret = clk_notifier_register(g12b_sys1_pll.hw.clk,
|
||||
&g12b_cpu_nb_data);
|
||||
/*parent_hw = clk_hw_get_parent(&g12b_cpu_clk1.mux.hw);
|
||||
*parent_clk = parent_hw->clk;
|
||||
*ret = clk_notifier_register(parent_clk,
|
||||
*&g12b_cpu_clk1.clk_nb);
|
||||
*/
|
||||
} else {
|
||||
parent_hw = clk_hw_get_parent(&g12a_cpu_clk.mux.hw);
|
||||
parent_clk = parent_hw->clk;
|
||||
ret = clk_notifier_register(parent_clk, &g12a_cpu_clk.clk_nb);
|
||||
ret = clk_notifier_register(g12a_sys_pll.hw.clk,
|
||||
&g12a_cpu_nb_data);
|
||||
/*parent_hw = clk_hw_get_parent(&g12a_cpu_clk.mux.hw);
|
||||
*parent_clk = parent_hw->clk;
|
||||
*ret = clk_notifier_register(parent_clk,
|
||||
* &g12a_cpu_clk.clk_nb);
|
||||
*/
|
||||
}
|
||||
|
||||
if (ret) {
|
||||
|
||||
@@ -21,6 +21,7 @@
|
||||
#include <linux/platform_device.h>
|
||||
#include <linux/init.h>
|
||||
#include <linux/slab.h>
|
||||
#include <linux/delay.h>
|
||||
#include <dt-bindings/clock/amlogic,g12a-clkc.h>
|
||||
|
||||
#include "../clkc.h"
|
||||
@@ -55,7 +56,7 @@ static struct meson_clk_pll g12b_sys_pll = {
|
||||
};
|
||||
|
||||
static u32 mux_table_cpu_p[] = { 0, 1, 2 };
|
||||
static u32 mux_table_cpu_px[] = { 0, 1 };
|
||||
/*static u32 mux_table_cpu_px[] = { 0, 1 };*/
|
||||
|
||||
static struct meson_cpu_mux_divider g12b_cpu_fclk_p = {
|
||||
.reg = (void *)HHI_SYS_CPUB_CLK_CNTL,
|
||||
@@ -106,22 +107,39 @@ static struct meson_cpu_mux_divider g12b_cpu_fclk_p = {
|
||||
},
|
||||
};
|
||||
|
||||
static struct meson_clk_cpu g12b_cpu_clk = {
|
||||
.reg_off = HHI_SYS_CPUB_CLK_CNTL,
|
||||
.clk_nb.notifier_call = meson_clk_cpu_notifier_cb,
|
||||
.mux.reg = (void *)HHI_SYS_CPUB_CLK_CNTL,
|
||||
.mux.shift = 11,
|
||||
.mux.mask = 0x1,
|
||||
.mux.lock = &clk_lock,
|
||||
.mux.table = mux_table_cpu_px,
|
||||
.mux.hw.init = &(struct clk_init_data){
|
||||
.name = "cpub_clk",
|
||||
.ops = &meson_clk_cpu_ops,
|
||||
.parent_names = (const char *[]){ "cpub_fixedpll_p",
|
||||
"sys_pll"},
|
||||
.num_parents = 2,
|
||||
.flags = CLK_GET_RATE_NOCACHE,
|
||||
},
|
||||
/*
|
||||
*static struct meson_clk_cpu g12b_cpu_clk = {
|
||||
* .reg_off = HHI_SYS_CPUB_CLK_CNTL,
|
||||
* .clk_nb.notifier_call = meson_clk_cpu_notifier_cb,
|
||||
* .mux.reg = (void *)HHI_SYS_CPUB_CLK_CNTL,
|
||||
* .mux.shift = 11,
|
||||
* .mux.mask = 0x1,
|
||||
* .mux.lock = &clk_lock,
|
||||
* .mux.table = mux_table_cpu_px,
|
||||
* .mux.hw.init = &(struct clk_init_data){
|
||||
* .name = "cpub_clk",
|
||||
* .ops = &meson_clk_cpu_ops,
|
||||
* .parent_names = (const char *[]){ "cpub_fixedpll_p",
|
||||
* "sys_pll"},
|
||||
* .num_parents = 2,
|
||||
* .flags = CLK_GET_RATE_NOCACHE,
|
||||
* },
|
||||
*};
|
||||
*/
|
||||
|
||||
static struct clk_mux g12b_cpu_clk = {
|
||||
.reg = (void *)HHI_SYS_CPUB_CLK_CNTL,
|
||||
.mask = 0x1,
|
||||
.shift = 11,
|
||||
.lock = &clk_lock,
|
||||
.hw.init = &(struct clk_init_data){
|
||||
.name = "cpub_clk",
|
||||
.ops = &clk_mux_ops,
|
||||
.parent_names = (const char *[]){ "cpu_fixedpll_p",
|
||||
"sys_pll" },
|
||||
.num_parents = 2,
|
||||
.flags = CLK_GET_RATE_NOCACHE,
|
||||
},
|
||||
};
|
||||
|
||||
static const char * const media_parent_names[] = { "xtal",
|
||||
@@ -463,7 +481,7 @@ static struct clk_gate *g12b_clk_gates[] = {
|
||||
|
||||
static struct clk_hw *g12b_clk_hws[] = {
|
||||
[CLKID_CPUB_FCLK_P - CLKID_G12B_ADD_BASE] = &g12b_cpu_fclk_p.hw,
|
||||
[CLKID_CPUB_CLK - CLKID_G12B_ADD_BASE] = &g12b_cpu_clk.mux.hw,
|
||||
[CLKID_CPUB_CLK - CLKID_G12B_ADD_BASE] = &g12b_cpu_clk.hw,
|
||||
[CLKID_CSI_DIG - CLKID_G12B_ADD_BASE] = &g12b_csi_dig.hw,
|
||||
[CLKID_VIPNANOQ - CLKID_G12B_ADD_BASE] = &g12b_vipnanoq.hw,
|
||||
[CLKID_GDC - CLKID_G12B_ADD_BASE] = &g12b_gdc.hw,
|
||||
@@ -472,12 +490,48 @@ static struct clk_hw *g12b_clk_hws[] = {
|
||||
[CLKID_CSI2_PHY0 - CLKID_G12B_ADD_BASE] = &g12b_csi2_phy0.hw,
|
||||
};
|
||||
|
||||
static int g12b_cpub_clk_notifier_cb(struct notifier_block *nb,
|
||||
unsigned long event, void *data)
|
||||
{
|
||||
struct clk_hw *cpu_clk_hw, *parent_clk_hw;
|
||||
struct clk *cpu_clk, *parent_clk;
|
||||
int ret;
|
||||
|
||||
switch (event) {
|
||||
case PRE_RATE_CHANGE:
|
||||
parent_clk_hw = &g12b_cpu_fclk_p.hw;
|
||||
break;
|
||||
case POST_RATE_CHANGE:
|
||||
parent_clk_hw = &g12b_sys_pll.hw;
|
||||
break;
|
||||
default:
|
||||
return NOTIFY_DONE;
|
||||
}
|
||||
|
||||
cpu_clk_hw = &g12b_cpu_clk.hw;
|
||||
cpu_clk = __clk_lookup(clk_hw_get_name(cpu_clk_hw));
|
||||
parent_clk = __clk_lookup(clk_hw_get_name(parent_clk_hw));
|
||||
|
||||
ret = clk_set_parent(cpu_clk, parent_clk);
|
||||
if (ret)
|
||||
return notifier_from_errno(ret);
|
||||
|
||||
udelay(80);
|
||||
|
||||
return NOTIFY_OK;
|
||||
}
|
||||
|
||||
static struct notifier_block g12b_cpub_nb_data = {
|
||||
.notifier_call = g12b_cpub_clk_notifier_cb,
|
||||
};
|
||||
|
||||
|
||||
static void __init g12b_clkc_init(struct device_node *np)
|
||||
{
|
||||
int ret = 0, clkid, i;
|
||||
struct clk_hw *parent_hw;
|
||||
struct clk *parent_clk;
|
||||
/*struct clk_hw *parent_hw;
|
||||
*struct clk *parent_clk;
|
||||
*/
|
||||
|
||||
/* Generic clocks and PLLs */
|
||||
if (!clk_base)
|
||||
@@ -488,11 +542,10 @@ static void __init g12b_clkc_init(struct device_node *np)
|
||||
return;
|
||||
}
|
||||
|
||||
g12b_cpu_clk.base = clk_base;
|
||||
g12b_cpu_fclk_p.reg = clk_base
|
||||
+ (unsigned long)g12b_cpu_fclk_p.reg;
|
||||
g12b_cpu_clk.mux.reg = clk_base
|
||||
+ (unsigned long)g12b_cpu_clk.mux.reg;
|
||||
g12b_cpu_clk.reg = clk_base
|
||||
+ (unsigned long)g12b_cpu_clk.reg;
|
||||
|
||||
cts_gdc_core_clk_mux.reg = clk_base
|
||||
+ (unsigned long)(cts_gdc_core_clk_mux.reg);
|
||||
@@ -699,9 +752,11 @@ static void __init g12b_clkc_init(struct device_node *np)
|
||||
* a new clk_hw, and this hack will no longer work. Releasing the ccr
|
||||
* feature before that time solves the problem :-)
|
||||
*/
|
||||
parent_hw = clk_hw_get_parent(&g12b_cpu_clk.mux.hw);
|
||||
parent_clk = parent_hw->clk;
|
||||
ret = clk_notifier_register(parent_clk, &g12b_cpu_clk.clk_nb);
|
||||
/*parent_hw = clk_hw_get_parent(&g12b_cpu_clk.mux.hw);
|
||||
*parent_clk = parent_hw->clk;
|
||||
*ret = clk_notifier_register(parent_clk, &g12b_cpu_clk.clk_nb);
|
||||
*/
|
||||
ret = clk_notifier_register(g12b_sys_pll.hw.clk, &g12b_cpub_nb_data);
|
||||
if (ret) {
|
||||
pr_err("%s: failed to register clock notifier for cpu_clk\n",
|
||||
__func__);
|
||||
|
||||
Reference in New Issue
Block a user