Commit c73b9099 authored by Janusz Krzysztofik's avatar Janusz Krzysztofik Committed by Arnd Bergmann

ARM: OMAP1: clock: Convert to CCF

OMAP1 still uses its own implementation of standard clock API defined in
include/linux/clk.h.  Internals of that implementation are not visible
outside OMAP1 directory.  As a consequence, device drivers are not able to
register clocks potentially provided by peripheral devices.

Drop OMAP1 implementation of the clock API and enable common clock
framework.  Modify the remaining low level code to be compatible with
clock provider API and register the clocks with CCF.

Move initialisation of clocks to omap1_timer_init() to avoid memory
allocation issues at early setup phase from where omap1_init_early() is
called.  Register the clocks after initialization of clock I/O registers,
local clock pointers used by OMAP1 clock ops, and local .rate fields of
clocks with no local implementation of .recalc ops, so CCF structures are
populated with correct data during clock registration.  Instead of
enabling some of the registered clocks, flag them for CCF as critical.
Introduce .is_enabled op using code that verifies hardware status of clock
enablement, split out from implementation of .disable_unused op, so the
latter is actually called by CCF for not requested but hardware enabled
clocks.  Add .round_rate ops where missing so .set_rate ops are called by
CCF as expected.  Since CCF allows parallel execution of .enable/.disable
and .set_rate ops, protect registers shared among those groups of ops from
concurrent access with spinlocks.  Drop local debugfs support in favor of
that provided by CCF.

v2: flag tc2_ck as CLK_IS_CRITICAL (Aaro)
v3: rebase on top of soc/omap1-multiplatform-5.18,
  - drop no longer needed includes from arch/arm/mach-omap1/io.c
