diff mbox

[U-Boot,1/2] at91: move a/a/c/arm926ejs/at91/clock.c -> a/a/lib/at91

Message ID 1307489278-96804-2-git-send-email-andreas.devel@googlemail.com
State RFC
Delegated to: Reinhard Meyer
Headers show

Commit Message

Andreas Bießmann June 7, 2011, 11:27 p.m. UTC
The at91/clock.c is copied from linux kernel and has support for both
arm920t and arm926ejs core devices. Therefore this patch moves this
generic at91/clock.c to a new place at arch/arm/lib/at91 to be used from
at91 family devices.

We build a new libat91-common.o to provide the required symbols to both
cpu types.

Signed-off-by: Andreas Bießmann <andreas.devel@googlemail.com>
---
 Makefile                             |    3 +
 arch/arm/cpu/arm926ejs/at91/Makefile |    1 -
 arch/arm/cpu/arm926ejs/at91/clock.c  |  215 ----------------------------------
 arch/arm/lib/at91/Makefile           |   45 +++++++
 arch/arm/lib/at91/clock.c            |  215 ++++++++++++++++++++++++++++++++++
 5 files changed, 263 insertions(+), 216 deletions(-)
 delete mode 100644 arch/arm/cpu/arm926ejs/at91/clock.c
 create mode 100644 arch/arm/lib/at91/Makefile
 create mode 100644 arch/arm/lib/at91/clock.c

Comments

Reinhard Meyer June 9, 2011, 9:38 a.m. UTC | #1
Dear Andreas Bießmann,
> The at91/clock.c is copied from linux kernel and has support for both
> arm920t and arm926ejs core devices. Therefore this patch moves this
> generic at91/clock.c to a new place at arch/arm/lib/at91 to be used from
> at91 family devices.
> 
> We build a new libat91-common.o to provide the required symbols to both
> cpu types.
> 
> Signed-off-by: Andreas Bießmann <andreas.devel@googlemail.com>
> ---
>  Makefile                             |    3 +
>  arch/arm/cpu/arm926ejs/at91/Makefile |    1 -
>  arch/arm/cpu/arm926ejs/at91/clock.c  |  215 ----------------------------------
>  arch/arm/lib/at91/Makefile           |   45 +++++++
>  arch/arm/lib/at91/clock.c            |  215 ++++++++++++++++++++++++++++++++++
>  5 files changed, 263 insertions(+), 216 deletions(-)
>  delete mode 100644 arch/arm/cpu/arm926ejs/at91/clock.c
>  create mode 100644 arch/arm/lib/at91/Makefile
>  create mode 100644 arch/arm/lib/at91/clock.c

Please use the "git-mv" command to move the file "clock.c".

On any account, we should discuss about opening a subdir lib scheme first.
Currently arch/arm/lib does not have any subdirectories.

I _do_ see the problems of the current ARM directory structure which makes
the ARM core a more important criteria than SoC family especially in the
Atmel context where the SoCs have identical peripheral blocks but can have
different ARM cores.

Since that change goes outside the "Atmel directory domain",
I consider this an RFC. Albert is already on CC.

Best Regards,
Reinhard
Andreas Bießmann June 9, 2011, 9:46 a.m. UTC | #2
Dear Reinhard Meyer,

Am 09.06.2011 11:38, schrieb Reinhard Meyer:
> Dear Andreas Bießmann,
>> The at91/clock.c is copied from linux kernel and has support for both
>> arm920t and arm926ejs core devices. Therefore this patch moves this
>> generic at91/clock.c to a new place at arch/arm/lib/at91 to be used from
>> at91 family devices.
>>
>> We build a new libat91-common.o to provide the required symbols to both
>> cpu types.
>>
>> Signed-off-by: Andreas Bießmann <andreas.devel@googlemail.com>
>> ---
>>  Makefile                             |    3 +
>>  arch/arm/cpu/arm926ejs/at91/Makefile |    1 -
>>  arch/arm/cpu/arm926ejs/at91/clock.c  |  215 ----------------------------------
>>  arch/arm/lib/at91/Makefile           |   45 +++++++
>>  arch/arm/lib/at91/clock.c            |  215 ++++++++++++++++++++++++++++++++++
>>  5 files changed, 263 insertions(+), 216 deletions(-)
>>  delete mode 100644 arch/arm/cpu/arm926ejs/at91/clock.c
>>  create mode 100644 arch/arm/lib/at91/Makefile
>>  create mode 100644 arch/arm/lib/at91/clock.c
> 
> Please use the "git-mv" command to move the file "clock.c".

