aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJon Hunter <jon-hunter@ti.com>2012-06-20 16:55:24 -0400
committerJon Hunter <jon-hunter@ti.com>2012-11-02 14:16:28 -0400
commitad24bde8f1025da014108493b595bed22d989efb (patch)
treea592f2174fee71fd8f16a2378cc1b35d28fb5bd0
parent8f0d8163b50e01f398b14bcd4dc039ac5ab18d64 (diff)
ARM: OMAP3: Dynamically disable secure timer nodes for secure devices
OMAP3 devices may or may not have security features enabled. Security enabled devices are known as high-secure (HS) and devices without security are known as general purpose (GP). For OMAP3 devices there are 12 general purpose timers available. On secure devices the 12th timer is reserved for secure usage and so cannot be used by the kernel, where as for a GP device it is available. We can detect the OMAP device type, secure or GP, at runtime via an on-chip register. Today, when not using DT, we do not register the 12th timer as a linux device if the device is secure. When using device tree, device tree is going to register all the timer devices it finds in the device tree blob. To prevent device tree from registering 12th timer on a secure OMAP3 device we can add a status property to the timer binding with the value "disabled" at boot time. Note that timer 12 on a OMAP3 device has a property "ti,timer-secure" to indicate that it will not be available on a secure device and so for secure OMAP3 devices, we search for timers with this property and then disable them. Using the prom_add_property() function to dynamically add a property was a recommended approach suggested by Rob Herring [1]. I have tested this on an OMAP3 GP device and faking it to pretend to be a secure device to ensure that any timers marked with "ti,timer-secure" are not registered on boot. I have also made sure that all timers are registered as expected on a GP device by default. [1] http://comments.gmane.org/gmane.linux.ports.arm.omap/79203 Signed-off-by: Jon Hunter <jon-hunter@ti.com>
-rw-r--r--arch/arm/mach-omap2/timer.c36
1 files changed, 36 insertions, 0 deletions
diff --git a/arch/arm/mach-omap2/timer.c b/arch/arm/mach-omap2/timer.c
index 69e46631a7cd..def9a0ebe42d 100644
--- a/arch/arm/mach-omap2/timer.c
+++ b/arch/arm/mach-omap2/timer.c
@@ -144,6 +144,41 @@ static struct clock_event_device clockevent_gpt = {
144 .set_mode = omap2_gp_timer_set_mode, 144 .set_mode = omap2_gp_timer_set_mode,
145}; 145};
146 146
147static struct property device_disabled = {
148 .name = "status",
149 .length = sizeof("disabled"),
150 .value = "disabled",
151};
152
153static struct of_device_id omap_timer_match[] __initdata = {
154 { .compatible = "ti,omap2-timer", },
155 { }
156};
157
158/**
159 * omap_dmtimer_init - initialisation function when device tree is used
160 *
161 * For secure OMAP3 devices, timers with device type "timer-secure" cannot
162 * be used by the kernel as they are reserved. Therefore, to prevent the
163 * kernel registering these devices remove them dynamically from the device
164 * tree on boot.
165 */
166void __init omap_dmtimer_init(void)
167{
168 struct device_node *np;
169
170 if (!cpu_is_omap34xx())
171 return;
172
173 /* If we are a secure device, remove any secure timer nodes */
174 if ((omap_type() != OMAP2_DEVICE_TYPE_GP)) {
175 for_each_matching_node(np, omap_timer_match) {
176 if (of_get_property(np, "ti,timer-secure", NULL))
177 prom_add_property(np, &device_disabled);
178 }
179 }
180}
181
147static int __init omap_dm_timer_init_one(struct omap_dm_timer *timer, 182static int __init omap_dm_timer_init_one(struct omap_dm_timer *timer,
148 int gptimer_id, 183 int gptimer_id,
149 const char *fck_source) 184 const char *fck_source)
@@ -437,6 +472,7 @@ static inline void __init realtime_counter_init(void)
437 clksrc_nr, clksrc_src) \ 472 clksrc_nr, clksrc_src) \
438static void __init omap##name##_timer_init(void) \ 473static void __init omap##name##_timer_init(void) \
439{ \ 474{ \
475 omap_dmtimer_init(); \
440 omap2_gp_clockevent_init((clkev_nr), clkev_src); \ 476 omap2_gp_clockevent_init((clkev_nr), clkev_src); \
441 omap2_clocksource_init((clksrc_nr), clksrc_src); \ 477 omap2_clocksource_init((clksrc_nr), clksrc_src); \
442} 478}