Commit 2ef9df7a authored by Nicolas Ferre's avatar Nicolas Ferre Committed by Russell King

[ARM] 5569/1: at91: Support for at91sam9g45: clocks management

Add the at91sam9g45 series support to the AT91 generic clock file.
This takes care of the particularities of the PMC for this series.
It also takes advantage of the management by functionalities of
those PLLs and clocks.
Signed-off-by: default avatarNicolas Ferre <nicolas.ferre@atmel.com>
Acked-by: default avatarAndrew Victor <linux@maxim.org.za>
Signed-off-by: default avatarRussell King <rmk+kernel@arm.linux.org.uk>
parent 7c6425a0
...@@ -47,20 +47,25 @@ ...@@ -47,20 +47,25 @@
* Chips have some kind of clocks : group them by functionality * Chips have some kind of clocks : group them by functionality
*/ */
#define cpu_has_utmi() ( cpu_is_at91cap9() \ #define cpu_has_utmi() ( cpu_is_at91cap9() \
|| cpu_is_at91sam9rl()) || cpu_is_at91sam9rl() \
|| cpu_is_at91sam9g45())
#define cpu_has_800M_plla() (cpu_is_at91sam9g20()) #define cpu_has_800M_plla() ( cpu_is_at91sam9g20() \
|| cpu_is_at91sam9g45())
#define cpu_has_pllb() (!cpu_is_at91sam9rl()) #define cpu_has_300M_plla() (0)
#define cpu_has_upll() (0) #define cpu_has_pllb() (!(cpu_is_at91sam9rl() \
|| cpu_is_at91sam9g45()))
#define cpu_has_upll() (cpu_is_at91sam9g45())
/* USB host HS & FS */ /* USB host HS & FS */
#define cpu_has_uhp() (!cpu_is_at91sam9rl()) #define cpu_has_uhp() (!cpu_is_at91sam9rl())
/* USB device FS only */ /* USB device FS only */
#define cpu_has_udpfs() (!cpu_is_at91sam9rl()) #define cpu_has_udpfs() (!(cpu_is_at91sam9rl() \
|| cpu_is_at91sam9g45()))
static LIST_HEAD(clocks); static LIST_HEAD(clocks);
static DEFINE_SPINLOCK(clk_lock); static DEFINE_SPINLOCK(clk_lock);
...@@ -133,6 +138,13 @@ static void pmc_uckr_mode(struct clk *clk, int is_on) ...@@ -133,6 +138,13 @@ static void pmc_uckr_mode(struct clk *clk, int is_on)
{ {
unsigned int uckr = at91_sys_read(AT91_CKGR_UCKR); unsigned int uckr = at91_sys_read(AT91_CKGR_UCKR);
if (cpu_is_at91sam9g45()) {
if (is_on)
uckr |= AT91_PMC_BIASEN;
else
uckr &= ~AT91_PMC_BIASEN;
}
if (is_on) { if (is_on) {
is_on = AT91_PMC_LOCKU; is_on = AT91_PMC_LOCKU;
at91_sys_write(AT91_CKGR_UCKR, uckr | clk->pmc_mask); at91_sys_write(AT91_CKGR_UCKR, uckr | clk->pmc_mask);
...@@ -310,6 +322,7 @@ long clk_round_rate(struct clk *clk, unsigned long rate) ...@@ -310,6 +322,7 @@ long clk_round_rate(struct clk *clk, unsigned long rate)
unsigned long flags; unsigned long flags;
unsigned prescale; unsigned prescale;
unsigned long actual; unsigned long actual;
unsigned long prev = ULONG_MAX;
if (!clk_is_programmable(clk)) if (!clk_is_programmable(clk))
return -EINVAL; return -EINVAL;
...@@ -317,8 +330,16 @@ long clk_round_rate(struct clk *clk, unsigned long rate) ...@@ -317,8 +330,16 @@ long clk_round_rate(struct clk *clk, unsigned long rate)
actual = clk->parent->rate_hz; actual = clk->parent->rate_hz;
for (prescale = 0; prescale < 7; prescale++) { for (prescale = 0; prescale < 7; prescale++) {
if (actual && actual <= rate) if (actual > rate)
prev = actual;
if (actual && actual <= rate) {
if ((prev - rate) < (rate - actual)) {
actual = prev;
prescale--;
}
break; break;
}
actual >>= 1; actual >>= 1;
} }
...@@ -373,6 +394,10 @@ int clk_set_parent(struct clk *clk, struct clk *parent) ...@@ -373,6 +394,10 @@ int clk_set_parent(struct clk *clk, struct clk *parent)
return -EBUSY; return -EBUSY;
if (!clk_is_primary(parent) || !clk_is_programmable(clk)) if (!clk_is_primary(parent) || !clk_is_programmable(clk))
return -EINVAL; return -EINVAL;
if (cpu_is_at91sam9rl() && parent->id == AT91_PMC_CSS_PLLB)
return -EINVAL;
spin_lock_irqsave(&clk_lock, flags); spin_lock_irqsave(&clk_lock, flags);
clk->rate_hz = parent->rate_hz; clk->rate_hz = parent->rate_hz;
...@@ -637,6 +662,7 @@ int __init at91_clock_init(unsigned long main_clock) ...@@ -637,6 +662,7 @@ int __init at91_clock_init(unsigned long main_clock)
{ {
unsigned tmp, freq, mckr; unsigned tmp, freq, mckr;
int i; int i;
int pll_overclock = false;
/* /*
* When the bootloader initialized the main oscillator correctly, * When the bootloader initialized the main oscillator correctly,
...@@ -654,12 +680,25 @@ int __init at91_clock_init(unsigned long main_clock) ...@@ -654,12 +680,25 @@ int __init at91_clock_init(unsigned long main_clock)
/* report if PLLA is more than mildly overclocked */ /* report if PLLA is more than mildly overclocked */
plla.rate_hz = at91_pll_rate(&plla, main_clock, at91_sys_read(AT91_CKGR_PLLAR)); plla.rate_hz = at91_pll_rate(&plla, main_clock, at91_sys_read(AT91_CKGR_PLLAR));
if ((!cpu_has_800M_plla() && plla.rate_hz > 209000000) if (cpu_has_300M_plla()) {
|| (cpu_has_800M_plla() && plla.rate_hz > 800000000)) if (plla.rate_hz > 300000000)
pll_overclock = true;
} else if (cpu_has_800M_plla()) {
if (plla.rate_hz > 800000000)
pll_overclock = true;
} else {
if (plla.rate_hz > 209000000)
pll_overclock = true;
}
if (pll_overclock)
pr_info("Clocks: PLLA overclocked, %ld MHz\n", plla.rate_hz / 1000000); pr_info("Clocks: PLLA overclocked, %ld MHz\n", plla.rate_hz / 1000000);
if (cpu_is_at91sam9g45()) {
mckr = at91_sys_read(AT91_PMC_MCKR);
plla.rate_hz /= (1 << ((mckr & AT91_PMC_PLLADIV2) >> 12)); /* plla divisor by 2 */
}
if (cpu_has_upll() && !cpu_has_pllb()) { if (!cpu_has_pllb() && cpu_has_upll()) {
/* setup UTMI clock as the fourth primary clock /* setup UTMI clock as the fourth primary clock
* (instead of pllb) */ * (instead of pllb) */
utmi_clk.type |= CLK_TYPE_PRIMARY; utmi_clk.type |= CLK_TYPE_PRIMARY;
...@@ -701,6 +740,9 @@ int __init at91_clock_init(unsigned long main_clock) ...@@ -701,6 +740,9 @@ int __init at91_clock_init(unsigned long main_clock)
freq / ((mckr & AT91_PMC_MDIV) >> 7) : freq; /* mdiv ; (x >> 7) = ((x >> 8) * 2) */ freq / ((mckr & AT91_PMC_MDIV) >> 7) : freq; /* mdiv ; (x >> 7) = ((x >> 8) * 2) */
if (mckr & AT91_PMC_PDIV) if (mckr & AT91_PMC_PDIV)
freq /= 2; /* processor clock division */ freq /= 2; /* processor clock division */
} else if (cpu_is_at91sam9g45()) {
mck.rate_hz = (mckr & AT91_PMC_MDIV) == AT91SAM9_PMC_MDIV_3 ?
freq / 3 : freq / (1 << ((mckr & AT91_PMC_MDIV) >> 8)); /* mdiv */
} else { } else {
mck.rate_hz = freq / (1 << ((mckr & AT91_PMC_MDIV) >> 8)); /* mdiv */ mck.rate_hz = freq / (1 << ((mckr & AT91_PMC_MDIV) >> 8)); /* mdiv */
} }
......
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