diff options
-rw-r--r-- | arch/arm/mach-s3c2443/Kconfig | 1 | ||||
-rw-r--r-- | arch/arm/mach-s3c2443/clock.c | 452 | ||||
-rw-r--r-- | arch/arm/plat-s3c24xx/Kconfig | 6 | ||||
-rw-r--r-- | arch/arm/plat-s3c24xx/Makefile | 1 | ||||
-rw-r--r-- | arch/arm/plat-s3c24xx/include/plat/s3c2443.h | 19 | ||||
-rw-r--r-- | arch/arm/plat-s3c24xx/s3c2443-clock.c | 472 |
6 files changed, 505 insertions, 446 deletions
diff --git a/arch/arm/mach-s3c2443/Kconfig b/arch/arm/mach-s3c2443/Kconfig index 698140af247c..4fef723126fa 100644 --- a/arch/arm/mach-s3c2443/Kconfig +++ b/arch/arm/mach-s3c2443/Kconfig | |||
@@ -8,6 +8,7 @@ config CPU_S3C2443 | |||
8 | select S3C2443_DMA if S3C2410_DMA | 8 | select S3C2443_DMA if S3C2410_DMA |
9 | select CPU_LLSERIAL_S3C2440 | 9 | select CPU_LLSERIAL_S3C2440 |
10 | select SAMSUNG_CLKSRC | 10 | select SAMSUNG_CLKSRC |
11 | select S3C2443_CLOCK | ||
11 | help | 12 | help |
12 | Support for the S3C2443 SoC from the S3C24XX line | 13 | Support for the S3C2443 SoC from the S3C24XX line |
13 | 14 | ||
diff --git a/arch/arm/mach-s3c2443/clock.c b/arch/arm/mach-s3c2443/clock.c index 76d8d66247c3..83b1aa63d778 100644 --- a/arch/arm/mach-s3c2443/clock.c +++ b/arch/arm/mach-s3c2443/clock.c | |||
@@ -21,6 +21,7 @@ | |||
21 | */ | 21 | */ |
22 | 22 | ||
23 | #include <linux/init.h> | 23 | #include <linux/init.h> |
24 | |||
24 | #include <linux/module.h> | 25 | #include <linux/module.h> |
25 | #include <linux/kernel.h> | 26 | #include <linux/kernel.h> |
26 | #include <linux/list.h> | 27 | #include <linux/list.h> |
@@ -54,111 +55,13 @@ | |||
54 | * set the correct muxing at initialisation | 55 | * set the correct muxing at initialisation |
55 | */ | 56 | */ |
56 | 57 | ||
57 | static int s3c2443_gate(void __iomem *reg, struct clk *clk, int enable) | ||
58 | { | ||
59 | u32 ctrlbit = clk->ctrlbit; | ||
60 | u32 con = __raw_readl(reg); | ||
61 | |||
62 | if (enable) | ||
63 | con |= ctrlbit; | ||
64 | else | ||
65 | con &= ~ctrlbit; | ||
66 | |||
67 | __raw_writel(con, reg); | ||
68 | return 0; | ||
69 | } | ||
70 | |||
71 | static int s3c2443_clkcon_enable_h(struct clk *clk, int enable) | ||
72 | { | ||
73 | return s3c2443_gate(S3C2443_HCLKCON, clk, enable); | ||
74 | } | ||
75 | |||
76 | static int s3c2443_clkcon_enable_p(struct clk *clk, int enable) | ||
77 | { | ||
78 | return s3c2443_gate(S3C2443_PCLKCON, clk, enable); | ||
79 | } | ||
80 | |||
81 | static int s3c2443_clkcon_enable_s(struct clk *clk, int enable) | ||
82 | { | ||
83 | return s3c2443_gate(S3C2443_SCLKCON, clk, enable); | ||
84 | } | ||
85 | |||
86 | /* clock selections */ | 58 | /* clock selections */ |
87 | 59 | ||
88 | /* mpllref is a direct descendant of clk_xtal by default, but it is not | ||
89 | * elided as the EPLL can be either sourced by the XTAL or EXTCLK and as | ||
90 | * such directly equating the two source clocks is impossible. | ||
91 | */ | ||
92 | static struct clk clk_mpllref = { | ||
93 | .name = "mpllref", | ||
94 | .parent = &clk_xtal, | ||
95 | .id = -1, | ||
96 | }; | ||
97 | |||
98 | static struct clk clk_i2s_ext = { | 60 | static struct clk clk_i2s_ext = { |
99 | .name = "i2s-ext", | 61 | .name = "i2s-ext", |
100 | .id = -1, | 62 | .id = -1, |
101 | }; | 63 | }; |
102 | 64 | ||
103 | static struct clk *clk_epllref_sources[] = { | ||
104 | [0] = &clk_mpllref, | ||
105 | [1] = &clk_mpllref, | ||
106 | [2] = &clk_xtal, | ||
107 | [3] = &clk_ext, | ||
108 | }; | ||
109 | |||
110 | static struct clksrc_clk clk_epllref = { | ||
111 | .clk = { | ||
112 | .name = "epllref", | ||
113 | .id = -1, | ||
114 | }, | ||
115 | .sources = &(struct clksrc_sources) { | ||
116 | .sources = clk_epllref_sources, | ||
117 | .nr_sources = ARRAY_SIZE(clk_epllref_sources), | ||
118 | }, | ||
119 | .reg_src = { .reg = S3C2443_CLKSRC, .size = 2, .shift = 7 }, | ||
120 | }; | ||
121 | |||
122 | static unsigned long s3c2443_getrate_mdivclk(struct clk *clk) | ||
123 | { | ||
124 | unsigned long parent_rate = clk_get_rate(clk->parent); | ||
125 | unsigned long div = __raw_readl(S3C2443_CLKDIV0); | ||
126 | |||
127 | div &= S3C2443_CLKDIV0_EXTDIV_MASK; | ||
128 | div >>= (S3C2443_CLKDIV0_EXTDIV_SHIFT-1); /* x2 */ | ||
129 | |||
130 | return parent_rate / (div + 1); | ||
131 | } | ||
132 | |||
133 | static struct clk clk_mdivclk = { | ||
134 | .name = "mdivclk", | ||
135 | .parent = &clk_mpllref, | ||
136 | .id = -1, | ||
137 | .ops = &(struct clk_ops) { | ||
138 | .get_rate = s3c2443_getrate_mdivclk, | ||
139 | }, | ||
140 | }; | ||
141 | |||
142 | static struct clk *clk_msysclk_sources[] = { | ||
143 | [0] = &clk_mpllref, | ||
144 | [1] = &clk_mpll, | ||
145 | [2] = &clk_mdivclk, | ||
146 | [3] = &clk_mpllref, | ||
147 | }; | ||
148 | |||
149 | static struct clksrc_clk clk_msysclk = { | ||
150 | .clk = { | ||
151 | .name = "msysclk", | ||
152 | .parent = &clk_xtal, | ||
153 | .id = -1, | ||
154 | }, | ||
155 | .sources = &(struct clksrc_sources) { | ||
156 | .sources = clk_msysclk_sources, | ||
157 | .nr_sources = ARRAY_SIZE(clk_msysclk_sources), | ||
158 | }, | ||
159 | .reg_src = { .reg = S3C2443_CLKSRC, .size = 2, .shift = 3 }, | ||
160 | }; | ||
161 | |||
162 | /* armdiv | 65 | /* armdiv |
163 | * | 66 | * |
164 | * this clock is sourced from msysclk and can have a number of | 67 | * this clock is sourced from msysclk and can have a number of |
@@ -266,44 +169,6 @@ static struct clksrc_clk clk_arm = { | |||
266 | .reg_src = { .reg = S3C2443_CLKDIV0, .size = 1, .shift = 13 }, | 169 | .reg_src = { .reg = S3C2443_CLKDIV0, .size = 1, .shift = 13 }, |
267 | }; | 170 | }; |
268 | 171 | ||
269 | /* esysclk | ||
270 | * | ||
271 | * this is sourced from either the EPLL or the EPLLref clock | ||
272 | */ | ||
273 | |||
274 | static struct clk *clk_sysclk_sources[] = { | ||
275 | [0] = &clk_epllref.clk, | ||
276 | [1] = &clk_epll, | ||
277 | }; | ||
278 | |||
279 | static struct clksrc_clk clk_esysclk = { | ||
280 | .clk = { | ||
281 | .name = "esysclk", | ||
282 | .parent = &clk_epll, | ||
283 | .id = -1, | ||
284 | }, | ||
285 | .sources = &(struct clksrc_sources) { | ||
286 | .sources = clk_sysclk_sources, | ||
287 | .nr_sources = ARRAY_SIZE(clk_sysclk_sources), | ||
288 | }, | ||
289 | .reg_src = { .reg = S3C2443_CLKSRC, .size = 1, .shift = 6 }, | ||
290 | }; | ||
291 | |||
292 | /* uartclk | ||
293 | * | ||
294 | * UART baud-rate clock sourced from esysclk via a divisor | ||
295 | */ | ||
296 | |||
297 | static struct clksrc_clk clk_uart = { | ||
298 | .clk = { | ||
299 | .name = "uartclk", | ||
300 | .id = -1, | ||
301 | .parent = &clk_esysclk.clk, | ||
302 | }, | ||
303 | .reg_div = { .reg = S3C2443_CLKDIV1, .size = 4, .shift = 8 }, | ||
304 | }; | ||
305 | |||
306 | |||
307 | /* hsspi | 172 | /* hsspi |
308 | * | 173 | * |
309 | * high-speed spi clock, sourced from esysclk | 174 | * high-speed spi clock, sourced from esysclk |
@@ -320,21 +185,6 @@ static struct clksrc_clk clk_hsspi = { | |||
320 | .reg_div = { .reg = S3C2443_CLKDIV1, .size = 2, .shift = 4 }, | 185 | .reg_div = { .reg = S3C2443_CLKDIV1, .size = 2, .shift = 4 }, |
321 | }; | 186 | }; |
322 | 187 | ||
323 | /* usbhost | ||
324 | * | ||
325 | * usb host bus-clock, usually 48MHz to provide USB bus clock timing | ||
326 | */ | ||
327 | |||
328 | static struct clksrc_clk clk_usb_bus_host = { | ||
329 | .clk = { | ||
330 | .name = "usb-bus-host-parent", | ||
331 | .id = -1, | ||
332 | .parent = &clk_esysclk.clk, | ||
333 | .ctrlbit = S3C2443_SCLKCON_USBHOST, | ||
334 | .enable = s3c2443_clkcon_enable_s, | ||
335 | }, | ||
336 | .reg_div = { .reg = S3C2443_CLKDIV1, .size = 2, .shift = 4 }, | ||
337 | }; | ||
338 | 188 | ||
339 | /* clk_hsmcc_div | 189 | /* clk_hsmcc_div |
340 | * | 190 | * |
@@ -433,89 +283,16 @@ static struct clksrc_clk clk_i2s = { | |||
433 | .reg_src = { .reg = S3C2443_CLKSRC, .size = 2, .shift = 14 }, | 283 | .reg_src = { .reg = S3C2443_CLKSRC, .size = 2, .shift = 14 }, |
434 | }; | 284 | }; |
435 | 285 | ||
436 | /* cam-if | ||
437 | * | ||
438 | * camera interface bus-clock, divided down from esysclk | ||
439 | */ | ||
440 | |||
441 | static struct clksrc_clk clk_cam = { | ||
442 | .clk = { | ||
443 | .name = "camif-upll", /* same as 2440 name */ | ||
444 | .id = -1, | ||
445 | .parent = &clk_esysclk.clk, | ||
446 | .ctrlbit = S3C2443_SCLKCON_CAMCLK, | ||
447 | .enable = s3c2443_clkcon_enable_s, | ||
448 | }, | ||
449 | .reg_div = { .reg = S3C2443_CLKDIV1, .size = 4, .shift = 26 }, | ||
450 | }; | ||
451 | |||
452 | /* display-if | ||
453 | * | ||
454 | * display interface clock, divided from esysclk | ||
455 | */ | ||
456 | |||
457 | static struct clksrc_clk clk_display = { | ||
458 | .clk = { | ||
459 | .name = "display-if", | ||
460 | .id = -1, | ||
461 | .parent = &clk_esysclk.clk, | ||
462 | .ctrlbit = S3C2443_SCLKCON_DISPCLK, | ||
463 | .enable = s3c2443_clkcon_enable_s, | ||
464 | }, | ||
465 | .reg_div = { .reg = S3C2443_CLKDIV1, .size = 8, .shift = 16 }, | ||
466 | }; | ||
467 | |||
468 | /* prediv | ||
469 | * | ||
470 | * this divides the msysclk down to pass to h/p/etc. | ||
471 | */ | ||
472 | |||
473 | static unsigned long s3c2443_prediv_getrate(struct clk *clk) | ||
474 | { | ||
475 | unsigned long rate = clk_get_rate(clk->parent); | ||
476 | unsigned long clkdiv0 = __raw_readl(S3C2443_CLKDIV0); | ||
477 | |||
478 | clkdiv0 &= S3C2443_CLKDIV0_PREDIV_MASK; | ||
479 | clkdiv0 >>= S3C2443_CLKDIV0_PREDIV_SHIFT; | ||
480 | |||
481 | return rate / (clkdiv0 + 1); | ||
482 | } | ||
483 | |||
484 | static struct clk clk_prediv = { | ||
485 | .name = "prediv", | ||
486 | .id = -1, | ||
487 | .parent = &clk_msysclk.clk, | ||
488 | .ops = &(struct clk_ops) { | ||
489 | .get_rate = s3c2443_prediv_getrate, | ||
490 | }, | ||
491 | }; | ||
492 | |||
493 | /* standard clock definitions */ | 286 | /* standard clock definitions */ |
494 | 287 | ||
495 | static struct clk init_clocks_off[] = { | 288 | static struct clk init_clocks_off[] = { |
496 | { | 289 | { |
497 | .name = "nand", | ||
498 | .id = -1, | ||
499 | .parent = &clk_h, | ||
500 | }, { | ||
501 | .name = "sdi", | 290 | .name = "sdi", |
502 | .id = -1, | 291 | .id = -1, |
503 | .parent = &clk_p, | 292 | .parent = &clk_p, |
504 | .enable = s3c2443_clkcon_enable_p, | 293 | .enable = s3c2443_clkcon_enable_p, |
505 | .ctrlbit = S3C2443_PCLKCON_SDI, | 294 | .ctrlbit = S3C2443_PCLKCON_SDI, |
506 | }, { | 295 | }, { |
507 | .name = "adc", | ||
508 | .id = -1, | ||
509 | .parent = &clk_p, | ||
510 | .enable = s3c2443_clkcon_enable_p, | ||
511 | .ctrlbit = S3C2443_PCLKCON_ADC, | ||
512 | }, { | ||
513 | .name = "i2c", | ||
514 | .id = -1, | ||
515 | .parent = &clk_p, | ||
516 | .enable = s3c2443_clkcon_enable_p, | ||
517 | .ctrlbit = S3C2443_PCLKCON_IIC, | ||
518 | }, { | ||
519 | .name = "iis", | 296 | .name = "iis", |
520 | .id = -1, | 297 | .id = -1, |
521 | .parent = &clk_p, | 298 | .parent = &clk_p, |
@@ -537,179 +314,12 @@ static struct clk init_clocks_off[] = { | |||
537 | }; | 314 | }; |
538 | 315 | ||
539 | static struct clk init_clocks[] = { | 316 | static struct clk init_clocks[] = { |
540 | { | ||
541 | .name = "dma", | ||
542 | .id = 0, | ||
543 | .parent = &clk_h, | ||
544 | .enable = s3c2443_clkcon_enable_h, | ||
545 | .ctrlbit = S3C2443_HCLKCON_DMA0, | ||
546 | }, { | ||
547 | .name = "dma", | ||
548 | .id = 1, | ||
549 | .parent = &clk_h, | ||
550 | .enable = s3c2443_clkcon_enable_h, | ||
551 | .ctrlbit = S3C2443_HCLKCON_DMA1, | ||
552 | }, { | ||
553 | .name = "dma", | ||
554 | .id = 2, | ||
555 | .parent = &clk_h, | ||
556 | .enable = s3c2443_clkcon_enable_h, | ||
557 | .ctrlbit = S3C2443_HCLKCON_DMA2, | ||
558 | }, { | ||
559 | .name = "dma", | ||
560 | .id = 3, | ||
561 | .parent = &clk_h, | ||
562 | .enable = s3c2443_clkcon_enable_h, | ||
563 | .ctrlbit = S3C2443_HCLKCON_DMA3, | ||
564 | }, { | ||
565 | .name = "dma", | ||
566 | .id = 4, | ||
567 | .parent = &clk_h, | ||
568 | .enable = s3c2443_clkcon_enable_h, | ||
569 | .ctrlbit = S3C2443_HCLKCON_DMA4, | ||
570 | }, { | ||
571 | .name = "dma", | ||
572 | .id = 5, | ||
573 | .parent = &clk_h, | ||
574 | .enable = s3c2443_clkcon_enable_h, | ||
575 | .ctrlbit = S3C2443_HCLKCON_DMA5, | ||
576 | }, { | ||
577 | .name = "lcd", | ||
578 | .id = -1, | ||
579 | .parent = &clk_h, | ||
580 | .enable = s3c2443_clkcon_enable_h, | ||
581 | .ctrlbit = S3C2443_HCLKCON_LCDC, | ||
582 | }, { | ||
583 | .name = "gpio", | ||
584 | .id = -1, | ||
585 | .parent = &clk_p, | ||
586 | .enable = s3c2443_clkcon_enable_p, | ||
587 | .ctrlbit = S3C2443_PCLKCON_GPIO, | ||
588 | }, { | ||
589 | .name = "usb-host", | ||
590 | .id = -1, | ||
591 | .parent = &clk_h, | ||
592 | .enable = s3c2443_clkcon_enable_h, | ||
593 | .ctrlbit = S3C2443_HCLKCON_USBH, | ||
594 | }, { | ||
595 | .name = "usb-device", | ||
596 | .id = -1, | ||
597 | .parent = &clk_h, | ||
598 | .enable = s3c2443_clkcon_enable_h, | ||
599 | .ctrlbit = S3C2443_HCLKCON_USBD, | ||
600 | }, { | ||
601 | .name = "hsmmc", | ||
602 | .id = -1, | ||
603 | .parent = &clk_h, | ||
604 | .enable = s3c2443_clkcon_enable_h, | ||
605 | .ctrlbit = S3C2443_HCLKCON_HSMMC, | ||
606 | }, { | ||
607 | .name = "cfc", | ||
608 | .id = -1, | ||
609 | .parent = &clk_h, | ||
610 | .enable = s3c2443_clkcon_enable_h, | ||
611 | .ctrlbit = S3C2443_HCLKCON_CFC, | ||
612 | }, { | ||
613 | .name = "ssmc", | ||
614 | .id = -1, | ||
615 | .parent = &clk_h, | ||
616 | .enable = s3c2443_clkcon_enable_h, | ||
617 | .ctrlbit = S3C2443_HCLKCON_SSMC, | ||
618 | }, { | ||
619 | .name = "timers", | ||
620 | .id = -1, | ||
621 | .parent = &clk_p, | ||
622 | .enable = s3c2443_clkcon_enable_p, | ||
623 | .ctrlbit = S3C2443_PCLKCON_PWMT, | ||
624 | }, { | ||
625 | .name = "uart", | ||
626 | .id = 0, | ||
627 | .parent = &clk_p, | ||
628 | .enable = s3c2443_clkcon_enable_p, | ||
629 | .ctrlbit = S3C2443_PCLKCON_UART0, | ||
630 | }, { | ||
631 | .name = "uart", | ||
632 | .id = 1, | ||
633 | .parent = &clk_p, | ||
634 | .enable = s3c2443_clkcon_enable_p, | ||
635 | .ctrlbit = S3C2443_PCLKCON_UART1, | ||
636 | }, { | ||
637 | .name = "uart", | ||
638 | .id = 2, | ||
639 | .parent = &clk_p, | ||
640 | .enable = s3c2443_clkcon_enable_p, | ||
641 | .ctrlbit = S3C2443_PCLKCON_UART2, | ||
642 | }, { | ||
643 | .name = "uart", | ||
644 | .id = 3, | ||
645 | .parent = &clk_p, | ||
646 | .enable = s3c2443_clkcon_enable_p, | ||
647 | .ctrlbit = S3C2443_PCLKCON_UART3, | ||
648 | }, { | ||
649 | .name = "rtc", | ||
650 | .id = -1, | ||
651 | .parent = &clk_p, | ||
652 | .enable = s3c2443_clkcon_enable_p, | ||
653 | .ctrlbit = S3C2443_PCLKCON_RTC, | ||
654 | }, { | ||
655 | .name = "watchdog", | ||
656 | .id = -1, | ||
657 | .parent = &clk_p, | ||
658 | .ctrlbit = S3C2443_PCLKCON_WDT, | ||
659 | }, { | ||
660 | .name = "usb-bus-host", | ||
661 | .id = -1, | ||
662 | .parent = &clk_usb_bus_host.clk, | ||
663 | }, { | ||
664 | .name = "ac97", | ||
665 | .id = -1, | ||
666 | .parent = &clk_p, | ||
667 | .ctrlbit = S3C2443_PCLKCON_AC97, | ||
668 | } | ||
669 | }; | 317 | }; |
670 | 318 | ||
671 | /* clocks to add where we need to check their parentage */ | ||
672 | |||
673 | static struct clksrc_clk __initdata *init_list[] = { | ||
674 | &clk_epllref, /* should be first */ | ||
675 | &clk_esysclk, | ||
676 | &clk_msysclk, | ||
677 | &clk_arm, | ||
678 | &clk_i2s_eplldiv, | ||
679 | &clk_i2s, | ||
680 | &clk_cam, | ||
681 | &clk_uart, | ||
682 | &clk_display, | ||
683 | &clk_hsmmc_div, | ||
684 | &clk_usb_bus_host, | ||
685 | }; | ||
686 | |||
687 | static void __init s3c2443_clk_initparents(void) | ||
688 | { | ||
689 | int ptr; | ||
690 | |||
691 | for (ptr = 0; ptr < ARRAY_SIZE(init_list); ptr++) | ||
692 | s3c_set_clksrc(init_list[ptr], true); | ||
693 | } | ||
694 | |||
695 | static inline unsigned long s3c2443_get_hdiv(unsigned long clkcon0) | ||
696 | { | ||
697 | clkcon0 &= S3C2443_CLKDIV0_HCLKDIV_MASK; | ||
698 | |||
699 | return clkcon0 + 1; | ||
700 | } | ||
701 | |||
702 | /* clocks to add straight away */ | 319 | /* clocks to add straight away */ |
703 | 320 | ||
704 | static struct clksrc_clk *clksrcs[] __initdata = { | 321 | static struct clksrc_clk *clksrcs[] __initdata = { |
705 | &clk_usb_bus_host, | ||
706 | &clk_epllref, | ||
707 | &clk_esysclk, | ||
708 | &clk_msysclk, | ||
709 | &clk_arm, | 322 | &clk_arm, |
710 | &clk_uart, | ||
711 | &clk_display, | ||
712 | &clk_cam, | ||
713 | &clk_i2s_eplldiv, | 323 | &clk_i2s_eplldiv, |
714 | &clk_i2s, | 324 | &clk_i2s, |
715 | &clk_hsspi, | 325 | &clk_hsspi, |
@@ -717,46 +327,13 @@ static struct clksrc_clk *clksrcs[] __initdata = { | |||
717 | }; | 327 | }; |
718 | 328 | ||
719 | static struct clk *clks[] __initdata = { | 329 | static struct clk *clks[] __initdata = { |
720 | &clk_ext, | ||
721 | &clk_epll, | ||
722 | &clk_usb_bus, | ||
723 | &clk_mpllref, | ||
724 | &clk_hsmmc, | 330 | &clk_hsmmc, |
725 | &clk_armdiv, | 331 | &clk_armdiv, |
726 | &clk_prediv, | ||
727 | }; | 332 | }; |
728 | 333 | ||
729 | void __init_or_cpufreq s3c2443_setup_clocks(void) | 334 | void __init_or_cpufreq s3c2443_setup_clocks(void) |
730 | { | 335 | { |
731 | unsigned long mpllcon = __raw_readl(S3C2443_MPLLCON); | 336 | s3c2443_common_setup_clocks(s3c2443_get_mpll, s3c2443_fclk_div); |
732 | unsigned long clkdiv0 = __raw_readl(S3C2443_CLKDIV0); | ||
733 | struct clk *xtal_clk; | ||
734 | unsigned long xtal; | ||
735 | unsigned long pll; | ||
736 | unsigned long fclk; | ||
737 | unsigned long hclk; | ||
738 | unsigned long pclk; | ||
739 | |||
740 | xtal_clk = clk_get(NULL, "xtal"); | ||
741 | xtal = clk_get_rate(xtal_clk); | ||
742 | clk_put(xtal_clk); | ||
743 | |||
744 | pll = s3c2443_get_mpll(mpllcon, xtal); | ||
745 | clk_msysclk.clk.rate = pll; | ||
746 | |||
747 | fclk = pll / s3c2443_fclk_div(clkdiv0); | ||
748 | hclk = s3c2443_prediv_getrate(&clk_prediv); | ||
749 | hclk /= s3c2443_get_hdiv(clkdiv0); | ||
750 | pclk = hclk / ((clkdiv0 & S3C2443_CLKDIV0_HALF_PCLK) ? 2 : 1); | ||
751 | |||
752 | s3c24xx_setup_clocks(fclk, hclk, pclk); | ||
753 | |||
754 | printk("S3C2443: mpll %s %ld.%03ld MHz, cpu %ld.%03ld MHz, mem %ld.%03ld MHz, pclk %ld.%03ld MHz\n", | ||
755 | (mpllcon & S3C2443_PLLCON_OFF) ? "off":"on", | ||
756 | print_mhz(pll), print_mhz(fclk), | ||
757 | print_mhz(hclk), print_mhz(pclk)); | ||
758 | |||
759 | s3c24xx_setup_clocks(fclk, hclk, pclk); | ||
760 | } | 337 | } |
761 | 338 | ||
762 | void __init s3c2443_init_clocks(int xtal) | 339 | void __init s3c2443_init_clocks(int xtal) |
@@ -764,35 +341,18 @@ void __init s3c2443_init_clocks(int xtal) | |||
764 | unsigned long epllcon = __raw_readl(S3C2443_EPLLCON); | 341 | unsigned long epllcon = __raw_readl(S3C2443_EPLLCON); |
765 | int ptr; | 342 | int ptr; |
766 | 343 | ||
767 | /* s3c2443 parents h and p clocks from prediv */ | 344 | clk_epll.rate = s3c2443_get_epll(epllcon, xtal); |
768 | clk_h.parent = &clk_prediv; | 345 | clk_epll.parent = &clk_epllref.clk; |
769 | clk_p.parent = &clk_prediv; | 346 | |
347 | s3c2443_common_init_clocks(xtal, s3c2443_get_mpll, s3c2443_fclk_div); | ||
770 | 348 | ||
771 | s3c24xx_register_baseclocks(xtal); | ||
772 | s3c2443_setup_clocks(); | 349 | s3c2443_setup_clocks(); |
773 | s3c2443_clk_initparents(); | ||
774 | 350 | ||
775 | s3c24xx_register_clocks(clks, ARRAY_SIZE(clks)); | 351 | s3c24xx_register_clocks(clks, ARRAY_SIZE(clks)); |
776 | 352 | ||
777 | for (ptr = 0; ptr < ARRAY_SIZE(clksrcs); ptr++) | 353 | for (ptr = 0; ptr < ARRAY_SIZE(clksrcs); ptr++) |
778 | s3c_register_clksrc(clksrcs[ptr], 1); | 354 | s3c_register_clksrc(clksrcs[ptr], 1); |
779 | 355 | ||
780 | clk_epll.rate = s3c2443_get_epll(epllcon, xtal); | ||
781 | clk_epll.parent = &clk_epllref.clk; | ||
782 | clk_usb_bus.parent = &clk_usb_bus_host.clk; | ||
783 | |||
784 | /* ensure usb bus clock is within correct rate of 48MHz */ | ||
785 | |||
786 | if (clk_get_rate(&clk_usb_bus_host.clk) != (48 * 1000 * 1000)) { | ||
787 | printk(KERN_INFO "Warning: USB host bus not at 48MHz\n"); | ||
788 | clk_set_rate(&clk_usb_bus_host.clk, 48*1000*1000); | ||
789 | } | ||
790 | |||
791 | printk("S3C2443: epll %s %ld.%03ld MHz, usb-bus %ld.%03ld MHz\n", | ||
792 | (epllcon & S3C2443_PLLCON_OFF) ? "off":"on", | ||
793 | print_mhz(clk_get_rate(&clk_epll)), | ||
794 | print_mhz(clk_get_rate(&clk_usb_bus))); | ||
795 | |||
796 | /* register clocks from clock array */ | 356 | /* register clocks from clock array */ |
797 | 357 | ||
798 | s3c_register_clocks(init_clocks, ARRAY_SIZE(init_clocks)); | 358 | s3c_register_clocks(init_clocks, ARRAY_SIZE(init_clocks)); |
diff --git a/arch/arm/plat-s3c24xx/Kconfig b/arch/arm/plat-s3c24xx/Kconfig index a830fad6f89e..3ce8f010b3c6 100644 --- a/arch/arm/plat-s3c24xx/Kconfig +++ b/arch/arm/plat-s3c24xx/Kconfig | |||
@@ -45,6 +45,12 @@ config S3C2410_CLOCK | |||
45 | Clock code for the S3C2410, and similar processors which | 45 | Clock code for the S3C2410, and similar processors which |
46 | is currently includes the S3C2410, S3C2440, S3C2442. | 46 | is currently includes the S3C2410, S3C2440, S3C2442. |
47 | 47 | ||
48 | config S3C2443_CLOCK | ||
49 | bool | ||
50 | help | ||
51 | Clock code for the S3C2443 and similar processors, which includes | ||
52 | the S3C2416 and S3C2450. | ||
53 | |||
48 | config S3C24XX_DCLK | 54 | config S3C24XX_DCLK |
49 | bool | 55 | bool |
50 | help | 56 | help |
diff --git a/arch/arm/plat-s3c24xx/Makefile b/arch/arm/plat-s3c24xx/Makefile index c2237c41141f..44aea8868f89 100644 --- a/arch/arm/plat-s3c24xx/Makefile +++ b/arch/arm/plat-s3c24xx/Makefile | |||
@@ -30,6 +30,7 @@ obj-$(CONFIG_PM) += pm.o | |||
30 | obj-$(CONFIG_PM) += irq-pm.o | 30 | obj-$(CONFIG_PM) += irq-pm.o |
31 | obj-$(CONFIG_PM) += sleep.o | 31 | obj-$(CONFIG_PM) += sleep.o |
32 | obj-$(CONFIG_S3C2410_CLOCK) += s3c2410-clock.o | 32 | obj-$(CONFIG_S3C2410_CLOCK) += s3c2410-clock.o |
33 | obj-$(CONFIG_S3C2443_CLOCK) += s3c2443-clock.o | ||
33 | obj-$(CONFIG_S3C2410_DMA) += dma.o | 34 | obj-$(CONFIG_S3C2410_DMA) += dma.o |
34 | obj-$(CONFIG_S3C2410_IOTIMING) += s3c2410-iotiming.o | 35 | obj-$(CONFIG_S3C2410_IOTIMING) += s3c2410-iotiming.o |
35 | obj-$(CONFIG_S3C2412_IOTIMING) += s3c2412-iotiming.o | 36 | obj-$(CONFIG_S3C2412_IOTIMING) += s3c2412-iotiming.o |
diff --git a/arch/arm/plat-s3c24xx/include/plat/s3c2443.h b/arch/arm/plat-s3c24xx/include/plat/s3c2443.h index 815b107ed890..a19715feb798 100644 --- a/arch/arm/plat-s3c24xx/include/plat/s3c2443.h +++ b/arch/arm/plat-s3c24xx/include/plat/s3c2443.h | |||
@@ -30,3 +30,22 @@ extern int s3c2443_baseclk_add(void); | |||
30 | #define s3c2443_map_io NULL | 30 | #define s3c2443_map_io NULL |
31 | #define s3c2443_init NULL | 31 | #define s3c2443_init NULL |
32 | #endif | 32 | #endif |
33 | |||
34 | /* common code used by s3c2443 and others. | ||
35 | * note, not to be used outside of arch/arm/mach-s3c* */ | ||
36 | |||
37 | struct clk; /* some files don't need clk.h otherwise */ | ||
38 | |||
39 | typedef unsigned int (*pll_fn)(unsigned int reg, unsigned int base); | ||
40 | typedef unsigned int (*fdiv_fn)(unsigned long clkcon0); | ||
41 | |||
42 | extern void s3c2443_common_setup_clocks(pll_fn get_mpll, fdiv_fn fdiv); | ||
43 | extern void s3c2443_common_init_clocks(int xtal, pll_fn get_mpll, fdiv_fn fdiv); | ||
44 | |||
45 | extern int s3c2443_clkcon_enable_h(struct clk *clk, int enable); | ||
46 | extern int s3c2443_clkcon_enable_p(struct clk *clk, int enable); | ||
47 | extern int s3c2443_clkcon_enable_s(struct clk *clk, int enable); | ||
48 | |||
49 | extern struct clksrc_clk clk_epllref; | ||
50 | extern struct clksrc_clk clk_esysclk; | ||
51 | extern struct clksrc_clk clk_msysclk; | ||
diff --git a/arch/arm/plat-s3c24xx/s3c2443-clock.c b/arch/arm/plat-s3c24xx/s3c2443-clock.c new file mode 100644 index 000000000000..461f070eb62d --- /dev/null +++ b/arch/arm/plat-s3c24xx/s3c2443-clock.c | |||
@@ -0,0 +1,472 @@ | |||
1 | /* linux/arch/arm/plat-s3c24xx/s3c2443-clock.c | ||
2 | * | ||
3 | * Copyright (c) 2007, 2010 Simtec Electronics | ||
4 | * Ben Dooks <ben@simtec.co.uk> | ||
5 | * | ||
6 | * S3C2443 Clock control suport - common code | ||
7 | */ | ||
8 | |||
9 | #include <linux/init.h> | ||
10 | #include <linux/clk.h> | ||
11 | #include <linux/io.h> | ||
12 | |||
13 | #include <mach/regs-s3c2443-clock.h> | ||
14 | |||
15 | #include <plat/s3c2443.h> | ||
16 | #include <plat/clock.h> | ||
17 | #include <plat/clock-clksrc.h> | ||
18 | #include <plat/cpu.h> | ||
19 | |||
20 | #include <plat/cpu-freq.h> | ||
21 | |||
22 | |||
23 | static int s3c2443_gate(void __iomem *reg, struct clk *clk, int enable) | ||
24 | { | ||
25 | u32 ctrlbit = clk->ctrlbit; | ||
26 | u32 con = __raw_readl(reg); | ||
27 | |||
28 | if (enable) | ||
29 | con |= ctrlbit; | ||
30 | else | ||
31 | con &= ~ctrlbit; | ||
32 | |||
33 | __raw_writel(con, reg); | ||
34 | return 0; | ||
35 | } | ||
36 | |||
37 | int s3c2443_clkcon_enable_h(struct clk *clk, int enable) | ||
38 | { | ||
39 | return s3c2443_gate(S3C2443_HCLKCON, clk, enable); | ||
40 | } | ||
41 | |||
42 | int s3c2443_clkcon_enable_p(struct clk *clk, int enable) | ||
43 | { | ||
44 | return s3c2443_gate(S3C2443_PCLKCON, clk, enable); | ||
45 | } | ||
46 | |||
47 | int s3c2443_clkcon_enable_s(struct clk *clk, int enable) | ||
48 | { | ||
49 | return s3c2443_gate(S3C2443_SCLKCON, clk, enable); | ||
50 | } | ||
51 | |||
52 | /* mpllref is a direct descendant of clk_xtal by default, but it is not | ||
53 | * elided as the EPLL can be either sourced by the XTAL or EXTCLK and as | ||
54 | * such directly equating the two source clocks is impossible. | ||
55 | */ | ||
56 | struct clk clk_mpllref = { | ||
57 | .name = "mpllref", | ||
58 | .parent = &clk_xtal, | ||
59 | .id = -1, | ||
60 | }; | ||
61 | |||
62 | static struct clk *clk_epllref_sources[] = { | ||
63 | [0] = &clk_mpllref, | ||
64 | [1] = &clk_mpllref, | ||
65 | [2] = &clk_xtal, | ||
66 | [3] = &clk_ext, | ||
67 | }; | ||
68 | |||
69 | struct clksrc_clk clk_epllref = { | ||
70 | .clk = { | ||
71 | .name = "epllref", | ||
72 | .id = -1, | ||
73 | }, | ||
74 | .sources = &(struct clksrc_sources) { | ||
75 | .sources = clk_epllref_sources, | ||
76 | .nr_sources = ARRAY_SIZE(clk_epllref_sources), | ||
77 | }, | ||
78 | .reg_src = { .reg = S3C2443_CLKSRC, .size = 2, .shift = 7 }, | ||
79 | }; | ||
80 | |||
81 | /* esysclk | ||
82 | * | ||
83 | * this is sourced from either the EPLL or the EPLLref clock | ||
84 | */ | ||
85 | |||
86 | static struct clk *clk_sysclk_sources[] = { | ||
87 | [0] = &clk_epllref.clk, | ||
88 | [1] = &clk_epll, | ||
89 | }; | ||
90 | |||
91 | struct clksrc_clk clk_esysclk = { | ||
92 | .clk = { | ||
93 | .name = "esysclk", | ||
94 | .parent = &clk_epll, | ||
95 | .id = -1, | ||
96 | }, | ||
97 | .sources = &(struct clksrc_sources) { | ||
98 | .sources = clk_sysclk_sources, | ||
99 | .nr_sources = ARRAY_SIZE(clk_sysclk_sources), | ||
100 | }, | ||
101 | .reg_src = { .reg = S3C2443_CLKSRC, .size = 1, .shift = 6 }, | ||
102 | }; | ||
103 | |||
104 | static unsigned long s3c2443_getrate_mdivclk(struct clk *clk) | ||
105 | { | ||
106 | unsigned long parent_rate = clk_get_rate(clk->parent); | ||
107 | unsigned long div = __raw_readl(S3C2443_CLKDIV0); | ||
108 | |||
109 | div &= S3C2443_CLKDIV0_EXTDIV_MASK; | ||
110 | div >>= (S3C2443_CLKDIV0_EXTDIV_SHIFT-1); /* x2 */ | ||
111 | |||
112 | return parent_rate / (div + 1); | ||
113 | } | ||
114 | |||
115 | static struct clk clk_mdivclk = { | ||
116 | .name = "mdivclk", | ||
117 | .parent = &clk_mpllref, | ||
118 | .id = -1, | ||
119 | .ops = &(struct clk_ops) { | ||
120 | .get_rate = s3c2443_getrate_mdivclk, | ||
121 | }, | ||
122 | }; | ||
123 | |||
124 | static struct clk *clk_msysclk_sources[] = { | ||
125 | [0] = &clk_mpllref, | ||
126 | [1] = &clk_mpll, | ||
127 | [2] = &clk_mdivclk, | ||
128 | [3] = &clk_mpllref, | ||
129 | }; | ||
130 | |||
131 | struct clksrc_clk clk_msysclk = { | ||
132 | .clk = { | ||
133 | .name = "msysclk", | ||
134 | .parent = &clk_xtal, | ||
135 | .id = -1, | ||
136 | }, | ||
137 | .sources = &(struct clksrc_sources) { | ||
138 | .sources = clk_msysclk_sources, | ||
139 | .nr_sources = ARRAY_SIZE(clk_msysclk_sources), | ||
140 | }, | ||
141 | .reg_src = { .reg = S3C2443_CLKSRC, .size = 2, .shift = 3 }, | ||
142 | }; | ||
143 | |||
144 | /* prediv | ||
145 | * | ||
146 | * this divides the msysclk down to pass to h/p/etc. | ||
147 | */ | ||
148 | |||
149 | static unsigned long s3c2443_prediv_getrate(struct clk *clk) | ||
150 | { | ||
151 | unsigned long rate = clk_get_rate(clk->parent); | ||
152 | unsigned long clkdiv0 = __raw_readl(S3C2443_CLKDIV0); | ||
153 | |||
154 | clkdiv0 &= S3C2443_CLKDIV0_PREDIV_MASK; | ||
155 | clkdiv0 >>= S3C2443_CLKDIV0_PREDIV_SHIFT; | ||
156 | |||
157 | return rate / (clkdiv0 + 1); | ||
158 | } | ||
159 | |||
160 | static struct clk clk_prediv = { | ||
161 | .name = "prediv", | ||
162 | .id = -1, | ||
163 | .parent = &clk_msysclk.clk, | ||
164 | .ops = &(struct clk_ops) { | ||
165 | .get_rate = s3c2443_prediv_getrate, | ||
166 | }, | ||
167 | }; | ||
168 | |||
169 | /* usbhost | ||
170 | * | ||
171 | * usb host bus-clock, usually 48MHz to provide USB bus clock timing | ||
172 | */ | ||
173 | |||
174 | static struct clksrc_clk clk_usb_bus_host = { | ||
175 | .clk = { | ||
176 | .name = "usb-bus-host-parent", | ||
177 | .id = -1, | ||
178 | .parent = &clk_esysclk.clk, | ||
179 | .ctrlbit = S3C2443_SCLKCON_USBHOST, | ||
180 | .enable = s3c2443_clkcon_enable_s, | ||
181 | }, | ||
182 | .reg_div = { .reg = S3C2443_CLKDIV1, .size = 2, .shift = 4 }, | ||
183 | }; | ||
184 | |||
185 | /* common clksrc clocks */ | ||
186 | |||
187 | static struct clksrc_clk clksrc_clks[] = { | ||
188 | { | ||
189 | /* ART baud-rate clock sourced from esysclk via a divisor */ | ||
190 | .clk = { | ||
191 | .name = "uartclk", | ||
192 | .id = -1, | ||
193 | .parent = &clk_esysclk.clk, | ||
194 | }, | ||
195 | .reg_div = { .reg = S3C2443_CLKDIV1, .size = 4, .shift = 8 }, | ||
196 | }, { | ||
197 | /* camera interface bus-clock, divided down from esysclk */ | ||
198 | .clk = { | ||
199 | .name = "camif-upll", /* same as 2440 name */ | ||
200 | .id = -1, | ||
201 | .parent = &clk_esysclk.clk, | ||
202 | .ctrlbit = S3C2443_SCLKCON_CAMCLK, | ||
203 | .enable = s3c2443_clkcon_enable_s, | ||
204 | }, | ||
205 | .reg_div = { .reg = S3C2443_CLKDIV1, .size = 4, .shift = 26 }, | ||
206 | }, { | ||
207 | .clk = { | ||
208 | .name = "display-if", | ||
209 | .id = -1, | ||
210 | .parent = &clk_esysclk.clk, | ||
211 | .ctrlbit = S3C2443_SCLKCON_DISPCLK, | ||
212 | .enable = s3c2443_clkcon_enable_s, | ||
213 | }, | ||
214 | .reg_div = { .reg = S3C2443_CLKDIV1, .size = 8, .shift = 16 }, | ||
215 | }, | ||
216 | }; | ||
217 | |||
218 | |||
219 | static struct clk init_clocks_off[] = { | ||
220 | { | ||
221 | .name = "adc", | ||
222 | .id = -1, | ||
223 | .parent = &clk_p, | ||
224 | .enable = s3c2443_clkcon_enable_p, | ||
225 | .ctrlbit = S3C2443_PCLKCON_ADC, | ||
226 | }, { | ||
227 | .name = "i2c", | ||
228 | .id = -1, | ||
229 | .parent = &clk_p, | ||
230 | .enable = s3c2443_clkcon_enable_p, | ||
231 | .ctrlbit = S3C2443_PCLKCON_IIC, | ||
232 | } | ||
233 | }; | ||
234 | |||
235 | static struct clk init_clocks[] = { | ||
236 | { | ||
237 | .name = "dma", | ||
238 | .id = 0, | ||
239 | .parent = &clk_h, | ||
240 | .enable = s3c2443_clkcon_enable_h, | ||
241 | .ctrlbit = S3C2443_HCLKCON_DMA0, | ||
242 | }, { | ||
243 | .name = "dma", | ||
244 | .id = 1, | ||
245 | .parent = &clk_h, | ||
246 | .enable = s3c2443_clkcon_enable_h, | ||
247 | .ctrlbit = S3C2443_HCLKCON_DMA1, | ||
248 | }, { | ||
249 | .name = "dma", | ||
250 | .id = 2, | ||
251 | .parent = &clk_h, | ||
252 | .enable = s3c2443_clkcon_enable_h, | ||
253 | .ctrlbit = S3C2443_HCLKCON_DMA2, | ||
254 | }, { | ||
255 | .name = "dma", | ||
256 | .id = 3, | ||
257 | .parent = &clk_h, | ||
258 | .enable = s3c2443_clkcon_enable_h, | ||
259 | .ctrlbit = S3C2443_HCLKCON_DMA3, | ||
260 | }, { | ||
261 | .name = "dma", | ||
262 | .id = 4, | ||
263 | .parent = &clk_h, | ||
264 | .enable = s3c2443_clkcon_enable_h, | ||
265 | .ctrlbit = S3C2443_HCLKCON_DMA4, | ||
266 | }, { | ||
267 | .name = "dma", | ||
268 | .id = 5, | ||
269 | .parent = &clk_h, | ||
270 | .enable = s3c2443_clkcon_enable_h, | ||
271 | .ctrlbit = S3C2443_HCLKCON_DMA5, | ||
272 | }, { | ||
273 | .name = "hsmmc", | ||
274 | .id = 0, | ||
275 | .parent = &clk_h, | ||
276 | .enable = s3c2443_clkcon_enable_h, | ||
277 | .ctrlbit = S3C2443_HCLKCON_HSMMC, | ||
278 | }, { | ||
279 | .name = "gpio", | ||
280 | .id = -1, | ||
281 | .parent = &clk_p, | ||
282 | .enable = s3c2443_clkcon_enable_p, | ||
283 | .ctrlbit = S3C2443_PCLKCON_GPIO, | ||
284 | }, { | ||
285 | .name = "usb-host", | ||
286 | .id = -1, | ||
287 | .parent = &clk_h, | ||
288 | .enable = s3c2443_clkcon_enable_h, | ||
289 | .ctrlbit = S3C2443_HCLKCON_USBH, | ||
290 | }, { | ||
291 | .name = "usb-device", | ||
292 | .id = -1, | ||
293 | .parent = &clk_h, | ||
294 | .enable = s3c2443_clkcon_enable_h, | ||
295 | .ctrlbit = S3C2443_HCLKCON_USBD, | ||
296 | }, { | ||
297 | .name = "lcd", | ||
298 | .id = -1, | ||
299 | .parent = &clk_h, | ||
300 | .enable = s3c2443_clkcon_enable_h, | ||
301 | .ctrlbit = S3C2443_HCLKCON_LCDC, | ||
302 | |||
303 | }, { | ||
304 | .name = "timers", | ||
305 | .id = -1, | ||
306 | .parent = &clk_p, | ||
307 | .enable = s3c2443_clkcon_enable_p, | ||
308 | .ctrlbit = S3C2443_PCLKCON_PWMT, | ||
309 | }, { | ||
310 | .name = "cfc", | ||
311 | .id = -1, | ||
312 | .parent = &clk_h, | ||
313 | .enable = s3c2443_clkcon_enable_h, | ||
314 | .ctrlbit = S3C2443_HCLKCON_CFC, | ||
315 | }, { | ||
316 | .name = "ssmc", | ||
317 | .id = -1, | ||
318 | .parent = &clk_h, | ||
319 | .enable = s3c2443_clkcon_enable_h, | ||
320 | .ctrlbit = S3C2443_HCLKCON_SSMC, | ||
321 | }, { | ||
322 | .name = "uart", | ||
323 | .id = 0, | ||
324 | .parent = &clk_p, | ||
325 | .enable = s3c2443_clkcon_enable_p, | ||
326 | .ctrlbit = S3C2443_PCLKCON_UART0, | ||
327 | }, { | ||
328 | .name = "uart", | ||
329 | .id = 1, | ||
330 | .parent = &clk_p, | ||
331 | .enable = s3c2443_clkcon_enable_p, | ||
332 | .ctrlbit = S3C2443_PCLKCON_UART1, | ||
333 | }, { | ||
334 | .name = "uart", | ||
335 | .id = 2, | ||
336 | .parent = &clk_p, | ||
337 | .enable = s3c2443_clkcon_enable_p, | ||
338 | .ctrlbit = S3C2443_PCLKCON_UART2, | ||
339 | }, { | ||
340 | .name = "uart", | ||
341 | .id = 3, | ||
342 | .parent = &clk_p, | ||
343 | .enable = s3c2443_clkcon_enable_p, | ||
344 | .ctrlbit = S3C2443_PCLKCON_UART3, | ||
345 | }, { | ||
346 | .name = "rtc", | ||
347 | .id = -1, | ||
348 | .parent = &clk_p, | ||
349 | .enable = s3c2443_clkcon_enable_p, | ||
350 | .ctrlbit = S3C2443_PCLKCON_RTC, | ||
351 | }, { | ||
352 | .name = "watchdog", | ||
353 | .id = -1, | ||
354 | .parent = &clk_p, | ||
355 | .ctrlbit = S3C2443_PCLKCON_WDT, | ||
356 | }, { | ||
357 | .name = "ac97", | ||
358 | .id = -1, | ||
359 | .parent = &clk_p, | ||
360 | .ctrlbit = S3C2443_PCLKCON_AC97, | ||
361 | }, { | ||
362 | .name = "nand", | ||
363 | .id = -1, | ||
364 | .parent = &clk_h, | ||
365 | }, { | ||
366 | .name = "usb-bus-host", | ||
367 | .id = -1, | ||
368 | .parent = &clk_usb_bus_host.clk, | ||
369 | } | ||
370 | }; | ||
371 | |||
372 | static inline unsigned long s3c2443_get_hdiv(unsigned long clkcon0) | ||
373 | { | ||
374 | clkcon0 &= S3C2443_CLKDIV0_HCLKDIV_MASK; | ||
375 | |||
376 | return clkcon0 + 1; | ||
377 | } | ||
378 | |||
379 | /* EPLLCON compatible enough to get on/off information */ | ||
380 | |||
381 | void __init_or_cpufreq s3c2443_common_setup_clocks(pll_fn get_mpll, | ||
382 | fdiv_fn get_fdiv) | ||
383 | { | ||
384 | unsigned long epllcon = __raw_readl(S3C2443_EPLLCON); | ||
385 | unsigned long mpllcon = __raw_readl(S3C2443_MPLLCON); | ||
386 | unsigned long clkdiv0 = __raw_readl(S3C2443_CLKDIV0); | ||
387 | struct clk *xtal_clk; | ||
388 | unsigned long xtal; | ||
389 | unsigned long pll; | ||
390 | unsigned long fclk; | ||
391 | unsigned long hclk; | ||
392 | unsigned long pclk; | ||
393 | int ptr; | ||
394 | |||
395 | xtal_clk = clk_get(NULL, "xtal"); | ||
396 | xtal = clk_get_rate(xtal_clk); | ||
397 | clk_put(xtal_clk); | ||
398 | |||
399 | pll = get_mpll(mpllcon, xtal); | ||
400 | clk_msysclk.clk.rate = pll; | ||
401 | |||
402 | fclk = pll / get_fdiv(clkdiv0); | ||
403 | hclk = s3c2443_prediv_getrate(&clk_prediv); | ||
404 | hclk /= s3c2443_get_hdiv(clkdiv0); | ||
405 | pclk = hclk / ((clkdiv0 & S3C2443_CLKDIV0_HALF_PCLK) ? 2 : 1); | ||
406 | |||
407 | s3c24xx_setup_clocks(fclk, hclk, pclk); | ||
408 | |||
409 | printk("CPU: MPLL %s %ld.%03ld MHz, cpu %ld.%03ld MHz, mem %ld.%03ld MHz, pclk %ld.%03ld MHz\n", | ||
410 | (mpllcon & S3C2443_PLLCON_OFF) ? "off":"on", | ||
411 | print_mhz(pll), print_mhz(fclk), | ||
412 | print_mhz(hclk), print_mhz(pclk)); | ||
413 | |||
414 | for (ptr = 0; ptr < ARRAY_SIZE(clksrc_clks); ptr++) | ||
415 | s3c_set_clksrc(&clksrc_clks[ptr], true); | ||
416 | |||
417 | /* ensure usb bus clock is within correct rate of 48MHz */ | ||
418 | |||
419 | if (clk_get_rate(&clk_usb_bus_host.clk) != (48 * 1000 * 1000)) { | ||
420 | printk(KERN_INFO "Warning: USB host bus not at 48MHz\n"); | ||
421 | clk_set_rate(&clk_usb_bus_host.clk, 48*1000*1000); | ||
422 | } | ||
423 | |||
424 | printk("CPU: EPLL %s %ld.%03ld MHz, usb-bus %ld.%03ld MHz\n", | ||
425 | (epllcon & S3C2443_PLLCON_OFF) ? "off":"on", | ||
426 | print_mhz(clk_get_rate(&clk_epll)), | ||
427 | print_mhz(clk_get_rate(&clk_usb_bus))); | ||
428 | } | ||
429 | |||
430 | static struct clk *clks[] __initdata = { | ||
431 | &clk_prediv, | ||
432 | &clk_mpllref, | ||
433 | &clk_mdivclk, | ||
434 | &clk_ext, | ||
435 | &clk_epll, | ||
436 | &clk_usb_bus, | ||
437 | }; | ||
438 | |||
439 | static struct clksrc_clk *clksrcs[] __initdata = { | ||
440 | &clk_usb_bus_host, | ||
441 | &clk_epllref, | ||
442 | &clk_esysclk, | ||
443 | &clk_msysclk, | ||
444 | }; | ||
445 | |||
446 | void __init s3c2443_common_init_clocks(int xtal, pll_fn get_mpll, | ||
447 | fdiv_fn get_fdiv) | ||
448 | { | ||
449 | int ptr; | ||
450 | |||
451 | /* s3c2443 parents h and p clocks from prediv */ | ||
452 | clk_h.parent = &clk_prediv; | ||
453 | clk_p.parent = &clk_prediv; | ||
454 | |||
455 | clk_usb_bus.parent = &clk_usb_bus_host.clk; | ||
456 | clk_epll.parent = &clk_epllref.clk; | ||
457 | |||
458 | s3c24xx_register_baseclocks(xtal); | ||
459 | s3c24xx_register_clocks(clks, ARRAY_SIZE(clks)); | ||
460 | |||
461 | for (ptr = 0; ptr < ARRAY_SIZE(clksrcs); ptr++) | ||
462 | s3c_register_clksrc(clksrcs[ptr], 1); | ||
463 | |||
464 | s3c_register_clksrc(clksrc_clks, ARRAY_SIZE(clksrc_clks)); | ||
465 | s3c_register_clocks(init_clocks, ARRAY_SIZE(init_clocks)); | ||
466 | |||
467 | /* See s3c2443/etc notes on disabling clocks at init time */ | ||
468 | s3c_register_clocks(init_clocks_off, ARRAY_SIZE(init_clocks_off)); | ||
469 | s3c_disable_clocks(init_clocks_off, ARRAY_SIZE(init_clocks_off)); | ||
470 | |||
471 | s3c2443_common_setup_clocks(get_mpll, get_fdiv); | ||
472 | } | ||