diff options
author | Russell King <rmk@dyn-67.arm.linux.org.uk> | 2009-09-12 07:02:26 -0400 |
---|---|---|
committer | Russell King <rmk+kernel@arm.linux.org.uk> | 2009-09-12 07:02:26 -0400 |
commit | ddd559b13f6d2fe3ad68c4b3f5235fd3c2eae4e3 (patch) | |
tree | d827bca3fc825a0ac33efbcd493713be40fcc812 /arch/arm/mach-omap2 | |
parent | cf7a2b4fb6a9b86779930a0a123b0df41aa9208f (diff) | |
parent | f17a1f06d2fa93f4825be572622eb02c4894db4e (diff) |
Merge branch 'devel-stable' into devel
Conflicts:
MAINTAINERS
arch/arm/mm/fault.c
Diffstat (limited to 'arch/arm/mach-omap2')
-rw-r--r-- | arch/arm/mach-omap2/board-4430sdp.c | 2 | ||||
-rw-r--r-- | arch/arm/mach-omap2/board-overo.c | 1 | ||||
-rw-r--r-- | arch/arm/mach-omap2/cm.h | 6 | ||||
-rw-r--r-- | arch/arm/mach-omap2/mcbsp.c | 41 | ||||
-rw-r--r-- | arch/arm/mach-omap2/pm.h | 3 | ||||
-rw-r--r-- | arch/arm/mach-omap2/pm24xx.c | 2 | ||||
-rw-r--r-- | arch/arm/mach-omap2/pm34xx.c | 51 | ||||
-rw-r--r-- | arch/arm/mach-omap2/serial.c | 207 |
8 files changed, 238 insertions, 75 deletions
diff --git a/arch/arm/mach-omap2/board-4430sdp.c b/arch/arm/mach-omap2/board-4430sdp.c index 57e477bd89c6..7e1e721f0324 100644 --- a/arch/arm/mach-omap2/board-4430sdp.c +++ b/arch/arm/mach-omap2/board-4430sdp.c | |||
@@ -39,7 +39,7 @@ static struct platform_device *sdp4430_devices[] __initdata = { | |||
39 | }; | 39 | }; |
40 | 40 | ||
41 | static struct omap_uart_config sdp4430_uart_config __initdata = { | 41 | static struct omap_uart_config sdp4430_uart_config __initdata = { |
42 | .enabled_uarts = (1 << 0) | (1 << 1) | (1 << 2), | 42 | .enabled_uarts = (1 << 0) | (1 << 1) | (1 << 2) | (1 << 3), |
43 | }; | 43 | }; |
44 | 44 | ||
45 | static struct omap_lcd_config sdp4430_lcd_config __initdata = { | 45 | static struct omap_lcd_config sdp4430_lcd_config __initdata = { |
diff --git a/arch/arm/mach-omap2/board-overo.c b/arch/arm/mach-omap2/board-overo.c index dff5528fbfb5..e26af837510b 100644 --- a/arch/arm/mach-omap2/board-overo.c +++ b/arch/arm/mach-omap2/board-overo.c | |||
@@ -51,6 +51,7 @@ | |||
51 | 51 | ||
52 | #define OVERO_GPIO_BT_XGATE 15 | 52 | #define OVERO_GPIO_BT_XGATE 15 |
53 | #define OVERO_GPIO_W2W_NRESET 16 | 53 | #define OVERO_GPIO_W2W_NRESET 16 |
54 | #define OVERO_GPIO_PENDOWN 114 | ||
54 | #define OVERO_GPIO_BT_NRESET 164 | 55 | #define OVERO_GPIO_BT_NRESET 164 |
55 | #define OVERO_GPIO_USBH_CPEN 168 | 56 | #define OVERO_GPIO_USBH_CPEN 168 |
56 | #define OVERO_GPIO_USBH_NRESET 183 | 57 | #define OVERO_GPIO_USBH_NRESET 183 |
diff --git a/arch/arm/mach-omap2/cm.h b/arch/arm/mach-omap2/cm.h index 1d3c93bf86d3..f3c91a1ca391 100644 --- a/arch/arm/mach-omap2/cm.h +++ b/arch/arm/mach-omap2/cm.h | |||
@@ -29,9 +29,9 @@ | |||
29 | * These registers appear once per CM module. | 29 | * These registers appear once per CM module. |
30 | */ | 30 | */ |
31 | 31 | ||
32 | #define OMAP3430_CM_REVISION OMAP_CM_REGADDR(OCP_MOD, 0x0000) | 32 | #define OMAP3430_CM_REVISION OMAP34XX_CM_REGADDR(OCP_MOD, 0x0000) |
33 | #define OMAP3430_CM_SYSCONFIG OMAP_CM_REGADDR(OCP_MOD, 0x0010) | 33 | #define OMAP3430_CM_SYSCONFIG OMAP34XX_CM_REGADDR(OCP_MOD, 0x0010) |
34 | #define OMAP3430_CM_POLCTRL OMAP_CM_REGADDR(OCP_MOD, 0x009c) | 34 | #define OMAP3430_CM_POLCTRL OMAP34XX_CM_REGADDR(OCP_MOD, 0x009c) |
35 | 35 | ||
36 | #define OMAP3_CM_CLKOUT_CTRL_OFFSET 0x0070 | 36 | #define OMAP3_CM_CLKOUT_CTRL_OFFSET 0x0070 |
37 | #define OMAP3430_CM_CLKOUT_CTRL OMAP_CM_REGADDR(OMAP3430_CCR_MOD, 0x0070) | 37 | #define OMAP3430_CM_CLKOUT_CTRL OMAP_CM_REGADDR(OMAP3430_CCR_MOD, 0x0070) |
diff --git a/arch/arm/mach-omap2/mcbsp.c b/arch/arm/mach-omap2/mcbsp.c index 99b6e1546311..d7288f1dc64f 100644 --- a/arch/arm/mach-omap2/mcbsp.c +++ b/arch/arm/mach-omap2/mcbsp.c | |||
@@ -168,6 +168,42 @@ static struct omap_mcbsp_platform_data omap34xx_mcbsp_pdata[] = { | |||
168 | #define OMAP34XX_MCBSP_PDATA_SZ 0 | 168 | #define OMAP34XX_MCBSP_PDATA_SZ 0 |
169 | #endif | 169 | #endif |
170 | 170 | ||
171 | static struct omap_mcbsp_platform_data omap44xx_mcbsp_pdata[] = { | ||
172 | { | ||
173 | .phys_base = OMAP44XX_MCBSP1_BASE, | ||
174 | .dma_rx_sync = OMAP44XX_DMA_MCBSP1_RX, | ||
175 | .dma_tx_sync = OMAP44XX_DMA_MCBSP1_TX, | ||
176 | .rx_irq = INT_24XX_MCBSP1_IRQ_RX, | ||
177 | .tx_irq = INT_24XX_MCBSP1_IRQ_TX, | ||
178 | .ops = &omap2_mcbsp_ops, | ||
179 | }, | ||
180 | { | ||
181 | .phys_base = OMAP44XX_MCBSP2_BASE, | ||
182 | .dma_rx_sync = OMAP44XX_DMA_MCBSP2_RX, | ||
183 | .dma_tx_sync = OMAP44XX_DMA_MCBSP2_TX, | ||
184 | .rx_irq = INT_24XX_MCBSP2_IRQ_RX, | ||
185 | .tx_irq = INT_24XX_MCBSP2_IRQ_TX, | ||
186 | .ops = &omap2_mcbsp_ops, | ||
187 | }, | ||
188 | { | ||
189 | .phys_base = OMAP44XX_MCBSP3_BASE, | ||
190 | .dma_rx_sync = OMAP44XX_DMA_MCBSP3_RX, | ||
191 | .dma_tx_sync = OMAP44XX_DMA_MCBSP3_TX, | ||
192 | .rx_irq = INT_24XX_MCBSP3_IRQ_RX, | ||
193 | .tx_irq = INT_24XX_MCBSP3_IRQ_TX, | ||
194 | .ops = &omap2_mcbsp_ops, | ||
195 | }, | ||
196 | { | ||
197 | .phys_base = OMAP44XX_MCBSP4_BASE, | ||
198 | .dma_rx_sync = OMAP44XX_DMA_MCBSP4_RX, | ||
199 | .dma_tx_sync = OMAP44XX_DMA_MCBSP4_TX, | ||
200 | .rx_irq = INT_24XX_MCBSP4_IRQ_RX, | ||
201 | .tx_irq = INT_24XX_MCBSP4_IRQ_TX, | ||
202 | .ops = &omap2_mcbsp_ops, | ||
203 | }, | ||
204 | }; | ||
205 | #define OMAP44XX_MCBSP_PDATA_SZ ARRAY_SIZE(omap44xx_mcbsp_pdata) | ||
206 | |||
171 | static int __init omap2_mcbsp_init(void) | 207 | static int __init omap2_mcbsp_init(void) |
172 | { | 208 | { |
173 | if (cpu_is_omap2420()) | 209 | if (cpu_is_omap2420()) |
@@ -176,6 +212,8 @@ static int __init omap2_mcbsp_init(void) | |||
176 | omap_mcbsp_count = OMAP2430_MCBSP_PDATA_SZ; | 212 | omap_mcbsp_count = OMAP2430_MCBSP_PDATA_SZ; |
177 | if (cpu_is_omap34xx()) | 213 | if (cpu_is_omap34xx()) |
178 | omap_mcbsp_count = OMAP34XX_MCBSP_PDATA_SZ; | 214 | omap_mcbsp_count = OMAP34XX_MCBSP_PDATA_SZ; |
215 | if (cpu_is_omap44xx()) | ||
216 | omap_mcbsp_count = OMAP44XX_MCBSP_PDATA_SZ; | ||
179 | 217 | ||
180 | mcbsp_ptr = kzalloc(omap_mcbsp_count * sizeof(struct omap_mcbsp *), | 218 | mcbsp_ptr = kzalloc(omap_mcbsp_count * sizeof(struct omap_mcbsp *), |
181 | GFP_KERNEL); | 219 | GFP_KERNEL); |
@@ -191,6 +229,9 @@ static int __init omap2_mcbsp_init(void) | |||
191 | if (cpu_is_omap34xx()) | 229 | if (cpu_is_omap34xx()) |
192 | omap_mcbsp_register_board_cfg(omap34xx_mcbsp_pdata, | 230 | omap_mcbsp_register_board_cfg(omap34xx_mcbsp_pdata, |
193 | OMAP34XX_MCBSP_PDATA_SZ); | 231 | OMAP34XX_MCBSP_PDATA_SZ); |
232 | if (cpu_is_omap44xx()) | ||
233 | omap_mcbsp_register_board_cfg(omap44xx_mcbsp_pdata, | ||
234 | OMAP44XX_MCBSP_PDATA_SZ); | ||
194 | 235 | ||
195 | return omap_mcbsp_init(); | 236 | return omap_mcbsp_init(); |
196 | } | 237 | } |
diff --git a/arch/arm/mach-omap2/pm.h b/arch/arm/mach-omap2/pm.h index f7b3baf76678..21201cd4117b 100644 --- a/arch/arm/mach-omap2/pm.h +++ b/arch/arm/mach-omap2/pm.h | |||
@@ -11,9 +11,6 @@ | |||
11 | #ifndef __ARCH_ARM_MACH_OMAP2_PM_H | 11 | #ifndef __ARCH_ARM_MACH_OMAP2_PM_H |
12 | #define __ARCH_ARM_MACH_OMAP2_PM_H | 12 | #define __ARCH_ARM_MACH_OMAP2_PM_H |
13 | 13 | ||
14 | extern int omap2_pm_init(void); | ||
15 | extern int omap3_pm_init(void); | ||
16 | |||
17 | #ifdef CONFIG_PM_DEBUG | 14 | #ifdef CONFIG_PM_DEBUG |
18 | extern void omap2_pm_dump(int mode, int resume, unsigned int us); | 15 | extern void omap2_pm_dump(int mode, int resume, unsigned int us); |
19 | extern int omap2_pm_debug; | 16 | extern int omap2_pm_debug; |
diff --git a/arch/arm/mach-omap2/pm24xx.c b/arch/arm/mach-omap2/pm24xx.c index db1025562fb0..528dbdc26e23 100644 --- a/arch/arm/mach-omap2/pm24xx.c +++ b/arch/arm/mach-omap2/pm24xx.c | |||
@@ -470,7 +470,7 @@ static void __init prcm_setup_regs(void) | |||
470 | WKUP_MOD, PM_WKEN); | 470 | WKUP_MOD, PM_WKEN); |
471 | } | 471 | } |
472 | 472 | ||
473 | int __init omap2_pm_init(void) | 473 | static int __init omap2_pm_init(void) |
474 | { | 474 | { |
475 | u32 l; | 475 | u32 l; |
476 | 476 | ||
diff --git a/arch/arm/mach-omap2/pm34xx.c b/arch/arm/mach-omap2/pm34xx.c index 841d4c5ed8be..488d595d8e4b 100644 --- a/arch/arm/mach-omap2/pm34xx.c +++ b/arch/arm/mach-omap2/pm34xx.c | |||
@@ -39,7 +39,9 @@ | |||
39 | struct power_state { | 39 | struct power_state { |
40 | struct powerdomain *pwrdm; | 40 | struct powerdomain *pwrdm; |
41 | u32 next_state; | 41 | u32 next_state; |
42 | #ifdef CONFIG_SUSPEND | ||
42 | u32 saved_state; | 43 | u32 saved_state; |
44 | #endif | ||
43 | struct list_head node; | 45 | struct list_head node; |
44 | }; | 46 | }; |
45 | 47 | ||
@@ -293,6 +295,9 @@ out: | |||
293 | local_irq_enable(); | 295 | local_irq_enable(); |
294 | } | 296 | } |
295 | 297 | ||
298 | #ifdef CONFIG_SUSPEND | ||
299 | static suspend_state_t suspend_state; | ||
300 | |||
296 | static int omap3_pm_prepare(void) | 301 | static int omap3_pm_prepare(void) |
297 | { | 302 | { |
298 | disable_hlt(); | 303 | disable_hlt(); |
@@ -321,7 +326,6 @@ static int omap3_pm_suspend(void) | |||
321 | restore: | 326 | restore: |
322 | /* Restore next_pwrsts */ | 327 | /* Restore next_pwrsts */ |
323 | list_for_each_entry(pwrst, &pwrst_list, node) { | 328 | list_for_each_entry(pwrst, &pwrst_list, node) { |
324 | set_pwrdm_state(pwrst->pwrdm, pwrst->saved_state); | ||
325 | state = pwrdm_read_prev_pwrst(pwrst->pwrdm); | 329 | state = pwrdm_read_prev_pwrst(pwrst->pwrdm); |
326 | if (state > pwrst->next_state) { | 330 | if (state > pwrst->next_state) { |
327 | printk(KERN_INFO "Powerdomain (%s) didn't enter " | 331 | printk(KERN_INFO "Powerdomain (%s) didn't enter " |
@@ -329,6 +333,7 @@ restore: | |||
329 | pwrst->pwrdm->name, pwrst->next_state); | 333 | pwrst->pwrdm->name, pwrst->next_state); |
330 | ret = -1; | 334 | ret = -1; |
331 | } | 335 | } |
336 | set_pwrdm_state(pwrst->pwrdm, pwrst->saved_state); | ||
332 | } | 337 | } |
333 | if (ret) | 338 | if (ret) |
334 | printk(KERN_ERR "Could not enter target state in pm_suspend\n"); | 339 | printk(KERN_ERR "Could not enter target state in pm_suspend\n"); |
@@ -339,11 +344,11 @@ restore: | |||
339 | return ret; | 344 | return ret; |
340 | } | 345 | } |
341 | 346 | ||
342 | static int omap3_pm_enter(suspend_state_t state) | 347 | static int omap3_pm_enter(suspend_state_t unused) |
343 | { | 348 | { |
344 | int ret = 0; | 349 | int ret = 0; |
345 | 350 | ||
346 | switch (state) { | 351 | switch (suspend_state) { |
347 | case PM_SUSPEND_STANDBY: | 352 | case PM_SUSPEND_STANDBY: |
348 | case PM_SUSPEND_MEM: | 353 | case PM_SUSPEND_MEM: |
349 | ret = omap3_pm_suspend(); | 354 | ret = omap3_pm_suspend(); |
@@ -360,12 +365,30 @@ static void omap3_pm_finish(void) | |||
360 | enable_hlt(); | 365 | enable_hlt(); |
361 | } | 366 | } |
362 | 367 | ||
368 | /* Hooks to enable / disable UART interrupts during suspend */ | ||
369 | static int omap3_pm_begin(suspend_state_t state) | ||
370 | { | ||
371 | suspend_state = state; | ||
372 | omap_uart_enable_irqs(0); | ||
373 | return 0; | ||
374 | } | ||
375 | |||
376 | static void omap3_pm_end(void) | ||
377 | { | ||
378 | suspend_state = PM_SUSPEND_ON; | ||
379 | omap_uart_enable_irqs(1); | ||
380 | return; | ||
381 | } | ||
382 | |||
363 | static struct platform_suspend_ops omap_pm_ops = { | 383 | static struct platform_suspend_ops omap_pm_ops = { |
384 | .begin = omap3_pm_begin, | ||
385 | .end = omap3_pm_end, | ||
364 | .prepare = omap3_pm_prepare, | 386 | .prepare = omap3_pm_prepare, |
365 | .enter = omap3_pm_enter, | 387 | .enter = omap3_pm_enter, |
366 | .finish = omap3_pm_finish, | 388 | .finish = omap3_pm_finish, |
367 | .valid = suspend_valid_only_mem, | 389 | .valid = suspend_valid_only_mem, |
368 | }; | 390 | }; |
391 | #endif /* CONFIG_SUSPEND */ | ||
369 | 392 | ||
370 | 393 | ||
371 | /** | 394 | /** |
@@ -613,6 +636,24 @@ static void __init prcm_setup_regs(void) | |||
613 | /* Clear any pending PRCM interrupts */ | 636 | /* Clear any pending PRCM interrupts */ |
614 | prm_write_mod_reg(0, OCP_MOD, OMAP3_PRM_IRQSTATUS_MPU_OFFSET); | 637 | prm_write_mod_reg(0, OCP_MOD, OMAP3_PRM_IRQSTATUS_MPU_OFFSET); |
615 | 638 | ||
639 | /* Don't attach IVA interrupts */ | ||
640 | prm_write_mod_reg(0, WKUP_MOD, OMAP3430_PM_IVAGRPSEL); | ||
641 | prm_write_mod_reg(0, CORE_MOD, OMAP3430_PM_IVAGRPSEL1); | ||
642 | prm_write_mod_reg(0, CORE_MOD, OMAP3430ES2_PM_IVAGRPSEL3); | ||
643 | prm_write_mod_reg(0, OMAP3430_PER_MOD, OMAP3430_PM_IVAGRPSEL); | ||
644 | |||
645 | /* Clear any pending 'reset' flags */ | ||
646 | prm_write_mod_reg(0xffffffff, MPU_MOD, RM_RSTST); | ||
647 | prm_write_mod_reg(0xffffffff, CORE_MOD, RM_RSTST); | ||
648 | prm_write_mod_reg(0xffffffff, OMAP3430_PER_MOD, RM_RSTST); | ||
649 | prm_write_mod_reg(0xffffffff, OMAP3430_EMU_MOD, RM_RSTST); | ||
650 | prm_write_mod_reg(0xffffffff, OMAP3430_NEON_MOD, RM_RSTST); | ||
651 | prm_write_mod_reg(0xffffffff, OMAP3430_DSS_MOD, RM_RSTST); | ||
652 | prm_write_mod_reg(0xffffffff, OMAP3430ES2_USBHOST_MOD, RM_RSTST); | ||
653 | |||
654 | /* Clear any pending PRCM interrupts */ | ||
655 | prm_write_mod_reg(0, OCP_MOD, OMAP3_PRM_IRQSTATUS_MPU_OFFSET); | ||
656 | |||
616 | omap3_iva_idle(); | 657 | omap3_iva_idle(); |
617 | omap3_d2d_idle(); | 658 | omap3_d2d_idle(); |
618 | } | 659 | } |
@@ -652,7 +693,7 @@ static int __init clkdms_setup(struct clockdomain *clkdm) | |||
652 | return 0; | 693 | return 0; |
653 | } | 694 | } |
654 | 695 | ||
655 | int __init omap3_pm_init(void) | 696 | static int __init omap3_pm_init(void) |
656 | { | 697 | { |
657 | struct power_state *pwrst, *tmp; | 698 | struct power_state *pwrst, *tmp; |
658 | int ret; | 699 | int ret; |
@@ -692,7 +733,9 @@ int __init omap3_pm_init(void) | |||
692 | _omap_sram_idle = omap_sram_push(omap34xx_cpu_suspend, | 733 | _omap_sram_idle = omap_sram_push(omap34xx_cpu_suspend, |
693 | omap34xx_cpu_suspend_sz); | 734 | omap34xx_cpu_suspend_sz); |
694 | 735 | ||
736 | #ifdef CONFIG_SUSPEND | ||
695 | suspend_set_ops(&omap_pm_ops); | 737 | suspend_set_ops(&omap_pm_ops); |
738 | #endif /* CONFIG_SUSPEND */ | ||
696 | 739 | ||
697 | pm_idle = omap3_pm_idle; | 740 | pm_idle = omap3_pm_idle; |
698 | 741 | ||
diff --git a/arch/arm/mach-omap2/serial.c b/arch/arm/mach-omap2/serial.c index b094c15bfe47..ce22344b94e7 100644 --- a/arch/arm/mach-omap2/serial.c +++ b/arch/arm/mach-omap2/serial.c | |||
@@ -54,6 +54,7 @@ struct omap_uart_state { | |||
54 | 54 | ||
55 | struct plat_serial8250_port *p; | 55 | struct plat_serial8250_port *p; |
56 | struct list_head node; | 56 | struct list_head node; |
57 | struct platform_device pdev; | ||
57 | 58 | ||
58 | #if defined(CONFIG_ARCH_OMAP3) && defined(CONFIG_PM) | 59 | #if defined(CONFIG_ARCH_OMAP3) && defined(CONFIG_PM) |
59 | int context_valid; | 60 | int context_valid; |
@@ -68,10 +69,9 @@ struct omap_uart_state { | |||
68 | #endif | 69 | #endif |
69 | }; | 70 | }; |
70 | 71 | ||
71 | static struct omap_uart_state omap_uart[OMAP_MAX_NR_PORTS]; | ||
72 | static LIST_HEAD(uart_list); | 72 | static LIST_HEAD(uart_list); |
73 | 73 | ||
74 | static struct plat_serial8250_port serial_platform_data[] = { | 74 | static struct plat_serial8250_port serial_platform_data0[] = { |
75 | { | 75 | { |
76 | .membase = IO_ADDRESS(OMAP_UART1_BASE), | 76 | .membase = IO_ADDRESS(OMAP_UART1_BASE), |
77 | .mapbase = OMAP_UART1_BASE, | 77 | .mapbase = OMAP_UART1_BASE, |
@@ -81,6 +81,12 @@ static struct plat_serial8250_port serial_platform_data[] = { | |||
81 | .regshift = 2, | 81 | .regshift = 2, |
82 | .uartclk = OMAP24XX_BASE_BAUD * 16, | 82 | .uartclk = OMAP24XX_BASE_BAUD * 16, |
83 | }, { | 83 | }, { |
84 | .flags = 0 | ||
85 | } | ||
86 | }; | ||
87 | |||
88 | static struct plat_serial8250_port serial_platform_data1[] = { | ||
89 | { | ||
84 | .membase = IO_ADDRESS(OMAP_UART2_BASE), | 90 | .membase = IO_ADDRESS(OMAP_UART2_BASE), |
85 | .mapbase = OMAP_UART2_BASE, | 91 | .mapbase = OMAP_UART2_BASE, |
86 | .irq = 73, | 92 | .irq = 73, |
@@ -89,6 +95,12 @@ static struct plat_serial8250_port serial_platform_data[] = { | |||
89 | .regshift = 2, | 95 | .regshift = 2, |
90 | .uartclk = OMAP24XX_BASE_BAUD * 16, | 96 | .uartclk = OMAP24XX_BASE_BAUD * 16, |
91 | }, { | 97 | }, { |
98 | .flags = 0 | ||
99 | } | ||
100 | }; | ||
101 | |||
102 | static struct plat_serial8250_port serial_platform_data2[] = { | ||
103 | { | ||
92 | .membase = IO_ADDRESS(OMAP_UART3_BASE), | 104 | .membase = IO_ADDRESS(OMAP_UART3_BASE), |
93 | .mapbase = OMAP_UART3_BASE, | 105 | .mapbase = OMAP_UART3_BASE, |
94 | .irq = 74, | 106 | .irq = 74, |
@@ -97,6 +109,16 @@ static struct plat_serial8250_port serial_platform_data[] = { | |||
97 | .regshift = 2, | 109 | .regshift = 2, |
98 | .uartclk = OMAP24XX_BASE_BAUD * 16, | 110 | .uartclk = OMAP24XX_BASE_BAUD * 16, |
99 | }, { | 111 | }, { |
112 | #ifdef CONFIG_ARCH_OMAP4 | ||
113 | .membase = IO_ADDRESS(OMAP_UART4_BASE), | ||
114 | .mapbase = OMAP_UART4_BASE, | ||
115 | .irq = 70, | ||
116 | .flags = UPF_BOOT_AUTOCONF, | ||
117 | .iotype = UPIO_MEM, | ||
118 | .regshift = 2, | ||
119 | .uartclk = OMAP24XX_BASE_BAUD * 16, | ||
120 | }, { | ||
121 | #endif | ||
100 | .flags = 0 | 122 | .flags = 0 |
101 | } | 123 | } |
102 | }; | 124 | }; |
@@ -217,6 +239,40 @@ static inline void omap_uart_disable_clocks(struct omap_uart_state *uart) | |||
217 | clk_disable(uart->fck); | 239 | clk_disable(uart->fck); |
218 | } | 240 | } |
219 | 241 | ||
242 | static void omap_uart_enable_wakeup(struct omap_uart_state *uart) | ||
243 | { | ||
244 | /* Set wake-enable bit */ | ||
245 | if (uart->wk_en && uart->wk_mask) { | ||
246 | u32 v = __raw_readl(uart->wk_en); | ||
247 | v |= uart->wk_mask; | ||
248 | __raw_writel(v, uart->wk_en); | ||
249 | } | ||
250 | |||
251 | /* Ensure IOPAD wake-enables are set */ | ||
252 | if (cpu_is_omap34xx() && uart->padconf) { | ||
253 | u16 v = omap_ctrl_readw(uart->padconf); | ||
254 | v |= OMAP3_PADCONF_WAKEUPENABLE0; | ||
255 | omap_ctrl_writew(v, uart->padconf); | ||
256 | } | ||
257 | } | ||
258 | |||
259 | static void omap_uart_disable_wakeup(struct omap_uart_state *uart) | ||
260 | { | ||
261 | /* Clear wake-enable bit */ | ||
262 | if (uart->wk_en && uart->wk_mask) { | ||
263 | u32 v = __raw_readl(uart->wk_en); | ||
264 | v &= ~uart->wk_mask; | ||
265 | __raw_writel(v, uart->wk_en); | ||
266 | } | ||
267 | |||
268 | /* Ensure IOPAD wake-enables are cleared */ | ||
269 | if (cpu_is_omap34xx() && uart->padconf) { | ||
270 | u16 v = omap_ctrl_readw(uart->padconf); | ||
271 | v &= ~OMAP3_PADCONF_WAKEUPENABLE0; | ||
272 | omap_ctrl_writew(v, uart->padconf); | ||
273 | } | ||
274 | } | ||
275 | |||
220 | static void omap_uart_smart_idle_enable(struct omap_uart_state *uart, | 276 | static void omap_uart_smart_idle_enable(struct omap_uart_state *uart, |
221 | int enable) | 277 | int enable) |
222 | { | 278 | { |
@@ -246,6 +302,11 @@ static void omap_uart_block_sleep(struct omap_uart_state *uart) | |||
246 | 302 | ||
247 | static void omap_uart_allow_sleep(struct omap_uart_state *uart) | 303 | static void omap_uart_allow_sleep(struct omap_uart_state *uart) |
248 | { | 304 | { |
305 | if (device_may_wakeup(&uart->pdev.dev)) | ||
306 | omap_uart_enable_wakeup(uart); | ||
307 | else | ||
308 | omap_uart_disable_wakeup(uart); | ||
309 | |||
249 | if (!uart->clocked) | 310 | if (!uart->clocked) |
250 | return; | 311 | return; |
251 | 312 | ||
@@ -292,7 +353,6 @@ void omap_uart_resume_idle(int num) | |||
292 | /* Check for normal UART wakeup */ | 353 | /* Check for normal UART wakeup */ |
293 | if (__raw_readl(uart->wk_st) & uart->wk_mask) | 354 | if (__raw_readl(uart->wk_st) & uart->wk_mask) |
294 | omap_uart_block_sleep(uart); | 355 | omap_uart_block_sleep(uart); |
295 | |||
296 | return; | 356 | return; |
297 | } | 357 | } |
298 | } | 358 | } |
@@ -346,16 +406,13 @@ static irqreturn_t omap_uart_interrupt(int irq, void *dev_id) | |||
346 | return IRQ_NONE; | 406 | return IRQ_NONE; |
347 | } | 407 | } |
348 | 408 | ||
349 | static u32 sleep_timeout = DEFAULT_TIMEOUT; | ||
350 | |||
351 | static void omap_uart_idle_init(struct omap_uart_state *uart) | 409 | static void omap_uart_idle_init(struct omap_uart_state *uart) |
352 | { | 410 | { |
353 | u32 v; | ||
354 | struct plat_serial8250_port *p = uart->p; | 411 | struct plat_serial8250_port *p = uart->p; |
355 | int ret; | 412 | int ret; |
356 | 413 | ||
357 | uart->can_sleep = 0; | 414 | uart->can_sleep = 0; |
358 | uart->timeout = sleep_timeout; | 415 | uart->timeout = DEFAULT_TIMEOUT; |
359 | setup_timer(&uart->timer, omap_uart_idle_timer, | 416 | setup_timer(&uart->timer, omap_uart_idle_timer, |
360 | (unsigned long) uart); | 417 | (unsigned long) uart); |
361 | mod_timer(&uart->timer, jiffies + uart->timeout); | 418 | mod_timer(&uart->timer, jiffies + uart->timeout); |
@@ -413,76 +470,101 @@ static void omap_uart_idle_init(struct omap_uart_state *uart) | |||
413 | uart->padconf = 0; | 470 | uart->padconf = 0; |
414 | } | 471 | } |
415 | 472 | ||
416 | /* Set wake-enable bit */ | ||
417 | if (uart->wk_en && uart->wk_mask) { | ||
418 | v = __raw_readl(uart->wk_en); | ||
419 | v |= uart->wk_mask; | ||
420 | __raw_writel(v, uart->wk_en); | ||
421 | } | ||
422 | |||
423 | /* Ensure IOPAD wake-enables are set */ | ||
424 | if (cpu_is_omap34xx() && uart->padconf) { | ||
425 | u16 v; | ||
426 | |||
427 | v = omap_ctrl_readw(uart->padconf); | ||
428 | v |= OMAP3_PADCONF_WAKEUPENABLE0; | ||
429 | omap_ctrl_writew(v, uart->padconf); | ||
430 | } | ||
431 | |||
432 | p->flags |= UPF_SHARE_IRQ; | 473 | p->flags |= UPF_SHARE_IRQ; |
433 | ret = request_irq(p->irq, omap_uart_interrupt, IRQF_SHARED, | 474 | ret = request_irq(p->irq, omap_uart_interrupt, IRQF_SHARED, |
434 | "serial idle", (void *)uart); | 475 | "serial idle", (void *)uart); |
435 | WARN_ON(ret); | 476 | WARN_ON(ret); |
436 | } | 477 | } |
437 | 478 | ||
438 | static ssize_t sleep_timeout_show(struct kobject *kobj, | 479 | void omap_uart_enable_irqs(int enable) |
439 | struct kobj_attribute *attr, | 480 | { |
481 | int ret; | ||
482 | struct omap_uart_state *uart; | ||
483 | |||
484 | list_for_each_entry(uart, &uart_list, node) { | ||
485 | if (enable) | ||
486 | ret = request_irq(uart->p->irq, omap_uart_interrupt, | ||
487 | IRQF_SHARED, "serial idle", (void *)uart); | ||
488 | else | ||
489 | free_irq(uart->p->irq, (void *)uart); | ||
490 | } | ||
491 | } | ||
492 | |||
493 | static ssize_t sleep_timeout_show(struct device *dev, | ||
494 | struct device_attribute *attr, | ||
440 | char *buf) | 495 | char *buf) |
441 | { | 496 | { |
442 | return sprintf(buf, "%u\n", sleep_timeout / HZ); | 497 | struct platform_device *pdev = container_of(dev, |
498 | struct platform_device, dev); | ||
499 | struct omap_uart_state *uart = container_of(pdev, | ||
500 | struct omap_uart_state, pdev); | ||
501 | |||
502 | return sprintf(buf, "%u\n", uart->timeout / HZ); | ||
443 | } | 503 | } |
444 | 504 | ||
445 | static ssize_t sleep_timeout_store(struct kobject *kobj, | 505 | static ssize_t sleep_timeout_store(struct device *dev, |
446 | struct kobj_attribute *attr, | 506 | struct device_attribute *attr, |
447 | const char *buf, size_t n) | 507 | const char *buf, size_t n) |
448 | { | 508 | { |
449 | struct omap_uart_state *uart; | 509 | struct platform_device *pdev = container_of(dev, |
510 | struct platform_device, dev); | ||
511 | struct omap_uart_state *uart = container_of(pdev, | ||
512 | struct omap_uart_state, pdev); | ||
450 | unsigned int value; | 513 | unsigned int value; |
451 | 514 | ||
452 | if (sscanf(buf, "%u", &value) != 1) { | 515 | if (sscanf(buf, "%u", &value) != 1) { |
453 | printk(KERN_ERR "sleep_timeout_store: Invalid value\n"); | 516 | printk(KERN_ERR "sleep_timeout_store: Invalid value\n"); |
454 | return -EINVAL; | 517 | return -EINVAL; |
455 | } | 518 | } |
456 | sleep_timeout = value * HZ; | 519 | |
457 | list_for_each_entry(uart, &uart_list, node) { | 520 | uart->timeout = value * HZ; |
458 | uart->timeout = sleep_timeout; | 521 | if (uart->timeout) |
459 | if (uart->timeout) | 522 | mod_timer(&uart->timer, jiffies + uart->timeout); |
460 | mod_timer(&uart->timer, jiffies + uart->timeout); | 523 | else |
461 | else | 524 | /* A zero value means disable timeout feature */ |
462 | /* A zero value means disable timeout feature */ | 525 | omap_uart_block_sleep(uart); |
463 | omap_uart_block_sleep(uart); | 526 | |
464 | } | ||
465 | return n; | 527 | return n; |
466 | } | 528 | } |
467 | 529 | ||
468 | static struct kobj_attribute sleep_timeout_attr = | 530 | DEVICE_ATTR(sleep_timeout, 0644, sleep_timeout_show, sleep_timeout_store); |
469 | __ATTR(sleep_timeout, 0644, sleep_timeout_show, sleep_timeout_store); | 531 | #define DEV_CREATE_FILE(dev, attr) WARN_ON(device_create_file(dev, attr)) |
470 | |||
471 | #else | 532 | #else |
472 | static inline void omap_uart_idle_init(struct omap_uart_state *uart) {} | 533 | static inline void omap_uart_idle_init(struct omap_uart_state *uart) {} |
534 | #define DEV_CREATE_FILE(dev, attr) | ||
473 | #endif /* CONFIG_PM */ | 535 | #endif /* CONFIG_PM */ |
474 | 536 | ||
475 | static struct platform_device serial_device = { | 537 | static struct omap_uart_state omap_uart[OMAP_MAX_NR_PORTS] = { |
476 | .name = "serial8250", | 538 | { |
477 | .id = PLAT8250_DEV_PLATFORM, | 539 | .pdev = { |
478 | .dev = { | 540 | .name = "serial8250", |
479 | .platform_data = serial_platform_data, | 541 | .id = PLAT8250_DEV_PLATFORM, |
542 | .dev = { | ||
543 | .platform_data = serial_platform_data0, | ||
544 | }, | ||
545 | }, | ||
546 | }, { | ||
547 | .pdev = { | ||
548 | .name = "serial8250", | ||
549 | .id = PLAT8250_DEV_PLATFORM1, | ||
550 | .dev = { | ||
551 | .platform_data = serial_platform_data1, | ||
552 | }, | ||
553 | }, | ||
554 | }, { | ||
555 | .pdev = { | ||
556 | .name = "serial8250", | ||
557 | .id = PLAT8250_DEV_PLATFORM2, | ||
558 | .dev = { | ||
559 | .platform_data = serial_platform_data2, | ||
560 | }, | ||
561 | }, | ||
480 | }, | 562 | }, |
481 | }; | 563 | }; |
482 | 564 | ||
483 | void __init omap_serial_init(void) | 565 | void __init omap_serial_init(void) |
484 | { | 566 | { |
485 | int i, err; | 567 | int i; |
486 | const struct omap_uart_config *info; | 568 | const struct omap_uart_config *info; |
487 | char name[16]; | 569 | char name[16]; |
488 | 570 | ||
@@ -496,14 +578,12 @@ void __init omap_serial_init(void) | |||
496 | 578 | ||
497 | if (info == NULL) | 579 | if (info == NULL) |
498 | return; | 580 | return; |
499 | if (cpu_is_omap44xx()) { | ||
500 | for (i = 0; i < OMAP_MAX_NR_PORTS; i++) | ||
501 | serial_platform_data[i].irq += 32; | ||
502 | } | ||
503 | 581 | ||
504 | for (i = 0; i < OMAP_MAX_NR_PORTS; i++) { | 582 | for (i = 0; i < OMAP_MAX_NR_PORTS; i++) { |
505 | struct plat_serial8250_port *p = serial_platform_data + i; | ||
506 | struct omap_uart_state *uart = &omap_uart[i]; | 583 | struct omap_uart_state *uart = &omap_uart[i]; |
584 | struct platform_device *pdev = &uart->pdev; | ||
585 | struct device *dev = &pdev->dev; | ||
586 | struct plat_serial8250_port *p = dev->platform_data; | ||
507 | 587 | ||
508 | if (!(info->enabled_uarts & (1 << i))) { | 588 | if (!(info->enabled_uarts & (1 << i))) { |
509 | p->membase = NULL; | 589 | p->membase = NULL; |
@@ -531,20 +611,21 @@ void __init omap_serial_init(void) | |||
531 | uart->num = i; | 611 | uart->num = i; |
532 | p->private_data = uart; | 612 | p->private_data = uart; |
533 | uart->p = p; | 613 | uart->p = p; |
534 | list_add(&uart->node, &uart_list); | 614 | list_add_tail(&uart->node, &uart_list); |
615 | |||
616 | if (cpu_is_omap44xx()) | ||
617 | p->irq += 32; | ||
535 | 618 | ||
536 | omap_uart_enable_clocks(uart); | 619 | omap_uart_enable_clocks(uart); |
537 | omap_uart_reset(uart); | 620 | omap_uart_reset(uart); |
538 | omap_uart_idle_init(uart); | 621 | omap_uart_idle_init(uart); |
539 | } | ||
540 | |||
541 | err = platform_device_register(&serial_device); | ||
542 | |||
543 | #ifdef CONFIG_PM | ||
544 | if (!err) | ||
545 | err = sysfs_create_file(&serial_device.dev.kobj, | ||
546 | &sleep_timeout_attr.attr); | ||
547 | #endif | ||
548 | 622 | ||
623 | if (WARN_ON(platform_device_register(pdev))) | ||
624 | continue; | ||
625 | if ((cpu_is_omap34xx() && uart->padconf) || | ||
626 | (uart->wk_en && uart->wk_mask)) { | ||
627 | device_init_wakeup(dev, true); | ||
628 | DEV_CREATE_FILE(dev, &dev_attr_sleep_timeout); | ||
629 | } | ||
630 | } | ||
549 | } | 631 | } |
550 | |||