Linux Kernel  3.7.1
 All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Groups Pages
pmu.c
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2011 Broadcom Corporation
3  *
4  * Permission to use, copy, modify, and/or distribute this software for any
5  * purpose with or without fee is hereby granted, provided that the above
6  * copyright notice and this permission notice appear in all copies.
7  *
8  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
11  * SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
13  * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
14  * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15  */
16 
17 #include <linux/delay.h>
18 #include <linux/io.h>
19 
20 #include <brcm_hw_ids.h>
21 #include <chipcommon.h>
22 #include <brcmu_utils.h>
23 #include "pub.h"
24 #include "aiutils.h"
25 #include "pmu.h"
26 #include "soc.h"
27 
28 /*
29  * external LPO crystal frequency
30  */
31 #define EXT_ILP_HZ 32768
32 
33 /*
34  * Duration for ILP clock frequency measurment in milliseconds
35  *
36  * remark: 1000 must be an integer multiple of this duration
37  */
38 #define ILP_CALC_DUR 10
39 
40 /* Fields in pmucontrol */
41 #define PCTL_ILP_DIV_MASK 0xffff0000
42 #define PCTL_ILP_DIV_SHIFT 16
43 #define PCTL_PLL_PLLCTL_UPD 0x00000400 /* rev 2 */
44 #define PCTL_NOILP_ON_WAIT 0x00000200 /* rev 1 */
45 #define PCTL_HT_REQ_EN 0x00000100
46 #define PCTL_ALP_REQ_EN 0x00000080
47 #define PCTL_XTALFREQ_MASK 0x0000007c
48 #define PCTL_XTALFREQ_SHIFT 2
49 #define PCTL_ILP_DIV_EN 0x00000002
50 #define PCTL_LPO_SEL 0x00000001
51 
52 /* ILP clock */
53 #define ILP_CLOCK 32000
54 
55 /* ALP clock on pre-PMU chips */
56 #define ALP_CLOCK 20000000
57 
58 /* pmustatus */
59 #define PST_EXTLPOAVAIL 0x0100
60 #define PST_WDRESET 0x0080
61 #define PST_INTPEND 0x0040
62 #define PST_SBCLKST 0x0030
63 #define PST_SBCLKST_ILP 0x0010
64 #define PST_SBCLKST_ALP 0x0020
65 #define PST_SBCLKST_HT 0x0030
66 #define PST_ALPAVAIL 0x0008
67 #define PST_HTAVAIL 0x0004
68 #define PST_RESINIT 0x0003
69 
70 /* PMU resource bit position */
71 #define PMURES_BIT(bit) (1 << (bit))
72 
73 /* PMU corerev and chip specific PLL controls.
74  * PMU<rev>_PLL<num>_XX where <rev> is PMU corerev and <num> is an arbitrary
75  * number to differentiate different PLLs controlled by the same PMU rev.
76  */
77 
78 /* pmu XtalFreqRatio */
79 #define PMU_XTALFREQ_REG_ILPCTR_MASK 0x00001FFF
80 #define PMU_XTALFREQ_REG_MEASURE_MASK 0x80000000
81 #define PMU_XTALFREQ_REG_MEASURE_SHIFT 31
82 
83 /* 4313 resources */
84 #define RES4313_BB_PU_RSRC 0
85 #define RES4313_ILP_REQ_RSRC 1
86 #define RES4313_XTAL_PU_RSRC 2
87 #define RES4313_ALP_AVAIL_RSRC 3
88 #define RES4313_RADIO_PU_RSRC 4
89 #define RES4313_BG_PU_RSRC 5
90 #define RES4313_VREG1P4_PU_RSRC 6
91 #define RES4313_AFE_PWRSW_RSRC 7
92 #define RES4313_RX_PWRSW_RSRC 8
93 #define RES4313_TX_PWRSW_RSRC 9
94 #define RES4313_BB_PWRSW_RSRC 10
95 #define RES4313_SYNTH_PWRSW_RSRC 11
96 #define RES4313_MISC_PWRSW_RSRC 12
97 #define RES4313_BB_PLL_PWRSW_RSRC 13
98 #define RES4313_HT_AVAIL_RSRC 14
99 #define RES4313_MACPHY_CLK_AVAIL_RSRC 15
100 
102 {
104 
105  switch (ai_get_chip_id(sih)) {
109  delay = 3700;
110  break;
111  default:
112  break;
113  }
114 
115  return (u16) delay;
116 }
117 
118 /* Read/write a chipcontrol reg */
120 {
121  ai_cc_reg(sih, offsetof(struct chipcregs, chipcontrol_addr), ~0, reg);
122  return ai_cc_reg(sih, offsetof(struct chipcregs, chipcontrol_data),
123  mask, val);
124 }
125 
126 /* Read/write a regcontrol reg */
128 {
129  ai_cc_reg(sih, offsetof(struct chipcregs, regcontrol_addr), ~0, reg);
130  return ai_cc_reg(sih, offsetof(struct chipcregs, regcontrol_data),
131  mask, val);
132 }
133 
134 /* Read/write a pllcontrol reg */
136 {
137  ai_cc_reg(sih, offsetof(struct chipcregs, pllcontrol_addr), ~0, reg);
138  return ai_cc_reg(sih, offsetof(struct chipcregs, pllcontrol_data),
139  mask, val);
140 }
141 
142 /* PMU PLL update */
143 void si_pmu_pllupd(struct si_pub *sih)
144 {
145  ai_cc_reg(sih, offsetof(struct chipcregs, pmucontrol),
147 }
148 
149 /* query alp/xtal clock frequency */
151 {
152  u32 clock = ALP_CLOCK;
153 
154  /* bail out with default */
155  if (!(ai_get_cccaps(sih) & CC_CAP_PMU))
156  return clock;
157 
158  switch (ai_get_chip_id(sih)) {
162  /* always 20Mhz */
163  clock = 20000 * 1000;
164  break;
165  default:
166  break;
167  }
168 
169  return clock;
170 }
171 
173 {
174  struct si_info *sii = container_of(sih, struct si_info, pub);
175  struct bcma_device *core;
176  u32 alp_khz;
177 
178  if (ai_get_pmurev(sih) < 10)
179  return 0;
180 
181  /* Remember original core before switch to chipc */
182  core = sii->icbus->drv_cc.core;
183 
184  if (bcma_read32(core, CHIPCREGOFFS(pmustatus)) & PST_EXTLPOAVAIL) {
185  u32 ilp_ctr, alp_hz;
186 
187  /*
188  * Enable the reg to measure the freq,
189  * in case it was disabled before
190  */
191  bcma_write32(core, CHIPCREGOFFS(pmu_xtalfreq),
193 
194  /* Delay for well over 4 ILP clocks */
195  udelay(1000);
196 
197  /* Read the latched number of ALP ticks per 4 ILP ticks */
198  ilp_ctr = bcma_read32(core, CHIPCREGOFFS(pmu_xtalfreq)) &
200 
201  /*
202  * Turn off the PMU_XTALFREQ_REG_MEASURE_SHIFT
203  * bit to save power
204  */
205  bcma_write32(core, CHIPCREGOFFS(pmu_xtalfreq), 0);
206 
207  /* Calculate ALP frequency */
208  alp_hz = (ilp_ctr * EXT_ILP_HZ) / 4;
209 
210  /*
211  * Round to nearest 100KHz, and at
212  * the same time convert to KHz
213  */
214  alp_khz = (alp_hz + 50000) / 100000 * 100;
215  } else
216  alp_khz = 0;
217 
218  return alp_khz;
219 }