diff options
author | Paul Walmsley <paul@pwsan.com> | 2011-10-06 19:18:45 -0400 |
---|---|---|
committer | Kevin Hilman <khilman@ti.com> | 2011-10-07 16:41:49 -0400 |
commit | b02b917211d50ad5dc13e49c933ef916b10e0d00 (patch) | |
tree | 138707c1aab31b0263cb1bd02ba4678c6a7d80f5 | |
parent | ff2f8e5ffb23de6e2284f31651447cb80a4c9d1b (diff) |
ARM: OMAP3: PM: fix I/O wakeup and I/O chain clock control detection
The way that we detect which OMAP3 chips support I/O wakeup and
software I/O chain clock control is broken.
Currently, I/O wakeup is marked as present for all OMAP3 SoCs other
than the AM3505/3517. The TI81xx family of SoCs are at present
considered to be OMAP3 SoCs, but don't support I/O wakeup. To resolve
this, convert the existing blacklist approach to an explicit,
whitelist support, in which only SoCs which are known to support I/O
wakeup are listed. (At present, this only includes OMAP34xx,
OMAP3503, OMAP3515, OMAP3525, OMAP3530, and OMAP36xx.)
Also, the current code incorrectly detects the presence of a
software-controllable I/O chain clock on several chips that don't
support it. This results in writes to reserved bitfields, unnecessary
delays, and console messages on kernels running on those chips:
http://www.spinics.net/lists/linux-omap/msg58735.html
Convert this test to a feature test with a chip-by-chip whitelist.
Thanks to Dave Hylands <dhylands@gmail.com> for reporting this problem
and doing some testing to help isolate the cause. Thanks to Steve
Sakoman <sakoman@gmail.com> for catching a bug in the first version of
this patch. Thanks to Russell King <linux@arm.linux.org.uk> for
comments.
Signed-off-by: Paul Walmsley <paul@pwsan.com>
Cc: Dave Hylands <dhylands@gmail.com>
Cc: Steve Sakoman <sakoman@gmail.com>
Tested-by: Steve Sakoman <sakoman@gmail.com>
Cc: Russell King - ARM Linux <linux@arm.linux.org.uk>
Signed-off-by: Kevin Hilman <khilman@ti.com>
-rw-r--r-- | arch/arm/mach-omap2/id.c | 5 | ||||
-rw-r--r-- | arch/arm/mach-omap2/pm34xx.c | 43 | ||||
-rw-r--r-- | arch/arm/plat-omap/include/plat/cpu.h | 17 |
3 files changed, 38 insertions, 27 deletions
diff --git a/arch/arm/mach-omap2/id.c b/arch/arm/mach-omap2/id.c index 37efb8696927..a1ccb66fbe75 100644 --- a/arch/arm/mach-omap2/id.c +++ b/arch/arm/mach-omap2/id.c | |||
@@ -201,8 +201,11 @@ static void __init omap3_check_features(void) | |||
201 | OMAP3_CHECK_FEATURE(status, ISP); | 201 | OMAP3_CHECK_FEATURE(status, ISP); |
202 | if (cpu_is_omap3630()) | 202 | if (cpu_is_omap3630()) |
203 | omap_features |= OMAP3_HAS_192MHZ_CLK; | 203 | omap_features |= OMAP3_HAS_192MHZ_CLK; |
204 | if (!cpu_is_omap3505() && !cpu_is_omap3517()) | 204 | if (cpu_is_omap3430() || cpu_is_omap3630()) |
205 | omap_features |= OMAP3_HAS_IO_WAKEUP; | 205 | omap_features |= OMAP3_HAS_IO_WAKEUP; |
206 | if (cpu_is_omap3630() || omap_rev() == OMAP3430_REV_ES3_1 || | ||
207 | omap_rev() == OMAP3430_REV_ES3_1_2) | ||
208 | omap_features |= OMAP3_HAS_IO_CHAIN_CTRL; | ||
206 | 209 | ||
207 | omap_features |= OMAP3_HAS_SDRC; | 210 | omap_features |= OMAP3_HAS_SDRC; |
208 | 211 | ||
diff --git a/arch/arm/mach-omap2/pm34xx.c b/arch/arm/mach-omap2/pm34xx.c index 1915050e9401..bfa8b8c8171a 100644 --- a/arch/arm/mach-omap2/pm34xx.c +++ b/arch/arm/mach-omap2/pm34xx.c | |||
@@ -99,31 +99,27 @@ static void omap3_enable_io_chain(void) | |||
99 | { | 99 | { |
100 | int timeout = 0; | 100 | int timeout = 0; |
101 | 101 | ||
102 | if (omap_rev() >= OMAP3430_REV_ES3_1) { | 102 | omap2_prm_set_mod_reg_bits(OMAP3430_EN_IO_CHAIN_MASK, WKUP_MOD, |
103 | omap2_prm_set_mod_reg_bits(OMAP3430_EN_IO_CHAIN_MASK, WKUP_MOD, | 103 | PM_WKEN); |
104 | PM_WKEN); | 104 | /* Do a readback to assure write has been done */ |
105 | /* Do a readback to assure write has been done */ | 105 | omap2_prm_read_mod_reg(WKUP_MOD, PM_WKEN); |
106 | omap2_prm_read_mod_reg(WKUP_MOD, PM_WKEN); | 106 | |
107 | 107 | while (!(omap2_prm_read_mod_reg(WKUP_MOD, PM_WKEN) & | |
108 | while (!(omap2_prm_read_mod_reg(WKUP_MOD, PM_WKEN) & | 108 | OMAP3430_ST_IO_CHAIN_MASK)) { |
109 | OMAP3430_ST_IO_CHAIN_MASK)) { | 109 | timeout++; |
110 | timeout++; | 110 | if (timeout > 1000) { |
111 | if (timeout > 1000) { | 111 | pr_err("Wake up daisy chain activation failed.\n"); |
112 | printk(KERN_ERR "Wake up daisy chain " | 112 | return; |
113 | "activation failed.\n"); | ||
114 | return; | ||
115 | } | ||
116 | omap2_prm_set_mod_reg_bits(OMAP3430_ST_IO_CHAIN_MASK, | ||
117 | WKUP_MOD, PM_WKEN); | ||
118 | } | 113 | } |
114 | omap2_prm_set_mod_reg_bits(OMAP3430_ST_IO_CHAIN_MASK, | ||
115 | WKUP_MOD, PM_WKEN); | ||
119 | } | 116 | } |
120 | } | 117 | } |
121 | 118 | ||
122 | static void omap3_disable_io_chain(void) | 119 | static void omap3_disable_io_chain(void) |
123 | { | 120 | { |
124 | if (omap_rev() >= OMAP3430_REV_ES3_1) | 121 | omap2_prm_clear_mod_reg_bits(OMAP3430_EN_IO_CHAIN_MASK, WKUP_MOD, |
125 | omap2_prm_clear_mod_reg_bits(OMAP3430_EN_IO_CHAIN_MASK, WKUP_MOD, | 122 | PM_WKEN); |
126 | PM_WKEN); | ||
127 | } | 123 | } |
128 | 124 | ||
129 | static void omap3_core_save_context(void) | 125 | static void omap3_core_save_context(void) |
@@ -375,7 +371,8 @@ void omap_sram_idle(void) | |||
375 | (per_next_state < PWRDM_POWER_ON || | 371 | (per_next_state < PWRDM_POWER_ON || |
376 | core_next_state < PWRDM_POWER_ON)) { | 372 | core_next_state < PWRDM_POWER_ON)) { |
377 | omap2_prm_set_mod_reg_bits(OMAP3430_EN_IO_MASK, WKUP_MOD, PM_WKEN); | 373 | omap2_prm_set_mod_reg_bits(OMAP3430_EN_IO_MASK, WKUP_MOD, PM_WKEN); |
378 | omap3_enable_io_chain(); | 374 | if (omap3_has_io_chain_ctrl()) |
375 | omap3_enable_io_chain(); | ||
379 | } | 376 | } |
380 | 377 | ||
381 | /* Block console output in case it is on one of the OMAP UARTs */ | 378 | /* Block console output in case it is on one of the OMAP UARTs */ |
@@ -478,7 +475,8 @@ console_still_active: | |||
478 | core_next_state < PWRDM_POWER_ON)) { | 475 | core_next_state < PWRDM_POWER_ON)) { |
479 | omap2_prm_clear_mod_reg_bits(OMAP3430_EN_IO_MASK, WKUP_MOD, | 476 | omap2_prm_clear_mod_reg_bits(OMAP3430_EN_IO_MASK, WKUP_MOD, |
480 | PM_WKEN); | 477 | PM_WKEN); |
481 | omap3_disable_io_chain(); | 478 | if (omap3_has_io_chain_ctrl()) |
479 | omap3_disable_io_chain(); | ||
482 | } | 480 | } |
483 | 481 | ||
484 | clkdm_allow_idle(mpu_pwrdm->pwrdm_clkdms[0]); | 482 | clkdm_allow_idle(mpu_pwrdm->pwrdm_clkdms[0]); |
@@ -871,6 +869,9 @@ static int __init omap3_pm_init(void) | |||
871 | if (!cpu_is_omap34xx()) | 869 | if (!cpu_is_omap34xx()) |
872 | return -ENODEV; | 870 | return -ENODEV; |
873 | 871 | ||
872 | if (!omap3_has_io_chain_ctrl()) | ||
873 | pr_warning("PM: no software I/O chain control; some wakeups may be lost\n"); | ||
874 | |||
874 | pm_errata_configure(); | 875 | pm_errata_configure(); |
875 | 876 | ||
876 | /* XXX prcm_setup_regs needs to be before enabling hw | 877 | /* XXX prcm_setup_regs needs to be before enabling hw |
diff --git a/arch/arm/plat-omap/include/plat/cpu.h b/arch/arm/plat-omap/include/plat/cpu.h index 67b3d75884cd..3a280aaf9675 100644 --- a/arch/arm/plat-omap/include/plat/cpu.h +++ b/arch/arm/plat-omap/include/plat/cpu.h | |||
@@ -477,6 +477,13 @@ void omap2_check_revision(void); | |||
477 | 477 | ||
478 | /* | 478 | /* |
479 | * Runtime detection of OMAP3 features | 479 | * Runtime detection of OMAP3 features |
480 | * | ||
481 | * OMAP3_HAS_IO_CHAIN_CTRL: Some later members of the OMAP3 chip | ||
482 | * family have OS-level control over the I/O chain clock. This is | ||
483 | * to avoid a window during which wakeups could potentially be lost | ||
484 | * during powerdomain transitions. If this bit is set, it | ||
485 | * indicates that the chip does support OS-level control of this | ||
486 | * feature. | ||
480 | */ | 487 | */ |
481 | extern u32 omap_features; | 488 | extern u32 omap_features; |
482 | 489 | ||
@@ -488,9 +495,10 @@ extern u32 omap_features; | |||
488 | #define OMAP3_HAS_192MHZ_CLK BIT(5) | 495 | #define OMAP3_HAS_192MHZ_CLK BIT(5) |
489 | #define OMAP3_HAS_IO_WAKEUP BIT(6) | 496 | #define OMAP3_HAS_IO_WAKEUP BIT(6) |
490 | #define OMAP3_HAS_SDRC BIT(7) | 497 | #define OMAP3_HAS_SDRC BIT(7) |
491 | #define OMAP4_HAS_MPU_1GHZ BIT(8) | 498 | #define OMAP3_HAS_IO_CHAIN_CTRL BIT(8) |
492 | #define OMAP4_HAS_MPU_1_2GHZ BIT(9) | 499 | #define OMAP4_HAS_MPU_1GHZ BIT(9) |
493 | #define OMAP4_HAS_MPU_1_5GHZ BIT(10) | 500 | #define OMAP4_HAS_MPU_1_2GHZ BIT(10) |
501 | #define OMAP4_HAS_MPU_1_5GHZ BIT(11) | ||
494 | 502 | ||
495 | 503 | ||
496 | #define OMAP3_HAS_FEATURE(feat,flag) \ | 504 | #define OMAP3_HAS_FEATURE(feat,flag) \ |
@@ -507,12 +515,11 @@ OMAP3_HAS_FEATURE(isp, ISP) | |||
507 | OMAP3_HAS_FEATURE(192mhz_clk, 192MHZ_CLK) | 515 | OMAP3_HAS_FEATURE(192mhz_clk, 192MHZ_CLK) |
508 | OMAP3_HAS_FEATURE(io_wakeup, IO_WAKEUP) | 516 | OMAP3_HAS_FEATURE(io_wakeup, IO_WAKEUP) |
509 | OMAP3_HAS_FEATURE(sdrc, SDRC) | 517 | OMAP3_HAS_FEATURE(sdrc, SDRC) |
518 | OMAP3_HAS_FEATURE(io_chain_ctrl, IO_CHAIN_CTRL) | ||
510 | 519 | ||
511 | /* | 520 | /* |
512 | * Runtime detection of OMAP4 features | 521 | * Runtime detection of OMAP4 features |
513 | */ | 522 | */ |
514 | extern u32 omap_features; | ||
515 | |||
516 | #define OMAP4_HAS_FEATURE(feat, flag) \ | 523 | #define OMAP4_HAS_FEATURE(feat, flag) \ |
517 | static inline unsigned int omap4_has_ ##feat(void) \ | 524 | static inline unsigned int omap4_has_ ##feat(void) \ |
518 | { \ | 525 | { \ |