Merge branch 'misc' into devel
[linux/fpc-iii.git] / arch / arm / mach-omap2 / prm44xx.c
bloba2a04bfa962855820c4f1a1576e9546ae87ef8c6
1 /*
2 * OMAP4 PRM module functions
4 * Copyright (C) 2010 Texas Instruments, Inc.
5 * Copyright (C) 2010 Nokia Corporation
6 * BenoƮt Cousson
7 * Paul Walmsley
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License version 2 as
11 * published by the Free Software Foundation.
14 #include <linux/kernel.h>
15 #include <linux/delay.h>
16 #include <linux/errno.h>
17 #include <linux/err.h>
18 #include <linux/io.h>
20 #include <plat/common.h>
21 #include <plat/cpu.h>
22 #include <plat/prcm.h>
24 #include "prm44xx.h"
25 #include "prm-regbits-44xx.h"
28 * Address offset (in bytes) between the reset control and the reset
29 * status registers: 4 bytes on OMAP4
31 #define OMAP4_RST_CTRL_ST_OFFSET 4
33 /* PRM low-level functions */
35 /* Read a register in a CM/PRM instance in the PRM module */
36 u32 omap4_prm_read_inst_reg(s16 inst, u16 reg)
38 return __raw_readl(OMAP44XX_PRM_REGADDR(inst, reg));
41 /* Write into a register in a CM/PRM instance in the PRM module */
42 void omap4_prm_write_inst_reg(u32 val, s16 inst, u16 reg)
44 __raw_writel(val, OMAP44XX_PRM_REGADDR(inst, reg));
47 /* Read-modify-write a register in a PRM module. Caller must lock */
48 u32 omap4_prm_rmw_inst_reg_bits(u32 mask, u32 bits, s16 inst, s16 reg)
50 u32 v;
52 v = omap4_prm_read_inst_reg(inst, reg);
53 v &= ~mask;
54 v |= bits;
55 omap4_prm_write_inst_reg(v, inst, reg);
57 return v;
60 /* Read a PRM register, AND it, and shift the result down to bit 0 */
61 /* XXX deprecated */
62 u32 omap4_prm_read_bits_shift(void __iomem *reg, u32 mask)
64 u32 v;
66 v = __raw_readl(reg);
67 v &= mask;
68 v >>= __ffs(mask);
70 return v;
73 /* Read-modify-write a register in a PRM module. Caller must lock */
74 /* XXX deprecated */
75 u32 omap4_prm_rmw_reg_bits(u32 mask, u32 bits, void __iomem *reg)
77 u32 v;
79 v = __raw_readl(reg);
80 v &= ~mask;
81 v |= bits;
82 __raw_writel(v, reg);
84 return v;
87 u32 omap4_prm_set_inst_reg_bits(u32 bits, s16 inst, s16 reg)
89 return omap4_prm_rmw_inst_reg_bits(bits, bits, inst, reg);
92 u32 omap4_prm_clear_inst_reg_bits(u32 bits, s16 inst, s16 reg)
94 return omap4_prm_rmw_inst_reg_bits(bits, 0x0, inst, reg);
97 /**
98 * omap4_prm_is_hardreset_asserted - read the HW reset line state of
99 * submodules contained in the hwmod module
100 * @rstctrl_reg: RM_RSTCTRL register address for this module
101 * @shift: register bit shift corresponding to the reset line to check
103 * Returns 1 if the (sub)module hardreset line is currently asserted,
104 * 0 if the (sub)module hardreset line is not currently asserted, or
105 * -EINVAL upon parameter error.
107 int omap4_prm_is_hardreset_asserted(void __iomem *rstctrl_reg, u8 shift)
109 if (!cpu_is_omap44xx() || !rstctrl_reg)
110 return -EINVAL;
112 return omap4_prm_read_bits_shift(rstctrl_reg, (1 << shift));
116 * omap4_prm_assert_hardreset - assert the HW reset line of a submodule
117 * @rstctrl_reg: RM_RSTCTRL register address for this module
118 * @shift: register bit shift corresponding to the reset line to assert
120 * Some IPs like dsp, ipu or iva contain processors that require an HW
121 * reset line to be asserted / deasserted in order to fully enable the
122 * IP. These modules may have multiple hard-reset lines that reset
123 * different 'submodules' inside the IP block. This function will
124 * place the submodule into reset. Returns 0 upon success or -EINVAL
125 * upon an argument error.
127 int omap4_prm_assert_hardreset(void __iomem *rstctrl_reg, u8 shift)
129 u32 mask;
131 if (!cpu_is_omap44xx() || !rstctrl_reg)
132 return -EINVAL;
134 mask = 1 << shift;
135 omap4_prm_rmw_reg_bits(mask, mask, rstctrl_reg);
137 return 0;
141 * omap4_prm_deassert_hardreset - deassert a submodule hardreset line and wait
142 * @rstctrl_reg: RM_RSTCTRL register address for this module
143 * @shift: register bit shift corresponding to the reset line to deassert
145 * Some IPs like dsp, ipu or iva contain processors that require an HW
146 * reset line to be asserted / deasserted in order to fully enable the
147 * IP. These modules may have multiple hard-reset lines that reset
148 * different 'submodules' inside the IP block. This function will
149 * take the submodule out of reset and wait until the PRCM indicates
150 * that the reset has completed before returning. Returns 0 upon success or
151 * -EINVAL upon an argument error, -EEXIST if the submodule was already out
152 * of reset, or -EBUSY if the submodule did not exit reset promptly.
154 int omap4_prm_deassert_hardreset(void __iomem *rstctrl_reg, u8 shift)
156 u32 mask;
157 void __iomem *rstst_reg;
158 int c;
160 if (!cpu_is_omap44xx() || !rstctrl_reg)
161 return -EINVAL;
163 rstst_reg = rstctrl_reg + OMAP4_RST_CTRL_ST_OFFSET;
165 mask = 1 << shift;
167 /* Check the current status to avoid de-asserting the line twice */
168 if (omap4_prm_read_bits_shift(rstctrl_reg, mask) == 0)
169 return -EEXIST;
171 /* Clear the reset status by writing 1 to the status bit */
172 omap4_prm_rmw_reg_bits(0xffffffff, mask, rstst_reg);
173 /* de-assert the reset control line */
174 omap4_prm_rmw_reg_bits(mask, 0, rstctrl_reg);
175 /* wait the status to be set */
176 omap_test_timeout(omap4_prm_read_bits_shift(rstst_reg, mask),
177 MAX_MODULE_HARDRESET_WAIT, c);
179 return (c == MAX_MODULE_HARDRESET_WAIT) ? -EBUSY : 0;
182 void omap4_prm_global_warm_sw_reset(void)
184 u32 v;
186 v = omap4_prm_read_inst_reg(OMAP4430_PRM_DEVICE_INST,
187 OMAP4_RM_RSTCTRL);
188 v |= OMAP4430_RST_GLOBAL_WARM_SW_MASK;
189 omap4_prm_write_inst_reg(v, OMAP4430_PRM_DEVICE_INST,
190 OMAP4_RM_RSTCTRL);
192 /* OCP barrier */
193 v = omap4_prm_read_inst_reg(OMAP4430_PRM_DEVICE_INST,
194 OMAP4_RM_RSTCTRL);