diff options
Diffstat (limited to 'arch/arm/mach-omap2/omap_hwmod.c')
| -rw-r--r-- | arch/arm/mach-omap2/omap_hwmod.c | 599 |
1 files changed, 496 insertions, 103 deletions
diff --git a/arch/arm/mach-omap2/omap_hwmod.c b/arch/arm/mach-omap2/omap_hwmod.c index cb911d7d1a3c..5a30658444d0 100644 --- a/arch/arm/mach-omap2/omap_hwmod.c +++ b/arch/arm/mach-omap2/omap_hwmod.c | |||
| @@ -13,10 +13,102 @@ | |||
| 13 | * it under the terms of the GNU General Public License version 2 as | 13 | * it under the terms of the GNU General Public License version 2 as |
| 14 | * published by the Free Software Foundation. | 14 | * published by the Free Software Foundation. |
| 15 | * | 15 | * |
| 16 | * This code manages "OMAP modules" (on-chip devices) and their | 16 | * Introduction |
| 17 | * integration with Linux device driver and bus code. | 17 | * ------------ |
| 18 | * | 18 | * One way to view an OMAP SoC is as a collection of largely unrelated |
| 19 | * References: | 19 | * IP blocks connected by interconnects. The IP blocks include |
| 20 | * devices such as ARM processors, audio serial interfaces, UARTs, | ||
| 21 | * etc. Some of these devices, like the DSP, are created by TI; | ||
| 22 | * others, like the SGX, largely originate from external vendors. In | ||
| 23 | * TI's documentation, on-chip devices are referred to as "OMAP | ||
| 24 | * modules." Some of these IP blocks are identical across several | ||
| 25 | * OMAP versions. Others are revised frequently. | ||
| 26 | * | ||
| 27 | * These OMAP modules are tied together by various interconnects. | ||
| 28 | * Most of the address and data flow between modules is via OCP-based | ||
| 29 | * interconnects such as the L3 and L4 buses; but there are other | ||
| 30 | * interconnects that distribute the hardware clock tree, handle idle | ||
| 31 | * and reset signaling, supply power, and connect the modules to | ||
| 32 | * various pads or balls on the OMAP package. | ||
| 33 | * | ||
| 34 | * OMAP hwmod provides a consistent way to describe the on-chip | ||
| 35 | * hardware blocks and their integration into the rest of the chip. | ||
| 36 | * This description can be automatically generated from the TI | ||
| 37 | * hardware database. OMAP hwmod provides a standard, consistent API | ||
| 38 | * to reset, enable, idle, and disable these hardware blocks. And | ||
| 39 | * hwmod provides a way for other core code, such as the Linux device | ||
| 40 | * code or the OMAP power management and address space mapping code, | ||
| 41 | * to query the hardware database. | ||
| 42 | * | ||
| 43 | * Using hwmod | ||
| 44 | * ----------- | ||
| 45 | * Drivers won't call hwmod functions directly. That is done by the | ||
| 46 | * omap_device code, and in rare occasions, by custom integration code | ||
| 47 | * in arch/arm/ *omap*. The omap_device code includes functions to | ||
| 48 | * build a struct platform_device using omap_hwmod data, and that is | ||
| 49 | * currently how hwmod data is communicated to drivers and to the | ||
| 50 | * Linux driver model. Most drivers will call omap_hwmod functions only | ||
| 51 | * indirectly, via pm_runtime*() functions. | ||
| 52 | * | ||
| 53 | * From a layering perspective, here is where the OMAP hwmod code | ||
| 54 | * fits into the kernel software stack: | ||
| 55 | * | ||
| 56 | * +-------------------------------+ | ||
| 57 | * | Device driver code | | ||
| 58 | * | (e.g., drivers/) | | ||
| 59 | * +-------------------------------+ | ||
| 60 | * | Linux driver model | | ||
| 61 | * | (platform_device / | | ||
| 62 | * | platform_driver data/code) | | ||
| 63 | * +-------------------------------+ | ||
| 64 | * | OMAP core-driver integration | | ||
| 65 | * |(arch/arm/mach-omap2/devices.c)| | ||
| 66 | * +-------------------------------+ | ||
| 67 | * | omap_device code | | ||
| 68 | * | (../plat-omap/omap_device.c) | | ||
| 69 | * +-------------------------------+ | ||
| 70 | * ----> | omap_hwmod code/data | <----- | ||
| 71 | * | (../mach-omap2/omap_hwmod*) | | ||
| 72 | * +-------------------------------+ | ||
| 73 | * | OMAP clock/PRCM/register fns | | ||
| 74 | * | (__raw_{read,write}l, clk*) | | ||
| 75 | * +-------------------------------+ | ||
| 76 | * | ||
| 77 | * Device drivers should not contain any OMAP-specific code or data in | ||
| 78 | * them. They should only contain code to operate the IP block that | ||
| 79 | * the driver is responsible for. This is because these IP blocks can | ||
| 80 | * also appear in other SoCs, either from TI (such as DaVinci) or from | ||
| 81 | * other manufacturers; and drivers should be reusable across other | ||
| 82 | * platforms. | ||
| 83 | * | ||
| 84 | * The OMAP hwmod code also will attempt to reset and idle all on-chip | ||
| 85 | * devices upon boot. The goal here is for the kernel to be | ||
| 86 | * completely self-reliant and independent from bootloaders. This is | ||
| 87 | * to ensure a repeatable configuration, both to ensure consistent | ||
| 88 | * runtime behavior, and to make it easier for others to reproduce | ||
| 89 | * bugs. | ||
| 90 | * | ||
| 91 | * OMAP module activity states | ||
| 92 | * --------------------------- | ||
| 93 | * The hwmod code considers modules to be in one of several activity | ||
| 94 | * states. IP blocks start out in an UNKNOWN state, then once they | ||
| 95 | * are registered via the hwmod code, proceed to the REGISTERED state. | ||
| 96 | * Once their clock names are resolved to clock pointers, the module | ||
| 97 | * enters the CLKS_INITED state; and finally, once the module has been | ||
| 98 | * reset and the integration registers programmed, the INITIALIZED state | ||
| 99 | * is entered. The hwmod code will then place the module into either | ||
| 100 | * the IDLE state to save power, or in the case of a critical system | ||
| 101 | * module, the ENABLED state. | ||
| 102 | * | ||
| 103 | * OMAP core integration code can then call omap_hwmod*() functions | ||
| 104 | * directly to move the module between the IDLE, ENABLED, and DISABLED | ||
| 105 | * states, as needed. This is done during both the PM idle loop, and | ||
| 106 | * in the OMAP core integration code's implementation of the PM runtime | ||
| 107 | * functions. | ||
| 108 | * | ||
| 109 | * References | ||
| 110 | * ---------- | ||
| 111 | * This is a partial list. | ||
| 20 | * - OMAP2420 Multimedia Processor Silicon Revision 2.1.1, 2.2 (SWPU064) | 112 | * - OMAP2420 Multimedia Processor Silicon Revision 2.1.1, 2.2 (SWPU064) |
| 21 | * - OMAP2430 Multimedia Device POP Silicon Revision 2.1 (SWPU090) | 113 | * - OMAP2430 Multimedia Device POP Silicon Revision 2.1 (SWPU090) |
| 22 | * - OMAP34xx Multimedia Device Silicon Revision 3.1 (SWPU108) | 114 | * - OMAP34xx Multimedia Device Silicon Revision 3.1 (SWPU108) |
| @@ -50,11 +142,13 @@ | |||
| 50 | #include <plat/powerdomain.h> | 142 | #include <plat/powerdomain.h> |
| 51 | #include <plat/clock.h> | 143 | #include <plat/clock.h> |
| 52 | #include <plat/omap_hwmod.h> | 144 | #include <plat/omap_hwmod.h> |
| 145 | #include <plat/prcm.h> | ||
| 53 | 146 | ||
| 54 | #include "cm.h" | 147 | #include "cm.h" |
| 148 | #include "prm.h" | ||
| 55 | 149 | ||
| 56 | /* Maximum microseconds to wait for OMAP module to reset */ | 150 | /* Maximum microseconds to wait for OMAP module to softreset */ |
| 57 | #define MAX_MODULE_RESET_WAIT 10000 | 151 | #define MAX_MODULE_SOFTRESET_WAIT 10000 |
| 58 | 152 | ||
| 59 | /* Name of the OMAP hwmod for the MPU */ | 153 | /* Name of the OMAP hwmod for the MPU */ |
| 60 | #define MPU_INITIATOR_NAME "mpu" | 154 | #define MPU_INITIATOR_NAME "mpu" |
| @@ -90,7 +184,7 @@ static int _update_sysc_cache(struct omap_hwmod *oh) | |||
| 90 | 184 | ||
| 91 | /* XXX ensure module interface clock is up */ | 185 | /* XXX ensure module interface clock is up */ |
| 92 | 186 | ||
| 93 | oh->_sysc_cache = omap_hwmod_readl(oh, oh->class->sysc->sysc_offs); | 187 | oh->_sysc_cache = omap_hwmod_read(oh, oh->class->sysc->sysc_offs); |
| 94 | 188 | ||
| 95 | if (!(oh->class->sysc->sysc_flags & SYSC_NO_CACHE)) | 189 | if (!(oh->class->sysc->sysc_flags & SYSC_NO_CACHE)) |
| 96 | oh->_int_flags |= _HWMOD_SYSCONFIG_LOADED; | 190 | oh->_int_flags |= _HWMOD_SYSCONFIG_LOADED; |
| @@ -117,7 +211,7 @@ static void _write_sysconfig(u32 v, struct omap_hwmod *oh) | |||
| 117 | 211 | ||
| 118 | if (oh->_sysc_cache != v) { | 212 | if (oh->_sysc_cache != v) { |
| 119 | oh->_sysc_cache = v; | 213 | oh->_sysc_cache = v; |
| 120 | omap_hwmod_writel(v, oh, oh->class->sysc->sysc_offs); | 214 | omap_hwmod_write(v, oh, oh->class->sysc->sysc_offs); |
| 121 | } | 215 | } |
| 122 | } | 216 | } |
| 123 | 217 | ||
| @@ -544,6 +638,36 @@ static int _disable_clocks(struct omap_hwmod *oh) | |||
| 544 | return 0; | 638 | return 0; |
| 545 | } | 639 | } |
| 546 | 640 | ||
| 641 | static void _enable_optional_clocks(struct omap_hwmod *oh) | ||
| 642 | { | ||
| 643 | struct omap_hwmod_opt_clk *oc; | ||
| 644 | int i; | ||
| 645 | |||
| 646 | pr_debug("omap_hwmod: %s: enabling optional clocks\n", oh->name); | ||
| 647 | |||
| 648 | for (i = oh->opt_clks_cnt, oc = oh->opt_clks; i > 0; i--, oc++) | ||
| 649 | if (oc->_clk) { | ||
| 650 | pr_debug("omap_hwmod: enable %s:%s\n", oc->role, | ||
| 651 | oc->_clk->name); | ||
| 652 | clk_enable(oc->_clk); | ||
| 653 | } | ||
| 654 | } | ||
| 655 | |||
| 656 | static void _disable_optional_clocks(struct omap_hwmod *oh) | ||
| 657 | { | ||
| 658 | struct omap_hwmod_opt_clk *oc; | ||
| 659 | int i; | ||
| 660 | |||
| 661 | pr_debug("omap_hwmod: %s: disabling optional clocks\n", oh->name); | ||
| 662 | |||
| 663 | for (i = oh->opt_clks_cnt, oc = oh->opt_clks; i > 0; i--, oc++) | ||
| 664 | if (oc->_clk) { | ||
| 665 | pr_debug("omap_hwmod: disable %s:%s\n", oc->role, | ||
| 666 | oc->_clk->name); | ||
| 667 | clk_disable(oc->_clk); | ||
| 668 | } | ||
| 669 | } | ||
| 670 | |||
| 547 | /** | 671 | /** |
| 548 | * _find_mpu_port_index - find hwmod OCP slave port ID intended for MPU use | 672 | * _find_mpu_port_index - find hwmod OCP slave port ID intended for MPU use |
| 549 | * @oh: struct omap_hwmod * | 673 | * @oh: struct omap_hwmod * |
| @@ -622,7 +746,7 @@ static void __iomem *_find_mpu_rt_base(struct omap_hwmod *oh, u8 index) | |||
| 622 | } | 746 | } |
| 623 | 747 | ||
| 624 | /** | 748 | /** |
| 625 | * _sysc_enable - try to bring a module out of idle via OCP_SYSCONFIG | 749 | * _enable_sysc - try to bring a module out of idle via OCP_SYSCONFIG |
| 626 | * @oh: struct omap_hwmod * | 750 | * @oh: struct omap_hwmod * |
| 627 | * | 751 | * |
| 628 | * If module is marked as SWSUP_SIDLE, force the module out of slave | 752 | * If module is marked as SWSUP_SIDLE, force the module out of slave |
| @@ -630,7 +754,7 @@ static void __iomem *_find_mpu_rt_base(struct omap_hwmod *oh, u8 index) | |||
| 630 | * as SWSUP_MSUSPEND, force the module out of master standby; | 754 | * as SWSUP_MSUSPEND, force the module out of master standby; |
| 631 | * otherwise, configure it for smart-standby. No return value. | 755 | * otherwise, configure it for smart-standby. No return value. |
| 632 | */ | 756 | */ |
| 633 | static void _sysc_enable(struct omap_hwmod *oh) | 757 | static void _enable_sysc(struct omap_hwmod *oh) |
| 634 | { | 758 | { |
| 635 | u8 idlemode, sf; | 759 | u8 idlemode, sf; |
| 636 | u32 v; | 760 | u32 v; |
| @@ -653,14 +777,6 @@ static void _sysc_enable(struct omap_hwmod *oh) | |||
| 653 | _set_master_standbymode(oh, idlemode, &v); | 777 | _set_master_standbymode(oh, idlemode, &v); |
| 654 | } | 778 | } |
| 655 | 779 | ||
| 656 | if (sf & SYSC_HAS_AUTOIDLE) { | ||
| 657 | idlemode = (oh->flags & HWMOD_NO_OCP_AUTOIDLE) ? | ||
| 658 | 0 : 1; | ||
| 659 | _set_module_autoidle(oh, idlemode, &v); | ||
| 660 | } | ||
| 661 | |||
| 662 | /* XXX OCP ENAWAKEUP bit? */ | ||
| 663 | |||
| 664 | /* | 780 | /* |
| 665 | * XXX The clock framework should handle this, by | 781 | * XXX The clock framework should handle this, by |
| 666 | * calling into this code. But this must wait until the | 782 | * calling into this code. But this must wait until the |
| @@ -671,10 +787,25 @@ static void _sysc_enable(struct omap_hwmod *oh) | |||
| 671 | _set_clockactivity(oh, oh->class->sysc->clockact, &v); | 787 | _set_clockactivity(oh, oh->class->sysc->clockact, &v); |
| 672 | 788 | ||
| 673 | _write_sysconfig(v, oh); | 789 | _write_sysconfig(v, oh); |
| 790 | |||
| 791 | /* If slave is in SMARTIDLE, also enable wakeup */ | ||
| 792 | if ((sf & SYSC_HAS_SIDLEMODE) && !(oh->flags & HWMOD_SWSUP_SIDLE)) | ||
| 793 | _enable_wakeup(oh); | ||
| 794 | |||
| 795 | /* | ||
| 796 | * Set the autoidle bit only after setting the smartidle bit | ||
| 797 | * Setting this will not have any impact on the other modules. | ||
| 798 | */ | ||
| 799 | if (sf & SYSC_HAS_AUTOIDLE) { | ||
| 800 | idlemode = (oh->flags & HWMOD_NO_OCP_AUTOIDLE) ? | ||
| 801 | 0 : 1; | ||
| 802 | _set_module_autoidle(oh, idlemode, &v); | ||
| 803 | _write_sysconfig(v, oh); | ||
| 804 | } | ||
| 674 | } | 805 | } |
| 675 | 806 | ||
| 676 | /** | 807 | /** |
| 677 | * _sysc_idle - try to put a module into idle via OCP_SYSCONFIG | 808 | * _idle_sysc - try to put a module into idle via OCP_SYSCONFIG |
| 678 | * @oh: struct omap_hwmod * | 809 | * @oh: struct omap_hwmod * |
| 679 | * | 810 | * |
| 680 | * If module is marked as SWSUP_SIDLE, force the module into slave | 811 | * If module is marked as SWSUP_SIDLE, force the module into slave |
| @@ -682,7 +813,7 @@ static void _sysc_enable(struct omap_hwmod *oh) | |||
| 682 | * as SWSUP_MSUSPEND, force the module into master standby; otherwise, | 813 | * as SWSUP_MSUSPEND, force the module into master standby; otherwise, |
| 683 | * configure it for smart-standby. No return value. | 814 | * configure it for smart-standby. No return value. |
| 684 | */ | 815 | */ |
| 685 | static void _sysc_idle(struct omap_hwmod *oh) | 816 | static void _idle_sysc(struct omap_hwmod *oh) |
| 686 | { | 817 | { |
| 687 | u8 idlemode, sf; | 818 | u8 idlemode, sf; |
| 688 | u32 v; | 819 | u32 v; |
| @@ -709,13 +840,13 @@ static void _sysc_idle(struct omap_hwmod *oh) | |||
| 709 | } | 840 | } |
| 710 | 841 | ||
| 711 | /** | 842 | /** |
| 712 | * _sysc_shutdown - force a module into idle via OCP_SYSCONFIG | 843 | * _shutdown_sysc - force a module into idle via OCP_SYSCONFIG |
| 713 | * @oh: struct omap_hwmod * | 844 | * @oh: struct omap_hwmod * |
| 714 | * | 845 | * |
| 715 | * Force the module into slave idle and master suspend. No return | 846 | * Force the module into slave idle and master suspend. No return |
| 716 | * value. | 847 | * value. |
| 717 | */ | 848 | */ |
| 718 | static void _sysc_shutdown(struct omap_hwmod *oh) | 849 | static void _shutdown_sysc(struct omap_hwmod *oh) |
| 719 | { | 850 | { |
| 720 | u32 v; | 851 | u32 v; |
| 721 | u8 sf; | 852 | u8 sf; |
| @@ -767,10 +898,10 @@ static struct omap_hwmod *_lookup(const char *name) | |||
| 767 | * @data: not used; pass NULL | 898 | * @data: not used; pass NULL |
| 768 | * | 899 | * |
| 769 | * Called by omap_hwmod_late_init() (after omap2_clk_init()). | 900 | * Called by omap_hwmod_late_init() (after omap2_clk_init()). |
| 770 | * Resolves all clock names embedded in the hwmod. Must be called | 901 | * Resolves all clock names embedded in the hwmod. Returns -EINVAL if |
| 771 | * with omap_hwmod_mutex held. Returns -EINVAL if the omap_hwmod | 902 | * the omap_hwmod has not yet been registered or if the clocks have |
| 772 | * has not yet been registered or if the clocks have already been | 903 | * already been initialized, 0 on success, or a non-zero error on |
| 773 | * initialized, 0 on success, or a non-zero error on failure. | 904 | * failure. |
| 774 | */ | 905 | */ |
| 775 | static int _init_clocks(struct omap_hwmod *oh, void *data) | 906 | static int _init_clocks(struct omap_hwmod *oh, void *data) |
| 776 | { | 907 | { |
| @@ -834,56 +965,202 @@ static int _wait_target_ready(struct omap_hwmod *oh) | |||
| 834 | } | 965 | } |
| 835 | 966 | ||
| 836 | /** | 967 | /** |
| 968 | * _lookup_hardreset - return the register bit shift for this hwmod/reset line | ||
| 969 | * @oh: struct omap_hwmod * | ||
| 970 | * @name: name of the reset line in the context of this hwmod | ||
| 971 | * | ||
| 972 | * Return the bit position of the reset line that match the | ||
| 973 | * input name. Return -ENOENT if not found. | ||
| 974 | */ | ||
| 975 | static u8 _lookup_hardreset(struct omap_hwmod *oh, const char *name) | ||
| 976 | { | ||
| 977 | int i; | ||
| 978 | |||
| 979 | for (i = 0; i < oh->rst_lines_cnt; i++) { | ||
| 980 | const char *rst_line = oh->rst_lines[i].name; | ||
| 981 | if (!strcmp(rst_line, name)) { | ||
| 982 | u8 shift = oh->rst_lines[i].rst_shift; | ||
| 983 | pr_debug("omap_hwmod: %s: _lookup_hardreset: %s: %d\n", | ||
| 984 | oh->name, rst_line, shift); | ||
| 985 | |||
| 986 | return shift; | ||
| 987 | } | ||
| 988 | } | ||
| 989 | |||
| 990 | return -ENOENT; | ||
| 991 | } | ||
| 992 | |||
| 993 | /** | ||
| 994 | * _assert_hardreset - assert the HW reset line of submodules | ||
| 995 | * contained in the hwmod module. | ||
| 996 | * @oh: struct omap_hwmod * | ||
| 997 | * @name: name of the reset line to lookup and assert | ||
| 998 | * | ||
| 999 | * Some IP like dsp, ipu or iva contain processor that require | ||
| 1000 | * an HW reset line to be assert / deassert in order to enable fully | ||
| 1001 | * the IP. | ||
| 1002 | */ | ||
| 1003 | static int _assert_hardreset(struct omap_hwmod *oh, const char *name) | ||
| 1004 | { | ||
| 1005 | u8 shift; | ||
| 1006 | |||
| 1007 | if (!oh) | ||
| 1008 | return -EINVAL; | ||
| 1009 | |||
| 1010 | shift = _lookup_hardreset(oh, name); | ||
| 1011 | if (IS_ERR_VALUE(shift)) | ||
| 1012 | return shift; | ||
| 1013 | |||
| 1014 | if (cpu_is_omap24xx() || cpu_is_omap34xx()) | ||
| 1015 | return omap2_prm_assert_hardreset(oh->prcm.omap2.module_offs, | ||
| 1016 | shift); | ||
| 1017 | else if (cpu_is_omap44xx()) | ||
| 1018 | return omap4_prm_assert_hardreset(oh->prcm.omap4.rstctrl_reg, | ||
| 1019 | shift); | ||
| 1020 | else | ||
| 1021 | return -EINVAL; | ||
| 1022 | } | ||
| 1023 | |||
| 1024 | /** | ||
| 1025 | * _deassert_hardreset - deassert the HW reset line of submodules contained | ||
| 1026 | * in the hwmod module. | ||
| 1027 | * @oh: struct omap_hwmod * | ||
| 1028 | * @name: name of the reset line to look up and deassert | ||
| 1029 | * | ||
| 1030 | * Some IP like dsp, ipu or iva contain processor that require | ||
| 1031 | * an HW reset line to be assert / deassert in order to enable fully | ||
| 1032 | * the IP. | ||
| 1033 | */ | ||
| 1034 | static int _deassert_hardreset(struct omap_hwmod *oh, const char *name) | ||
| 1035 | { | ||
| 1036 | u8 shift; | ||
| 1037 | int r; | ||
| 1038 | |||
| 1039 | if (!oh) | ||
| 1040 | return -EINVAL; | ||
| 1041 | |||
| 1042 | shift = _lookup_hardreset(oh, name); | ||
| 1043 | if (IS_ERR_VALUE(shift)) | ||
| 1044 | return shift; | ||
| 1045 | |||
| 1046 | if (cpu_is_omap24xx() || cpu_is_omap34xx()) | ||
| 1047 | r = omap2_prm_deassert_hardreset(oh->prcm.omap2.module_offs, | ||
| 1048 | shift); | ||
| 1049 | else if (cpu_is_omap44xx()) | ||
| 1050 | r = omap4_prm_deassert_hardreset(oh->prcm.omap4.rstctrl_reg, | ||
| 1051 | shift); | ||
| 1052 | else | ||
| 1053 | return -EINVAL; | ||
| 1054 | |||
| 1055 | if (r == -EBUSY) | ||
| 1056 | pr_warning("omap_hwmod: %s: failed to hardreset\n", oh->name); | ||
| 1057 | |||
| 1058 | return r; | ||
| 1059 | } | ||
| 1060 | |||
| 1061 | /** | ||
| 1062 | * _read_hardreset - read the HW reset line state of submodules | ||
| 1063 | * contained in the hwmod module | ||
| 1064 | * @oh: struct omap_hwmod * | ||
| 1065 | * @name: name of the reset line to look up and read | ||
| 1066 | * | ||
| 1067 | * Return the state of the reset line. | ||
| 1068 | */ | ||
| 1069 | static int _read_hardreset(struct omap_hwmod *oh, const char *name) | ||
| 1070 | { | ||
| 1071 | u8 shift; | ||
| 1072 | |||
| 1073 | if (!oh) | ||
| 1074 | return -EINVAL; | ||
| 1075 | |||
| 1076 | shift = _lookup_hardreset(oh, name); | ||
| 1077 | if (IS_ERR_VALUE(shift)) | ||
| 1078 | return shift; | ||
| 1079 | |||
| 1080 | if (cpu_is_omap24xx() || cpu_is_omap34xx()) { | ||
| 1081 | return omap2_prm_is_hardreset_asserted(oh->prcm.omap2.module_offs, | ||
| 1082 | shift); | ||
| 1083 | } else if (cpu_is_omap44xx()) { | ||
| 1084 | return omap4_prm_is_hardreset_asserted(oh->prcm.omap4.rstctrl_reg, | ||
| 1085 | shift); | ||
| 1086 | } else { | ||
| 1087 | return -EINVAL; | ||
| 1088 | } | ||
| 1089 | } | ||
| 1090 | |||
| 1091 | /** | ||
| 837 | * _reset - reset an omap_hwmod | 1092 | * _reset - reset an omap_hwmod |
| 838 | * @oh: struct omap_hwmod * | 1093 | * @oh: struct omap_hwmod * |
| 839 | * | 1094 | * |
| 840 | * Resets an omap_hwmod @oh via the OCP_SYSCONFIG bit. hwmod must be | 1095 | * Resets an omap_hwmod @oh via the OCP_SYSCONFIG bit. hwmod must be |
| 841 | * enabled for this to work. Must be called with omap_hwmod_mutex | 1096 | * enabled for this to work. Returns -EINVAL if the hwmod cannot be |
| 842 | * held. Returns -EINVAL if the hwmod cannot be reset this way or if | 1097 | * reset this way or if the hwmod is in the wrong state, -ETIMEDOUT if |
| 843 | * the hwmod is in the wrong state, -ETIMEDOUT if the module did not | 1098 | * the module did not reset in time, or 0 upon success. |
| 844 | * reset in time, or 0 upon success. | 1099 | * |
| 1100 | * In OMAP3 a specific SYSSTATUS register is used to get the reset status. | ||
| 1101 | * Starting in OMAP4, some IPs does not have SYSSTATUS register and instead | ||
| 1102 | * use the SYSCONFIG softreset bit to provide the status. | ||
| 1103 | * | ||
| 1104 | * Note that some IP like McBSP does have a reset control but no reset status. | ||
| 845 | */ | 1105 | */ |
| 846 | static int _reset(struct omap_hwmod *oh) | 1106 | static int _reset(struct omap_hwmod *oh) |
| 847 | { | 1107 | { |
| 848 | u32 r, v; | 1108 | u32 v; |
| 849 | int c = 0; | 1109 | int c = 0; |
| 1110 | int ret = 0; | ||
| 850 | 1111 | ||
| 851 | if (!oh->class->sysc || | 1112 | if (!oh->class->sysc || |
| 852 | !(oh->class->sysc->sysc_flags & SYSC_HAS_SOFTRESET) || | 1113 | !(oh->class->sysc->sysc_flags & SYSC_HAS_SOFTRESET)) |
| 853 | (oh->class->sysc->sysc_flags & SYSS_MISSING)) | ||
| 854 | return -EINVAL; | 1114 | return -EINVAL; |
| 855 | 1115 | ||
| 856 | /* clocks must be on for this operation */ | 1116 | /* clocks must be on for this operation */ |
| 857 | if (oh->_state != _HWMOD_STATE_ENABLED) { | 1117 | if (oh->_state != _HWMOD_STATE_ENABLED) { |
| 858 | WARN(1, "omap_hwmod: %s: reset can only be entered from " | 1118 | pr_warning("omap_hwmod: %s: reset can only be entered from " |
| 859 | "enabled state\n", oh->name); | 1119 | "enabled state\n", oh->name); |
| 860 | return -EINVAL; | 1120 | return -EINVAL; |
| 861 | } | 1121 | } |
| 862 | 1122 | ||
| 1123 | /* For some modules, all optionnal clocks need to be enabled as well */ | ||
| 1124 | if (oh->flags & HWMOD_CONTROL_OPT_CLKS_IN_RESET) | ||
| 1125 | _enable_optional_clocks(oh); | ||
| 1126 | |||
| 863 | pr_debug("omap_hwmod: %s: resetting\n", oh->name); | 1127 | pr_debug("omap_hwmod: %s: resetting\n", oh->name); |
| 864 | 1128 | ||
| 865 | v = oh->_sysc_cache; | 1129 | v = oh->_sysc_cache; |
| 866 | r = _set_softreset(oh, &v); | 1130 | ret = _set_softreset(oh, &v); |
| 867 | if (r) | 1131 | if (ret) |
| 868 | return r; | 1132 | goto dis_opt_clks; |
| 869 | _write_sysconfig(v, oh); | 1133 | _write_sysconfig(v, oh); |
| 870 | 1134 | ||
| 871 | omap_test_timeout((omap_hwmod_readl(oh, oh->class->sysc->syss_offs) & | 1135 | if (oh->class->sysc->sysc_flags & SYSS_HAS_RESET_STATUS) |
| 872 | SYSS_RESETDONE_MASK), | 1136 | omap_test_timeout((omap_hwmod_read(oh, |
| 873 | MAX_MODULE_RESET_WAIT, c); | 1137 | oh->class->sysc->syss_offs) |
| 874 | 1138 | & SYSS_RESETDONE_MASK), | |
| 875 | if (c == MAX_MODULE_RESET_WAIT) | 1139 | MAX_MODULE_SOFTRESET_WAIT, c); |
| 876 | WARN(1, "omap_hwmod: %s: failed to reset in %d usec\n", | 1140 | else if (oh->class->sysc->sysc_flags & SYSC_HAS_RESET_STATUS) |
| 877 | oh->name, MAX_MODULE_RESET_WAIT); | 1141 | omap_test_timeout(!(omap_hwmod_read(oh, |
| 1142 | oh->class->sysc->sysc_offs) | ||
| 1143 | & SYSC_TYPE2_SOFTRESET_MASK), | ||
| 1144 | MAX_MODULE_SOFTRESET_WAIT, c); | ||
| 1145 | |||
| 1146 | if (c == MAX_MODULE_SOFTRESET_WAIT) | ||
| 1147 | pr_warning("omap_hwmod: %s: softreset failed (waited %d usec)\n", | ||
| 1148 | oh->name, MAX_MODULE_SOFTRESET_WAIT); | ||
| 878 | else | 1149 | else |
| 879 | pr_debug("omap_hwmod: %s: reset in %d usec\n", oh->name, c); | 1150 | pr_debug("omap_hwmod: %s: softreset in %d usec\n", oh->name, c); |
| 880 | 1151 | ||
| 881 | /* | 1152 | /* |
| 882 | * XXX add _HWMOD_STATE_WEDGED for modules that don't come back from | 1153 | * XXX add _HWMOD_STATE_WEDGED for modules that don't come back from |
| 883 | * _wait_target_ready() or _reset() | 1154 | * _wait_target_ready() or _reset() |
| 884 | */ | 1155 | */ |
| 885 | 1156 | ||
| 886 | return (c == MAX_MODULE_RESET_WAIT) ? -ETIMEDOUT : 0; | 1157 | ret = (c == MAX_MODULE_SOFTRESET_WAIT) ? -ETIMEDOUT : 0; |
| 1158 | |||
| 1159 | dis_opt_clks: | ||
| 1160 | if (oh->flags & HWMOD_CONTROL_OPT_CLKS_IN_RESET) | ||
| 1161 | _disable_optional_clocks(oh); | ||
| 1162 | |||
| 1163 | return ret; | ||
| 887 | } | 1164 | } |
| 888 | 1165 | ||
| 889 | /** | 1166 | /** |
| @@ -891,9 +1168,11 @@ static int _reset(struct omap_hwmod *oh) | |||
| 891 | * @oh: struct omap_hwmod * | 1168 | * @oh: struct omap_hwmod * |
| 892 | * | 1169 | * |
| 893 | * Enables an omap_hwmod @oh such that the MPU can access the hwmod's | 1170 | * Enables an omap_hwmod @oh such that the MPU can access the hwmod's |
| 894 | * register target. Must be called with omap_hwmod_mutex held. | 1171 | * register target. (This function has a full name -- |
| 895 | * Returns -EINVAL if the hwmod is in the wrong state or passes along | 1172 | * _omap_hwmod_enable() rather than simply _enable() -- because it is |
| 896 | * the return value of _wait_target_ready(). | 1173 | * currently required by the pm34xx.c idle loop.) Returns -EINVAL if |
| 1174 | * the hwmod is in the wrong state or passes along the return value of | ||
| 1175 | * _wait_target_ready(). | ||
| 897 | */ | 1176 | */ |
| 898 | int _omap_hwmod_enable(struct omap_hwmod *oh) | 1177 | int _omap_hwmod_enable(struct omap_hwmod *oh) |
| 899 | { | 1178 | { |
| @@ -909,6 +1188,15 @@ int _omap_hwmod_enable(struct omap_hwmod *oh) | |||
| 909 | 1188 | ||
| 910 | pr_debug("omap_hwmod: %s: enabling\n", oh->name); | 1189 | pr_debug("omap_hwmod: %s: enabling\n", oh->name); |
| 911 | 1190 | ||
| 1191 | /* | ||
| 1192 | * If an IP contains only one HW reset line, then de-assert it in order | ||
| 1193 | * to allow to enable the clocks. Otherwise the PRCM will return | ||
| 1194 | * Intransition status, and the init will failed. | ||
| 1195 | */ | ||
| 1196 | if ((oh->_state == _HWMOD_STATE_INITIALIZED || | ||
| 1197 | oh->_state == _HWMOD_STATE_DISABLED) && oh->rst_lines_cnt == 1) | ||
| 1198 | _deassert_hardreset(oh, oh->rst_lines[0].name); | ||
| 1199 | |||
| 912 | /* XXX mux balls */ | 1200 | /* XXX mux balls */ |
| 913 | 1201 | ||
| 914 | _add_initiator_dep(oh, mpu_oh); | 1202 | _add_initiator_dep(oh, mpu_oh); |
| @@ -922,7 +1210,7 @@ int _omap_hwmod_enable(struct omap_hwmod *oh) | |||
| 922 | if (oh->class->sysc) { | 1210 | if (oh->class->sysc) { |
| 923 | if (!(oh->_int_flags & _HWMOD_SYSCONFIG_LOADED)) | 1211 | if (!(oh->_int_flags & _HWMOD_SYSCONFIG_LOADED)) |
| 924 | _update_sysc_cache(oh); | 1212 | _update_sysc_cache(oh); |
| 925 | _sysc_enable(oh); | 1213 | _enable_sysc(oh); |
| 926 | } | 1214 | } |
| 927 | } else { | 1215 | } else { |
| 928 | pr_debug("omap_hwmod: %s: _wait_target_ready: %d\n", | 1216 | pr_debug("omap_hwmod: %s: _wait_target_ready: %d\n", |
| @@ -933,12 +1221,14 @@ int _omap_hwmod_enable(struct omap_hwmod *oh) | |||
| 933 | } | 1221 | } |
| 934 | 1222 | ||
| 935 | /** | 1223 | /** |
| 936 | * _idle - idle an omap_hwmod | 1224 | * _omap_hwmod_idle - idle an omap_hwmod |
| 937 | * @oh: struct omap_hwmod * | 1225 | * @oh: struct omap_hwmod * |
| 938 | * | 1226 | * |
| 939 | * Idles an omap_hwmod @oh. This should be called once the hwmod has | 1227 | * Idles an omap_hwmod @oh. This should be called once the hwmod has |
| 940 | * no further work. Returns -EINVAL if the hwmod is in the wrong | 1228 | * no further work. (This function has a full name -- |
| 941 | * state or returns 0. | 1229 | * _omap_hwmod_idle() rather than simply _idle() -- because it is |
| 1230 | * currently required by the pm34xx.c idle loop.) Returns -EINVAL if | ||
| 1231 | * the hwmod is in the wrong state or returns 0. | ||
| 942 | */ | 1232 | */ |
| 943 | int _omap_hwmod_idle(struct omap_hwmod *oh) | 1233 | int _omap_hwmod_idle(struct omap_hwmod *oh) |
| 944 | { | 1234 | { |
| @@ -951,7 +1241,7 @@ int _omap_hwmod_idle(struct omap_hwmod *oh) | |||
| 951 | pr_debug("omap_hwmod: %s: idling\n", oh->name); | 1241 | pr_debug("omap_hwmod: %s: idling\n", oh->name); |
| 952 | 1242 | ||
| 953 | if (oh->class->sysc) | 1243 | if (oh->class->sysc) |
| 954 | _sysc_idle(oh); | 1244 | _idle_sysc(oh); |
| 955 | _del_initiator_dep(oh, mpu_oh); | 1245 | _del_initiator_dep(oh, mpu_oh); |
| 956 | _disable_clocks(oh); | 1246 | _disable_clocks(oh); |
| 957 | 1247 | ||
| @@ -981,10 +1271,21 @@ static int _shutdown(struct omap_hwmod *oh) | |||
| 981 | pr_debug("omap_hwmod: %s: disabling\n", oh->name); | 1271 | pr_debug("omap_hwmod: %s: disabling\n", oh->name); |
| 982 | 1272 | ||
| 983 | if (oh->class->sysc) | 1273 | if (oh->class->sysc) |
| 984 | _sysc_shutdown(oh); | 1274 | _shutdown_sysc(oh); |
| 985 | _del_initiator_dep(oh, mpu_oh); | 1275 | |
| 986 | /* XXX what about the other system initiators here? DMA, tesla, d2d */ | 1276 | /* |
| 987 | _disable_clocks(oh); | 1277 | * If an IP contains only one HW reset line, then assert it |
| 1278 | * before disabling the clocks and shutting down the IP. | ||
| 1279 | */ | ||
| 1280 | if (oh->rst_lines_cnt == 1) | ||
| 1281 | _assert_hardreset(oh, oh->rst_lines[0].name); | ||
| 1282 | |||
| 1283 | /* clocks and deps are already disabled in idle */ | ||
| 1284 | if (oh->_state == _HWMOD_STATE_ENABLED) { | ||
| 1285 | _del_initiator_dep(oh, mpu_oh); | ||
| 1286 | /* XXX what about the other system initiators here? dma, dsp */ | ||
| 1287 | _disable_clocks(oh); | ||
| 1288 | } | ||
| 988 | /* XXX Should this code also force-disable the optional clocks? */ | 1289 | /* XXX Should this code also force-disable the optional clocks? */ |
| 989 | 1290 | ||
| 990 | /* XXX mux any associated balls to safe mode */ | 1291 | /* XXX mux any associated balls to safe mode */ |
| @@ -1000,11 +1301,10 @@ static int _shutdown(struct omap_hwmod *oh) | |||
| 1000 | * @skip_setup_idle_p: do not idle hwmods at the end of the fn if 1 | 1301 | * @skip_setup_idle_p: do not idle hwmods at the end of the fn if 1 |
| 1001 | * | 1302 | * |
| 1002 | * Writes the CLOCKACTIVITY bits @clockact to the hwmod @oh | 1303 | * Writes the CLOCKACTIVITY bits @clockact to the hwmod @oh |
| 1003 | * OCP_SYSCONFIG register. Must be called with omap_hwmod_mutex held. | 1304 | * OCP_SYSCONFIG register. @skip_setup_idle is intended to be used on |
| 1004 | * @skip_setup_idle is intended to be used on a system that will not | 1305 | * a system that will not call omap_hwmod_enable() to enable devices |
| 1005 | * call omap_hwmod_enable() to enable devices (e.g., a system without | 1306 | * (e.g., a system without PM runtime). Returns -EINVAL if the hwmod |
| 1006 | * PM runtime). Returns -EINVAL if the hwmod is in the wrong state or | 1307 | * is in the wrong state or returns 0. |
| 1007 | * returns 0. | ||
| 1008 | */ | 1308 | */ |
| 1009 | static int _setup(struct omap_hwmod *oh, void *data) | 1309 | static int _setup(struct omap_hwmod *oh, void *data) |
| 1010 | { | 1310 | { |
| @@ -1034,8 +1334,19 @@ static int _setup(struct omap_hwmod *oh, void *data) | |||
| 1034 | } | 1334 | } |
| 1035 | } | 1335 | } |
| 1036 | 1336 | ||
| 1337 | mutex_init(&oh->_mutex); | ||
| 1037 | oh->_state = _HWMOD_STATE_INITIALIZED; | 1338 | oh->_state = _HWMOD_STATE_INITIALIZED; |
| 1038 | 1339 | ||
| 1340 | /* | ||
| 1341 | * In the case of hwmod with hardreset that should not be | ||
| 1342 | * de-assert at boot time, we have to keep the module | ||
| 1343 | * initialized, because we cannot enable it properly with the | ||
| 1344 | * reset asserted. Exit without warning because that behavior is | ||
| 1345 | * expected. | ||
| 1346 | */ | ||
| 1347 | if ((oh->flags & HWMOD_INIT_NO_RESET) && oh->rst_lines_cnt == 1) | ||
| 1348 | return 0; | ||
| 1349 | |||
| 1039 | r = _omap_hwmod_enable(oh); | 1350 | r = _omap_hwmod_enable(oh); |
| 1040 | if (r) { | 1351 | if (r) { |
| 1041 | pr_warning("omap_hwmod: %s: cannot be enabled (%d)\n", | 1352 | pr_warning("omap_hwmod: %s: cannot be enabled (%d)\n", |
| @@ -1044,16 +1355,16 @@ static int _setup(struct omap_hwmod *oh, void *data) | |||
| 1044 | } | 1355 | } |
| 1045 | 1356 | ||
| 1046 | if (!(oh->flags & HWMOD_INIT_NO_RESET)) { | 1357 | if (!(oh->flags & HWMOD_INIT_NO_RESET)) { |
| 1358 | _reset(oh); | ||
| 1359 | |||
| 1047 | /* | 1360 | /* |
| 1048 | * XXX Do the OCP_SYSCONFIG bits need to be | 1361 | * OCP_SYSCONFIG bits need to be reprogrammed after a softreset. |
| 1049 | * reprogrammed after a reset? If not, then this can | 1362 | * The _omap_hwmod_enable() function should be split to |
| 1050 | * be removed. If they do, then probably the | 1363 | * avoid the rewrite of the OCP_SYSCONFIG register. |
| 1051 | * _omap_hwmod_enable() function should be split to avoid the | ||
| 1052 | * rewrite of the OCP_SYSCONFIG register. | ||
| 1053 | */ | 1364 | */ |
| 1054 | if (oh->class->sysc) { | 1365 | if (oh->class->sysc) { |
| 1055 | _update_sysc_cache(oh); | 1366 | _update_sysc_cache(oh); |
| 1056 | _sysc_enable(oh); | 1367 | _enable_sysc(oh); |
| 1057 | } | 1368 | } |
| 1058 | } | 1369 | } |
| 1059 | 1370 | ||
| @@ -1067,14 +1378,20 @@ static int _setup(struct omap_hwmod *oh, void *data) | |||
| 1067 | 1378 | ||
| 1068 | /* Public functions */ | 1379 | /* Public functions */ |
| 1069 | 1380 | ||
| 1070 | u32 omap_hwmod_readl(struct omap_hwmod *oh, u16 reg_offs) | 1381 | u32 omap_hwmod_read(struct omap_hwmod *oh, u16 reg_offs) |
| 1071 | { | 1382 | { |
| 1072 | return __raw_readl(oh->_mpu_rt_va + reg_offs); | 1383 | if (oh->flags & HWMOD_16BIT_REG) |
| 1384 | return __raw_readw(oh->_mpu_rt_va + reg_offs); | ||
| 1385 | else | ||
| 1386 | return __raw_readl(oh->_mpu_rt_va + reg_offs); | ||
| 1073 | } | 1387 | } |
| 1074 | 1388 | ||
| 1075 | void omap_hwmod_writel(u32 v, struct omap_hwmod *oh, u16 reg_offs) | 1389 | void omap_hwmod_write(u32 v, struct omap_hwmod *oh, u16 reg_offs) |
| 1076 | { | 1390 | { |
| 1077 | __raw_writel(v, oh->_mpu_rt_va + reg_offs); | 1391 | if (oh->flags & HWMOD_16BIT_REG) |
| 1392 | __raw_writew(v, oh->_mpu_rt_va + reg_offs); | ||
| 1393 | else | ||
| 1394 | __raw_writel(v, oh->_mpu_rt_va + reg_offs); | ||
| 1078 | } | 1395 | } |
| 1079 | 1396 | ||
| 1080 | /** | 1397 | /** |
| @@ -1309,7 +1626,7 @@ int omap_hwmod_unregister(struct omap_hwmod *oh) | |||
| 1309 | * omap_hwmod_enable - enable an omap_hwmod | 1626 | * omap_hwmod_enable - enable an omap_hwmod |
| 1310 | * @oh: struct omap_hwmod * | 1627 | * @oh: struct omap_hwmod * |
| 1311 | * | 1628 | * |
| 1312 | * Enable an omap_hwomd @oh. Intended to be called by omap_device_enable(). | 1629 | * Enable an omap_hwmod @oh. Intended to be called by omap_device_enable(). |
| 1313 | * Returns -EINVAL on error or passes along the return value from _enable(). | 1630 | * Returns -EINVAL on error or passes along the return value from _enable(). |
| 1314 | */ | 1631 | */ |
| 1315 | int omap_hwmod_enable(struct omap_hwmod *oh) | 1632 | int omap_hwmod_enable(struct omap_hwmod *oh) |
| @@ -1319,9 +1636,9 @@ int omap_hwmod_enable(struct omap_hwmod *oh) | |||
| 1319 | if (!oh) | 1636 | if (!oh) |
| 1320 | return -EINVAL; | 1637 | return -EINVAL; |
| 1321 | 1638 | ||
| 1322 | mutex_lock(&omap_hwmod_mutex); | 1639 | mutex_lock(&oh->_mutex); |
| 1323 | r = _omap_hwmod_enable(oh); | 1640 | r = _omap_hwmod_enable(oh); |
| 1324 | mutex_unlock(&omap_hwmod_mutex); | 1641 | mutex_unlock(&oh->_mutex); |
| 1325 | 1642 | ||
| 1326 | return r; | 1643 | return r; |
| 1327 | } | 1644 | } |
| @@ -1331,7 +1648,7 @@ int omap_hwmod_enable(struct omap_hwmod *oh) | |||
| 1331 | * omap_hwmod_idle - idle an omap_hwmod | 1648 | * omap_hwmod_idle - idle an omap_hwmod |
| 1332 | * @oh: struct omap_hwmod * | 1649 | * @oh: struct omap_hwmod * |
| 1333 | * | 1650 | * |
| 1334 | * Idle an omap_hwomd @oh. Intended to be called by omap_device_idle(). | 1651 | * Idle an omap_hwmod @oh. Intended to be called by omap_device_idle(). |
| 1335 | * Returns -EINVAL on error or passes along the return value from _idle(). | 1652 | * Returns -EINVAL on error or passes along the return value from _idle(). |
| 1336 | */ | 1653 | */ |
| 1337 | int omap_hwmod_idle(struct omap_hwmod *oh) | 1654 | int omap_hwmod_idle(struct omap_hwmod *oh) |
| @@ -1339,9 +1656,9 @@ int omap_hwmod_idle(struct omap_hwmod *oh) | |||
| 1339 | if (!oh) | 1656 | if (!oh) |
| 1340 | return -EINVAL; | 1657 | return -EINVAL; |
| 1341 | 1658 | ||
| 1342 | mutex_lock(&omap_hwmod_mutex); | 1659 | mutex_lock(&oh->_mutex); |
| 1343 | _omap_hwmod_idle(oh); | 1660 | _omap_hwmod_idle(oh); |
| 1344 | mutex_unlock(&omap_hwmod_mutex); | 1661 | mutex_unlock(&oh->_mutex); |
| 1345 | 1662 | ||
| 1346 | return 0; | 1663 | return 0; |
| 1347 | } | 1664 | } |
| @@ -1350,7 +1667,7 @@ int omap_hwmod_idle(struct omap_hwmod *oh) | |||
| 1350 | * omap_hwmod_shutdown - shutdown an omap_hwmod | 1667 | * omap_hwmod_shutdown - shutdown an omap_hwmod |
| 1351 | * @oh: struct omap_hwmod * | 1668 | * @oh: struct omap_hwmod * |
| 1352 | * | 1669 | * |
| 1353 | * Shutdown an omap_hwomd @oh. Intended to be called by | 1670 | * Shutdown an omap_hwmod @oh. Intended to be called by |
| 1354 | * omap_device_shutdown(). Returns -EINVAL on error or passes along | 1671 | * omap_device_shutdown(). Returns -EINVAL on error or passes along |
| 1355 | * the return value from _shutdown(). | 1672 | * the return value from _shutdown(). |
| 1356 | */ | 1673 | */ |
| @@ -1359,9 +1676,9 @@ int omap_hwmod_shutdown(struct omap_hwmod *oh) | |||
| 1359 | if (!oh) | 1676 | if (!oh) |
| 1360 | return -EINVAL; | 1677 | return -EINVAL; |
| 1361 | 1678 | ||
| 1362 | mutex_lock(&omap_hwmod_mutex); | 1679 | mutex_lock(&oh->_mutex); |
| 1363 | _shutdown(oh); | 1680 | _shutdown(oh); |
| 1364 | mutex_unlock(&omap_hwmod_mutex); | 1681 | mutex_unlock(&oh->_mutex); |
| 1365 | 1682 | ||
| 1366 | return 0; | 1683 | return 0; |
| 1367 | } | 1684 | } |
| @@ -1374,9 +1691,9 @@ int omap_hwmod_shutdown(struct omap_hwmod *oh) | |||
| 1374 | */ | 1691 | */ |
| 1375 | int omap_hwmod_enable_clocks(struct omap_hwmod *oh) | 1692 | int omap_hwmod_enable_clocks(struct omap_hwmod *oh) |
| 1376 | { | 1693 | { |
| 1377 | mutex_lock(&omap_hwmod_mutex); | 1694 | mutex_lock(&oh->_mutex); |
| 1378 | _enable_clocks(oh); | 1695 | _enable_clocks(oh); |
| 1379 | mutex_unlock(&omap_hwmod_mutex); | 1696 | mutex_unlock(&oh->_mutex); |
| 1380 | 1697 | ||
| 1381 | return 0; | 1698 | return 0; |
| 1382 | } | 1699 | } |
| @@ -1389,9 +1706,9 @@ int omap_hwmod_enable_clocks(struct omap_hwmod *oh) | |||
| 1389 | */ | 1706 | */ |
| 1390 | int omap_hwmod_disable_clocks(struct omap_hwmod *oh) | 1707 | int omap_hwmod_disable_clocks(struct omap_hwmod *oh) |
| 1391 | { | 1708 | { |
| 1392 | mutex_lock(&omap_hwmod_mutex); | 1709 | mutex_lock(&oh->_mutex); |
| 1393 | _disable_clocks(oh); | 1710 | _disable_clocks(oh); |
| 1394 | mutex_unlock(&omap_hwmod_mutex); | 1711 | mutex_unlock(&oh->_mutex); |
| 1395 | 1712 | ||
| 1396 | return 0; | 1713 | return 0; |
| 1397 | } | 1714 | } |
| @@ -1421,7 +1738,7 @@ void omap_hwmod_ocp_barrier(struct omap_hwmod *oh) | |||
| 1421 | * Forces posted writes to complete on the OCP thread handling | 1738 | * Forces posted writes to complete on the OCP thread handling |
| 1422 | * register writes | 1739 | * register writes |
| 1423 | */ | 1740 | */ |
| 1424 | omap_hwmod_readl(oh, oh->class->sysc->sysc_offs); | 1741 | omap_hwmod_read(oh, oh->class->sysc->sysc_offs); |
| 1425 | } | 1742 | } |
| 1426 | 1743 | ||
| 1427 | /** | 1744 | /** |
| @@ -1430,20 +1747,18 @@ void omap_hwmod_ocp_barrier(struct omap_hwmod *oh) | |||
| 1430 | * | 1747 | * |
| 1431 | * Under some conditions, a driver may wish to reset the entire device. | 1748 | * Under some conditions, a driver may wish to reset the entire device. |
| 1432 | * Called from omap_device code. Returns -EINVAL on error or passes along | 1749 | * Called from omap_device code. Returns -EINVAL on error or passes along |
| 1433 | * the return value from _reset()/_enable(). | 1750 | * the return value from _reset(). |
| 1434 | */ | 1751 | */ |
| 1435 | int omap_hwmod_reset(struct omap_hwmod *oh) | 1752 | int omap_hwmod_reset(struct omap_hwmod *oh) |
| 1436 | { | 1753 | { |
| 1437 | int r; | 1754 | int r; |
| 1438 | 1755 | ||
| 1439 | if (!oh || !(oh->_state & _HWMOD_STATE_ENABLED)) | 1756 | if (!oh) |
| 1440 | return -EINVAL; | 1757 | return -EINVAL; |
| 1441 | 1758 | ||
| 1442 | mutex_lock(&omap_hwmod_mutex); | 1759 | mutex_lock(&oh->_mutex); |
| 1443 | r = _reset(oh); | 1760 | r = _reset(oh); |
| 1444 | if (!r) | 1761 | mutex_unlock(&oh->_mutex); |
| 1445 | r = _omap_hwmod_enable(oh); | ||
| 1446 | mutex_unlock(&omap_hwmod_mutex); | ||
| 1447 | 1762 | ||
| 1448 | return r; | 1763 | return r; |
| 1449 | } | 1764 | } |
| @@ -1468,7 +1783,7 @@ int omap_hwmod_count_resources(struct omap_hwmod *oh) | |||
| 1468 | { | 1783 | { |
| 1469 | int ret, i; | 1784 | int ret, i; |
| 1470 | 1785 | ||
| 1471 | ret = oh->mpu_irqs_cnt + oh->sdma_chs_cnt; | 1786 | ret = oh->mpu_irqs_cnt + oh->sdma_reqs_cnt; |
| 1472 | 1787 | ||
| 1473 | for (i = 0; i < oh->slaves_cnt; i++) | 1788 | for (i = 0; i < oh->slaves_cnt; i++) |
| 1474 | ret += oh->slaves[i]->addr_cnt; | 1789 | ret += oh->slaves[i]->addr_cnt; |
| @@ -1501,10 +1816,10 @@ int omap_hwmod_fill_resources(struct omap_hwmod *oh, struct resource *res) | |||
| 1501 | r++; | 1816 | r++; |
| 1502 | } | 1817 | } |
| 1503 | 1818 | ||
| 1504 | for (i = 0; i < oh->sdma_chs_cnt; i++) { | 1819 | for (i = 0; i < oh->sdma_reqs_cnt; i++) { |
| 1505 | (res + r)->name = (oh->sdma_chs + i)->name; | 1820 | (res + r)->name = (oh->sdma_reqs + i)->name; |
| 1506 | (res + r)->start = (oh->sdma_chs + i)->dma_ch; | 1821 | (res + r)->start = (oh->sdma_reqs + i)->dma_req; |
| 1507 | (res + r)->end = (oh->sdma_chs + i)->dma_ch; | 1822 | (res + r)->end = (oh->sdma_reqs + i)->dma_req; |
| 1508 | (res + r)->flags = IORESOURCE_DMA; | 1823 | (res + r)->flags = IORESOURCE_DMA; |
| 1509 | r++; | 1824 | r++; |
| 1510 | } | 1825 | } |
| @@ -1644,9 +1959,9 @@ int omap_hwmod_enable_wakeup(struct omap_hwmod *oh) | |||
| 1644 | !(oh->class->sysc->sysc_flags & SYSC_HAS_ENAWAKEUP)) | 1959 | !(oh->class->sysc->sysc_flags & SYSC_HAS_ENAWAKEUP)) |
| 1645 | return -EINVAL; | 1960 | return -EINVAL; |
| 1646 | 1961 | ||
| 1647 | mutex_lock(&omap_hwmod_mutex); | 1962 | mutex_lock(&oh->_mutex); |
| 1648 | _enable_wakeup(oh); | 1963 | _enable_wakeup(oh); |
| 1649 | mutex_unlock(&omap_hwmod_mutex); | 1964 | mutex_unlock(&oh->_mutex); |
| 1650 | 1965 | ||
| 1651 | return 0; | 1966 | return 0; |
| 1652 | } | 1967 | } |
| @@ -1669,14 +1984,92 @@ int omap_hwmod_disable_wakeup(struct omap_hwmod *oh) | |||
| 1669 | !(oh->class->sysc->sysc_flags & SYSC_HAS_ENAWAKEUP)) | 1984 | !(oh->class->sysc->sysc_flags & SYSC_HAS_ENAWAKEUP)) |
| 1670 | return -EINVAL; | 1985 | return -EINVAL; |
| 1671 | 1986 | ||
| 1672 | mutex_lock(&omap_hwmod_mutex); | 1987 | mutex_lock(&oh->_mutex); |
| 1673 | _disable_wakeup(oh); | 1988 | _disable_wakeup(oh); |
| 1674 | mutex_unlock(&omap_hwmod_mutex); | 1989 | mutex_unlock(&oh->_mutex); |
| 1675 | 1990 | ||
| 1676 | return 0; | 1991 | return 0; |
| 1677 | } | 1992 | } |
| 1678 | 1993 | ||
| 1679 | /** | 1994 | /** |
| 1995 | * omap_hwmod_assert_hardreset - assert the HW reset line of submodules | ||
| 1996 | * contained in the hwmod module. | ||
| 1997 | * @oh: struct omap_hwmod * | ||
| 1998 | * @name: name of the reset line to lookup and assert | ||
| 1999 | * | ||
| 2000 | * Some IP like dsp, ipu or iva contain processor that require | ||
| 2001 | * an HW reset line to be assert / deassert in order to enable fully | ||
| 2002 | * the IP. Returns -EINVAL if @oh is null or if the operation is not | ||
| 2003 | * yet supported on this OMAP; otherwise, passes along the return value | ||
| 2004 | * from _assert_hardreset(). | ||
| 2005 | */ | ||
| 2006 | int omap_hwmod_assert_hardreset(struct omap_hwmod *oh, const char *name) | ||
| 2007 | { | ||
| 2008 | int ret; | ||
| 2009 | |||
| 2010 | if (!oh) | ||
| 2011 | return -EINVAL; | ||
| 2012 | |||
| 2013 | mutex_lock(&oh->_mutex); | ||
| 2014 | ret = _assert_hardreset(oh, name); | ||
| 2015 | mutex_unlock(&oh->_mutex); | ||
| 2016 | |||
| 2017 | return ret; | ||
| 2018 | } | ||
| 2019 | |||
| 2020 | /** | ||
| 2021 | * omap_hwmod_deassert_hardreset - deassert the HW reset line of submodules | ||
| 2022 | * contained in the hwmod module. | ||
| 2023 | * @oh: struct omap_hwmod * | ||
| 2024 | * @name: name of the reset line to look up and deassert | ||
| 2025 | * | ||
| 2026 | * Some IP like dsp, ipu or iva contain processor that require | ||
| 2027 | * an HW reset line to be assert / deassert in order to enable fully | ||
| 2028 | * the IP. Returns -EINVAL if @oh is null or if the operation is not | ||
| 2029 | * yet supported on this OMAP; otherwise, passes along the return value | ||
| 2030 | * from _deassert_hardreset(). | ||
| 2031 | */ | ||
| 2032 | int omap_hwmod_deassert_hardreset(struct omap_hwmod *oh, const char *name) | ||
| 2033 | { | ||
| 2034 | int ret; | ||
| 2035 | |||
| 2036 | if (!oh) | ||
| 2037 | return -EINVAL; | ||
| 2038 | |||
| 2039 | mutex_lock(&oh->_mutex); | ||
| 2040 | ret = _deassert_hardreset(oh, name); | ||
| 2041 | mutex_unlock(&oh->_mutex); | ||
| 2042 | |||
| 2043 | return ret; | ||
| 2044 | } | ||
| 2045 | |||
| 2046 | /** | ||
| 2047 | * omap_hwmod_read_hardreset - read the HW reset line state of submodules | ||
| 2048 | * contained in the hwmod module | ||
| 2049 | * @oh: struct omap_hwmod * | ||
| 2050 | * @name: name of the reset line to look up and read | ||
| 2051 | * | ||
| 2052 | * Return the current state of the hwmod @oh's reset line named @name: | ||
| 2053 | * returns -EINVAL upon parameter error or if this operation | ||
| 2054 | * is unsupported on the current OMAP; otherwise, passes along the return | ||
| 2055 | * value from _read_hardreset(). | ||
| 2056 | */ | ||
| 2057 | int omap_hwmod_read_hardreset(struct omap_hwmod *oh, const char *name) | ||
| 2058 | { | ||
| 2059 | int ret; | ||
| 2060 | |||
| 2061 | if (!oh) | ||
| 2062 | return -EINVAL; | ||
| 2063 | |||
| 2064 | mutex_lock(&oh->_mutex); | ||
| 2065 | ret = _read_hardreset(oh, name); | ||
| 2066 | mutex_unlock(&oh->_mutex); | ||
| 2067 | |||
| 2068 | return ret; | ||
| 2069 | } | ||
| 2070 | |||
| 2071 | |||
| 2072 | /** | ||
| 1680 | * omap_hwmod_for_each_by_class - call @fn for each hwmod of class @classname | 2073 | * omap_hwmod_for_each_by_class - call @fn for each hwmod of class @classname |
| 1681 | * @classname: struct omap_hwmod_class name to search for | 2074 | * @classname: struct omap_hwmod_class name to search for |
| 1682 | * @fn: callback function pointer to call for each hwmod in class @classname | 2075 | * @fn: callback function pointer to call for each hwmod in class @classname |
