aboutsummaryrefslogtreecommitdiffstats
path: root/arch
diff options
context:
space:
mode:
authorJon Hunter <jon-hunter@ti.com>2009-06-27 01:07:25 -0400
committerKevin Hilman <khilman@deeprootsystems.com>2009-10-05 13:50:59 -0400
commit77da2d910a17e1e6a7e949578723d5aab58568d5 (patch)
tree4d66e52b5644931c6d1c4808053cf26a0dd39fc1 /arch
parent17d857be649a21ca90008c6dc425d849fa83db5c (diff)
OMAP3: PM: Prevent hang in prcm_interrupt_handler
There are two scenarios where a race condition could result in a hang in the prcm_interrupt handler. These are: 1). Waiting for PRM_IRQSTATUS_MPU register to clear. Bit 0 of the PRM_IRQSTATUS_MPU register indicates that a wake-up event is pending for the MPU. This bit can only be cleared if the all the wake-up events latched in the various PM_WKST_x registers have been cleared. If a wake-up event occurred during the processing of the prcm interrupt handler, after the corresponding PM_WKST_x register was checked but before the PRM_IRQSTATUS_MPU was cleared, then the CPU would be stuck forever waiting for bit 0 in PRM_IRQSTATUS_MPU to be cleared. 2). Waiting for the PM_WKST_x register to clear. Some power domains have more than one wake-up source. The PM_WKST_x registers indicate the source of a wake-up event and need to be cleared after a wake-up event occurs. When the PM_WKST_x registers are read and before they are cleared, it is possible that another wake-up event could occur causing another bit to be set in one of the PM_WKST_x registers. If this did occur after reading a PM_WKST_x register then the CPU would miss this event and get stuck forever in a loop waiting for that PM_WKST_x register to clear. This patch address the above race conditions that would result in a hang. Signed-off-by: Jon Hunter <jon-hunter@ti.com> Reviewed-by: Paul Walmsley <paul@pwsan.com> Signed-off-by: Kevin Hilman <khilman@deeprootsystems.com>
Diffstat (limited to 'arch')
-rw-r--r--arch/arm/mach-omap2/pm34xx.c143
1 files changed, 60 insertions, 83 deletions
diff --git a/arch/arm/mach-omap2/pm34xx.c b/arch/arm/mach-omap2/pm34xx.c
index 0ff5a6c53aa0..1e7aae2ce5ed 100644
--- a/arch/arm/mach-omap2/pm34xx.c
+++ b/arch/arm/mach-omap2/pm34xx.c
@@ -51,97 +51,74 @@ static void (*_omap_sram_idle)(u32 *addr, int save_state);
51 51
52static struct powerdomain *mpu_pwrdm; 52static struct powerdomain *mpu_pwrdm;
53 53
54/* PRCM Interrupt Handler for wakeups */ 54/*
55static irqreturn_t prcm_interrupt_handler (int irq, void *dev_id) 55 * PRCM Interrupt Handler Helper Function
56 *
57 * The purpose of this function is to clear any wake-up events latched
58 * in the PRCM PM_WKST_x registers. It is possible that a wake-up event
59 * may occur whilst attempting to clear a PM_WKST_x register and thus
60 * set another bit in this register. A while loop is used to ensure
61 * that any peripheral wake-up events occurring while attempting to
62 * clear the PM_WKST_x are detected and cleared.
63 */
64static void prcm_clear_mod_irqs(s16 module, u8 regs)
56{ 65{
57 u32 wkst, irqstatus_mpu; 66 u32 wkst, fclk, iclk;
58 u32 fclk, iclk; 67 u16 wkst_off = (regs == 3) ? OMAP3430ES2_PM_WKST3 : PM_WKST1;
68 u16 fclk_off = (regs == 3) ? OMAP3430ES2_CM_FCLKEN3 : CM_FCLKEN1;
69 u16 iclk_off = (regs == 3) ? CM_ICLKEN3 : CM_ICLKEN1;
59 70
60 /* WKUP */ 71 wkst = prm_read_mod_reg(module, wkst_off);
61 wkst = prm_read_mod_reg(WKUP_MOD, PM_WKST);
62 if (wkst) { 72 if (wkst) {
63 iclk = cm_read_mod_reg(WKUP_MOD, CM_ICLKEN); 73 iclk = cm_read_mod_reg(module, iclk_off);
64 fclk = cm_read_mod_reg(WKUP_MOD, CM_FCLKEN); 74 fclk = cm_read_mod_reg(module, fclk_off);
65 cm_set_mod_reg_bits(wkst, WKUP_MOD, CM_ICLKEN); 75 while (wkst) {
66 cm_set_mod_reg_bits(wkst, WKUP_MOD, CM_FCLKEN); 76 cm_set_mod_reg_bits(wkst, module, iclk_off);
67 prm_write_mod_reg(wkst, WKUP_MOD, PM_WKST); 77 cm_set_mod_reg_bits(wkst, module, fclk_off);
68 while (prm_read_mod_reg(WKUP_MOD, PM_WKST)) 78 prm_write_mod_reg(wkst, module, wkst_off);
69 cpu_relax(); 79 wkst = prm_read_mod_reg(module, wkst_off);
70 cm_write_mod_reg(iclk, WKUP_MOD, CM_ICLKEN); 80 }
71 cm_write_mod_reg(fclk, WKUP_MOD, CM_FCLKEN); 81 cm_write_mod_reg(iclk, module, iclk_off);
72 } 82 cm_write_mod_reg(fclk, module, fclk_off);
73
74 /* CORE */
75 wkst = prm_read_mod_reg(CORE_MOD, PM_WKST1);
76 if (wkst) {
77 iclk = cm_read_mod_reg(CORE_MOD, CM_ICLKEN1);
78 fclk = cm_read_mod_reg(CORE_MOD, CM_FCLKEN1);
79 cm_set_mod_reg_bits(wkst, CORE_MOD, CM_ICLKEN1);
80 cm_set_mod_reg_bits(wkst, CORE_MOD, CM_FCLKEN1);
81 prm_write_mod_reg(wkst, CORE_MOD, PM_WKST1);
82 while (prm_read_mod_reg(CORE_MOD, PM_WKST1))
83 cpu_relax();
84 cm_write_mod_reg(iclk, CORE_MOD, CM_ICLKEN1);
85 cm_write_mod_reg(fclk, CORE_MOD, CM_FCLKEN1);
86 }
87 wkst = prm_read_mod_reg(CORE_MOD, OMAP3430ES2_PM_WKST3);
88 if (wkst) {
89 iclk = cm_read_mod_reg(CORE_MOD, CM_ICLKEN3);
90 fclk = cm_read_mod_reg(CORE_MOD, OMAP3430ES2_CM_FCLKEN3);
91 cm_set_mod_reg_bits(wkst, CORE_MOD, CM_ICLKEN3);
92 cm_set_mod_reg_bits(wkst, CORE_MOD, OMAP3430ES2_CM_FCLKEN3);
93 prm_write_mod_reg(wkst, CORE_MOD, OMAP3430ES2_PM_WKST3);
94 while (prm_read_mod_reg(CORE_MOD, OMAP3430ES2_PM_WKST3))
95 cpu_relax();
96 cm_write_mod_reg(iclk, CORE_MOD, CM_ICLKEN3);
97 cm_write_mod_reg(fclk, CORE_MOD, OMAP3430ES2_CM_FCLKEN3);
98 }
99
100 /* PER */
101 wkst = prm_read_mod_reg(OMAP3430_PER_MOD, PM_WKST);
102 if (wkst) {
103 iclk = cm_read_mod_reg(OMAP3430_PER_MOD, CM_ICLKEN);
104 fclk = cm_read_mod_reg(OMAP3430_PER_MOD, CM_FCLKEN);
105 cm_set_mod_reg_bits(wkst, OMAP3430_PER_MOD, CM_ICLKEN);
106 cm_set_mod_reg_bits(wkst, OMAP3430_PER_MOD, CM_FCLKEN);
107 prm_write_mod_reg(wkst, OMAP3430_PER_MOD, PM_WKST);
108 while (prm_read_mod_reg(OMAP3430_PER_MOD, PM_WKST))
109 cpu_relax();
110 cm_write_mod_reg(iclk, OMAP3430_PER_MOD, CM_ICLKEN);
111 cm_write_mod_reg(fclk, OMAP3430_PER_MOD, CM_FCLKEN);
112 } 83 }
84}
113 85
114 if (omap_rev() > OMAP3430_REV_ES1_0) { 86/*
115 /* USBHOST */ 87 * PRCM Interrupt Handler
116 wkst = prm_read_mod_reg(OMAP3430ES2_USBHOST_MOD, PM_WKST); 88 *
117 if (wkst) { 89 * The PRM_IRQSTATUS_MPU register indicates if there are any pending
118 iclk = cm_read_mod_reg(OMAP3430ES2_USBHOST_MOD, 90 * interrupts from the PRCM for the MPU. These bits must be cleared in
119 CM_ICLKEN); 91 * order to clear the PRCM interrupt. The PRCM interrupt handler is
120 fclk = cm_read_mod_reg(OMAP3430ES2_USBHOST_MOD, 92 * implemented to simply clear the PRM_IRQSTATUS_MPU in order to clear
121 CM_FCLKEN); 93 * the PRCM interrupt. Please note that bit 0 of the PRM_IRQSTATUS_MPU
122 cm_set_mod_reg_bits(wkst, OMAP3430ES2_USBHOST_MOD, 94 * register indicates that a wake-up event is pending for the MPU and
123 CM_ICLKEN); 95 * this bit can only be cleared if the all the wake-up events latched
124 cm_set_mod_reg_bits(wkst, OMAP3430ES2_USBHOST_MOD, 96 * in the various PM_WKST_x registers have been cleared. The interrupt
125 CM_FCLKEN); 97 * handler is implemented using a do-while loop so that if a wake-up
126 prm_write_mod_reg(wkst, OMAP3430ES2_USBHOST_MOD, 98 * event occurred during the processing of the prcm interrupt handler
127 PM_WKST); 99 * (setting a bit in the corresponding PM_WKST_x register and thus
128 while (prm_read_mod_reg(OMAP3430ES2_USBHOST_MOD, 100 * preventing us from clearing bit 0 of the PRM_IRQSTATUS_MPU register)
129 PM_WKST)) 101 * this would be handled.
130 cpu_relax(); 102 */
131 cm_write_mod_reg(iclk, OMAP3430ES2_USBHOST_MOD, 103static irqreturn_t prcm_interrupt_handler (int irq, void *dev_id)
132 CM_ICLKEN); 104{
133 cm_write_mod_reg(fclk, OMAP3430ES2_USBHOST_MOD, 105 u32 irqstatus_mpu;
134 CM_FCLKEN); 106
107 do {
108 prcm_clear_mod_irqs(WKUP_MOD, 1);
109 prcm_clear_mod_irqs(CORE_MOD, 1);
110 prcm_clear_mod_irqs(OMAP3430_PER_MOD, 1);
111 if (omap_rev() > OMAP3430_REV_ES1_0) {
112 prcm_clear_mod_irqs(CORE_MOD, 3);
113 prcm_clear_mod_irqs(OMAP3430ES2_USBHOST_MOD, 1);
135 } 114 }
136 }
137 115
138 irqstatus_mpu = prm_read_mod_reg(OCP_MOD, 116 irqstatus_mpu = prm_read_mod_reg(OCP_MOD,
139 OMAP3_PRM_IRQSTATUS_MPU_OFFSET); 117 OMAP3_PRM_IRQSTATUS_MPU_OFFSET);
140 prm_write_mod_reg(irqstatus_mpu, OCP_MOD, 118 prm_write_mod_reg(irqstatus_mpu, OCP_MOD,
141 OMAP3_PRM_IRQSTATUS_MPU_OFFSET); 119 OMAP3_PRM_IRQSTATUS_MPU_OFFSET);
142 120
143 while (prm_read_mod_reg(OCP_MOD, OMAP3_PRM_IRQSTATUS_MPU_OFFSET)) 121 } while (prm_read_mod_reg(OCP_MOD, OMAP3_PRM_IRQSTATUS_MPU_OFFSET));
144 cpu_relax();
145 122
146 return IRQ_HANDLED; 123 return IRQ_HANDLED;
147} 124}