aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorRajendra Nayak <rnayak@ti.com>2011-02-25 18:06:47 -0500
committerPaul Walmsley <paul@pwsan.com>2011-02-25 18:06:47 -0500
commit4aef7a2a5aad52b60ac1a2f3cee055b8271b70d5 (patch)
treeee1714e1dd13d9beecd886ba40e9ebea169faf53
parent32d4034eea9c5c2b2edc365e1a0787c8f5b84c3c (diff)
OMAP: clockdomain: Arch specific funcs to handle deps
Define the following architecture specific funtions for omap2/3 .clkdm_add_wkdep .clkdm_del_wkdep .clkdm_read_wkdep .clkdm_clear_all_wkdeps .clkdm_add_sleepdep .clkdm_del_sleepdep .clkdm_read_sleepdep .clkdm_clear_all_sleepdeps Convert the platform-independent framework to call these functions. With this also move the clkdm lookups for all wkdep_srcs and sleepdep_srcs at clkdm_init. Signed-off-by: Rajendra Nayak <rnayak@ti.com> [paul@pwsan.com: fixed loop termination conditions in omap*_clkdm_clear_all_*(); thanks to Kevin Hilman for finding and helping fix those bugs; also avoid re-resolving clockdomains during init; abstracted out clkdm_dep walk] Cc: Kevin Hilman <khilman@ti.com> Signed-off-by: Paul Walmsley <paul@pwsan.com>
-rw-r--r--arch/arm/mach-omap2/Makefile2
-rw-r--r--arch/arm/mach-omap2/clockdomain.c177
-rw-r--r--arch/arm/mach-omap2/clockdomain.h6
-rw-r--r--arch/arm/mach-omap2/clockdomain2xxx_3xxx.c130
-rw-r--r--arch/arm/mach-omap2/clockdomains2xxx_3xxx_data.c9
-rw-r--r--arch/arm/mach-omap2/io.c6
6 files changed, 246 insertions, 84 deletions
diff --git a/arch/arm/mach-omap2/Makefile b/arch/arm/mach-omap2/Makefile
index ee72a9787bf1..759d9ebd79c6 100644
--- a/arch/arm/mach-omap2/Makefile
+++ b/arch/arm/mach-omap2/Makefile
@@ -102,8 +102,10 @@ obj-$(CONFIG_ARCH_OMAP4) += $(powerdomain-common) \
102 102
103# PRCM clockdomain control 103# PRCM clockdomain control
104obj-$(CONFIG_ARCH_OMAP2) += clockdomain.o \ 104obj-$(CONFIG_ARCH_OMAP2) += clockdomain.o \
105 clockdomain2xxx_3xxx.o \
105 clockdomains2xxx_3xxx_data.o 106 clockdomains2xxx_3xxx_data.o
106obj-$(CONFIG_ARCH_OMAP3) += clockdomain.o \ 107obj-$(CONFIG_ARCH_OMAP3) += clockdomain.o \
108 clockdomain2xxx_3xxx.o \
107 clockdomains2xxx_3xxx_data.o 109 clockdomains2xxx_3xxx_data.o
108obj-$(CONFIG_ARCH_OMAP4) += clockdomain.o \ 110obj-$(CONFIG_ARCH_OMAP4) += clockdomain.o \
109 clockdomains44xx_data.o 111 clockdomains44xx_data.o
diff --git a/arch/arm/mach-omap2/clockdomain.c b/arch/arm/mach-omap2/clockdomain.c
index f70b06ae8664..895c153c18e0 100644
--- a/arch/arm/mach-omap2/clockdomain.c
+++ b/arch/arm/mach-omap2/clockdomain.c
@@ -287,6 +287,32 @@ static void _disable_hwsup(struct clockdomain *clkdm)
287 BUG(); 287 BUG();
288} 288}
289 289
290/**
291 * _resolve_clkdm_deps() - resolve clkdm_names in @clkdm_deps to clkdms
292 * @clkdm: clockdomain that we are resolving dependencies for
293 * @clkdm_deps: ptr to array of struct clkdm_deps to resolve
294 *
295 * Iterates through @clkdm_deps, looking up the struct clockdomain named by
296 * clkdm_name and storing the clockdomain pointer in the struct clkdm_dep.
297 * No return value.
298 */
299static void _resolve_clkdm_deps(struct clockdomain *clkdm,
300 struct clkdm_dep *clkdm_deps)
301{
302 struct clkdm_dep *cd;
303
304 for (cd = clkdm_deps; cd && cd->clkdm_name; cd++) {
305 if (!omap_chip_is(cd->omap_chip))
306 continue;
307 if (cd->clkdm)
308 continue;
309 cd->clkdm = _clkdm_lookup(cd->clkdm_name);
310
311 WARN(!cd->clkdm, "clockdomain: %s: could not find clkdm %s while resolving dependencies - should never happen",
312 clkdm->name, cd->clkdm_name);
313 }
314}
315
290/* Public functions */ 316/* Public functions */
291 317
292/** 318/**
@@ -333,7 +359,10 @@ void clkdm_init(struct clockdomain **clkdms,
333 else if (clkdm->flags & CLKDM_CAN_DISABLE_AUTO) 359 else if (clkdm->flags & CLKDM_CAN_DISABLE_AUTO)
334 omap2_clkdm_deny_idle(clkdm); 360 omap2_clkdm_deny_idle(clkdm);
335 361
362 _resolve_clkdm_deps(clkdm, clkdm->wkdep_srcs);
336 clkdm_clear_all_wkdeps(clkdm); 363 clkdm_clear_all_wkdeps(clkdm);
364
365 _resolve_clkdm_deps(clkdm, clkdm->sleepdep_srcs);
337 clkdm_clear_all_sleepdeps(clkdm); 366 clkdm_clear_all_sleepdeps(clkdm);
338 } 367 }
339} 368}
@@ -430,6 +459,7 @@ struct powerdomain *clkdm_get_pwrdm(struct clockdomain *clkdm)
430int clkdm_add_wkdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2) 459int clkdm_add_wkdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
431{ 460{
432 struct clkdm_dep *cd; 461 struct clkdm_dep *cd;
462 int ret = 0;
433 463
434 if (!cpu_is_omap24xx() && !cpu_is_omap34xx()) { 464 if (!cpu_is_omap24xx() && !cpu_is_omap34xx()) {
435 pr_err("clockdomain: %s/%s: %s: not yet implemented\n", 465 pr_err("clockdomain: %s/%s: %s: not yet implemented\n",
@@ -441,21 +471,26 @@ int clkdm_add_wkdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
441 return -EINVAL; 471 return -EINVAL;
442 472
443 cd = _clkdm_deps_lookup(clkdm2, clkdm1->wkdep_srcs); 473 cd = _clkdm_deps_lookup(clkdm2, clkdm1->wkdep_srcs);
444 if (IS_ERR(cd)) { 474 if (IS_ERR(cd))
475 ret = PTR_ERR(cd);
476
477 if (!arch_clkdm || !arch_clkdm->clkdm_add_wkdep)
478 ret = -EINVAL;
479
480 if (ret) {
445 pr_debug("clockdomain: hardware cannot set/clear wake up of " 481 pr_debug("clockdomain: hardware cannot set/clear wake up of "
446 "%s when %s wakes up\n", clkdm1->name, clkdm2->name); 482 "%s when %s wakes up\n", clkdm1->name, clkdm2->name);
447 return PTR_ERR(cd); 483 return ret;
448 } 484 }
449 485
450 if (atomic_inc_return(&cd->wkdep_usecount) == 1) { 486 if (atomic_inc_return(&cd->wkdep_usecount) == 1) {
451 pr_debug("clockdomain: hardware will wake up %s when %s wakes " 487 pr_debug("clockdomain: hardware will wake up %s when %s wakes "
452 "up\n", clkdm1->name, clkdm2->name); 488 "up\n", clkdm1->name, clkdm2->name);
453 489
454 omap2_prm_set_mod_reg_bits((1 << clkdm2->dep_bit), 490 ret = arch_clkdm->clkdm_add_wkdep(clkdm1, clkdm2);
455 clkdm1->pwrdm.ptr->prcm_offs, PM_WKDEP);
456 } 491 }
457 492
458 return 0; 493 return ret;
459} 494}
460 495
461/** 496/**
@@ -471,6 +506,7 @@ int clkdm_add_wkdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
471int clkdm_del_wkdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2) 506int clkdm_del_wkdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
472{ 507{
473 struct clkdm_dep *cd; 508 struct clkdm_dep *cd;
509 int ret = 0;
474 510
475 if (!cpu_is_omap24xx() && !cpu_is_omap34xx()) { 511 if (!cpu_is_omap24xx() && !cpu_is_omap34xx()) {
476 pr_err("clockdomain: %s/%s: %s: not yet implemented\n", 512 pr_err("clockdomain: %s/%s: %s: not yet implemented\n",
@@ -482,21 +518,26 @@ int clkdm_del_wkdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
482 return -EINVAL; 518 return -EINVAL;
483 519
484 cd = _clkdm_deps_lookup(clkdm2, clkdm1->wkdep_srcs); 520 cd = _clkdm_deps_lookup(clkdm2, clkdm1->wkdep_srcs);
485 if (IS_ERR(cd)) { 521 if (IS_ERR(cd))
522 ret = PTR_ERR(cd);
523
524 if (!arch_clkdm || !arch_clkdm->clkdm_del_wkdep)
525 ret = -EINVAL;
526
527 if (ret) {
486 pr_debug("clockdomain: hardware cannot set/clear wake up of " 528 pr_debug("clockdomain: hardware cannot set/clear wake up of "
487 "%s when %s wakes up\n", clkdm1->name, clkdm2->name); 529 "%s when %s wakes up\n", clkdm1->name, clkdm2->name);
488 return PTR_ERR(cd); 530 return ret;
489 } 531 }
490 532
491 if (atomic_dec_return(&cd->wkdep_usecount) == 0) { 533 if (atomic_dec_return(&cd->wkdep_usecount) == 0) {
492 pr_debug("clockdomain: hardware will no longer wake up %s " 534 pr_debug("clockdomain: hardware will no longer wake up %s "
493 "after %s wakes up\n", clkdm1->name, clkdm2->name); 535 "after %s wakes up\n", clkdm1->name, clkdm2->name);
494 536
495 omap2_prm_clear_mod_reg_bits((1 << clkdm2->dep_bit), 537 ret = arch_clkdm->clkdm_del_wkdep(clkdm1, clkdm2);
496 clkdm1->pwrdm.ptr->prcm_offs, PM_WKDEP);
497 } 538 }
498 539
499 return 0; 540 return ret;
500} 541}
501 542
502/** 543/**
@@ -516,6 +557,7 @@ int clkdm_del_wkdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
516int clkdm_read_wkdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2) 557int clkdm_read_wkdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
517{ 558{
518 struct clkdm_dep *cd; 559 struct clkdm_dep *cd;
560 int ret = 0;
519 561
520 if (!clkdm1 || !clkdm2) 562 if (!clkdm1 || !clkdm2)
521 return -EINVAL; 563 return -EINVAL;
@@ -527,15 +569,20 @@ int clkdm_read_wkdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
527 } 569 }
528 570
529 cd = _clkdm_deps_lookup(clkdm2, clkdm1->wkdep_srcs); 571 cd = _clkdm_deps_lookup(clkdm2, clkdm1->wkdep_srcs);
530 if (IS_ERR(cd)) { 572 if (IS_ERR(cd))
573 ret = PTR_ERR(cd);
574
575 if (!arch_clkdm || !arch_clkdm->clkdm_read_wkdep)
576 ret = -EINVAL;
577
578 if (ret) {
531 pr_debug("clockdomain: hardware cannot set/clear wake up of " 579 pr_debug("clockdomain: hardware cannot set/clear wake up of "
532 "%s when %s wakes up\n", clkdm1->name, clkdm2->name); 580 "%s when %s wakes up\n", clkdm1->name, clkdm2->name);
533 return PTR_ERR(cd); 581 return ret;
534 } 582 }
535 583
536 /* XXX It's faster to return the atomic wkdep_usecount */ 584 /* XXX It's faster to return the atomic wkdep_usecount */
537 return omap2_prm_read_mod_bits_shift(clkdm1->pwrdm.ptr->prcm_offs, PM_WKDEP, 585 return arch_clkdm->clkdm_read_wkdep(clkdm1, clkdm2);
538 (1 << clkdm2->dep_bit));
539} 586}
540 587
541/** 588/**
@@ -550,9 +597,6 @@ int clkdm_read_wkdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
550 */ 597 */
551int clkdm_clear_all_wkdeps(struct clockdomain *clkdm) 598int clkdm_clear_all_wkdeps(struct clockdomain *clkdm)
552{ 599{
553 struct clkdm_dep *cd;
554 u32 mask = 0;
555
556 if (!cpu_is_omap24xx() && !cpu_is_omap34xx()) { 600 if (!cpu_is_omap24xx() && !cpu_is_omap34xx()) {
557 pr_err("clockdomain: %s: %s: not yet implemented\n", 601 pr_err("clockdomain: %s: %s: not yet implemented\n",
558 clkdm->name, __func__); 602 clkdm->name, __func__);
@@ -562,21 +606,10 @@ int clkdm_clear_all_wkdeps(struct clockdomain *clkdm)
562 if (!clkdm) 606 if (!clkdm)
563 return -EINVAL; 607 return -EINVAL;
564 608
565 for (cd = clkdm->wkdep_srcs; cd && cd->clkdm_name; cd++) { 609 if (!arch_clkdm || !arch_clkdm->clkdm_clear_all_wkdeps)
566 if (!omap_chip_is(cd->omap_chip)) 610 return -EINVAL;
567 continue;
568
569 if (!cd->clkdm && cd->clkdm_name)
570 cd->clkdm = _clkdm_lookup(cd->clkdm_name);
571
572 /* PRM accesses are slow, so minimize them */
573 mask |= 1 << cd->clkdm->dep_bit;
574 atomic_set(&cd->wkdep_usecount, 0);
575 }
576
577 omap2_prm_clear_mod_reg_bits(mask, clkdm->pwrdm.ptr->prcm_offs, PM_WKDEP);
578 611
579 return 0; 612 return arch_clkdm->clkdm_clear_all_wkdeps(clkdm);
580} 613}
581 614
582/** 615/**
@@ -594,31 +627,33 @@ int clkdm_clear_all_wkdeps(struct clockdomain *clkdm)
594int clkdm_add_sleepdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2) 627int clkdm_add_sleepdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
595{ 628{
596 struct clkdm_dep *cd; 629 struct clkdm_dep *cd;
597 630 int ret = 0;
598 if (!cpu_is_omap34xx())
599 return -EINVAL;
600 631
601 if (!clkdm1 || !clkdm2) 632 if (!clkdm1 || !clkdm2)
602 return -EINVAL; 633 return -EINVAL;
603 634
604 cd = _clkdm_deps_lookup(clkdm2, clkdm1->sleepdep_srcs); 635 cd = _clkdm_deps_lookup(clkdm2, clkdm1->sleepdep_srcs);
605 if (IS_ERR(cd)) { 636 if (IS_ERR(cd))
637 ret = PTR_ERR(cd);
638
639 if (!arch_clkdm || !arch_clkdm->clkdm_add_sleepdep)
640 ret = -EINVAL;
641
642 if (ret) {
606 pr_debug("clockdomain: hardware cannot set/clear sleep " 643 pr_debug("clockdomain: hardware cannot set/clear sleep "
607 "dependency affecting %s from %s\n", clkdm1->name, 644 "dependency affecting %s from %s\n", clkdm1->name,
608 clkdm2->name); 645 clkdm2->name);
609 return PTR_ERR(cd); 646 return ret;
610 } 647 }
611 648
612 if (atomic_inc_return(&cd->sleepdep_usecount) == 1) { 649 if (atomic_inc_return(&cd->sleepdep_usecount) == 1) {
613 pr_debug("clockdomain: will prevent %s from sleeping if %s " 650 pr_debug("clockdomain: will prevent %s from sleeping if %s "
614 "is active\n", clkdm1->name, clkdm2->name); 651 "is active\n", clkdm1->name, clkdm2->name);
615 652
616 omap2_cm_set_mod_reg_bits((1 << clkdm2->dep_bit), 653 ret = arch_clkdm->clkdm_add_sleepdep(clkdm1, clkdm2);
617 clkdm1->pwrdm.ptr->prcm_offs,
618 OMAP3430_CM_SLEEPDEP);
619 } 654 }
620 655
621 return 0; 656 return ret;
622} 657}
623 658
624/** 659/**
@@ -636,19 +671,23 @@ int clkdm_add_sleepdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
636int clkdm_del_sleepdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2) 671int clkdm_del_sleepdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
637{ 672{
638 struct clkdm_dep *cd; 673 struct clkdm_dep *cd;
639 674 int ret = 0;
640 if (!cpu_is_omap34xx())
641 return -EINVAL;
642 675
643 if (!clkdm1 || !clkdm2) 676 if (!clkdm1 || !clkdm2)
644 return -EINVAL; 677 return -EINVAL;
645 678
646 cd = _clkdm_deps_lookup(clkdm2, clkdm1->sleepdep_srcs); 679 cd = _clkdm_deps_lookup(clkdm2, clkdm1->sleepdep_srcs);
647 if (IS_ERR(cd)) { 680 if (IS_ERR(cd))
681 ret = PTR_ERR(cd);
682
683 if (!arch_clkdm || !arch_clkdm->clkdm_del_sleepdep)
684 ret = -EINVAL;
685
686 if (ret) {
648 pr_debug("clockdomain: hardware cannot set/clear sleep " 687 pr_debug("clockdomain: hardware cannot set/clear sleep "
649 "dependency affecting %s from %s\n", clkdm1->name, 688 "dependency affecting %s from %s\n", clkdm1->name,
650 clkdm2->name); 689 clkdm2->name);
651 return PTR_ERR(cd); 690 return ret;
652 } 691 }
653 692
654 if (atomic_dec_return(&cd->sleepdep_usecount) == 0) { 693 if (atomic_dec_return(&cd->sleepdep_usecount) == 0) {
@@ -656,12 +695,10 @@ int clkdm_del_sleepdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
656 "sleeping if %s is active\n", clkdm1->name, 695 "sleeping if %s is active\n", clkdm1->name,
657 clkdm2->name); 696 clkdm2->name);
658 697
659 omap2_cm_clear_mod_reg_bits((1 << clkdm2->dep_bit), 698 ret = arch_clkdm->clkdm_del_sleepdep(clkdm1, clkdm2);
660 clkdm1->pwrdm.ptr->prcm_offs,
661 OMAP3430_CM_SLEEPDEP);
662 } 699 }
663 700
664 return 0; 701 return ret;
665} 702}
666 703
667/** 704/**
@@ -683,25 +720,27 @@ int clkdm_del_sleepdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
683int clkdm_read_sleepdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2) 720int clkdm_read_sleepdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
684{ 721{
685 struct clkdm_dep *cd; 722 struct clkdm_dep *cd;
686 723 int ret = 0;
687 if (!cpu_is_omap34xx())
688 return -EINVAL;
689 724
690 if (!clkdm1 || !clkdm2) 725 if (!clkdm1 || !clkdm2)
691 return -EINVAL; 726 return -EINVAL;
692 727
693 cd = _clkdm_deps_lookup(clkdm2, clkdm1->sleepdep_srcs); 728 cd = _clkdm_deps_lookup(clkdm2, clkdm1->sleepdep_srcs);
694 if (IS_ERR(cd)) { 729 if (IS_ERR(cd))
730 ret = PTR_ERR(cd);
731
732 if (!arch_clkdm || !arch_clkdm->clkdm_read_sleepdep)
733 ret = -EINVAL;
734
735 if (ret) {
695 pr_debug("clockdomain: hardware cannot set/clear sleep " 736 pr_debug("clockdomain: hardware cannot set/clear sleep "
696 "dependency affecting %s from %s\n", clkdm1->name, 737 "dependency affecting %s from %s\n", clkdm1->name,
697 clkdm2->name); 738 clkdm2->name);
698 return PTR_ERR(cd); 739 return ret;
699 } 740 }
700 741
701 /* XXX It's faster to return the atomic sleepdep_usecount */ 742 /* XXX It's faster to return the atomic sleepdep_usecount */
702 return omap2_prm_read_mod_bits_shift(clkdm1->pwrdm.ptr->prcm_offs, 743 return arch_clkdm->clkdm_read_sleepdep(clkdm1, clkdm2);
703 OMAP3430_CM_SLEEPDEP,
704 (1 << clkdm2->dep_bit));
705} 744}
706 745
707/** 746/**
@@ -716,31 +755,13 @@ int clkdm_read_sleepdep(struct clockdomain *clkdm1, struct clockdomain *clkdm2)
716 */ 755 */
717int clkdm_clear_all_sleepdeps(struct clockdomain *clkdm) 756int clkdm_clear_all_sleepdeps(struct clockdomain *clkdm)
718{ 757{
719 struct clkdm_dep *cd;
720 u32 mask = 0;
721
722 if (!cpu_is_omap34xx())
723 return -EINVAL;
724
725 if (!clkdm) 758 if (!clkdm)
726 return -EINVAL; 759 return -EINVAL;
727 760
728 for (cd = clkdm->sleepdep_srcs; cd && cd->clkdm_name; cd++) { 761 if (!arch_clkdm || !arch_clkdm->clkdm_clear_all_sleepdeps)
729 if (!omap_chip_is(cd->omap_chip)) 762 return -EINVAL;
730 continue;
731
732 if (!cd->clkdm && cd->clkdm_name)
733 cd->clkdm = _clkdm_lookup(cd->clkdm_name);
734
735 /* PRM accesses are slow, so minimize them */
736 mask |= 1 << cd->clkdm->dep_bit;
737 atomic_set(&cd->sleepdep_usecount, 0);
738 }
739
740 omap2_prm_clear_mod_reg_bits(mask, clkdm->pwrdm.ptr->prcm_offs,
741 OMAP3430_CM_SLEEPDEP);
742 763
743 return 0; 764 return arch_clkdm->clkdm_clear_all_sleepdeps(clkdm);
744} 765}
745 766
746/** 767/**
diff --git a/arch/arm/mach-omap2/clockdomain.h b/arch/arm/mach-omap2/clockdomain.h
index 71ad265cf133..90b6d6a50862 100644
--- a/arch/arm/mach-omap2/clockdomain.h
+++ b/arch/arm/mach-omap2/clockdomain.h
@@ -176,7 +176,11 @@ int omap2_clkdm_sleep(struct clockdomain *clkdm);
176int omap2_clkdm_clk_enable(struct clockdomain *clkdm, struct clk *clk); 176int omap2_clkdm_clk_enable(struct clockdomain *clkdm, struct clk *clk);
177int omap2_clkdm_clk_disable(struct clockdomain *clkdm, struct clk *clk); 177int omap2_clkdm_clk_disable(struct clockdomain *clkdm, struct clk *clk);
178 178
179extern void __init omap2_clockdomains_init(void); 179extern void __init omap2xxx_clockdomains_init(void);
180extern void __init omap3xxx_clockdomains_init(void);
180extern void __init omap44xx_clockdomains_init(void); 181extern void __init omap44xx_clockdomains_init(void);
181 182
183extern struct clkdm_ops omap2_clkdm_operations;
184extern struct clkdm_ops omap3_clkdm_operations;
185
182#endif 186#endif
diff --git a/arch/arm/mach-omap2/clockdomain2xxx_3xxx.c b/arch/arm/mach-omap2/clockdomain2xxx_3xxx.c
new file mode 100644
index 000000000000..a1fd6fd5a466
--- /dev/null
+++ b/arch/arm/mach-omap2/clockdomain2xxx_3xxx.c
@@ -0,0 +1,130 @@
1/*
2 * OMAP2 and OMAP3 clockdomain control
3 *
4 * Copyright (C) 2008-2010 Texas Instruments, Inc.
5 * Copyright (C) 2008-2010 Nokia Corporation
6 *
7 * Derived from mach-omap2/clockdomain.c written by Paul Walmsley
8 * Rajendra Nayak <rnayak@ti.com>
9 *
10 * This program is free software; you can redistribute it and/or modify
11 * it under the terms of the GNU General Public License version 2 as
12 * published by the Free Software Foundation.
13 */
14
15#include <linux/types.h>
16#include <plat/prcm.h>
17#include "prm.h"
18#include "prm2xxx_3xxx.h"
19#include "cm.h"
20#include "cm2xxx_3xxx.h"
21#include "cm-regbits-24xx.h"
22#include "cm-regbits-34xx.h"
23#include "clockdomain.h"
24
25static int omap2_clkdm_add_wkdep(struct clockdomain *clkdm1,
26 struct clockdomain *clkdm2)
27{
28 omap2_prm_set_mod_reg_bits((1 << clkdm2->dep_bit),
29 clkdm1->pwrdm.ptr->prcm_offs, PM_WKDEP);
30 return 0;
31}
32
33static int omap2_clkdm_del_wkdep(struct clockdomain *clkdm1,
34 struct clockdomain *clkdm2)
35{
36 omap2_prm_clear_mod_reg_bits((1 << clkdm2->dep_bit),
37 clkdm1->pwrdm.ptr->prcm_offs, PM_WKDEP);
38 return 0;
39}
40
41static int omap2_clkdm_read_wkdep(struct clockdomain *clkdm1,
42 struct clockdomain *clkdm2)
43{
44 return omap2_prm_read_mod_bits_shift(clkdm1->pwrdm.ptr->prcm_offs,
45 PM_WKDEP, (1 << clkdm2->dep_bit));
46}
47
48static int omap2_clkdm_clear_all_wkdeps(struct clockdomain *clkdm)
49{
50 struct clkdm_dep *cd;
51 u32 mask = 0;
52
53 for (cd = clkdm->wkdep_srcs; cd && cd->clkdm_name; cd++) {
54 if (!omap_chip_is(cd->omap_chip))
55 continue;
56 if (!cd->clkdm)
57 continue; /* only happens if data is erroneous */
58
59 /* PRM accesses are slow, so minimize them */
60 mask |= 1 << cd->clkdm->dep_bit;
61 atomic_set(&cd->wkdep_usecount, 0);
62 }
63
64 omap2_prm_clear_mod_reg_bits(mask, clkdm->pwrdm.ptr->prcm_offs,
65 PM_WKDEP);
66 return 0;
67}
68
69static int omap3_clkdm_add_sleepdep(struct clockdomain *clkdm1,
70 struct clockdomain *clkdm2)
71{
72 omap2_cm_set_mod_reg_bits((1 << clkdm2->dep_bit),
73 clkdm1->pwrdm.ptr->prcm_offs,
74 OMAP3430_CM_SLEEPDEP);
75 return 0;
76}
77
78static int omap3_clkdm_del_sleepdep(struct clockdomain *clkdm1,
79 struct clockdomain *clkdm2)
80{
81 omap2_cm_clear_mod_reg_bits((1 << clkdm2->dep_bit),
82 clkdm1->pwrdm.ptr->prcm_offs,
83 OMAP3430_CM_SLEEPDEP);
84 return 0;
85}
86
87static int omap3_clkdm_read_sleepdep(struct clockdomain *clkdm1,
88 struct clockdomain *clkdm2)
89{
90 return omap2_prm_read_mod_bits_shift(clkdm1->pwrdm.ptr->prcm_offs,
91 OMAP3430_CM_SLEEPDEP, (1 << clkdm2->dep_bit));
92}
93
94static int omap3_clkdm_clear_all_sleepdeps(struct clockdomain *clkdm)
95{
96 struct clkdm_dep *cd;
97 u32 mask = 0;
98
99 for (cd = clkdm->sleepdep_srcs; cd && cd->clkdm_name; cd++) {
100 if (!omap_chip_is(cd->omap_chip))
101 continue;
102 if (!cd->clkdm)
103 continue; /* only happens if data is erroneous */
104
105 /* PRM accesses are slow, so minimize them */
106 mask |= 1 << cd->clkdm->dep_bit;
107 atomic_set(&cd->sleepdep_usecount, 0);
108 }
109 omap2_prm_clear_mod_reg_bits(mask, clkdm->pwrdm.ptr->prcm_offs,
110 OMAP3430_CM_SLEEPDEP);
111 return 0;
112}
113
114struct clkdm_ops omap2_clkdm_operations = {
115 .clkdm_add_wkdep = omap2_clkdm_add_wkdep,
116 .clkdm_del_wkdep = omap2_clkdm_del_wkdep,
117 .clkdm_read_wkdep = omap2_clkdm_read_wkdep,
118 .clkdm_clear_all_wkdeps = omap2_clkdm_clear_all_wkdeps,
119};
120
121struct clkdm_ops omap3_clkdm_operations = {
122 .clkdm_add_wkdep = omap2_clkdm_add_wkdep,
123 .clkdm_del_wkdep = omap2_clkdm_del_wkdep,
124 .clkdm_read_wkdep = omap2_clkdm_read_wkdep,
125 .clkdm_clear_all_wkdeps = omap2_clkdm_clear_all_wkdeps,
126 .clkdm_add_sleepdep = omap3_clkdm_add_sleepdep,
127 .clkdm_del_sleepdep = omap3_clkdm_del_sleepdep,
128 .clkdm_read_sleepdep = omap3_clkdm_read_sleepdep,
129 .clkdm_clear_all_sleepdeps = omap3_clkdm_clear_all_sleepdeps,
130};
diff --git a/arch/arm/mach-omap2/clockdomains2xxx_3xxx_data.c b/arch/arm/mach-omap2/clockdomains2xxx_3xxx_data.c
index e2a959eab312..ffdfe54f3264 100644
--- a/arch/arm/mach-omap2/clockdomains2xxx_3xxx_data.c
+++ b/arch/arm/mach-omap2/clockdomains2xxx_3xxx_data.c
@@ -854,7 +854,12 @@ static struct clockdomain *clockdomains_omap2[] __initdata = {
854 NULL, 854 NULL,
855}; 855};
856 856
857void __init omap2_clockdomains_init(void) 857void __init omap2xxx_clockdomains_init(void)
858{ 858{
859 clkdm_init(clockdomains_omap2, clkdm_autodeps, NULL); 859 clkdm_init(clockdomains_omap2, clkdm_autodeps, &omap2_clkdm_operations);
860}
861
862void __init omap3xxx_clockdomains_init(void)
863{
864 clkdm_init(clockdomains_omap2, clkdm_autodeps, &omap3_clkdm_operations);
860} 865}
diff --git a/arch/arm/mach-omap2/io.c b/arch/arm/mach-omap2/io.c
index b8b49e4ae928..03f71ec3cd82 100644
--- a/arch/arm/mach-omap2/io.c
+++ b/arch/arm/mach-omap2/io.c
@@ -357,15 +357,15 @@ void __init omap2_init_common_infrastructure(void)
357 357
358 if (cpu_is_omap242x()) { 358 if (cpu_is_omap242x()) {
359 omap2xxx_powerdomains_init(); 359 omap2xxx_powerdomains_init();
360 omap2_clockdomains_init(); 360 omap2xxx_clockdomains_init();
361 omap2420_hwmod_init(); 361 omap2420_hwmod_init();
362 } else if (cpu_is_omap243x()) { 362 } else if (cpu_is_omap243x()) {
363 omap2xxx_powerdomains_init(); 363 omap2xxx_powerdomains_init();
364 omap2_clockdomains_init(); 364 omap2xxx_clockdomains_init();
365 omap2430_hwmod_init(); 365 omap2430_hwmod_init();
366 } else if (cpu_is_omap34xx()) { 366 } else if (cpu_is_omap34xx()) {
367 omap3xxx_powerdomains_init(); 367 omap3xxx_powerdomains_init();
368 omap2_clockdomains_init(); 368 omap3xxx_clockdomains_init();
369 omap3xxx_hwmod_init(); 369 omap3xxx_hwmod_init();
370 } else if (cpu_is_omap44xx()) { 370 } else if (cpu_is_omap44xx()) {
371 omap44xx_powerdomains_init(); 371 omap44xx_powerdomains_init();