I did it that way ... but I think there was another trick for
git-format-patch ... will have a look for that

> On any account, we should discuss about opening a subdir lib scheme first.
> Currently arch/arm/lib does not have any subdirectories.

Well as described in cover letter this was discussed before. See the
other discussion on gmane:
http://mid.gmane.org/BANLkTimn29vmAYGb5csMdCYS-xx6zD_6Zw@mail.gmail.com

I think this solution was preferred by Albert too. But I'm open for
another solution, so lets see what possibilities you have.

> I _do_ see the problems of the current ARM directory structure which makes
> the ARM core a more important criteria than SoC family especially in the
> Atmel context where the SoCs have identical peripheral blocks but can have
> different ARM cores.

correct

regards

Andreas Bießmann
Reinhard Meyer June 9, 2011, 10:23 a.m. UTC | #3
Dear Andreas Bießmann,
> Dear Reinhard Meyer,
> 
> Am 09.06.2011 11:38, schrieb Reinhard Meyer:
>> Dear Andreas Bießmann,
>>> The at91/clock.c is copied from linux kernel and has support for both
>>> arm920t and arm926ejs core devices. Therefore this patch moves this
>>> generic at91/clock.c to a new place at arch/arm/lib/at91 to be used from
>>> at91 family devices.
>>>
>>> We build a new libat91-common.o to provide the required symbols to both
>>> cpu types.
>>>
>>> Signed-off-by: Andreas Bießmann <andreas.devel@googlemail.com>
>>> ---
>>>  Makefile                             |    3 +
>>>  arch/arm/cpu/arm926ejs/at91/Makefile |    1 -
>>>  arch/arm/cpu/arm926ejs/at91/clock.c  |  215 ----------------------------------
>>>  arch/arm/lib/at91/Makefile           |   45 +++++++
>>>  arch/arm/lib/at91/clock.c            |  215 ++++++++++++++++++++++++++++++++++
>>>  5 files changed, 263 insertions(+), 216 deletions(-)
>>>  delete mode 100644 arch/arm/cpu/arm926ejs/at91/clock.c
>>>  create mode 100644 arch/arm/lib/at91/Makefile
>>>  create mode 100644 arch/arm/lib/at91/clock.c
>>
>> Please use the "git-mv" command to move the file "clock.c".
> 
> I did it that way ... but I think there was another trick for
> git-format-patch ... will have a look for that
> 
>> On any account, we should discuss about opening a subdir lib scheme first.
>> Currently arch/arm/lib does not have any subdirectories.
> 
> Well as described in cover letter this was discussed before. See the
> other discussion on gmane:
> http://mid.gmane.org/BANLkTimn29vmAYGb5csMdCYS-xx6zD_6Zw@mail.gmail.com

Sorry, I had forgotten already about that discussion, it was even
marked "important" in my e-Mail program... and I wanted to comment then
but was too busy with other things ;)

> 
> I think this solution was preferred by Albert too. But I'm open for
> another solution, so lets see what possibilities you have.

What other files do you see (in the future) to be in arch/arm/lib/at91?
For just clock.c, it seems like an overkill... Can we (on the long run)
move _all_ files from the at91 subdirs there, like *_devices.c, reset.c,
timer.c, at91sam9_watchdog.c? Then it would make sense to me...

