Commit 7a110b91 authored by Claudiu Beznea's avatar Claudiu Beznea Committed by Stephen Boyd

clk: at91: clk-master: re-factor master clock

Re-factor master clock driver by splitting it into 2 clocks: prescaller
and divider clocks. Based on registered clock flags the prescaler's rate
could be changed at runtime. This is necessary for platforms supporting
DVFS (e.g. SAMA7G5) where master clock could be changed at run-time.
Signed-off-by: default avatarClaudiu Beznea <claudiu.beznea@microchip.com>
Link: https://lore.kernel.org/r/1605800597-16720-11-git-send-email-claudiu.beznea@microchip.comSigned-off-by: default avatarStephen Boyd <sboyd@kernel.org>
parent 120d5d8b
...@@ -7,6 +7,8 @@ ...@@ -7,6 +7,8 @@
#include "pmc.h" #include "pmc.h"
static DEFINE_SPINLOCK(rm9200_mck_lock);
struct sck { struct sck {
char *n; char *n;
char *p; char *p;
...@@ -137,9 +139,20 @@ static void __init at91rm9200_pmc_setup(struct device_node *np) ...@@ -137,9 +139,20 @@ static void __init at91rm9200_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "pllack"; parent_names[2] = "pllack";
parent_names[3] = "pllbck"; parent_names[3] = "pllbck";
hw = at91_clk_register_master(regmap, "masterck", 4, parent_names, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
parent_names,
&at91rm9200_master_layout,
&rm9200_mck_characteristics,
&rm9200_mck_lock, CLK_SET_RATE_GATE,
INT_MIN);
if (IS_ERR(hw))
goto err_free;
hw = at91_clk_register_master_div(regmap, "masterck_div",
"masterck_pres",
&at91rm9200_master_layout, &at91rm9200_master_layout,
&rm9200_mck_characteristics); &rm9200_mck_characteristics,
&rm9200_mck_lock, CLK_SET_RATE_GATE);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
...@@ -181,7 +194,7 @@ static void __init at91rm9200_pmc_setup(struct device_node *np) ...@@ -181,7 +194,7 @@ static void __init at91rm9200_pmc_setup(struct device_node *np)
for (i = 0; i < ARRAY_SIZE(at91rm9200_periphck); i++) { for (i = 0; i < ARRAY_SIZE(at91rm9200_periphck); i++) {
hw = at91_clk_register_peripheral(regmap, hw = at91_clk_register_peripheral(regmap,
at91rm9200_periphck[i].n, at91rm9200_periphck[i].n,
"masterck", "masterck_div",
at91rm9200_periphck[i].id); at91rm9200_periphck[i].id);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -32,6 +32,8 @@ struct at91sam926x_data { ...@@ -32,6 +32,8 @@ struct at91sam926x_data {
bool has_slck; bool has_slck;
}; };
static DEFINE_SPINLOCK(at91sam9260_mck_lock);
static const struct clk_master_characteristics sam9260_mck_characteristics = { static const struct clk_master_characteristics sam9260_mck_characteristics = {
.output = { .min = 0, .max = 105000000 }, .output = { .min = 0, .max = 105000000 },
.divisors = { 1, 2, 4, 0 }, .divisors = { 1, 2, 4, 0 },
...@@ -218,8 +220,8 @@ static const struct sck at91sam9261_systemck[] = { ...@@ -218,8 +220,8 @@ static const struct sck at91sam9261_systemck[] = {
{ .n = "pck1", .p = "prog1", .id = 9 }, { .n = "pck1", .p = "prog1", .id = 9 },
{ .n = "pck2", .p = "prog2", .id = 10 }, { .n = "pck2", .p = "prog2", .id = 10 },
{ .n = "pck3", .p = "prog3", .id = 11 }, { .n = "pck3", .p = "prog3", .id = 11 },
{ .n = "hclk0", .p = "masterck", .id = 16 }, { .n = "hclk0", .p = "masterck_div", .id = 16 },
{ .n = "hclk1", .p = "masterck", .id = 17 }, { .n = "hclk1", .p = "masterck_div", .id = 17 },
}; };
static const struct pck at91sam9261_periphck[] = { static const struct pck at91sam9261_periphck[] = {
...@@ -413,9 +415,21 @@ static void __init at91sam926x_pmc_setup(struct device_node *np, ...@@ -413,9 +415,21 @@ static void __init at91sam926x_pmc_setup(struct device_node *np,
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "pllack"; parent_names[2] = "pllack";
parent_names[3] = "pllbck"; parent_names[3] = "pllbck";
hw = at91_clk_register_master(regmap, "masterck", 4, parent_names, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
parent_names,
&at91rm9200_master_layout,
data->mck_characteristics,
&at91sam9260_mck_lock,
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw))
goto err_free;
hw = at91_clk_register_master_div(regmap, "masterck_div",
"masterck_pres",
&at91rm9200_master_layout, &at91rm9200_master_layout,
data->mck_characteristics); data->mck_characteristics,
&at91sam9260_mck_lock,
CLK_SET_RATE_GATE);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
...@@ -457,7 +471,7 @@ static void __init at91sam926x_pmc_setup(struct device_node *np, ...@@ -457,7 +471,7 @@ static void __init at91sam926x_pmc_setup(struct device_node *np,
for (i = 0; i < data->num_pck; i++) { for (i = 0; i < data->num_pck; i++) {
hw = at91_clk_register_peripheral(regmap, hw = at91_clk_register_peripheral(regmap,
data->pck[i].n, data->pck[i].n,
"masterck", "masterck_div",
data->pck[i].id); data->pck[i].id);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -7,6 +7,8 @@ ...@@ -7,6 +7,8 @@
#include "pmc.h" #include "pmc.h"
static DEFINE_SPINLOCK(at91sam9g45_mck_lock);
static const struct clk_master_characteristics mck_characteristics = { static const struct clk_master_characteristics mck_characteristics = {
.output = { .min = 0, .max = 133333333 }, .output = { .min = 0, .max = 133333333 },
.divisors = { 1, 2, 4, 3 }, .divisors = { 1, 2, 4, 3 },
...@@ -40,7 +42,7 @@ static const struct { ...@@ -40,7 +42,7 @@ static const struct {
char *p; char *p;
u8 id; u8 id;
} at91sam9g45_systemck[] = { } at91sam9g45_systemck[] = {
{ .n = "ddrck", .p = "masterck", .id = 2 }, { .n = "ddrck", .p = "masterck_div", .id = 2 },
{ .n = "uhpck", .p = "usbck", .id = 6 }, { .n = "uhpck", .p = "usbck", .id = 6 },
{ .n = "pck0", .p = "prog0", .id = 8 }, { .n = "pck0", .p = "prog0", .id = 8 },
{ .n = "pck1", .p = "prog1", .id = 9 }, { .n = "pck1", .p = "prog1", .id = 9 },
...@@ -148,9 +150,21 @@ static void __init at91sam9g45_pmc_setup(struct device_node *np) ...@@ -148,9 +150,21 @@ static void __init at91sam9g45_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "plladivck"; parent_names[2] = "plladivck";
parent_names[3] = "utmick"; parent_names[3] = "utmick";
hw = at91_clk_register_master(regmap, "masterck", 4, parent_names, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
parent_names,
&at91rm9200_master_layout,
&mck_characteristics,
&at91sam9g45_mck_lock,
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw))
goto err_free;
hw = at91_clk_register_master_div(regmap, "masterck_div",
"masterck_pres",
&at91rm9200_master_layout, &at91rm9200_master_layout,
&mck_characteristics); &mck_characteristics,
&at91sam9g45_mck_lock,
CLK_SET_RATE_GATE);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
...@@ -166,7 +180,7 @@ static void __init at91sam9g45_pmc_setup(struct device_node *np) ...@@ -166,7 +180,7 @@ static void __init at91sam9g45_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "plladivck"; parent_names[2] = "plladivck";
parent_names[3] = "utmick"; parent_names[3] = "utmick";
parent_names[4] = "masterck"; parent_names[4] = "masterck_div";
for (i = 0; i < 2; i++) { for (i = 0; i < 2; i++) {
char name[6]; char name[6];
...@@ -195,7 +209,7 @@ static void __init at91sam9g45_pmc_setup(struct device_node *np) ...@@ -195,7 +209,7 @@ static void __init at91sam9g45_pmc_setup(struct device_node *np)
for (i = 0; i < ARRAY_SIZE(at91sam9g45_periphck); i++) { for (i = 0; i < ARRAY_SIZE(at91sam9g45_periphck); i++) {
hw = at91_clk_register_peripheral(regmap, hw = at91_clk_register_peripheral(regmap,
at91sam9g45_periphck[i].n, at91sam9g45_periphck[i].n,
"masterck", "masterck_div",
at91sam9g45_periphck[i].id); at91sam9g45_periphck[i].id);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -7,6 +7,8 @@ ...@@ -7,6 +7,8 @@
#include "pmc.h" #include "pmc.h"
static DEFINE_SPINLOCK(at91sam9n12_mck_lock);
static const struct clk_master_characteristics mck_characteristics = { static const struct clk_master_characteristics mck_characteristics = {
.output = { .min = 0, .max = 133333333 }, .output = { .min = 0, .max = 133333333 },
.divisors = { 1, 2, 4, 3 }, .divisors = { 1, 2, 4, 3 },
...@@ -54,8 +56,8 @@ static const struct { ...@@ -54,8 +56,8 @@ static const struct {
char *p; char *p;
u8 id; u8 id;
} at91sam9n12_systemck[] = { } at91sam9n12_systemck[] = {
{ .n = "ddrck", .p = "masterck", .id = 2 }, { .n = "ddrck", .p = "masterck_div", .id = 2 },
{ .n = "lcdck", .p = "masterck", .id = 3 }, { .n = "lcdck", .p = "masterck_div", .id = 3 },
{ .n = "uhpck", .p = "usbck", .id = 6 }, { .n = "uhpck", .p = "usbck", .id = 6 },
{ .n = "udpck", .p = "usbck", .id = 7 }, { .n = "udpck", .p = "usbck", .id = 7 },
{ .n = "pck0", .p = "prog0", .id = 8 }, { .n = "pck0", .p = "prog0", .id = 8 },
...@@ -175,9 +177,21 @@ static void __init at91sam9n12_pmc_setup(struct device_node *np) ...@@ -175,9 +177,21 @@ static void __init at91sam9n12_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "plladivck"; parent_names[2] = "plladivck";
parent_names[3] = "pllbck"; parent_names[3] = "pllbck";
hw = at91_clk_register_master(regmap, "masterck", 4, parent_names, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
parent_names,
&at91sam9x5_master_layout,
&mck_characteristics,
&at91sam9n12_mck_lock,
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw))
goto err_free;
hw = at91_clk_register_master_div(regmap, "masterck_div",
"masterck_pres",
&at91sam9x5_master_layout, &at91sam9x5_master_layout,
&mck_characteristics); &mck_characteristics,
&at91sam9n12_mck_lock,
CLK_SET_RATE_GATE);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
...@@ -191,7 +205,7 @@ static void __init at91sam9n12_pmc_setup(struct device_node *np) ...@@ -191,7 +205,7 @@ static void __init at91sam9n12_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "plladivck"; parent_names[2] = "plladivck";
parent_names[3] = "pllbck"; parent_names[3] = "pllbck";
parent_names[4] = "masterck"; parent_names[4] = "masterck_div";
for (i = 0; i < 2; i++) { for (i = 0; i < 2; i++) {
char name[6]; char name[6];
...@@ -221,7 +235,7 @@ static void __init at91sam9n12_pmc_setup(struct device_node *np) ...@@ -221,7 +235,7 @@ static void __init at91sam9n12_pmc_setup(struct device_node *np)
hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock, hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock,
&at91sam9n12_pcr_layout, &at91sam9n12_pcr_layout,
at91sam9n12_periphck[i].n, at91sam9n12_periphck[i].n,
"masterck", "masterck_div",
at91sam9n12_periphck[i].id, at91sam9n12_periphck[i].id,
&range, INT_MIN); &range, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
......
...@@ -7,6 +7,8 @@ ...@@ -7,6 +7,8 @@
#include "pmc.h" #include "pmc.h"
static DEFINE_SPINLOCK(sam9rl_mck_lock);
static const struct clk_master_characteristics sam9rl_mck_characteristics = { static const struct clk_master_characteristics sam9rl_mck_characteristics = {
.output = { .min = 0, .max = 94000000 }, .output = { .min = 0, .max = 94000000 },
.divisors = { 1, 2, 4, 0 }, .divisors = { 1, 2, 4, 0 },
...@@ -117,9 +119,20 @@ static void __init at91sam9rl_pmc_setup(struct device_node *np) ...@@ -117,9 +119,20 @@ static void __init at91sam9rl_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "pllack"; parent_names[2] = "pllack";
parent_names[3] = "utmick"; parent_names[3] = "utmick";
hw = at91_clk_register_master(regmap, "masterck", 4, parent_names, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
parent_names,
&at91rm9200_master_layout,
&sam9rl_mck_characteristics,
&sam9rl_mck_lock, CLK_SET_RATE_GATE,
INT_MIN);
if (IS_ERR(hw))
goto err_free;
hw = at91_clk_register_master_div(regmap, "masterck_div",
"masterck_pres",
&at91rm9200_master_layout, &at91rm9200_master_layout,
&sam9rl_mck_characteristics); &sam9rl_mck_characteristics,
&sam9rl_mck_lock, CLK_SET_RATE_GATE);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
...@@ -129,7 +142,7 @@ static void __init at91sam9rl_pmc_setup(struct device_node *np) ...@@ -129,7 +142,7 @@ static void __init at91sam9rl_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "pllack"; parent_names[2] = "pllack";
parent_names[3] = "utmick"; parent_names[3] = "utmick";
parent_names[4] = "masterck"; parent_names[4] = "masterck_div";
for (i = 0; i < 2; i++) { for (i = 0; i < 2; i++) {
char name[6]; char name[6];
...@@ -158,7 +171,7 @@ static void __init at91sam9rl_pmc_setup(struct device_node *np) ...@@ -158,7 +171,7 @@ static void __init at91sam9rl_pmc_setup(struct device_node *np)
for (i = 0; i < ARRAY_SIZE(at91sam9rl_periphck); i++) { for (i = 0; i < ARRAY_SIZE(at91sam9rl_periphck); i++) {
hw = at91_clk_register_peripheral(regmap, hw = at91_clk_register_peripheral(regmap,
at91sam9rl_periphck[i].n, at91sam9rl_periphck[i].n,
"masterck", "masterck_div",
at91sam9rl_periphck[i].id); at91sam9rl_periphck[i].id);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
...@@ -7,6 +7,8 @@ ...@@ -7,6 +7,8 @@
#include "pmc.h" #include "pmc.h"
static DEFINE_SPINLOCK(mck_lock);
static const struct clk_master_characteristics mck_characteristics = { static const struct clk_master_characteristics mck_characteristics = {
.output = { .min = 0, .max = 133333333 }, .output = { .min = 0, .max = 133333333 },
.divisors = { 1, 2, 4, 3 }, .divisors = { 1, 2, 4, 3 },
...@@ -41,7 +43,7 @@ static const struct { ...@@ -41,7 +43,7 @@ static const struct {
char *p; char *p;
u8 id; u8 id;
} at91sam9x5_systemck[] = { } at91sam9x5_systemck[] = {
{ .n = "ddrck", .p = "masterck", .id = 2 }, { .n = "ddrck", .p = "masterck_div", .id = 2 },
{ .n = "smdck", .p = "smdclk", .id = 4 }, { .n = "smdck", .p = "smdclk", .id = 4 },
{ .n = "uhpck", .p = "usbck", .id = 6 }, { .n = "uhpck", .p = "usbck", .id = 6 },
{ .n = "udpck", .p = "usbck", .id = 7 }, { .n = "udpck", .p = "usbck", .id = 7 },
...@@ -196,9 +198,19 @@ static void __init at91sam9x5_pmc_setup(struct device_node *np, ...@@ -196,9 +198,19 @@ static void __init at91sam9x5_pmc_setup(struct device_node *np,
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "plladivck"; parent_names[2] = "plladivck";
parent_names[3] = "utmick"; parent_names[3] = "utmick";
hw = at91_clk_register_master(regmap, "masterck", 4, parent_names, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
parent_names,
&at91sam9x5_master_layout,
&mck_characteristics, &mck_lock,
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw))
goto err_free;
hw = at91_clk_register_master_div(regmap, "masterck_div",
"masterck_pres",
&at91sam9x5_master_layout, &at91sam9x5_master_layout,
&mck_characteristics); &mck_characteristics, &mck_lock,
CLK_SET_RATE_GATE);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
...@@ -218,7 +230,7 @@ static void __init at91sam9x5_pmc_setup(struct device_node *np, ...@@ -218,7 +230,7 @@ static void __init at91sam9x5_pmc_setup(struct device_node *np,
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "plladivck"; parent_names[2] = "plladivck";
parent_names[3] = "utmick"; parent_names[3] = "utmick";
parent_names[4] = "masterck"; parent_names[4] = "masterck_div";
for (i = 0; i < 2; i++) { for (i = 0; i < 2; i++) {
char name[6]; char name[6];
...@@ -245,7 +257,7 @@ static void __init at91sam9x5_pmc_setup(struct device_node *np, ...@@ -245,7 +257,7 @@ static void __init at91sam9x5_pmc_setup(struct device_node *np,
} }
if (has_lcdck) { if (has_lcdck) {
hw = at91_clk_register_system(regmap, "lcdck", "masterck", 3); hw = at91_clk_register_system(regmap, "lcdck", "masterck_div", 3);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
...@@ -256,7 +268,7 @@ static void __init at91sam9x5_pmc_setup(struct device_node *np, ...@@ -256,7 +268,7 @@ static void __init at91sam9x5_pmc_setup(struct device_node *np,
hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock, hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock,
&at91sam9x5_pcr_layout, &at91sam9x5_pcr_layout,
at91sam9x5_periphck[i].n, at91sam9x5_periphck[i].n,
"masterck", "masterck_div",
at91sam9x5_periphck[i].id, at91sam9x5_periphck[i].id,
&range, INT_MIN); &range, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
...@@ -269,7 +281,7 @@ static void __init at91sam9x5_pmc_setup(struct device_node *np, ...@@ -269,7 +281,7 @@ static void __init at91sam9x5_pmc_setup(struct device_node *np,
hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock, hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock,
&at91sam9x5_pcr_layout, &at91sam9x5_pcr_layout,
extra_pcks[i].n, extra_pcks[i].n,
"masterck", "masterck_div",
extra_pcks[i].id, extra_pcks[i].id,
&range, INT_MIN); &range, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
......
This diff is collapsed.
...@@ -24,6 +24,8 @@ ...@@ -24,6 +24,8 @@
#define GCK_INDEX_DT_AUDIO_PLL 5 #define GCK_INDEX_DT_AUDIO_PLL 5
static DEFINE_SPINLOCK(mck_lock);
#ifdef CONFIG_HAVE_AT91_AUDIO_PLL #ifdef CONFIG_HAVE_AT91_AUDIO_PLL
static void __init of_sama5d2_clk_audio_pll_frac_setup(struct device_node *np) static void __init of_sama5d2_clk_audio_pll_frac_setup(struct device_node *np)
{ {
...@@ -388,9 +390,16 @@ of_at91_clk_master_setup(struct device_node *np, ...@@ -388,9 +390,16 @@ of_at91_clk_master_setup(struct device_node *np,
if (IS_ERR(regmap)) if (IS_ERR(regmap))
return; return;
hw = at91_clk_register_master(regmap, name, num_parents, hw = at91_clk_register_master_pres(regmap, "masterck_pres", num_parents,
parent_names, layout, parent_names, layout,
characteristics); characteristics, &mck_lock,
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw))
goto out_free_characteristics;
hw = at91_clk_register_master_div(regmap, name, "masterck_pres",
layout, characteristics,
&mck_lock, CLK_SET_RATE_GATE);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto out_free_characteristics; goto out_free_characteristics;
......
...@@ -155,10 +155,18 @@ at91_clk_register_sam9x5_main(struct regmap *regmap, const char *name, ...@@ -155,10 +155,18 @@ at91_clk_register_sam9x5_main(struct regmap *regmap, const char *name,
const char **parent_names, int num_parents); const char **parent_names, int num_parents);
struct clk_hw * __init struct clk_hw * __init
at91_clk_register_master(struct regmap *regmap, const char *name, at91_clk_register_master_pres(struct regmap *regmap, const char *name,
int num_parents, const char **parent_names, int num_parents, const char **parent_names,
const struct clk_master_layout *layout, const struct clk_master_layout *layout,
const struct clk_master_characteristics *characteristics); const struct clk_master_characteristics *characteristics,
spinlock_t *lock, u32 flags, int chg_pid);
struct clk_hw * __init
at91_clk_register_master_div(struct regmap *regmap, const char *name,
const char *parent_names,
const struct clk_master_layout *layout,
const struct clk_master_characteristics *characteristics,
spinlock_t *lock, u32 flags);
struct clk_hw * __init struct clk_hw * __init
at91_clk_sama7g5_register_master(struct regmap *regmap, at91_clk_sama7g5_register_master(struct regmap *regmap,
......
...@@ -8,6 +8,7 @@ ...@@ -8,6 +8,7 @@
#include "pmc.h" #include "pmc.h"
static DEFINE_SPINLOCK(pmc_pll_lock); static DEFINE_SPINLOCK(pmc_pll_lock);
static DEFINE_SPINLOCK(mck_lock);
static const struct clk_master_characteristics mck_characteristics = { static const struct clk_master_characteristics mck_characteristics = {
.output = { .min = 140000000, .max = 200000000 }, .output = { .min = 140000000, .max = 200000000 },
...@@ -76,11 +77,11 @@ static const struct { ...@@ -76,11 +77,11 @@ static const struct {
char *p; char *p;
u8 id; u8 id;
} sam9x60_systemck[] = { } sam9x60_systemck[] = {
{ .n = "ddrck", .p = "masterck", .id = 2 }, { .n = "ddrck", .p = "masterck_div", .id = 2 },
{ .n = "uhpck", .p = "usbck", .id = 6 }, { .n = "uhpck", .p = "usbck", .id = 6 },
{ .n = "pck0", .p = "prog0", .id = 8 }, { .n = "pck0", .p = "prog0", .id = 8 },
{ .n = "pck1", .p = "prog1", .id = 9 }, { .n = "pck1", .p = "prog1", .id = 9 },
{ .n = "qspick", .p = "masterck", .id = 19 }, { .n = "qspick", .p = "masterck_div", .id = 19 },
}; };
static const struct { static const struct {
...@@ -272,9 +273,17 @@ static void __init sam9x60_pmc_setup(struct device_node *np) ...@@ -272,9 +273,17 @@ static void __init sam9x60_pmc_setup(struct device_node *np)
parent_names[0] = md_slck_name; parent_names[0] = md_slck_name;
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "pllack_divck"; parent_names[2] = "pllack_divck";
hw = at91_clk_register_master(regmap, "masterck", 3, parent_names, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 3,
&sam9x60_master_layout, parent_names, &sam9x60_master_layout,
&mck_characteristics); &mck_characteristics, &mck_lock,
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw))
goto err_free;
hw = at91_clk_register_master_div(regmap, "masterck_div",
"masterck_pres", &sam9x60_master_layout,
&mck_characteristics, &mck_lock,
CLK_SET_RATE_GATE);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
...@@ -290,7 +299,7 @@ static void __init sam9x60_pmc_setup(struct device_node *np) ...@@ -290,7 +299,7 @@ static void __init sam9x60_pmc_setup(struct device_node *np)
parent_names[0] = md_slck_name; parent_names[0] = md_slck_name;
parent_names[1] = td_slck_name; parent_names[1] = td_slck_name;
parent_names[2] = "mainck"; parent_names[2] = "mainck";
parent_names[3] = "masterck"; parent_names[3] = "masterck_div";
parent_names[4] = "pllack_divck"; parent_names[4] = "pllack_divck";
parent_names[5] = "upllck_divck"; parent_names[5] = "upllck_divck";
for (i = 0; i < 2; i++) { for (i = 0; i < 2; i++) {
...@@ -322,7 +331,7 @@ static void __init sam9x60_pmc_setup(struct device_node *np) ...@@ -322,7 +331,7 @@ static void __init sam9x60_pmc_setup(struct device_node *np)
hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock, hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock,
&sam9x60_pcr_layout, &sam9x60_pcr_layout,
sam9x60_periphck[i].n, sam9x60_periphck[i].n,
"masterck", "masterck_div",
sam9x60_periphck[i].id, sam9x60_periphck[i].id,
&range, INT_MIN); &range, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
......
...@@ -7,6 +7,8 @@ ...@@ -7,6 +7,8 @@
#include "pmc.h" #include "pmc.h"
static DEFINE_SPINLOCK(mck_lock);
static const struct clk_master_characteristics mck_characteristics = { static const struct clk_master_characteristics mck_characteristics = {
.output = { .min = 124000000, .max = 166000000 }, .output = { .min = 124000000, .max = 166000000 },
.divisors = { 1, 2, 4, 3 }, .divisors = { 1, 2, 4, 3 },
...@@ -40,14 +42,14 @@ static const struct { ...@@ -40,14 +42,14 @@ static const struct {
char *p; char *p;
u8 id; u8 id;
} sama5d2_systemck[] = { } sama5d2_systemck[] = {
{ .n = "ddrck", .p = "masterck", .id = 2 }, { .n = "ddrck", .p = "masterck_div", .id = 2 },
{ .n = "lcdck", .p = "masterck", .id = 3 }, { .n = "lcdck", .p = "masterck_div", .id = 3 },
{ .n = "uhpck", .p = "usbck", .id = 6 }, { .n = "uhpck", .p = "usbck", .id = 6 },
{ .n = "udpck", .p = "usbck", .id = 7 }, { .n = "udpck", .p = "usbck", .id = 7 },
{ .n = "pck0", .p = "prog0", .id = 8 }, { .n = "pck0", .p = "prog0", .id = 8 },
{ .n = "pck1", .p = "prog1", .id = 9 }, { .n = "pck1", .p = "prog1", .id = 9 },
{ .n = "pck2", .p = "prog2", .id = 10 }, { .n = "pck2", .p = "prog2", .id = 10 },
{ .n = "iscck", .p = "masterck", .id = 18 }, { .n = "iscck", .p = "masterck_div", .id = 18 },
}; };
static const struct { static const struct {
...@@ -235,15 +237,25 @@ static void __init sama5d2_pmc_setup(struct device_node *np) ...@@ -235,15 +237,25 @@ static void __init sama5d2_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "plladivck"; parent_names[2] = "plladivck";
parent_names[3] = "utmick"; parent_names[3] = "utmick";
hw = at91_clk_register_master(regmap, "masterck", 4, parent_names, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
parent_names,
&at91sam9x5_master_layout,
&mck_characteristics, &mck_lock,
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw))
goto err_free;
hw = at91_clk_register_master_div(regmap, "masterck_div",
"masterck_pres",
&at91sam9x5_master_layout, &at91sam9x5_master_layout,
&mck_characteristics); &mck_characteristics, &mck_lock,
CLK_SET_RATE_GATE);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
sama5d2_pmc->chws[PMC_MCK] = hw; sama5d2_pmc->chws[PMC_MCK] = hw;
hw = at91_clk_register_h32mx(regmap, "h32mxck", "masterck"); hw = at91_clk_register_h32mx(regmap, "h32mxck", "masterck_div");
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
...@@ -259,7 +271,7 @@ static void __init sama5d2_pmc_setup(struct device_node *np) ...@@ -259,7 +271,7 @@ static void __init sama5d2_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "plladivck"; parent_names[2] = "plladivck";
parent_names[3] = "utmick"; parent_names[3] = "utmick";
parent_names[4] = "masterck"; parent_names[4] = "masterck_div";
parent_names[5] = "audiopll_pmcck"; parent_names[5] = "audiopll_pmcck";
for (i = 0; i < 3; i++) { for (i = 0; i < 3; i++) {
char name[6]; char name[6];
...@@ -290,7 +302,7 @@ static void __init sama5d2_pmc_setup(struct device_node *np) ...@@ -290,7 +302,7 @@ static void __init sama5d2_pmc_setup(struct device_node *np)
hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock, hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock,
&sama5d2_pcr_layout, &sama5d2_pcr_layout,
sama5d2_periphck[i].n, sama5d2_periphck[i].n,
"masterck", "masterck_div",
sama5d2_periphck[i].id, sama5d2_periphck[i].id,
&range, INT_MIN); &range, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
...@@ -317,7 +329,7 @@ static void __init sama5d2_pmc_setup(struct device_node *np) ...@@ -317,7 +329,7 @@ static void __init sama5d2_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "plladivck"; parent_names[2] = "plladivck";
parent_names[3] = "utmick"; parent_names[3] = "utmick";
parent_names[4] = "masterck"; parent_names[4] = "masterck_div";
parent_names[5] = "audiopll_pmcck"; parent_names[5] = "audiopll_pmcck";
for (i = 0; i < ARRAY_SIZE(sama5d2_gck); i++) { for (i = 0; i < ARRAY_SIZE(sama5d2_gck); i++) {
hw = at91_clk_register_generated(regmap, &pmc_pcr_lock, hw = at91_clk_register_generated(regmap, &pmc_pcr_lock,
......
...@@ -7,6 +7,8 @@ ...@@ -7,6 +7,8 @@
#include "pmc.h" #include "pmc.h"
static DEFINE_SPINLOCK(mck_lock);
static const struct clk_master_characteristics mck_characteristics = { static const struct clk_master_characteristics mck_characteristics = {
.output = { .min = 0, .max = 166000000 }, .output = { .min = 0, .max = 166000000 },
.divisors = { 1, 2, 4, 3 }, .divisors = { 1, 2, 4, 3 },
...@@ -40,8 +42,8 @@ static const struct { ...@@ -40,8 +42,8 @@ static const struct {
char *p; char *p;
u8 id; u8 id;
} sama5d3_systemck[] = { } sama5d3_systemck[] = {
{ .n = "ddrck", .p = "masterck", .id = 2 }, { .n = "ddrck", .p = "masterck_div", .id = 2 },
{ .n = "lcdck", .p = "masterck", .id = 3 }, { .n = "lcdck", .p = "masterck_div", .id = 3 },
{ .n = "smdck", .p = "smdclk", .id = 4 }, { .n = "smdck", .p = "smdclk", .id = 4 },
{ .n = "uhpck", .p = "usbck", .id = 6 }, { .n = "uhpck", .p = "usbck", .id = 6 },
{ .n = "udpck", .p = "usbck", .id = 7 }, { .n = "udpck", .p = "usbck", .id = 7 },
...@@ -170,9 +172,19 @@ static void __init sama5d3_pmc_setup(struct device_node *np) ...@@ -170,9 +172,19 @@ static void __init sama5d3_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "plladivck"; parent_names[2] = "plladivck";
parent_names[3] = "utmick"; parent_names[3] = "utmick";
hw = at91_clk_register_master(regmap, "masterck", 4, parent_names, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
parent_names,
&at91sam9x5_master_layout,
&mck_characteristics, &mck_lock,
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw))
goto err_free;
hw = at91_clk_register_master_div(regmap, "masterck_div",
"masterck_pres",
&at91sam9x5_master_layout, &at91sam9x5_master_layout,
&mck_characteristics); &mck_characteristics, &mck_lock,
CLK_SET_RATE_GATE);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
...@@ -192,7 +204,7 @@ static void __init sama5d3_pmc_setup(struct device_node *np) ...@@ -192,7 +204,7 @@ static void __init sama5d3_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "plladivck"; parent_names[2] = "plladivck";
parent_names[3] = "utmick"; parent_names[3] = "utmick";
parent_names[4] = "masterck"; parent_names[4] = "masterck_div";
for (i = 0; i < 3; i++) { for (i = 0; i < 3; i++) {
char name[6]; char name[6];
...@@ -222,7 +234,7 @@ static void __init sama5d3_pmc_setup(struct device_node *np) ...@@ -222,7 +234,7 @@ static void __init sama5d3_pmc_setup(struct device_node *np)
hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock, hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock,
&sama5d3_pcr_layout, &sama5d3_pcr_layout,
sama5d3_periphck[i].n, sama5d3_periphck[i].n,
"masterck", "masterck_div",
sama5d3_periphck[i].id, sama5d3_periphck[i].id,
&sama5d3_periphck[i].r, &sama5d3_periphck[i].r,
INT_MIN); INT_MIN);
......
...@@ -7,6 +7,8 @@ ...@@ -7,6 +7,8 @@
#include "pmc.h" #include "pmc.h"
static DEFINE_SPINLOCK(mck_lock);
static const struct clk_master_characteristics mck_characteristics = { static const struct clk_master_characteristics mck_characteristics = {
.output = { .min = 125000000, .max = 200000000 }, .output = { .min = 125000000, .max = 200000000 },
.divisors = { 1, 2, 4, 3 }, .divisors = { 1, 2, 4, 3 },
...@@ -39,8 +41,8 @@ static const struct { ...@@ -39,8 +41,8 @@ static const struct {
char *p; char *p;
u8 id; u8 id;
} sama5d4_systemck[] = { } sama5d4_systemck[] = {
{ .n = "ddrck", .p = "masterck", .id = 2 }, { .n = "ddrck", .p = "masterck_div", .id = 2 },
{ .n = "lcdck", .p = "masterck", .id = 3 }, { .n = "lcdck", .p = "masterck_div", .id = 3 },
{ .n = "smdck", .p = "smdclk", .id = 4 }, { .n = "smdck", .p = "smdclk", .id = 4 },
{ .n = "uhpck", .p = "usbck", .id = 6 }, { .n = "uhpck", .p = "usbck", .id = 6 },
{ .n = "udpck", .p = "usbck", .id = 7 }, { .n = "udpck", .p = "usbck", .id = 7 },
...@@ -185,15 +187,25 @@ static void __init sama5d4_pmc_setup(struct device_node *np) ...@@ -185,15 +187,25 @@ static void __init sama5d4_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "plladivck"; parent_names[2] = "plladivck";
parent_names[3] = "utmick"; parent_names[3] = "utmick";
hw = at91_clk_register_master(regmap, "masterck", 4, parent_names, hw = at91_clk_register_master_pres(regmap, "masterck_pres", 4,
parent_names,
&at91sam9x5_master_layout,
&mck_characteristics, &mck_lock,
CLK_SET_RATE_GATE, INT_MIN);
if (IS_ERR(hw))
goto err_free;
hw = at91_clk_register_master_div(regmap, "masterck_div",
"masterck_pres",
&at91sam9x5_master_layout, &at91sam9x5_master_layout,
&mck_characteristics); &mck_characteristics, &mck_lock,
CLK_SET_RATE_GATE);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
sama5d4_pmc->chws[PMC_MCK] = hw; sama5d4_pmc->chws[PMC_MCK] = hw;
hw = at91_clk_register_h32mx(regmap, "h32mxck", "masterck"); hw = at91_clk_register_h32mx(regmap, "h32mxck", "masterck_div");
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
...@@ -215,7 +227,7 @@ static void __init sama5d4_pmc_setup(struct device_node *np) ...@@ -215,7 +227,7 @@ static void __init sama5d4_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "plladivck"; parent_names[2] = "plladivck";
parent_names[3] = "utmick"; parent_names[3] = "utmick";
parent_names[4] = "masterck"; parent_names[4] = "masterck_div";
for (i = 0; i < 3; i++) { for (i = 0; i < 3; i++) {
char name[6]; char name[6];
...@@ -245,7 +257,7 @@ static void __init sama5d4_pmc_setup(struct device_node *np) ...@@ -245,7 +257,7 @@ static void __init sama5d4_pmc_setup(struct device_node *np)
hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock, hw = at91_clk_register_sam9x5_peripheral(regmap, &pmc_pcr_lock,
&sama5d4_pcr_layout, &sama5d4_pcr_layout,
sama5d4_periphck[i].n, sama5d4_periphck[i].n,
"masterck", "masterck_div",
sama5d4_periphck[i].id, sama5d4_periphck[i].id,
&range, INT_MIN); &range, INT_MIN);
if (IS_ERR(hw)) if (IS_ERR(hw))
......
...@@ -32,6 +32,7 @@ ...@@ -32,6 +32,7 @@
} while (0) } while (0)
static DEFINE_SPINLOCK(pmc_pll_lock); static DEFINE_SPINLOCK(pmc_pll_lock);
static DEFINE_SPINLOCK(pmc_mck0_lock);
static DEFINE_SPINLOCK(pmc_mckX_lock); static DEFINE_SPINLOCK(pmc_mckX_lock);
/** /**
...@@ -984,8 +985,16 @@ static void __init sama7g5_pmc_setup(struct device_node *np) ...@@ -984,8 +985,16 @@ static void __init sama7g5_pmc_setup(struct device_node *np)
parent_names[1] = "mainck"; parent_names[1] = "mainck";
parent_names[2] = "cpupll_divpmcck"; parent_names[2] = "cpupll_divpmcck";
parent_names[3] = "syspll_divpmcck"; parent_names[3] = "syspll_divpmcck";
hw = at91_clk_register_master(regmap, "mck0", 4, parent_names, hw = at91_clk_register_master_pres(regmap, "mck0_pres", 4, parent_names,
&mck0_layout, &mck0_characteristics); &mck0_layout, &mck0_characteristics,
&pmc_mck0_lock,
CLK_SET_RATE_PARENT, 0);
if (IS_ERR(hw))
goto err_free;
hw = at91_clk_register_master_div(regmap, "mck0_div", "mck0_pres",
&mck0_layout, &mck0_characteristics,
&pmc_mck0_lock, 0);
if (IS_ERR(hw)) if (IS_ERR(hw))
goto err_free; goto err_free;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment