diff options
-rw-r--r-- | arch/arm/mach-ux500/clock.c | 49 |
1 files changed, 9 insertions, 40 deletions
diff --git a/arch/arm/mach-ux500/clock.c b/arch/arm/mach-ux500/clock.c index 1675047daf20..a7cd0d55828e 100644 --- a/arch/arm/mach-ux500/clock.c +++ b/arch/arm/mach-ux500/clock.c | |||
@@ -133,7 +133,7 @@ static unsigned long clk_mtu_get_rate(struct clk *clk) | |||
133 | { | 133 | { |
134 | void __iomem *addr = __io_address(UX500_PRCMU_BASE) | 134 | void __iomem *addr = __io_address(UX500_PRCMU_BASE) |
135 | + PRCM_TCR; | 135 | + PRCM_TCR; |
136 | u32 tcr = readl(addr); | 136 | u32 tcr; |
137 | int mtu = (int) clk->data; | 137 | int mtu = (int) clk->data; |
138 | /* | 138 | /* |
139 | * One of these is selected eventually | 139 | * One of these is selected eventually |
@@ -144,6 +144,14 @@ static unsigned long clk_mtu_get_rate(struct clk *clk) | |||
144 | unsigned long mturate; | 144 | unsigned long mturate; |
145 | unsigned long retclk; | 145 | unsigned long retclk; |
146 | 146 | ||
147 | /* | ||
148 | * On a startup, always conifgure the TCR to the doze mode; | ||
149 | * bootloaders do it for us. Do this in the kernel too. | ||
150 | */ | ||
151 | writel(PRCM_TCR_DOZE_MODE, addr); | ||
152 | |||
153 | tcr = readl(addr); | ||
154 | |||
147 | /* Get the rate from the parent as a default */ | 155 | /* Get the rate from the parent as a default */ |
148 | if (clk->parent_periph) | 156 | if (clk->parent_periph) |
149 | mturate = clk_get_rate(clk->parent_periph); | 157 | mturate = clk_get_rate(clk->parent_periph); |
@@ -153,45 +161,6 @@ static unsigned long clk_mtu_get_rate(struct clk *clk) | |||
153 | /* We need to be connected SOMEWHERE */ | 161 | /* We need to be connected SOMEWHERE */ |
154 | BUG(); | 162 | BUG(); |
155 | 163 | ||
156 | /* | ||
157 | * Are we in doze mode? | ||
158 | * In this mode the parent peripheral or the fixed 32768 Hz | ||
159 | * clock is fed into the block. | ||
160 | */ | ||
161 | if (!(tcr & PRCM_TCR_DOZE_MODE)) { | ||
162 | /* | ||
163 | * Here we're using the clock input from the APE ULP | ||
164 | * clock domain. But first: are the timers stopped? | ||
165 | */ | ||
166 | if (tcr & PRCM_TCR_STOPPED) { | ||
167 | clk32k = 0; | ||
168 | mturate = 0; | ||
169 | } else { | ||
170 | /* Else default mode: 0 and 2.4 MHz */ | ||
171 | clk32k = 0; | ||
172 | if (cpu_is_u5500()) | ||
173 | /* DB5500 divides by 8 */ | ||
174 | mturate /= 8; | ||
175 | else if (cpu_is_u8500ed()) { | ||
176 | /* | ||
177 | * This clocking setting must not be used | ||
178 | * in the ED chip, it is simply not | ||
179 | * connected anywhere! | ||
180 | */ | ||
181 | mturate = 0; | ||
182 | BUG(); | ||
183 | } else | ||
184 | /* | ||
185 | * In this mode the ulp38m4 clock is divided | ||
186 | * by a factor 16, on the DB8500 typically | ||
187 | * 38400000 / 16 ~ 2.4 MHz. | ||
188 | * TODO: Replace the constant with a reference | ||
189 | * to the ULP source once this is modeled. | ||
190 | */ | ||
191 | mturate = 38400000 / 16; | ||
192 | } | ||
193 | } | ||
194 | |||
195 | /* Return the clock selected for this MTU */ | 164 | /* Return the clock selected for this MTU */ |
196 | if (tcr & (1 << mtu)) | 165 | if (tcr & (1 << mtu)) |
197 | retclk = clk32k; | 166 | retclk = clk32k; |