if YES, ignore the following ideas:
{
When I look at the #ifdef's near the end of clock.c I get an uneasy feeling;)

With more Atmel SoCs coming up which must have additional clock domains
(e.g. with 2x EMAC, CAN, etc) I can imagine this file to become more and
more ugly.

I currently have rough ideas to add to the discussion:

1. move the SoC specific clock stuff into <SoC>_devices.c - even if that
means some duplication (the get_*_clk_rate() functions could possibly be
inlined and put into the header file ?)

2. introduce <SoC>_clock.c (if one does not like the clock stuff in "devices")

3. similar to drivers/watchdog we could have drivers/clock and put
at91sam9_clock.c there. On that account, the hardware specific timer
functions of any new timer scheme could be placed there as well.
}

> 
>> I _do_ see the problems of the current ARM directory structure which makes
>> the ARM core a more important criteria than SoC family especially in the
>> Atmel context where the SoCs have identical peripheral blocks but can have
>> different ARM cores.
> 
> correct

Best Regards,
Reinhard
Andreas Bießmann June 9, 2011, 11:07 a.m. UTC | #4
Dear Reinhard Meyer,

Am 09.06.2011 12:23, schrieb Reinhard Meyer:
> Dear Andreas Bießmann,
>> Dear Reinhard Meyer,
>>
>> Am 09.06.2011 11:38, schrieb Reinhard Meyer:
>>> Dear Andreas Bießmann,

<snip patch header>

>>> On any account, we should discuss about opening a subdir lib scheme first.
>>> Currently arch/arm/lib does not have any subdirectories.
>>
>> Well as described in cover letter this was discussed before. See the
>> other discussion on gmane:
>> http://mid.gmane.org/BANLkTimn29vmAYGb5csMdCYS-xx6zD_6Zw@mail.gmail.com
> 
> Sorry, I had forgotten already about that discussion, it was even
> marked "important" in my e-Mail program... and I wanted to comment then
> but was too busy with other things ;)

no problem, we can do it here ...

>>
>> I think this solution was preferred by Albert too. But I'm open for
>> another solution, so lets see what possibilities you have.
> 
> What other files do you see (in the future) to be in arch/arm/lib/at91?

Well, if I look at the arm920t/at91 I do see following files:

 timer.c
  timer could not be shared between arm920t and arm926ejs at91 SoC
  cause at91rm9200 do not have a 32 bit PITC!
 reset.c
  reset could not be shared cause at91rm9200 has an input only NRST pin.
  We do need the watchdog workaround to do the reset

A future commit will add cpu.c as in arm926ejs but with less features
(just arch_cpu_init() for at91_clock_init()).

If I look at ar926ejs/at91 files I do see additionally these files:

 eflash.c
  new feature in some sam9XE devices, not related to at91rm9200
 led.c
  I do hate this ... we have dozen of led API ... but it is really
  helpful for assembler (lowlevel_init) stage debugging without JTAG,
  could be useful for at91rm9200
 <SoC>_devices.c
  ...

> For just clock.c, it seems like an overkill...

You are right here ..

> Can we (on the long run)
> move _all_ files from the at91 subdirs there, like *_devices.c, reset.c,
> timer.c, at91sam9_watchdog.c? Then it would make sense to me...

This makes no sense to me in the current structure. If we have some
arch/arm/mach-XXX like the linux kernel it would make sense to make the
effort to unify those files.

The main aim for this patch was to not duplicate the clock.c from
arm926ejs to arm920t (remember, the file is already prepared for
at92rm9200 cause it is a copy from linux source tree).

> if YES, ignore the following ideas:
> {
> When I look at the #ifdef's near the end of clock.c I get an uneasy feeling;)

I see ... Therefore it would be better to rewrite the clock.c in
arm926ejs (remove at91rm9200 support) and arm920t (do not add support
for other SoC).

> With more Atmel SoCs coming up which must have additional clock domains
> (e.g. with 2x EMAC, CAN, etc) I can imagine this file to become more and
> more ugly.

That may be true.

> I currently have rough ideas to add to the discussion:
> 
> 1. move the SoC specific clock stuff into <SoC>_devices.c - even if that
> means some duplication (the get_*_clk_rate() functions could possibly be
> inlined and put into the header file ?)

We should inline the get_*_clk_rate() functions in header in any case.
That is a good idea.

> 2. introduce <SoC>_clock.c (if one does not like the clock stuff in "devices")

But this needs also an rewrite of the clock.c driver (which was copied
from linux kernel).

> 3. similar to drivers/watchdog we could have drivers/clock and put
> at91sam9_clock.c there. On that account, the hardware specific timer
> functions of any new timer scheme could be placed there as well.

That could be another solution.

The conclusion is that lib/at91 is not the correct way, cause there are
only a few commonalities between arm920t and arm926ejs. Therefore we have
 a) include new clock.c in arm920t
 b) move current clock.c to driver/clock (or similar)