Signed-off-by: default avatarJanusz Krzysztofik <jmkrzyszt@gmail.com>
Acked-by: default avatarTony Lindgren <tony@atomide.com>
Signed-off-by: default avatarArnd Bergmann <arnd@arndb.de>
parent 6f9b5ed8
This diff is collapsed.
...@@ -11,12 +11,11 @@ ...@@ -11,12 +11,11 @@
#define __ARCH_ARM_MACH_OMAP1_CLOCK_H #define __ARCH_ARM_MACH_OMAP1_CLOCK_H
#include <linux/clk.h> #include <linux/clk.h>
#include <linux/list.h>
#include <linux/clkdev.h> #include <linux/clkdev.h>
#include <linux/clk-provider.h>
struct module; struct module;
struct clk; struct omap1_clk;
struct omap_clk { struct omap_clk {
u16 cpu; u16 cpu;
...@@ -29,7 +28,7 @@ struct omap_clk { ...@@ -29,7 +28,7 @@ struct omap_clk {
.lk = { \ .lk = { \
.dev_id = dev, \ .dev_id = dev, \
.con_id = con, \ .con_id = con, \
.clk = ck, \ .clk_hw = ck, \
}, \ }, \
} }
...@@ -40,10 +39,6 @@ struct omap_clk { ...@@ -40,10 +39,6 @@ struct omap_clk {
#define CK_16XX (1 << 3) /* 16xx, 17xx, 5912 */ #define CK_16XX (1 << 3) /* 16xx, 17xx, 5912 */
#define CK_1710 (1 << 4) /* 1710 extra for rate selection */ #define CK_1710 (1 << 4) /* 1710 extra for rate selection */
/* Temporary, needed during the common clock framework conversion */
#define __clk_get_name(clk) (clk->name)
/** /**
* struct clkops - some clock function pointers * struct clkops - some clock function pointers
* @enable: fn ptr that enables the current clock in hardware * @enable: fn ptr that enables the current clock in hardware
...@@ -51,8 +46,8 @@ struct omap_clk { ...@@ -51,8 +46,8 @@ struct omap_clk {
* @allow_idle: fn ptr that enables autoidle for the current clock in hardware * @allow_idle: fn ptr that enables autoidle for the current clock in hardware
*/ */
struct clkops { struct clkops {
int (*enable)(struct clk *); int (*enable)(struct omap1_clk *clk);
void (*disable)(struct clk *); void (*disable)(struct omap1_clk *clk);
}; };
/* /*
...@@ -65,13 +60,9 @@ struct clkops { ...@@ -65,13 +60,9 @@ struct clkops {
#define CLOCK_NO_IDLE_PARENT (1 << 2) #define CLOCK_NO_IDLE_PARENT (1 << 2)
/** /**
* struct clk - OMAP struct clk * struct omap1_clk - OMAP1 struct clk
* @node: list_head connecting this clock into the full clock list * @hw: struct clk_hw for common clock framework integration
* @ops: struct clkops * for this clock * @ops: struct clkops * for this clock
* @name: the name of the clock in the hardware (used in hwmod data and debug)
* @parent: pointer to this clock's parent struct clk
* @children: list_head connecting to the child clks' @sibling list_heads
* @sibling: list_head connecting this clk to its parent clk's @children
* @rate: current clock rate * @rate: current clock rate
* @enable_reg: register to write to enable the clock (see @enable_bit) * @enable_reg: register to write to enable the clock (see @enable_bit)
* @recalc: fn ptr that returns the clock's current rate * @recalc: fn ptr that returns the clock's current rate
...@@ -79,102 +70,65 @@ struct clkops { ...@@ -79,102 +70,65 @@ struct clkops {
* @round_rate: fn ptr that can round the clock's current rate * @round_rate: fn ptr that can round the clock's current rate
* @init: fn ptr to do clock-specific initialization * @init: fn ptr to do clock-specific initialization
* @enable_bit: bitshift to write to enable/disable the clock (see @enable_reg) * @enable_bit: bitshift to write to enable/disable the clock (see @enable_reg)
* @usecount: number of users that have requested this clock to be enabled
* @fixed_div: when > 0, this clock's rate is its parent's rate / @fixed_div * @fixed_div: when > 0, this clock's rate is its parent's rate / @fixed_div
* @flags: see "struct clk.flags possibilities" above * @flags: see "struct clk.flags possibilities" above
* @rate_offset: bitshift for rate selection bitfield (OMAP1 only) * @rate_offset: bitshift for rate selection bitfield (OMAP1 only)
*
* XXX @rate_offset should probably be removed and OMAP1
* clock code converted to use clksel.
*
* XXX @usecount is poorly named. It should be "enable_count" or
* something similar. "users" in the description refers to kernel
* code (core code or drivers) that have called clk_enable() and not
* yet called clk_disable(); the usecount of parent clocks is also
* incremented by the clock code when clk_enable() is called on child
* clocks and decremented by the clock code when clk_disable() is
* called on child clocks.
*
* XXX @usecount, @children, @sibling should be marked for
* internal use only.
*
* @children and @sibling are used to optimize parent-to-child clock
* tree traversals. (child-to-parent traversals use @parent.)
*
* XXX The notion of the clock's current rate probably needs to be
* separated from the clock's target rate.
*/ */
struct clk { struct omap1_clk {
struct list_head node; struct clk_hw hw;
const struct clkops *ops; const struct clkops *ops;
const char *name;
struct clk *parent;
struct list_head children;
struct list_head sibling; /* node for children */
unsigned long rate; unsigned long rate;
void __iomem *enable_reg; void __iomem *enable_reg;
unsigned long (*recalc)(struct clk *); unsigned long (*recalc)(struct omap1_clk *clk, unsigned long rate);
int (*set_rate)(struct clk *, unsigned long); int (*set_rate)(struct omap1_clk *clk, unsigned long rate,
long (*round_rate)(struct clk *, unsigned long); unsigned long p_rate);
void (*init)(struct clk *); long (*round_rate)(struct omap1_clk *clk, unsigned long rate,
unsigned long *p_rate);
int (*init)(struct omap1_clk *clk);
u8 enable_bit; u8 enable_bit;
s8 usecount;
u8 fixed_div; u8 fixed_div;
u8 flags; u8 flags;
u8 rate_offset; u8 rate_offset;
#if defined(CONFIG_PM_DEBUG) && defined(CONFIG_DEBUG_FS)
struct dentry *dent; /* For visible tree hierarchy */
#endif
}; };
#define to_omap1_clk(_hw) container_of(_hw, struct omap1_clk, hw)
extern void clk_preinit(struct clk *clk); void propagate_rate(struct omap1_clk *clk);
extern int clk_register(struct clk *clk); unsigned long followparent_recalc(struct omap1_clk *clk, unsigned long p_rate);
extern void clk_unregister(struct clk *clk); unsigned long omap_fixed_divisor_recalc(struct omap1_clk *clk, unsigned long p_rate);
extern void propagate_rate(struct clk *clk);
extern unsigned long followparent_recalc(struct clk *clk);
unsigned long omap_fixed_divisor_recalc(struct clk *clk);
extern const struct clkops clkops_null; extern struct omap1_clk dummy_ck;
extern struct clk dummy_ck;
int omap1_clk_init(void); int omap1_clk_init(void);
void omap1_clk_late_init(void); void omap1_clk_late_init(void);
extern int omap1_clk_enable(struct clk *clk); unsigned long omap1_ckctl_recalc(struct omap1_clk *clk, unsigned long p_rate);
extern void omap1_clk_disable(struct clk *clk); long omap1_round_sossi_rate(struct omap1_clk *clk, unsigned long rate, unsigned long *p_rate);
extern long omap1_clk_round_rate(struct clk *clk, unsigned long rate); int omap1_set_sossi_rate(struct omap1_clk *clk, unsigned long rate, unsigned long p_rate);
extern int omap1_clk_set_rate(struct clk *clk, unsigned long rate); unsigned long omap1_sossi_recalc(struct omap1_clk *clk, unsigned long p_rate);
extern unsigned long omap1_ckctl_recalc(struct clk *clk); unsigned long omap1_ckctl_recalc_dsp_domain(struct omap1_clk *clk, unsigned long p_rate);
extern int omap1_set_sossi_rate(struct clk *clk, unsigned long rate); int omap1_clk_set_rate_dsp_domain(struct omap1_clk *clk, unsigned long rate,
extern unsigned long omap1_sossi_recalc(struct clk *clk); unsigned long p_rate);
extern unsigned long omap1_ckctl_recalc_dsp_domain(struct clk *clk); long omap1_round_uart_rate(struct omap1_clk *clk, unsigned long rate, unsigned long *p_rate);
extern int omap1_clk_set_rate_dsp_domain(struct clk *clk, unsigned long rate); int omap1_set_uart_rate(struct omap1_clk *clk, unsigned long rate, unsigned long p_rate);
extern int omap1_set_uart_rate(struct clk *clk, unsigned long rate); unsigned long omap1_uart_recalc(struct omap1_clk *clk, unsigned long p_rate);
extern unsigned long omap1_uart_recalc(struct clk *clk); int omap1_set_ext_clk_rate(struct omap1_clk *clk, unsigned long rate, unsigned long p_rate);
extern int omap1_set_ext_clk_rate(struct clk *clk, unsigned long rate); long omap1_round_ext_clk_rate(struct omap1_clk *clk, unsigned long rate, unsigned long *p_rate);
extern long omap1_round_ext_clk_rate(struct clk *clk, unsigned long rate); int omap1_init_ext_clk(struct omap1_clk *clk);
extern void omap1_init_ext_clk(struct clk *clk); int omap1_select_table_rate(struct omap1_clk *clk, unsigned long rate, unsigned long p_rate);
extern int omap1_select_table_rate(struct clk *clk, unsigned long rate); long omap1_round_to_table_rate(struct omap1_clk *clk, unsigned long rate, unsigned long *p_rate);
extern long omap1_round_to_table_rate(struct clk *clk, unsigned long rate); int omap1_clk_set_rate_ckctl_arm(struct omap1_clk *clk, unsigned long rate, unsigned long p_rate);
extern int omap1_clk_set_rate_ckctl_arm(struct clk *clk, unsigned long rate); long omap1_clk_round_rate_ckctl_arm(struct omap1_clk *clk, unsigned long rate,
extern long omap1_clk_round_rate_ckctl_arm(struct clk *clk, unsigned long rate); unsigned long *p_rate);
#ifdef CONFIG_OMAP_RESET_CLOCKS
extern void omap1_clk_disable_unused(struct clk *clk);
#else
#define omap1_clk_disable_unused NULL
#endif
struct uart_clk { struct uart_clk {
struct clk clk; struct omap1_clk clk;
unsigned long sysc_addr; unsigned long sysc_addr;
}; };
/* Provide a method for preventing idling some ARM IDLECT clocks */ /* Provide a method for preventing idling some ARM IDLECT clocks */
struct arm_idlect1_clk { struct arm_idlect1_clk {
struct clk clk; struct omap1_clk clk;
unsigned long no_idle_count; unsigned long no_idle_count;
__u8 idlect_shift; __u8 idlect_shift;
}; };
/* ARM_CKCTL bit shifts */ /* ARM_CKCTL bit shifts */
...@@ -224,7 +178,7 @@ struct arm_idlect1_clk { ...@@ -224,7 +178,7 @@ struct arm_idlect1_clk {
#define SOFT_REQ_REG2 0xfffe0880 #define SOFT_REQ_REG2 0xfffe0880
extern __u32 arm_idlect1_mask; extern __u32 arm_idlect1_mask;
extern struct clk *api_ck_p, *ck_dpll1_p, *ck_ref_p; extern struct omap1_clk *api_ck_p, *ck_dpll1_p, *ck_ref_p;
extern const struct clkops clkops_dspck; extern const struct clkops clkops_dspck;
extern const struct clkops clkops_uart_16xx; extern const struct clkops clkops_uart_16xx;
...@@ -233,4 +187,9 @@ extern const struct clkops clkops_generic; ...@@ -233,4 +187,9 @@ extern const struct clkops clkops_generic;
/* used for passing SoC type to omap1_{select,round_to}_table_rate() */ /* used for passing SoC type to omap1_{select,round_to}_table_rate() */
extern u32 cpu_mask; extern u32 cpu_mask;
extern const struct clk_ops omap1_clk_null_ops;
extern const struct clk_ops omap1_clk_gate_ops;
extern const struct clk_ops omap1_clk_rate_ops;
extern const struct clk_ops omap1_clk_full_ops;
#endif #endif
This diff is collapsed.
...@@ -15,10 +15,8 @@ ...@@ -15,10 +15,8 @@
#include <asm/mach/map.h> #include <asm/mach/map.h>
#include "tc.h" #include "tc.h"
#include "mux.h"
#include "iomap.h" #include "iomap.h"
#include "common.h" #include "common.h"
#include "clock.h"
/* /*
* The machine specific code may provide the extra mapping besides the * The machine specific code may provide the extra mapping besides the
...@@ -125,11 +123,6 @@ void __init omap1_init_early(void) ...@@ -125,11 +123,6 @@ void __init omap1_init_early(void)
*/ */
omap_writew(0x0, MPU_PUBLIC_TIPB_CNTL); omap_writew(0x0, MPU_PUBLIC_TIPB_CNTL);
omap_writew(0x0, MPU_PRIVATE_TIPB_CNTL); omap_writew(0x0, MPU_PRIVATE_TIPB_CNTL);
/* Must init clocks early to assure that timer interrupt works
*/
omap1_clk_init();
omap1_mux_init();
} }
void __init omap1_init_late(void) void __init omap1_init_late(void)
......
...@@ -51,8 +51,10 @@ ...@@ -51,8 +51,10 @@
#include <asm/mach/time.h> #include <asm/mach/time.h>
#include "hardware.h" #include "hardware.h"
#include "mux.h"
#include "iomap.h" #include "iomap.h"
#include "common.h" #include "common.h"
#include "clock.h"
#ifdef CONFIG_OMAP_MPU_TIMER #ifdef CONFIG_OMAP_MPU_TIMER
...@@ -224,6 +226,9 @@ static inline void omap_mpu_timer_init(void) ...@@ -224,6 +226,9 @@ static inline void omap_mpu_timer_init(void)
*/ */
void __init omap1_timer_init(void) void __init omap1_timer_init(void)
{ {
omap1_clk_init();
omap1_mux_init();
if (omap_32k_timer_init() != 0) if (omap_32k_timer_init() != 0)
omap_mpu_timer_init(); omap_mpu_timer_init();
} }
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