regards

Andreas Bießmann
diff mbox

Patch

diff --git a/Makefile b/Makefile
index 76124e5..64e47c7 100644
--- a/Makefile
+++ b/Makefile
@@ -189,6 +189,9 @@  LIBS += $(shell if [ -f board/$(VENDOR)/common/Makefile ]; then echo \
 LIBS += $(CPUDIR)/lib$(CPU).o
 ifdef SOC
 LIBS += $(CPUDIR)/$(SOC)/lib$(SOC).o
+ifeq ($(SOC),at91)
+LIBS += arch/arm/lib/at91/libat91-common.o
+endif
 endif
 ifeq ($(CPU),ixp)
 LIBS += arch/arm/cpu/ixp/npe/libnpe.o
diff --git a/arch/arm/cpu/arm926ejs/at91/Makefile b/arch/arm/cpu/arm926ejs/at91/Makefile
index f333753..ee66b3d 100644
--- a/arch/arm/cpu/arm926ejs/at91/Makefile
+++ b/arch/arm/cpu/arm926ejs/at91/Makefile
@@ -37,7 +37,6 @@  COBJS-$(CONFIG_AT91SAM9M10G45)	+= at91sam9m10g45_devices.o
 COBJS-$(CONFIG_AT91SAM9G45)	+= at91sam9m10g45_devices.o
 COBJS-$(CONFIG_AT91_EFLASH)	+= eflash.o
 COBJS-$(CONFIG_AT91_LED)	+= led.o
-COBJS-y += clock.o
 COBJS-y += cpu.o
 COBJS-y	+= reset.o
 COBJS-y	+= timer.o
diff --git a/arch/arm/cpu/arm926ejs/at91/clock.c b/arch/arm/cpu/arm926ejs/at91/clock.c
deleted file mode 100644
index 608af2c..0000000
--- a/arch/arm/cpu/arm926ejs/at91/clock.c
+++ /dev/null
@@ -1,215 +0,0 @@ 
-/*
- * [origin: Linux kernel linux/arch/arm/mach-at91/clock.c]
- *
- * Copyright (C) 2005 David Brownell
- * Copyright (C) 2005 Ivan Kokshaysky
- * Copyright (C) 2009 Jean-Christophe PLAGNIOL-VILLARD <plagnioj@jcrosoft.com>
- *
- * 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.
- */
-
-#include <common.h>
-#include <asm/io.h>
-#include <asm/arch/hardware.h>
-#include <asm/arch/at91_pmc.h>
-#include <asm/arch/clk.h>
-
-#if !defined(CONFIG_AT91FAMILY)
-# error You need to define CONFIG_AT91FAMILY in your board config!
-#endif
-
-DECLARE_GLOBAL_DATA_PTR;
-
-unsigned long get_cpu_clk_rate(void)
-{
-	return gd->cpu_clk_rate_hz;
-}
-
-unsigned long get_main_clk_rate(void)
-{
-	return gd->main_clk_rate_hz;
-}
-
-unsigned long get_mck_clk_rate(void)
-{
-	return gd->mck_rate_hz;
-}
-
-unsigned long get_plla_clk_rate(void)
-{
-	return gd->plla_rate_hz;
-}
-
-unsigned long get_pllb_clk_rate(void)
-{
-	return gd->pllb_rate_hz;
-}
-
-u32 get_pllb_init(void)
-{
-	return gd->at91_pllb_usb_init;
-}
-
-static unsigned long at91_css_to_rate(unsigned long css)
-{
-	switch (css) {
-	case AT91_PMC_MCKR_CSS_SLOW:
-		return CONFIG_SYS_AT91_SLOW_CLOCK;
-	case AT91_PMC_MCKR_CSS_MAIN:
-		return gd->main_clk_rate_hz;
-	case AT91_PMC_MCKR_CSS_PLLA:
-		return gd->plla_rate_hz;
-	case AT91_PMC_MCKR_CSS_PLLB:
-		return gd->pllb_rate_hz;
-	}
-
-	return 0;
-}
-
-#ifdef CONFIG_USB_ATMEL
-static unsigned at91_pll_calc(unsigned main_freq, unsigned out_freq)
-{
-	unsigned i, div = 0, mul = 0, diff = 1 << 30;
-	unsigned ret = (out_freq > 155000000) ? 0xbe00 : 0x3e00;
-
-	/* PLL output max 240 MHz (or 180 MHz per errata) */
-	if (out_freq > 240000000)
-		goto fail;
-
-	for (i = 1; i < 256; i++) {
-		int diff1;
-		unsigned input, mul1;
-
-		/*
-		 * PLL input between 1MHz and 32MHz per spec, but lower
-		 * frequences seem necessary in some cases so allow 100K.
-		 * Warning: some newer products need 2MHz min.
-		 */
-		input = main_freq / i;
-#if defined(CONFIG_AT91SAM9G20)
-		if (input < 2000000)
-			continue;
-#endif
-		if (input < 100000)
-			continue;
-		if (input > 32000000)
-			continue;
-
-		mul1 = out_freq / input;
-#if defined(CONFIG_AT91SAM9G20)
-		if (mul > 63)
-			continue;
-#endif
-		if (mul1 > 2048)
-			continue;
-		if (mul1 < 2)
-			goto fail;
-
-		diff1 = out_freq - input * mul1;
-		if (diff1 < 0)
-			diff1 = -diff1;
-		if (diff > diff1) {
-			diff = diff1;
-			div = i;
-			mul = mul1;
-			if (diff == 0)
-				break;
-		}
-	}
-	if (i == 256 && diff > (out_freq >> 5))
-		goto fail;
-	return ret | ((mul - 1) << 16) | div;
-fail:
-	return 0;
-}
-#endif
-
-static u32 at91_pll_rate(u32 freq, u32 reg)
-{
-	unsigned mul, div;
-
-	div = reg & 0xff;
-	mul = (reg >> 16) & 0x7ff;
-	if (div && mul) {
-		freq /= div;
-		freq *= mul + 1;
-	} else
-		freq = 0;
-
-	return freq;
-}
-
-int at91_clock_init(unsigned long main_clock)
-{
-	unsigned freq, mckr;
-	at91_pmc_t *pmc = (at91_pmc_t *) ATMEL_BASE_PMC;
-#ifndef CONFIG_SYS_AT91_MAIN_CLOCK
-	unsigned tmp;
-	/*
-	 * When the bootloader initialized the main oscillator correctly,
-	 * there's no problem using the cycle counter.  But if it didn't,
-	 * or when using oscillator bypass mode, we must be told the speed
-	 * of the main clock.
-	 */
-	if (!main_clock) {
-		do {
-			tmp = readl(&pmc->mcfr);
-		} while (!(tmp & AT91_PMC_MCFR_MAINRDY));
-		tmp &= AT91_PMC_MCFR_MAINF_MASK;
-		main_clock = tmp * (CONFIG_SYS_AT91_SLOW_CLOCK / 16);
-	}
-#endif
-	gd->main_clk_rate_hz = main_clock;
-
-	/* report if PLLA is more than mildly overclocked */
-	gd->plla_rate_hz = at91_pll_rate(main_clock, readl(&pmc->pllar));
-
-#ifdef CONFIG_USB_ATMEL
-	/*
-	 * USB clock init:  choose 48 MHz PLLB value,
-	 * disable 48MHz clock during usb peripheral suspend.
-	 *
-	 * REVISIT:  assumes MCK doesn't derive from PLLB!
-	 */
-	gd->at91_pllb_usb_init = at91_pll_calc(main_clock, 48000000 * 2) |
-			     AT91_PMC_PLLBR_USBDIV_2;
-	gd->pllb_rate_hz = at91_pll_rate(main_clock, gd->at91_pllb_usb_init);
-#endif
-
-	/*
-	 * MCK and CPU derive from one of those primary clocks.
-	 * For now, assume this parentage won't change.
-	 */
-	mckr = readl(&pmc->mckr);
-#if defined(CONFIG_AT91SAM9G45) || defined(CONFIG_AT91SAM9M10G45)
-	/* plla divisor by 2 */
-	gd->plla_rate_hz /= (1 << ((mckr & 1 << 12) >> 12));
-#endif
-	gd->mck_rate_hz = at91_css_to_rate(mckr & AT91_PMC_MCKR_CSS_MASK);
-	freq = gd->mck_rate_hz;
-
-	freq /= (1 << ((mckr & AT91_PMC_MCKR_PRES_MASK) >> 2));	/* prescale */
-#if defined(CONFIG_AT91RM9200)
-	/* mdiv */
-	gd->mck_rate_hz = freq / (1 + ((mckr & AT91_PMC_MCKR_MDIV_MASK) >> 8));
-#elif defined(CONFIG_AT91SAM9G20)
-	/* mdiv ; (x >> 7) = ((x >> 8) * 2) */
-	gd->mck_rate_hz = (mckr & AT91_PMC_MCKR_MDIV_MASK) ?
-		freq / ((mckr & AT91_PMC_MCKR_MDIV_MASK) >> 7) : freq;
-	if (mckr & AT91_PMC_MCKR_MDIV_MASK)
-		freq /= 2;			/* processor clock division */
-#elif defined(CONFIG_AT91SAM9G45) || defined(CONFIG_AT91SAM9M10G45)
-	gd->mck_rate_hz = (mckr & AT91_PMC_MCKR_MDIV_MASK) ==
-		(AT91_PMC_MCKR_MDIV_2 | AT91_PMC_MCKR_MDIV_4)
-		? freq / 3
-		: freq / (1 << ((mckr & AT91_PMC_MCKR_MDIV_MASK) >> 8));
-#else
-	gd->mck_rate_hz = freq / (1 << ((mckr & AT91_PMC_MCKR_MDIV_MASK) >> 8));
-#endif
-	gd->cpu_clk_rate_hz = freq;
-
-	return 0;
-}
diff --git a/arch/arm/lib/at91/Makefile b/arch/arm/lib/at91/Makefile
new file mode 100644
index 0000000..a264630
--- /dev/null
+++ b/arch/arm/lib/at91/Makefile
@@ -0,0 +1,45 @@ 
+#
+# (C) Copyright 2000-2008
+# Wolfgang Denk, DENX Software Engineering, wd@denx.de.
+#
+# See file CREDITS for list of people who contributed to this
+# project.
+#
+# 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., 59 Temple Place, Suite 330, Boston,
+# MA 02111-1307 USA
+#
+
+include $(TOPDIR)/config.mk
+
+LIB	= $(obj)libat91-common.o
+
+COBJS-y	+= clock.o
+
+SRCS    := $(SOBJS-y:.o=.S) $(COBJS-y:.o=.c)
+OBJS    := $(addprefix $(obj),$(SOBJS-y) $(COBJS-y))
+
+all:	$(obj).depend $(LIB)
+
+$(LIB):	$(OBJS)
+	$(call cmd_link_o_target, $(OBJS))
+
+#########################################################################
+
+# defines $(obj).depend target
+include $(SRCTREE)/rules.mk
+
+sinclude $(obj).depend
+
+#########################################################################
diff --git a/arch/arm/lib/at91/clock.c b/arch/arm/lib/at91/clock.c
new file mode 100644
index 0000000..608af2c
--- /dev/null
+++ b/arch/arm/lib/at91/clock.c
@@ -0,0 +1,215 @@ 
+/*
+ * [origin: Linux kernel linux/arch/arm/mach-at91/clock.c]
+ *
+ * Copyright (C) 2005 David Brownell
+ * Copyright (C) 2005 Ivan Kokshaysky
+ * Copyright (C) 2009 Jean-Christophe PLAGNIOL-VILLARD <plagnioj@jcrosoft.com>
+ *
+ * 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.
+ */
+
+#include <common.h>
+#include <asm/io.h>
+#include <asm/arch/hardware.h>
+#include <asm/arch/at91_pmc.h>
+#include <asm/arch/clk.h>
+
+#if !defined(CONFIG_AT91FAMILY)
+# error You need to define CONFIG_AT91FAMILY in your board config!
+#endif
+
+DECLARE_GLOBAL_DATA_PTR;
+
+unsigned long get_cpu_clk_rate(void)
+{
+	return gd->cpu_clk_rate_hz;
+}
+
+unsigned long get_main_clk_rate(void)
+{
+	return gd->main_clk_rate_hz;
+}
+
+unsigned long get_mck_clk_rate(void)
+{
+	return gd->mck_rate_hz;
+}
+
+unsigned long get_plla_clk_rate(void)
+{
+	return gd->plla_rate_hz;
+}
+
+unsigned long get_pllb_clk_rate(void)
+{
+	return gd->pllb_rate_hz;
+}
+
+u32 get_pllb_init(void)
+{
+	return gd->at91_pllb_usb_init;
+}
+
+static unsigned long at91_css_to_rate(unsigned long css)
+{
+	switch (css) {
+	case AT91_PMC_MCKR_CSS_SLOW:
+		return CONFIG_SYS_AT91_SLOW_CLOCK;
+	case AT91_PMC_MCKR_CSS_MAIN:
+		return gd->main_clk_rate_hz;
+	case AT91_PMC_MCKR_CSS_PLLA:
+		return gd->plla_rate_hz;
+	case AT91_PMC_MCKR_CSS_PLLB:
+		return gd->pllb_rate_hz;
+	}
+
+	return 0;
+}
+
+#ifdef CONFIG_USB_ATMEL
+static unsigned at91_pll_calc(unsigned main_freq, unsigned out_freq)
+{
+	unsigned i, div = 0, mul = 0, diff = 1 << 30;
+	unsigned ret = (out_freq > 155000000) ? 0xbe00 : 0x3e00;
+
+	/* PLL output max 240 MHz (or 180 MHz per errata) */
+	if (out_freq > 240000000)
+		goto fail;
+
+	for (i = 1; i < 256; i++) {
+		int diff1;
+		unsigned input, mul1;
+
+		/*
+		 * PLL input between 1MHz and 32MHz per spec, but lower
+		 * frequences seem necessary in some cases so allow 100K.
+		 * Warning: some newer products need 2MHz min.
+		 */
+		input = main_freq / i;
+#if defined(CONFIG_AT91SAM9G20)
+		if (input < 2000000)
+			continue;
+#endif
+		if (input < 100000)
+			continue;
+		if (input > 32000000)
+			continue;
+
+		mul1 = out_freq / input;
+#if defined(CONFIG_AT91SAM9G20)
+		if (mul > 63)
+			continue;
+#endif
+		if (mul1 > 2048)
+			continue;
+		if (mul1 < 2)
+			goto fail;
+
+		diff1 = out_freq - input * mul1;
+		if (diff1 < 0)
+			diff1 = -diff1;
+		if (diff > diff1) {
+			diff = diff1;
+			div = i;
+			mul = mul1;
+			if (diff == 0)
+				break;
+		}
+	}
+	if (i == 256 && diff > (out_freq >> 5))
+		goto fail;
+	return ret | ((mul - 1) << 16) | div;
+fail:
+	return 0;
+}
+#endif
+
+static u32 at91_pll_rate(u32 freq, u32 reg)
+{
+	unsigned mul, div;
+
+	div = reg & 0xff;
+	mul = (reg >> 16) & 0x7ff;
+	if (div && mul) {
+		freq /= div;
+		freq *= mul + 1;
+	} else
+		freq = 0;
+
+	return freq;
+}
+
+int at91_clock_init(unsigned long main_clock)
+{
+	unsigned freq, mckr;
+	at91_pmc_t *pmc = (at91_pmc_t *) ATMEL_BASE_PMC;
+#ifndef CONFIG_SYS_AT91_MAIN_CLOCK
+	unsigned tmp;
+	/*
+	 * When the bootloader initialized the main oscillator correctly,
+	 * there's no problem using the cycle counter.  But if it didn't,
+	 * or when using oscillator bypass mode, we must be told the speed
+	 * of the main clock.
+	 */
+	if (!main_clock) {
+		do {
+			tmp = readl(&pmc->mcfr);
+		} while (!(tmp & AT91_PMC_MCFR_MAINRDY));
+		tmp &= AT91_PMC_MCFR_MAINF_MASK;
+		main_clock = tmp * (CONFIG_SYS_AT91_SLOW_CLOCK / 16);
+	}
+#endif
+	gd->main_clk_rate_hz = main_clock;
+
+	/* report if PLLA is more than mildly overclocked */
+	gd->plla_rate_hz = at91_pll_rate(main_clock, readl(&pmc->pllar));
+
+#ifdef CONFIG_USB_ATMEL
+	/*
+	 * USB clock init:  choose 48 MHz PLLB value,
+	 * disable 48MHz clock during usb peripheral suspend.
+	 *
+	 * REVISIT:  assumes MCK doesn't derive from PLLB!
+	 */
+	gd->at91_pllb_usb_init = at91_pll_calc(main_clock, 48000000 * 2) |
+			     AT91_PMC_PLLBR_USBDIV_2;
+	gd->pllb_rate_hz = at91_pll_rate(main_clock, gd->at91_pllb_usb_init);
+#endif
+
+	/*
+	 * MCK and CPU derive from one of those primary clocks.
+	 * For now, assume this parentage won't change.
+	 */
+	mckr = readl(&pmc->mckr);
+#if defined(CONFIG_AT91SAM9G45) || defined(CONFIG_AT91SAM9M10G45)
+	/* plla divisor by 2 */
+	gd->plla_rate_hz /= (1 << ((mckr & 1 << 12) >> 12));
+#endif
+	gd->mck_rate_hz = at91_css_to_rate(mckr & AT91_PMC_MCKR_CSS_MASK);
+	freq = gd->mck_rate_hz;
+
+	freq /= (1 << ((mckr & AT91_PMC_MCKR_PRES_MASK) >> 2));	/* prescale */
+#if defined(CONFIG_AT91RM9200)
+	/* mdiv */
+	gd->mck_rate_hz = freq / (1 + ((mckr & AT91_PMC_MCKR_MDIV_MASK) >> 8));
+#elif defined(CONFIG_AT91SAM9G20)
+	/* mdiv ; (x >> 7) = ((x >> 8) * 2) */
+	gd->mck_rate_hz = (mckr & AT91_PMC_MCKR_MDIV_MASK) ?
+		freq / ((mckr & AT91_PMC_MCKR_MDIV_MASK) >> 7) : freq;
+	if (mckr & AT91_PMC_MCKR_MDIV_MASK)
+		freq /= 2;			/* processor clock division */
+#elif defined(CONFIG_AT91SAM9G45) || defined(CONFIG_AT91SAM9M10G45)
+	gd->mck_rate_hz = (mckr & AT91_PMC_MCKR_MDIV_MASK) ==
+		(AT91_PMC_MCKR_MDIV_2 | AT91_PMC_MCKR_MDIV_4)
+		? freq / 3
+		: freq / (1 << ((mckr & AT91_PMC_MCKR_MDIV_MASK) >> 8));
+#else
+	gd->mck_rate_hz = freq / (1 << ((mckr & AT91_PMC_MCKR_MDIV_MASK) >> 8));
+#endif
+	gd->cpu_clk_rate_hz = freq;
+
+	return 0;
+}