blob: 6e7a8150ce767aed94f29b12e7ff5d6c31cd1a9c [file] [log] [blame]
Sagar Dharia33beca02012-10-22 16:21:46 -06001/* Copyright (c) 2011-2013, The Linux Foundation. All rights reserved.
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12#include <linux/irq.h>
13#include <linux/kernel.h>
14#include <linux/init.h>
15#include <linux/slab.h>
16#include <linux/io.h>
17#include <linux/interrupt.h>
18#include <linux/platform_device.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070019#include <linux/slimbus/slimbus.h>
20#include <linux/delay.h>
21#include <linux/kthread.h>
22#include <linux/clk.h>
Sagar Dharia45ee38a2011-08-03 17:01:31 -060023#include <linux/pm_runtime.h>
Sagar Dhariaf8f603b2012-03-21 15:25:17 -060024#include <linux/of.h>
25#include <linux/of_slimbus.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070026#include <mach/sps.h>
Joonwoo Parkf69f77a2012-08-28 15:26:11 -070027#include <mach/qdsp6v2/apr.h>
Sagar Dharia2754ab42012-08-21 18:07:39 -060028#include "slim-msm.h"
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070029
30#define MSM_SLIM_NAME "msm_slim_ctrl"
31#define SLIM_ROOT_FREQ 24576000
32
Sagar Dharia45ee38a2011-08-03 17:01:31 -060033#define QC_MSM_DEVS 5
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070034
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070035/* Manager registers */
36enum mgr_reg {
37 MGR_CFG = 0x200,
38 MGR_STATUS = 0x204,
39 MGR_RX_MSGQ_CFG = 0x208,
40 MGR_INT_EN = 0x210,
41 MGR_INT_STAT = 0x214,
42 MGR_INT_CLR = 0x218,
43 MGR_TX_MSG = 0x230,
44 MGR_RX_MSG = 0x270,
Sagar Dhariaac913452012-09-04 11:27:26 -060045 MGR_IE_STAT = 0x2F0,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070046 MGR_VE_STAT = 0x300,
47};
48
49enum msg_cfg {
50 MGR_CFG_ENABLE = 1,
51 MGR_CFG_RX_MSGQ_EN = 1 << 1,
52 MGR_CFG_TX_MSGQ_EN_HIGH = 1 << 2,
53 MGR_CFG_TX_MSGQ_EN_LOW = 1 << 3,
54};
55/* Message queue types */
56enum msm_slim_msgq_type {
57 MSGQ_RX = 0,
58 MSGQ_TX_LOW = 1,
59 MSGQ_TX_HIGH = 2,
60};
61/* Framer registers */
62enum frm_reg {
63 FRM_CFG = 0x400,
64 FRM_STAT = 0x404,
65 FRM_INT_EN = 0x410,
66 FRM_INT_STAT = 0x414,
67 FRM_INT_CLR = 0x418,
68 FRM_WAKEUP = 0x41C,
69 FRM_CLKCTL_DONE = 0x420,
70 FRM_IE_STAT = 0x430,
71 FRM_VE_STAT = 0x440,
72};
73
74/* Interface registers */
75enum intf_reg {
76 INTF_CFG = 0x600,
77 INTF_STAT = 0x604,
78 INTF_INT_EN = 0x610,
79 INTF_INT_STAT = 0x614,
80 INTF_INT_CLR = 0x618,
81 INTF_IE_STAT = 0x630,
82 INTF_VE_STAT = 0x640,
83};
84
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070085enum mgr_intr {
86 MGR_INT_RECFG_DONE = 1 << 24,
87 MGR_INT_TX_NACKED_2 = 1 << 25,
88 MGR_INT_MSG_BUF_CONTE = 1 << 26,
89 MGR_INT_RX_MSG_RCVD = 1 << 30,
90 MGR_INT_TX_MSG_SENT = 1 << 31,
91};
92
93enum frm_cfg {
94 FRM_ACTIVE = 1,
95 CLK_GEAR = 7,
96 ROOT_FREQ = 11,
97 REF_CLK_GEAR = 15,
Sagar Dhariadebc8b72012-08-11 15:02:12 -060098 INTR_WAKE = 19,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070099};
100
Sagar Dharia790cfd02011-09-25 17:56:24 -0600101static struct msm_slim_sat *msm_slim_alloc_sat(struct msm_slim_ctrl *dev);
102
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700103static int msm_sat_enqueue(struct msm_slim_sat *sat, u32 *buf, u8 len)
104{
105 struct msm_slim_ctrl *dev = sat->dev;
Stephen Boyd3cdfc712012-10-31 17:41:43 -0700106 unsigned long flags;
107 spin_lock_irqsave(&sat->lock, flags);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700108 if ((sat->stail + 1) % SAT_CONCUR_MSG == sat->shead) {
Stephen Boyd3cdfc712012-10-31 17:41:43 -0700109 spin_unlock_irqrestore(&sat->lock, flags);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700110 dev_err(dev->dev, "SAT QUEUE full!");
111 return -EXFULL;
112 }
113 memcpy(sat->sat_msgs[sat->stail], (u8 *)buf, len);
114 sat->stail = (sat->stail + 1) % SAT_CONCUR_MSG;
Stephen Boyd3cdfc712012-10-31 17:41:43 -0700115 spin_unlock_irqrestore(&sat->lock, flags);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700116 return 0;
117}
118
119static int msm_sat_dequeue(struct msm_slim_sat *sat, u8 *buf)
120{
121 unsigned long flags;
122 spin_lock_irqsave(&sat->lock, flags);
123 if (sat->stail == sat->shead) {
124 spin_unlock_irqrestore(&sat->lock, flags);
125 return -ENODATA;
126 }
127 memcpy(buf, sat->sat_msgs[sat->shead], 40);
128 sat->shead = (sat->shead + 1) % SAT_CONCUR_MSG;
129 spin_unlock_irqrestore(&sat->lock, flags);
130 return 0;
131}
132
133static void msm_get_eaddr(u8 *e_addr, u32 *buffer)
134{
135 e_addr[0] = (buffer[1] >> 24) & 0xff;
136 e_addr[1] = (buffer[1] >> 16) & 0xff;
137 e_addr[2] = (buffer[1] >> 8) & 0xff;
138 e_addr[3] = buffer[1] & 0xff;
139 e_addr[4] = (buffer[0] >> 24) & 0xff;
140 e_addr[5] = (buffer[0] >> 16) & 0xff;
141}
142
143static bool msm_is_sat_dev(u8 *e_addr)
144{
145 if (e_addr[5] == QC_MFGID_LSB && e_addr[4] == QC_MFGID_MSB &&
146 e_addr[2] != QC_CHIPID_SL &&
147 (e_addr[1] == QC_DEVID_SAT1 || e_addr[1] == QC_DEVID_SAT2))
148 return true;
149 return false;
150}
151
Sagar Dharia790cfd02011-09-25 17:56:24 -0600152static struct msm_slim_sat *addr_to_sat(struct msm_slim_ctrl *dev, u8 laddr)
153{
154 struct msm_slim_sat *sat = NULL;
155 int i = 0;
156 while (!sat && i < dev->nsats) {
157 if (laddr == dev->satd[i]->satcl.laddr)
158 sat = dev->satd[i];
159 i++;
160 }
161 return sat;
162}
163
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700164static irqreturn_t msm_slim_interrupt(int irq, void *d)
165{
166 struct msm_slim_ctrl *dev = d;
167 u32 pstat;
168 u32 stat = readl_relaxed(dev->base + MGR_INT_STAT);
169
170 if (stat & MGR_INT_TX_MSG_SENT || stat & MGR_INT_TX_NACKED_2) {
171 if (stat & MGR_INT_TX_MSG_SENT)
172 writel_relaxed(MGR_INT_TX_MSG_SENT,
173 dev->base + MGR_INT_CLR);
174 else {
Sagar Dhariaac913452012-09-04 11:27:26 -0600175 u32 mgr_stat = readl_relaxed(dev->base + MGR_STATUS);
176 u32 mgr_ie_stat = readl_relaxed(dev->base +
177 MGR_IE_STAT);
178 u32 frm_stat = readl_relaxed(dev->base + FRM_STAT);
179 u32 frm_cfg = readl_relaxed(dev->base + FRM_CFG);
180 u32 frm_intr_stat = readl_relaxed(dev->base +
181 FRM_INT_STAT);
182 u32 frm_ie_stat = readl_relaxed(dev->base +
183 FRM_IE_STAT);
184 u32 intf_stat = readl_relaxed(dev->base + INTF_STAT);
185 u32 intf_intr_stat = readl_relaxed(dev->base +
186 INTF_INT_STAT);
187 u32 intf_ie_stat = readl_relaxed(dev->base +
188 INTF_IE_STAT);
189
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700190 writel_relaxed(MGR_INT_TX_NACKED_2,
191 dev->base + MGR_INT_CLR);
Sagar Dhariaac913452012-09-04 11:27:26 -0600192 pr_err("TX Nack MGR dump:int_stat:0x%x, mgr_stat:0x%x",
193 stat, mgr_stat);
194 pr_err("TX Nack MGR dump:ie_stat:0x%x", mgr_ie_stat);
195 pr_err("TX Nack FRM dump:int_stat:0x%x, frm_stat:0x%x",
196 frm_intr_stat, frm_stat);
197 pr_err("TX Nack FRM dump:frm_cfg:0x%x, ie_stat:0x%x",
198 frm_cfg, frm_ie_stat);
199 pr_err("TX Nack INTF dump:intr_st:0x%x, intf_stat:0x%x",
200 intf_intr_stat, intf_stat);
201 pr_err("TX Nack INTF dump:ie_stat:0x%x", intf_ie_stat);
202
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700203 dev->err = -EIO;
204 }
205 /*
206 * Guarantee that interrupt clear bit write goes through before
207 * signalling completion/exiting ISR
208 */
209 mb();
210 if (dev->wr_comp)
211 complete(dev->wr_comp);
212 }
213 if (stat & MGR_INT_RX_MSG_RCVD) {
214 u32 rx_buf[10];
215 u32 mc, mt;
216 u8 len, i;
217 rx_buf[0] = readl_relaxed(dev->base + MGR_RX_MSG);
218 len = rx_buf[0] & 0x1F;
219 for (i = 1; i < ((len + 3) >> 2); i++) {
220 rx_buf[i] = readl_relaxed(dev->base + MGR_RX_MSG +
221 (4 * i));
222 dev_dbg(dev->dev, "reading data: %x\n", rx_buf[i]);
223 }
224 mt = (rx_buf[0] >> 5) & 0x7;
225 mc = (rx_buf[0] >> 8) & 0xff;
226 dev_dbg(dev->dev, "MC: %x, MT: %x\n", mc, mt);
227 if (mt == SLIM_MSG_MT_DEST_REFERRED_USER ||
228 mt == SLIM_MSG_MT_SRC_REFERRED_USER) {
Sagar Dharia790cfd02011-09-25 17:56:24 -0600229 u8 laddr = (u8)((rx_buf[0] >> 16) & 0xFF);
230 struct msm_slim_sat *sat = addr_to_sat(dev, laddr);
231 if (sat)
232 msm_sat_enqueue(sat, rx_buf, len);
233 else
234 dev_err(dev->dev, "unknown sat:%d message",
235 laddr);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700236 writel_relaxed(MGR_INT_RX_MSG_RCVD,
237 dev->base + MGR_INT_CLR);
238 /*
239 * Guarantee that CLR bit write goes through before
240 * queuing work
241 */
242 mb();
Sagar Dharia790cfd02011-09-25 17:56:24 -0600243 if (sat)
244 queue_work(sat->wq, &sat->wd);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700245 } else if (mt == SLIM_MSG_MT_CORE &&
246 mc == SLIM_MSG_MC_REPORT_PRESENT) {
247 u8 e_addr[6];
248 msm_get_eaddr(e_addr, rx_buf);
Sagar Dharia790cfd02011-09-25 17:56:24 -0600249 msm_slim_rx_enqueue(dev, rx_buf, len);
250 writel_relaxed(MGR_INT_RX_MSG_RCVD, dev->base +
251 MGR_INT_CLR);
252 /*
253 * Guarantee that CLR bit write goes through
254 * before signalling completion
255 */
256 mb();
257 complete(&dev->rx_msgq_notify);
Kiran Gunda33a23a92013-01-10 17:03:33 +0530258 } else if (mt == SLIM_MSG_MT_CORE &&
259 mc == SLIM_MSG_MC_REPORT_ABSENT) {
260 writel_relaxed(MGR_INT_RX_MSG_RCVD, dev->base +
261 MGR_INT_CLR);
262 /*
263 * Guarantee that CLR bit write goes through
264 * before signalling completion
265 */
266 mb();
267 complete(&dev->rx_msgq_notify);
268
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700269 } else if (mc == SLIM_MSG_MC_REPLY_INFORMATION ||
270 mc == SLIM_MSG_MC_REPLY_VALUE) {
271 msm_slim_rx_enqueue(dev, rx_buf, len);
272 writel_relaxed(MGR_INT_RX_MSG_RCVD, dev->base +
273 MGR_INT_CLR);
274 /*
275 * Guarantee that CLR bit write goes through
276 * before signalling completion
277 */
278 mb();
279 complete(&dev->rx_msgq_notify);
Sagar Dharia144e5e02011-08-08 17:30:11 -0600280 } else if (mc == SLIM_MSG_MC_REPORT_INFORMATION) {
281 u8 *buf = (u8 *)rx_buf;
282 u8 l_addr = buf[2];
283 u16 ele = (u16)buf[4] << 4;
284 ele |= ((buf[3] & 0xf0) >> 4);
285 dev_err(dev->dev, "Slim-dev:%d report inf element:0x%x",
286 l_addr, ele);
287 for (i = 0; i < len - 5; i++)
288 dev_err(dev->dev, "offset:0x%x:bit mask:%x",
289 i, buf[i+5]);
290 writel_relaxed(MGR_INT_RX_MSG_RCVD, dev->base +
291 MGR_INT_CLR);
292 /*
293 * Guarantee that CLR bit write goes through
294 * before exiting
295 */
296 mb();
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700297 } else {
298 dev_err(dev->dev, "Unexpected MC,%x MT:%x, len:%d",
299 mc, mt, len);
300 for (i = 0; i < ((len + 3) >> 2); i++)
301 dev_err(dev->dev, "error msg: %x", rx_buf[i]);
302 writel_relaxed(MGR_INT_RX_MSG_RCVD, dev->base +
303 MGR_INT_CLR);
304 /*
305 * Guarantee that CLR bit write goes through
306 * before exiting
307 */
308 mb();
309 }
310 }
311 if (stat & MGR_INT_RECFG_DONE) {
312 writel_relaxed(MGR_INT_RECFG_DONE, dev->base + MGR_INT_CLR);
313 /*
314 * Guarantee that CLR bit write goes through
315 * before exiting ISR
316 */
317 mb();
318 complete(&dev->reconf);
319 }
Sagar Dharia82e516f2012-03-16 16:01:23 -0600320 pstat = readl_relaxed(PGD_THIS_EE(PGD_PORT_INT_ST_EEn, dev->ver));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700321 if (pstat != 0) {
Sagar Dharia2b8a4b52013-05-15 20:01:45 -0600322 return msm_slim_port_irq_handler(dev, pstat);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700323 }
324
325 return IRQ_HANDLED;
326}
327
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700328static int msm_xfer_msg(struct slim_controller *ctrl, struct slim_msg_txn *txn)
329{
330 DECLARE_COMPLETION_ONSTACK(done);
331 struct msm_slim_ctrl *dev = slim_get_ctrldata(ctrl);
332 u32 *pbuf;
333 u8 *puc;
334 int timeout;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700335 int msgv = -1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700336 u8 la = txn->la;
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600337 u8 mc = (u8)(txn->mc & 0xFF);
338 /*
339 * Voting for runtime PM: Slimbus has 2 possible use cases:
340 * 1. messaging
341 * 2. Data channels
342 * Messaging case goes through messaging slots and data channels
343 * use their own slots
344 * This "get" votes for messaging bandwidth
345 */
346 if (!(txn->mc & SLIM_MSG_CLK_PAUSE_SEQ_FLG))
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700347 msgv = msm_slim_get_ctrl(dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700348 mutex_lock(&dev->tx_lock);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700349 if (dev->state == MSM_CTRL_ASLEEP ||
350 ((!(txn->mc & SLIM_MSG_CLK_PAUSE_SEQ_FLG)) &&
351 dev->state == MSM_CTRL_SLEEPING)) {
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600352 dev_err(dev->dev, "runtime or system PM suspended state");
353 mutex_unlock(&dev->tx_lock);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700354 if (msgv >= 0)
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600355 msm_slim_put_ctrl(dev);
356 return -EBUSY;
357 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700358 if (txn->mt == SLIM_MSG_MT_CORE &&
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600359 mc == SLIM_MSG_MC_BEGIN_RECONFIGURATION) {
360 if (dev->reconf_busy) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700361 wait_for_completion(&dev->reconf);
362 dev->reconf_busy = false;
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600363 }
364 /* This "get" votes for data channels */
365 if (dev->ctrl.sched.usedslots != 0 &&
366 !dev->chan_active) {
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700367 int chv = msm_slim_get_ctrl(dev);
368 if (chv >= 0)
369 dev->chan_active = true;
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600370 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700371 }
372 txn->rl--;
Sagar Dharia2754ab42012-08-21 18:07:39 -0600373 pbuf = msm_get_msg_buf(dev, txn->rl);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700374 dev->wr_comp = NULL;
375 dev->err = 0;
376
377 if (txn->dt == SLIM_MSG_DEST_ENUMADDR) {
378 mutex_unlock(&dev->tx_lock);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700379 if (msgv >= 0)
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600380 msm_slim_put_ctrl(dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700381 return -EPROTONOSUPPORT;
382 }
383 if (txn->mt == SLIM_MSG_MT_CORE && txn->la == 0xFF &&
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600384 (mc == SLIM_MSG_MC_CONNECT_SOURCE ||
385 mc == SLIM_MSG_MC_CONNECT_SINK ||
386 mc == SLIM_MSG_MC_DISCONNECT_PORT))
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700387 la = dev->pgdla;
388 if (txn->dt == SLIM_MSG_DEST_LOGICALADDR)
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600389 *pbuf = SLIM_MSG_ASM_FIRST_WORD(txn->rl, txn->mt, mc, 0, la);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700390 else
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600391 *pbuf = SLIM_MSG_ASM_FIRST_WORD(txn->rl, txn->mt, mc, 1, la);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700392 if (txn->dt == SLIM_MSG_DEST_LOGICALADDR)
393 puc = ((u8 *)pbuf) + 3;
394 else
395 puc = ((u8 *)pbuf) + 2;
396 if (txn->rbuf)
397 *(puc++) = txn->tid;
398 if ((txn->mt == SLIM_MSG_MT_CORE) &&
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600399 ((mc >= SLIM_MSG_MC_REQUEST_INFORMATION &&
400 mc <= SLIM_MSG_MC_REPORT_INFORMATION) ||
401 (mc >= SLIM_MSG_MC_REQUEST_VALUE &&
402 mc <= SLIM_MSG_MC_CHANGE_VALUE))) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700403 *(puc++) = (txn->ec & 0xFF);
404 *(puc++) = (txn->ec >> 8)&0xFF;
405 }
406 if (txn->wbuf)
407 memcpy(puc, txn->wbuf, txn->len);
408 if (txn->mt == SLIM_MSG_MT_CORE && txn->la == 0xFF &&
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600409 (mc == SLIM_MSG_MC_CONNECT_SOURCE ||
410 mc == SLIM_MSG_MC_CONNECT_SINK ||
411 mc == SLIM_MSG_MC_DISCONNECT_PORT)) {
412 if (mc != SLIM_MSG_MC_DISCONNECT_PORT)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700413 dev->err = msm_slim_connect_pipe_port(dev, *puc);
414 else {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700415 /*
416 * Remove channel disconnects master-side ports from
417 * channel. No need to send that again on the bus
Sagar Dharia2d74c142013-05-17 02:24:06 -0600418 * Only disable port
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700419 */
Sagar Dharia2d74c142013-05-17 02:24:06 -0600420 writel_relaxed(0, PGD_PORT(PGD_PORT_CFGn,
421 (*puc + dev->port_b), dev->ver));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700422 mutex_unlock(&dev->tx_lock);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700423 if (msgv >= 0)
424 msm_slim_put_ctrl(dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700425 return 0;
426 }
427 if (dev->err) {
428 dev_err(dev->dev, "pipe-port connect err:%d", dev->err);
429 mutex_unlock(&dev->tx_lock);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700430 if (msgv >= 0)
431 msm_slim_put_ctrl(dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700432 return dev->err;
433 }
Sagar Dharia2b8a4b52013-05-15 20:01:45 -0600434 *(puc) = *(puc) + dev->port_b;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700435 }
436 if (txn->mt == SLIM_MSG_MT_CORE &&
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600437 mc == SLIM_MSG_MC_BEGIN_RECONFIGURATION)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700438 dev->reconf_busy = true;
439 dev->wr_comp = &done;
Sagar Dharia2754ab42012-08-21 18:07:39 -0600440 msm_send_msg_buf(dev, pbuf, txn->rl, MGR_TX_MSG);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700441 timeout = wait_for_completion_timeout(&done, HZ);
Sagar Dhariaac913452012-09-04 11:27:26 -0600442 if (!timeout)
443 dev->wr_comp = NULL;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700444 if (mc == SLIM_MSG_MC_RECONFIGURE_NOW) {
445 if ((txn->mc == (SLIM_MSG_MC_RECONFIGURE_NOW |
446 SLIM_MSG_CLK_PAUSE_SEQ_FLG)) &&
447 timeout) {
448 timeout = wait_for_completion_timeout(&dev->reconf, HZ);
449 dev->reconf_busy = false;
450 if (timeout) {
Sagar Dharia9acf7f42012-03-08 09:45:30 -0700451 clk_disable_unprepare(dev->rclk);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700452 disable_irq(dev->irq);
453 }
454 }
455 if ((txn->mc == (SLIM_MSG_MC_RECONFIGURE_NOW |
456 SLIM_MSG_CLK_PAUSE_SEQ_FLG)) &&
457 !timeout) {
458 dev->reconf_busy = false;
459 dev_err(dev->dev, "clock pause failed");
460 mutex_unlock(&dev->tx_lock);
461 return -ETIMEDOUT;
462 }
463 if (txn->mt == SLIM_MSG_MT_CORE &&
464 txn->mc == SLIM_MSG_MC_RECONFIGURE_NOW) {
465 if (dev->ctrl.sched.usedslots == 0 &&
466 dev->chan_active) {
467 dev->chan_active = false;
468 msm_slim_put_ctrl(dev);
469 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600470 }
471 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600472 mutex_unlock(&dev->tx_lock);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700473 if (msgv >= 0)
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600474 msm_slim_put_ctrl(dev);
475
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700476 if (!timeout)
477 dev_err(dev->dev, "TX timed out:MC:0x%x,mt:0x%x", txn->mc,
478 txn->mt);
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600479
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700480 return timeout ? dev->err : -ETIMEDOUT;
481}
482
Sagar Dhariaac913452012-09-04 11:27:26 -0600483static void msm_slim_wait_retry(struct msm_slim_ctrl *dev)
484{
485 int msec_per_frm = 0;
486 int sfr_per_sec;
487 /* Wait for 1 superframe, or default time and then retry */
488 sfr_per_sec = dev->framer.superfreq /
489 (1 << (SLIM_MAX_CLK_GEAR - dev->ctrl.clkgear));
490 if (sfr_per_sec)
491 msec_per_frm = MSEC_PER_SEC / sfr_per_sec;
492 if (msec_per_frm < DEF_RETRY_MS)
493 msec_per_frm = DEF_RETRY_MS;
494 msleep(msec_per_frm);
495}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700496static int msm_set_laddr(struct slim_controller *ctrl, const u8 *ea,
497 u8 elen, u8 laddr)
498{
499 struct msm_slim_ctrl *dev = slim_get_ctrldata(ctrl);
Sagar Dhariaac913452012-09-04 11:27:26 -0600500 struct completion done;
501 int timeout, ret, retries = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700502 u32 *buf;
Sagar Dhariaac913452012-09-04 11:27:26 -0600503retry_laddr:
504 init_completion(&done);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700505 mutex_lock(&dev->tx_lock);
Sagar Dharia2754ab42012-08-21 18:07:39 -0600506 buf = msm_get_msg_buf(dev, 9);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700507 buf[0] = SLIM_MSG_ASM_FIRST_WORD(9, SLIM_MSG_MT_CORE,
508 SLIM_MSG_MC_ASSIGN_LOGICAL_ADDRESS,
509 SLIM_MSG_DEST_LOGICALADDR,
510 ea[5] | ea[4] << 8);
511 buf[1] = ea[3] | (ea[2] << 8) | (ea[1] << 16) | (ea[0] << 24);
512 buf[2] = laddr;
513
514 dev->wr_comp = &done;
Sagar Dharia2754ab42012-08-21 18:07:39 -0600515 ret = msm_send_msg_buf(dev, buf, 9, MGR_TX_MSG);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700516 timeout = wait_for_completion_timeout(&done, HZ);
Sagar Dhariaac913452012-09-04 11:27:26 -0600517 if (!timeout)
518 dev->err = -ETIMEDOUT;
519 if (dev->err) {
520 ret = dev->err;
521 dev->err = 0;
522 dev->wr_comp = NULL;
523 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700524 mutex_unlock(&dev->tx_lock);
Sagar Dhariaac913452012-09-04 11:27:26 -0600525 if (ret) {
526 pr_err("set LADDR:0x%x failed:ret:%d, retrying", laddr, ret);
527 if (retries < INIT_MX_RETRIES) {
528 msm_slim_wait_retry(dev);
529 retries++;
530 goto retry_laddr;
531 } else {
532 pr_err("set LADDR failed after retrying:ret:%d", ret);
533 }
534 }
535 return ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700536}
537
Sagar Dharia144e5e02011-08-08 17:30:11 -0600538static int msm_clk_pause_wakeup(struct slim_controller *ctrl)
539{
540 struct msm_slim_ctrl *dev = slim_get_ctrldata(ctrl);
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600541 enable_irq(dev->irq);
Sagar Dharia9acf7f42012-03-08 09:45:30 -0700542 clk_prepare_enable(dev->rclk);
Sagar Dharia144e5e02011-08-08 17:30:11 -0600543 writel_relaxed(1, dev->base + FRM_WAKEUP);
544 /* Make sure framer wakeup write goes through before exiting function */
545 mb();
546 /*
547 * Workaround: Currently, slave is reporting lost-sync messages
548 * after slimbus comes out of clock pause.
549 * Transaction with slave fail before slave reports that message
550 * Give some time for that report to come
551 * Slimbus wakes up in clock gear 10 at 24.576MHz. With each superframe
552 * being 250 usecs, we wait for 20 superframes here to ensure
553 * we get the message
554 */
555 usleep_range(5000, 5000);
556 return 0;
557}
558
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700559static int msm_sat_define_ch(struct msm_slim_sat *sat, u8 *buf, u8 len, u8 mc)
560{
561 struct msm_slim_ctrl *dev = sat->dev;
562 enum slim_ch_control oper;
563 int i;
564 int ret = 0;
565 if (mc == SLIM_USR_MC_CHAN_CTRL) {
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600566 for (i = 0; i < sat->nsatch; i++) {
567 if (buf[5] == sat->satch[i].chan)
568 break;
569 }
570 if (i >= sat->nsatch)
571 return -ENOTCONN;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700572 oper = ((buf[3] & 0xC0) >> 6);
573 /* part of grp. activating/removing 1 will take care of rest */
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600574 ret = slim_control_ch(&sat->satcl, sat->satch[i].chanh, oper,
575 false);
576 if (!ret) {
577 for (i = 5; i < len; i++) {
578 int j;
579 for (j = 0; j < sat->nsatch; j++) {
580 if (buf[i] == sat->satch[j].chan) {
581 if (oper == SLIM_CH_REMOVE)
582 sat->satch[j].req_rem++;
583 else
584 sat->satch[j].req_def++;
585 break;
586 }
587 }
588 }
589 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700590 } else {
591 u16 chh[40];
592 struct slim_ch prop;
593 u32 exp;
Sagar Dhariab886e042012-10-17 22:41:57 -0600594 u16 *grph = NULL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700595 u8 coeff, cc;
596 u8 prrate = buf[6];
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600597 if (len <= 8)
598 return -EINVAL;
599 for (i = 8; i < len; i++) {
600 int j = 0;
601 for (j = 0; j < sat->nsatch; j++) {
602 if (sat->satch[j].chan == buf[i]) {
603 chh[i - 8] = sat->satch[j].chanh;
604 break;
605 }
606 }
607 if (j < sat->nsatch) {
608 u16 dummy;
609 ret = slim_query_ch(&sat->satcl, buf[i],
610 &dummy);
611 if (ret)
612 return ret;
613 if (mc == SLIM_USR_MC_DEF_ACT_CHAN)
614 sat->satch[j].req_def++;
Sagar Dhariab886e042012-10-17 22:41:57 -0600615 /* First channel in group from satellite */
616 if (i == 8)
617 grph = &sat->satch[j].chanh;
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600618 continue;
619 }
620 if (sat->nsatch >= MSM_MAX_SATCH)
621 return -EXFULL;
622 ret = slim_query_ch(&sat->satcl, buf[i], &chh[i - 8]);
623 if (ret)
624 return ret;
625 sat->satch[j].chan = buf[i];
626 sat->satch[j].chanh = chh[i - 8];
627 if (mc == SLIM_USR_MC_DEF_ACT_CHAN)
628 sat->satch[j].req_def++;
Sagar Dhariab886e042012-10-17 22:41:57 -0600629 if (i == 8)
630 grph = &sat->satch[j].chanh;
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600631 sat->nsatch++;
632 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700633 prop.dataf = (enum slim_ch_dataf)((buf[3] & 0xE0) >> 5);
634 prop.auxf = (enum slim_ch_auxf)((buf[4] & 0xC0) >> 5);
635 prop.baser = SLIM_RATE_4000HZ;
636 if (prrate & 0x8)
637 prop.baser = SLIM_RATE_11025HZ;
638 else
639 prop.baser = SLIM_RATE_4000HZ;
640 prop.prot = (enum slim_ch_proto)(buf[5] & 0x0F);
641 prop.sampleszbits = (buf[4] & 0x1F)*SLIM_CL_PER_SL;
642 exp = (u32)((buf[5] & 0xF0) >> 4);
643 coeff = (buf[4] & 0x20) >> 5;
644 cc = (coeff ? 3 : 1);
645 prop.ratem = cc * (1 << exp);
646 if (i > 9)
647 ret = slim_define_ch(&sat->satcl, &prop, chh, len - 8,
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600648 true, &chh[0]);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700649 else
650 ret = slim_define_ch(&sat->satcl, &prop,
Sagar Dhariab886e042012-10-17 22:41:57 -0600651 chh, 1, true, &chh[0]);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700652 dev_dbg(dev->dev, "define sat grp returned:%d", ret);
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600653 if (ret)
654 return ret;
Sagar Dhariab886e042012-10-17 22:41:57 -0600655 else if (grph)
656 *grph = chh[0];
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700657
658 /* part of group so activating 1 will take care of rest */
659 if (mc == SLIM_USR_MC_DEF_ACT_CHAN)
660 ret = slim_control_ch(&sat->satcl,
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600661 chh[0],
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700662 SLIM_CH_ACTIVATE, false);
663 }
664 return ret;
665}
666
667static void msm_slim_rxwq(struct msm_slim_ctrl *dev)
668{
669 u8 buf[40];
670 u8 mc, mt, len;
671 int i, ret;
672 if ((msm_slim_rx_dequeue(dev, (u8 *)buf)) != -ENODATA) {
673 len = buf[0] & 0x1F;
674 mt = (buf[0] >> 5) & 0x7;
675 mc = buf[1];
676 if (mt == SLIM_MSG_MT_CORE &&
677 mc == SLIM_MSG_MC_REPORT_PRESENT) {
678 u8 laddr;
679 u8 e_addr[6];
680 for (i = 0; i < 6; i++)
681 e_addr[i] = buf[7-i];
682
Sagar Dhariaf0b9c752012-09-09 17:32:46 -0600683 ret = slim_assign_laddr(&dev->ctrl, e_addr, 6, &laddr,
684 false);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700685 /* Is this Qualcomm ported generic device? */
686 if (!ret && e_addr[5] == QC_MFGID_LSB &&
687 e_addr[4] == QC_MFGID_MSB &&
688 e_addr[1] == QC_DEVID_PGD &&
689 e_addr[2] != QC_CHIPID_SL)
690 dev->pgdla = laddr;
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600691 if (!ret && !pm_runtime_enabled(dev->dev) &&
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700692 laddr == (QC_MSM_DEVS - 1))
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600693 pm_runtime_enable(dev->dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700694
Sagar Dharia790cfd02011-09-25 17:56:24 -0600695 if (!ret && msm_is_sat_dev(e_addr)) {
696 struct msm_slim_sat *sat = addr_to_sat(dev,
697 laddr);
698 if (!sat)
699 sat = msm_slim_alloc_sat(dev);
700 if (!sat)
701 return;
702
703 sat->satcl.laddr = laddr;
704 msm_sat_enqueue(sat, (u32 *)buf, len);
705 queue_work(sat->wq, &sat->wd);
706 }
Sagar Dhariaac913452012-09-04 11:27:26 -0600707 if (ret)
708 pr_err("assign laddr failed, error:%d", ret);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700709 } else if (mc == SLIM_MSG_MC_REPLY_INFORMATION ||
710 mc == SLIM_MSG_MC_REPLY_VALUE) {
711 u8 tid = buf[3];
712 dev_dbg(dev->dev, "tid:%d, len:%d\n", tid, len - 4);
713 slim_msg_response(&dev->ctrl, &buf[4], tid,
714 len - 4);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700715 pm_runtime_mark_last_busy(dev->dev);
Sagar Dharia144e5e02011-08-08 17:30:11 -0600716 } else if (mc == SLIM_MSG_MC_REPORT_INFORMATION) {
717 u8 l_addr = buf[2];
718 u16 ele = (u16)buf[4] << 4;
719 ele |= ((buf[3] & 0xf0) >> 4);
720 dev_err(dev->dev, "Slim-dev:%d report inf element:0x%x",
721 l_addr, ele);
722 for (i = 0; i < len - 5; i++)
723 dev_err(dev->dev, "offset:0x%x:bit mask:%x",
724 i, buf[i+5]);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700725 } else {
726 dev_err(dev->dev, "unexpected message:mc:%x, mt:%x",
727 mc, mt);
728 for (i = 0; i < len; i++)
729 dev_err(dev->dev, "error msg: %x", buf[i]);
730
731 }
732 } else
733 dev_err(dev->dev, "rxwq called and no dequeue");
734}
735
736static void slim_sat_rxprocess(struct work_struct *work)
737{
738 struct msm_slim_sat *sat = container_of(work, struct msm_slim_sat, wd);
739 struct msm_slim_ctrl *dev = sat->dev;
740 u8 buf[40];
741
742 while ((msm_sat_dequeue(sat, buf)) != -ENODATA) {
743 struct slim_msg_txn txn;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700744 u8 len, mc, mt;
745 u32 bw_sl;
746 int ret = 0;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700747 int satv = -1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700748 bool gen_ack = false;
749 u8 tid;
750 u8 wbuf[8];
Sagar Dhariaac913452012-09-04 11:27:26 -0600751 int i, retries = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700752 txn.mt = SLIM_MSG_MT_SRC_REFERRED_USER;
753 txn.dt = SLIM_MSG_DEST_LOGICALADDR;
754 txn.ec = 0;
755 txn.rbuf = NULL;
756 txn.la = sat->satcl.laddr;
757 /* satellite handling */
758 len = buf[0] & 0x1F;
759 mc = buf[1];
760 mt = (buf[0] >> 5) & 0x7;
761
762 if (mt == SLIM_MSG_MT_CORE &&
763 mc == SLIM_MSG_MC_REPORT_PRESENT) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700764 u8 e_addr[6];
765 for (i = 0; i < 6; i++)
766 e_addr[i] = buf[7-i];
767
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600768 if (pm_runtime_enabled(dev->dev)) {
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700769 satv = msm_slim_get_ctrl(dev);
770 if (satv >= 0)
771 sat->pending_capability = true;
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600772 }
Sagar Dharia69bf5572012-02-21 14:45:35 -0700773 /*
774 * Since capability message is already sent, present
775 * message will indicate subsystem hosting this
776 * satellite has restarted.
777 * Remove all active channels of this satellite
778 * when this is detected
779 */
780 if (sat->sent_capability) {
781 for (i = 0; i < sat->nsatch; i++) {
Sagar Dharia1944a132012-08-15 00:01:57 -0600782 if (sat->satch[i].reconf) {
783 pr_err("SSR, sat:%d, rm ch:%d",
Sagar Dhariaac913452012-09-04 11:27:26 -0600784 sat->satcl.laddr,
Sagar Dharia69bf5572012-02-21 14:45:35 -0700785 sat->satch[i].chan);
Sagar Dharia69bf5572012-02-21 14:45:35 -0700786 slim_control_ch(&sat->satcl,
787 sat->satch[i].chanh,
788 SLIM_CH_REMOVE, true);
Sagar Dhariab886e042012-10-17 22:41:57 -0600789 slim_dealloc_ch(&sat->satcl,
790 sat->satch[i].chanh);
Sagar Dharia1944a132012-08-15 00:01:57 -0600791 sat->satch[i].reconf = false;
792 }
Sagar Dharia69bf5572012-02-21 14:45:35 -0700793 }
794 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600795 } else if (mt != SLIM_MSG_MT_CORE &&
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700796 mc != SLIM_MSG_MC_REPORT_PRESENT) {
797 satv = msm_slim_get_ctrl(dev);
798 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700799 switch (mc) {
800 case SLIM_MSG_MC_REPORT_PRESENT:
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600801 /* Remove runtime_pm vote once satellite acks */
802 if (mt != SLIM_MSG_MT_CORE) {
803 if (pm_runtime_enabled(dev->dev) &&
804 sat->pending_capability) {
805 msm_slim_put_ctrl(dev);
806 sat->pending_capability = false;
807 }
808 continue;
809 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700810 /* send a Manager capability msg */
Sagar Dharia790cfd02011-09-25 17:56:24 -0600811 if (sat->sent_capability) {
812 if (mt == SLIM_MSG_MT_CORE)
813 goto send_capability;
814 else
815 continue;
816 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700817 ret = slim_add_device(&dev->ctrl, &sat->satcl);
818 if (ret) {
819 dev_err(dev->dev,
820 "Satellite-init failed");
821 continue;
822 }
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600823 /* Satellite-channels */
824 sat->satch = kzalloc(MSM_MAX_SATCH *
825 sizeof(struct msm_sat_chan),
826 GFP_KERNEL);
Sagar Dharia790cfd02011-09-25 17:56:24 -0600827send_capability:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700828 txn.mc = SLIM_USR_MC_MASTER_CAPABILITY;
829 txn.mt = SLIM_MSG_MT_SRC_REFERRED_USER;
830 txn.la = sat->satcl.laddr;
831 txn.rl = 8;
832 wbuf[0] = SAT_MAGIC_LSB;
833 wbuf[1] = SAT_MAGIC_MSB;
834 wbuf[2] = SAT_MSG_VER;
835 wbuf[3] = SAT_MSG_PROT;
836 txn.wbuf = wbuf;
837 txn.len = 4;
Sagar Dhariaac913452012-09-04 11:27:26 -0600838 ret = msm_xfer_msg(&dev->ctrl, &txn);
839 if (ret) {
840 pr_err("capability for:0x%x fail:%d, retry:%d",
841 sat->satcl.laddr, ret, retries);
842 if (retries < INIT_MX_RETRIES) {
843 msm_slim_wait_retry(dev);
844 retries++;
845 goto send_capability;
846 } else {
847 pr_err("failed after all retries:%d",
848 ret);
849 }
850 } else {
851 sat->sent_capability = true;
852 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700853 break;
854 case SLIM_USR_MC_ADDR_QUERY:
855 memcpy(&wbuf[1], &buf[4], 6);
856 ret = slim_get_logical_addr(&sat->satcl,
857 &wbuf[1], 6, &wbuf[7]);
858 if (ret)
859 memset(&wbuf[1], 0, 6);
860 wbuf[0] = buf[3];
861 txn.mc = SLIM_USR_MC_ADDR_REPLY;
862 txn.rl = 12;
863 txn.len = 8;
864 txn.wbuf = wbuf;
865 msm_xfer_msg(&dev->ctrl, &txn);
866 break;
867 case SLIM_USR_MC_DEFINE_CHAN:
868 case SLIM_USR_MC_DEF_ACT_CHAN:
869 case SLIM_USR_MC_CHAN_CTRL:
870 if (mc != SLIM_USR_MC_CHAN_CTRL)
871 tid = buf[7];
872 else
873 tid = buf[4];
874 gen_ack = true;
875 ret = msm_sat_define_ch(sat, buf, len, mc);
876 if (ret) {
877 dev_err(dev->dev,
878 "SAT define_ch returned:%d",
879 ret);
880 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600881 if (!sat->pending_reconf) {
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700882 int chv = msm_slim_get_ctrl(dev);
883 if (chv >= 0)
884 sat->pending_reconf = true;
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600885 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700886 break;
887 case SLIM_USR_MC_RECONFIG_NOW:
888 tid = buf[3];
889 gen_ack = true;
890 ret = slim_reconfigure_now(&sat->satcl);
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600891 for (i = 0; i < sat->nsatch; i++) {
892 struct msm_sat_chan *sch = &sat->satch[i];
Sagar Dhariaef444892012-09-05 12:19:24 -0600893 if (sch->req_rem && sch->reconf) {
Sagar Dharia1944a132012-08-15 00:01:57 -0600894 if (!ret) {
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600895 slim_dealloc_ch(&sat->satcl,
896 sch->chanh);
Sagar Dharia1944a132012-08-15 00:01:57 -0600897 sch->reconf = false;
898 }
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600899 sch->req_rem--;
900 } else if (sch->req_def) {
901 if (ret)
902 slim_dealloc_ch(&sat->satcl,
903 sch->chanh);
Sagar Dharia1944a132012-08-15 00:01:57 -0600904 else
905 sch->reconf = true;
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600906 sch->req_def--;
907 }
908 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600909 if (sat->pending_reconf) {
910 msm_slim_put_ctrl(dev);
911 sat->pending_reconf = false;
912 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700913 break;
914 case SLIM_USR_MC_REQ_BW:
915 /* what we get is in SLOTS */
916 bw_sl = (u32)buf[4] << 3 |
917 ((buf[3] & 0xE0) >> 5);
918 sat->satcl.pending_msgsl = bw_sl;
919 tid = buf[5];
920 gen_ack = true;
921 break;
922 case SLIM_USR_MC_CONNECT_SRC:
923 case SLIM_USR_MC_CONNECT_SINK:
924 if (mc == SLIM_USR_MC_CONNECT_SRC)
925 txn.mc = SLIM_MSG_MC_CONNECT_SOURCE;
926 else
927 txn.mc = SLIM_MSG_MC_CONNECT_SINK;
928 wbuf[0] = buf[4] & 0x1F;
929 wbuf[1] = buf[5];
930 tid = buf[6];
931 txn.la = buf[3];
932 txn.mt = SLIM_MSG_MT_CORE;
933 txn.rl = 6;
934 txn.len = 2;
935 txn.wbuf = wbuf;
936 gen_ack = true;
937 ret = msm_xfer_msg(&dev->ctrl, &txn);
938 break;
939 case SLIM_USR_MC_DISCONNECT_PORT:
940 txn.mc = SLIM_MSG_MC_DISCONNECT_PORT;
941 wbuf[0] = buf[4] & 0x1F;
942 tid = buf[5];
943 txn.la = buf[3];
944 txn.rl = 5;
945 txn.len = 1;
946 txn.mt = SLIM_MSG_MT_CORE;
947 txn.wbuf = wbuf;
948 gen_ack = true;
949 ret = msm_xfer_msg(&dev->ctrl, &txn);
Kiran Gunda33a23a92013-01-10 17:03:33 +0530950 break;
951 case SLIM_MSG_MC_REPORT_ABSENT:
952 dev_info(dev->dev, "Received Report Absent Message\n");
953 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700954 default:
955 break;
956 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600957 if (!gen_ack) {
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700958 if (mc != SLIM_MSG_MC_REPORT_PRESENT && satv >= 0)
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600959 msm_slim_put_ctrl(dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700960 continue;
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600961 }
962
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700963 wbuf[0] = tid;
964 if (!ret)
965 wbuf[1] = MSM_SAT_SUCCSS;
966 else
967 wbuf[1] = 0;
968 txn.mc = SLIM_USR_MC_GENERIC_ACK;
969 txn.la = sat->satcl.laddr;
970 txn.rl = 6;
971 txn.len = 2;
972 txn.wbuf = wbuf;
973 txn.mt = SLIM_MSG_MT_SRC_REFERRED_USER;
974 msm_xfer_msg(&dev->ctrl, &txn);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700975 if (satv >= 0)
976 msm_slim_put_ctrl(dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700977 }
978}
979
Sagar Dharia790cfd02011-09-25 17:56:24 -0600980static struct msm_slim_sat *msm_slim_alloc_sat(struct msm_slim_ctrl *dev)
981{
982 struct msm_slim_sat *sat;
983 char *name;
984 if (dev->nsats >= MSM_MAX_NSATS)
985 return NULL;
986
987 sat = kzalloc(sizeof(struct msm_slim_sat), GFP_KERNEL);
988 if (!sat) {
989 dev_err(dev->dev, "no memory for satellite");
990 return NULL;
991 }
992 name = kzalloc(SLIMBUS_NAME_SIZE, GFP_KERNEL);
993 if (!name) {
994 dev_err(dev->dev, "no memory for satellite name");
995 kfree(sat);
996 return NULL;
997 }
998 dev->satd[dev->nsats] = sat;
999 sat->dev = dev;
1000 snprintf(name, SLIMBUS_NAME_SIZE, "msm_sat%d", dev->nsats);
1001 sat->satcl.name = name;
1002 spin_lock_init(&sat->lock);
1003 INIT_WORK(&sat->wd, slim_sat_rxprocess);
1004 sat->wq = create_singlethread_workqueue(sat->satcl.name);
1005 if (!sat->wq) {
1006 kfree(name);
1007 kfree(sat);
1008 return NULL;
1009 }
1010 /*
1011 * Both sats will be allocated from RX thread and RX thread will
1012 * process messages sequentially. No synchronization necessary
1013 */
1014 dev->nsats++;
1015 return sat;
1016}
1017
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001018static int msm_slim_rx_msgq_thread(void *data)
1019{
1020 struct msm_slim_ctrl *dev = (struct msm_slim_ctrl *)data;
1021 struct completion *notify = &dev->rx_msgq_notify;
1022 struct msm_slim_sat *sat = NULL;
1023 u32 mc = 0;
1024 u32 mt = 0;
1025 u32 buffer[10];
1026 int index = 0;
1027 u8 msg_len = 0;
1028 int ret;
1029
1030 dev_dbg(dev->dev, "rx thread started");
1031
1032 while (!kthread_should_stop()) {
1033 set_current_state(TASK_INTERRUPTIBLE);
1034 ret = wait_for_completion_interruptible(notify);
1035
1036 if (ret)
1037 dev_err(dev->dev, "rx thread wait error:%d", ret);
1038
1039 /* 1 irq notification per message */
Sagar Dharia24419e32013-01-14 17:56:32 -07001040 if (dev->use_rx_msgqs != MSM_MSGQ_ENABLED) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001041 msm_slim_rxwq(dev);
1042 continue;
1043 }
1044
1045 ret = msm_slim_rx_msgq_get(dev, buffer, index);
1046 if (ret) {
1047 dev_err(dev->dev, "rx_msgq_get() failed 0x%x\n", ret);
1048 continue;
1049 }
1050
1051 pr_debug("message[%d] = 0x%x\n", index, *buffer);
1052
1053 /* Decide if we use generic RX or satellite RX */
1054 if (index++ == 0) {
1055 msg_len = *buffer & 0x1F;
1056 pr_debug("Start of new message, len = %d\n", msg_len);
1057 mt = (buffer[0] >> 5) & 0x7;
1058 mc = (buffer[0] >> 8) & 0xff;
1059 dev_dbg(dev->dev, "MC: %x, MT: %x\n", mc, mt);
1060 if (mt == SLIM_MSG_MT_DEST_REFERRED_USER ||
Sagar Dharia790cfd02011-09-25 17:56:24 -06001061 mt == SLIM_MSG_MT_SRC_REFERRED_USER) {
1062 u8 laddr;
1063 laddr = (u8)((buffer[0] >> 16) & 0xff);
1064 sat = addr_to_sat(dev, laddr);
1065 }
Kiran Gunda33a23a92013-01-10 17:03:33 +05301066 }
1067 if ((index * 4) >= msg_len) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001068 index = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001069 if (sat) {
1070 msm_sat_enqueue(sat, buffer, msg_len);
1071 queue_work(sat->wq, &sat->wd);
1072 sat = NULL;
1073 } else {
1074 msm_slim_rx_enqueue(dev, buffer, msg_len);
1075 msm_slim_rxwq(dev);
1076 }
1077 }
1078 }
1079
1080 return 0;
1081}
1082
Sagar Dhariacc969452011-09-19 10:34:30 -06001083static void msm_slim_prg_slew(struct platform_device *pdev,
1084 struct msm_slim_ctrl *dev)
1085{
1086 struct resource *slew_io;
1087 void __iomem *slew_reg;
1088 /* SLEW RATE register for this slimbus */
1089 dev->slew_mem = platform_get_resource_byname(pdev, IORESOURCE_MEM,
1090 "slimbus_slew_reg");
1091 if (!dev->slew_mem) {
1092 dev_dbg(&pdev->dev, "no slimbus slew resource\n");
1093 return;
1094 }
1095 slew_io = request_mem_region(dev->slew_mem->start,
1096 resource_size(dev->slew_mem), pdev->name);
1097 if (!slew_io) {
1098 dev_dbg(&pdev->dev, "slimbus-slew mem claimed\n");
1099 dev->slew_mem = NULL;
1100 return;
1101 }
1102
1103 slew_reg = ioremap(dev->slew_mem->start, resource_size(dev->slew_mem));
1104 if (!slew_reg) {
1105 dev_dbg(dev->dev, "slew register mapping failed");
1106 release_mem_region(dev->slew_mem->start,
1107 resource_size(dev->slew_mem));
1108 dev->slew_mem = NULL;
1109 return;
1110 }
1111 writel_relaxed(1, slew_reg);
1112 /* Make sure slimbus-slew rate enabling goes through */
1113 wmb();
1114 iounmap(slew_reg);
1115}
1116
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001117static int __devinit msm_slim_probe(struct platform_device *pdev)
1118{
1119 struct msm_slim_ctrl *dev;
1120 int ret;
Joonwoo Parkf69f77a2012-08-28 15:26:11 -07001121 enum apr_subsys_state q6_state;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001122 struct resource *bam_mem, *bam_io;
1123 struct resource *slim_mem, *slim_io;
1124 struct resource *irq, *bam_irq;
Sagar Dharia1beb2202012-07-31 19:06:21 -06001125 bool rxreg_access = false;
Joonwoo Parkf69f77a2012-08-28 15:26:11 -07001126
1127 q6_state = apr_get_q6_state();
1128 if (q6_state == APR_SUBSYS_DOWN) {
1129 dev_dbg(&pdev->dev, "defering %s, adsp_state %d\n", __func__,
1130 q6_state);
1131 return -EPROBE_DEFER;
1132 } else
1133 dev_dbg(&pdev->dev, "adsp is ready\n");
1134
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001135 slim_mem = platform_get_resource_byname(pdev, IORESOURCE_MEM,
1136 "slimbus_physical");
1137 if (!slim_mem) {
1138 dev_err(&pdev->dev, "no slimbus physical memory resource\n");
1139 return -ENODEV;
1140 }
1141 slim_io = request_mem_region(slim_mem->start, resource_size(slim_mem),
1142 pdev->name);
1143 if (!slim_io) {
1144 dev_err(&pdev->dev, "slimbus memory already claimed\n");
1145 return -EBUSY;
1146 }
1147
1148 bam_mem = platform_get_resource_byname(pdev, IORESOURCE_MEM,
1149 "slimbus_bam_physical");
1150 if (!bam_mem) {
1151 dev_err(&pdev->dev, "no slimbus BAM memory resource\n");
1152 ret = -ENODEV;
1153 goto err_get_res_bam_failed;
1154 }
1155 bam_io = request_mem_region(bam_mem->start, resource_size(bam_mem),
1156 pdev->name);
1157 if (!bam_io) {
1158 release_mem_region(slim_mem->start, resource_size(slim_mem));
1159 dev_err(&pdev->dev, "slimbus BAM memory already claimed\n");
1160 ret = -EBUSY;
1161 goto err_get_res_bam_failed;
1162 }
1163 irq = platform_get_resource_byname(pdev, IORESOURCE_IRQ,
1164 "slimbus_irq");
1165 if (!irq) {
1166 dev_err(&pdev->dev, "no slimbus IRQ resource\n");
1167 ret = -ENODEV;
1168 goto err_get_res_failed;
1169 }
1170 bam_irq = platform_get_resource_byname(pdev, IORESOURCE_IRQ,
1171 "slimbus_bam_irq");
1172 if (!bam_irq) {
1173 dev_err(&pdev->dev, "no slimbus BAM IRQ resource\n");
1174 ret = -ENODEV;
1175 goto err_get_res_failed;
1176 }
1177
1178 dev = kzalloc(sizeof(struct msm_slim_ctrl), GFP_KERNEL);
1179 if (!dev) {
1180 dev_err(&pdev->dev, "no memory for MSM slimbus controller\n");
1181 ret = -ENOMEM;
1182 goto err_get_res_failed;
1183 }
1184 dev->dev = &pdev->dev;
1185 platform_set_drvdata(pdev, dev);
1186 slim_set_ctrldata(&dev->ctrl, dev);
1187 dev->base = ioremap(slim_mem->start, resource_size(slim_mem));
1188 if (!dev->base) {
1189 dev_err(&pdev->dev, "IOremap failed\n");
1190 ret = -ENOMEM;
1191 goto err_ioremap_failed;
1192 }
1193 dev->bam.base = ioremap(bam_mem->start, resource_size(bam_mem));
1194 if (!dev->bam.base) {
1195 dev_err(&pdev->dev, "BAM IOremap failed\n");
1196 ret = -ENOMEM;
1197 goto err_ioremap_bam_failed;
1198 }
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001199 if (pdev->dev.of_node) {
1200
1201 ret = of_property_read_u32(pdev->dev.of_node, "cell-index",
1202 &dev->ctrl.nr);
1203 if (ret) {
1204 dev_err(&pdev->dev, "Cell index not specified:%d", ret);
1205 goto err_of_init_failed;
1206 }
Sagar Dharia1beb2202012-07-31 19:06:21 -06001207 rxreg_access = of_property_read_bool(pdev->dev.of_node,
1208 "qcom,rxreg-access");
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001209 /* Optional properties */
1210 ret = of_property_read_u32(pdev->dev.of_node,
1211 "qcom,min-clk-gear", &dev->ctrl.min_cg);
1212 ret = of_property_read_u32(pdev->dev.of_node,
1213 "qcom,max-clk-gear", &dev->ctrl.max_cg);
Sagar Dharia1beb2202012-07-31 19:06:21 -06001214 pr_debug("min_cg:%d, max_cg:%d, rxreg: %d", dev->ctrl.min_cg,
1215 dev->ctrl.max_cg, rxreg_access);
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001216 } else {
1217 dev->ctrl.nr = pdev->id;
1218 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001219 dev->ctrl.nchans = MSM_SLIM_NCHANS;
1220 dev->ctrl.nports = MSM_SLIM_NPORTS;
1221 dev->ctrl.set_laddr = msm_set_laddr;
1222 dev->ctrl.xfer_msg = msm_xfer_msg;
Sagar Dharia144e5e02011-08-08 17:30:11 -06001223 dev->ctrl.wakeup = msm_clk_pause_wakeup;
Sagar Dharia100e7212013-05-17 18:20:57 -06001224 dev->ctrl.alloc_port = msm_alloc_port;
1225 dev->ctrl.dealloc_port = msm_dealloc_port;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001226 dev->ctrl.port_xfer = msm_slim_port_xfer;
1227 dev->ctrl.port_xfer_status = msm_slim_port_xfer_status;
1228 /* Reserve some messaging BW for satellite-apps driver communication */
1229 dev->ctrl.sched.pending_msgsl = 30;
1230
1231 init_completion(&dev->reconf);
1232 mutex_init(&dev->tx_lock);
1233 spin_lock_init(&dev->rx_lock);
1234 dev->ee = 1;
Sagar Dharia1beb2202012-07-31 19:06:21 -06001235 if (rxreg_access)
Sagar Dharia24419e32013-01-14 17:56:32 -07001236 dev->use_rx_msgqs = MSM_MSGQ_DISABLED;
Sagar Dharia1beb2202012-07-31 19:06:21 -06001237 else
Sagar Dharia24419e32013-01-14 17:56:32 -07001238 dev->use_rx_msgqs = MSM_MSGQ_RESET;
Sagar Dharia1beb2202012-07-31 19:06:21 -06001239
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001240 dev->irq = irq->start;
1241 dev->bam.irq = bam_irq->start;
1242
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001243 dev->hclk = clk_get(dev->dev, "iface_clk");
1244 if (IS_ERR(dev->hclk))
1245 dev->hclk = NULL;
1246 else
1247 clk_prepare_enable(dev->hclk);
1248
Sagar Dharia60f59a72012-10-17 12:42:03 -06001249 ret = msm_slim_sps_init(dev, bam_mem, MGR_STATUS, false);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001250 if (ret != 0) {
1251 dev_err(dev->dev, "error SPS init\n");
1252 goto err_sps_init_failed;
1253 }
1254
Sagar Dharia2754ab42012-08-21 18:07:39 -06001255 /* Fire up the Rx message queue thread */
1256 dev->rx_msgq_thread = kthread_run(msm_slim_rx_msgq_thread, dev,
1257 MSM_SLIM_NAME "_rx_msgq_thread");
1258 if (IS_ERR(dev->rx_msgq_thread)) {
1259 ret = PTR_ERR(dev->rx_msgq_thread);
1260 dev_err(dev->dev, "Failed to start Rx message queue thread\n");
1261 goto err_thread_create_failed;
1262 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001263
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001264 dev->framer.rootfreq = SLIM_ROOT_FREQ >> 3;
1265 dev->framer.superfreq =
1266 dev->framer.rootfreq / SLIM_CL_PER_SUPERFRAME_DIV8;
1267 dev->ctrl.a_framer = &dev->framer;
1268 dev->ctrl.clkgear = SLIM_MAX_CLK_GEAR;
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001269 dev->ctrl.dev.parent = &pdev->dev;
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001270 dev->ctrl.dev.of_node = pdev->dev.of_node;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001271
1272 ret = request_irq(dev->irq, msm_slim_interrupt, IRQF_TRIGGER_HIGH,
1273 "msm_slim_irq", dev);
1274 if (ret) {
1275 dev_err(&pdev->dev, "request IRQ failed\n");
1276 goto err_request_irq_failed;
1277 }
1278
Sagar Dhariacc969452011-09-19 10:34:30 -06001279 msm_slim_prg_slew(pdev, dev);
Sagar Dhariab1c0acf2012-02-06 18:16:58 -07001280
1281 /* Register with framework before enabling frame, clock */
1282 ret = slim_add_numbered_controller(&dev->ctrl);
1283 if (ret) {
1284 dev_err(dev->dev, "error adding controller\n");
1285 goto err_ctrl_failed;
1286 }
1287
1288
Tianyi Gou44a81b02012-02-06 17:49:07 -08001289 dev->rclk = clk_get(dev->dev, "core_clk");
Sagar Dhariab1c0acf2012-02-06 18:16:58 -07001290 if (!dev->rclk) {
1291 dev_err(dev->dev, "slimbus clock not found");
1292 goto err_clk_get_failed;
1293 }
Sagar Dhariacc969452011-09-19 10:34:30 -06001294 clk_set_rate(dev->rclk, SLIM_ROOT_FREQ);
Sagar Dharia9acf7f42012-03-08 09:45:30 -07001295 clk_prepare_enable(dev->rclk);
Sagar Dhariacc969452011-09-19 10:34:30 -06001296
Sagar Dharia82e516f2012-03-16 16:01:23 -06001297 dev->ver = readl_relaxed(dev->base);
1298 /* Version info in 16 MSbits */
1299 dev->ver >>= 16;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001300 /* Component register initialization */
Sagar Dharia82e516f2012-03-16 16:01:23 -06001301 writel_relaxed(1, dev->base + CFG_PORT(COMP_CFG, dev->ver));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001302 writel_relaxed((EE_MGR_RSC_GRP | EE_NGD_2 | EE_NGD_1),
Sagar Dharia82e516f2012-03-16 16:01:23 -06001303 dev->base + CFG_PORT(COMP_TRUST_CFG, dev->ver));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001304
1305 /*
1306 * Manager register initialization
1307 * If RX msg Q is used, disable RX_MSG_RCVD interrupt
1308 */
Sagar Dharia24419e32013-01-14 17:56:32 -07001309 if (dev->use_rx_msgqs == MSM_MSGQ_ENABLED)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001310 writel_relaxed((MGR_INT_RECFG_DONE | MGR_INT_TX_NACKED_2 |
1311 MGR_INT_MSG_BUF_CONTE | /* MGR_INT_RX_MSG_RCVD | */
1312 MGR_INT_TX_MSG_SENT), dev->base + MGR_INT_EN);
1313 else
1314 writel_relaxed((MGR_INT_RECFG_DONE | MGR_INT_TX_NACKED_2 |
1315 MGR_INT_MSG_BUF_CONTE | MGR_INT_RX_MSG_RCVD |
1316 MGR_INT_TX_MSG_SENT), dev->base + MGR_INT_EN);
1317 writel_relaxed(1, dev->base + MGR_CFG);
1318 /*
1319 * Framer registers are beyond 1K memory region after Manager and/or
1320 * component registers. Make sure those writes are ordered
1321 * before framer register writes
1322 */
1323 wmb();
1324
1325 /* Framer register initialization */
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001326 writel_relaxed((1 << INTR_WAKE) | (0xA << REF_CLK_GEAR) |
1327 (0xA << CLK_GEAR) | (1 << ROOT_FREQ) | (1 << FRM_ACTIVE) | 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001328 dev->base + FRM_CFG);
1329 /*
1330 * Make sure that framer wake-up and enabling writes go through
1331 * before any other component is enabled. Framer is responsible for
1332 * clocking the bus and enabling framer first will ensure that other
1333 * devices can report presence when they are enabled
1334 */
1335 mb();
1336
1337 /* Enable RX msg Q */
Sagar Dharia24419e32013-01-14 17:56:32 -07001338 if (dev->use_rx_msgqs == MSM_MSGQ_ENABLED)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001339 writel_relaxed(MGR_CFG_ENABLE | MGR_CFG_RX_MSGQ_EN,
1340 dev->base + MGR_CFG);
1341 else
1342 writel_relaxed(MGR_CFG_ENABLE, dev->base + MGR_CFG);
1343 /*
1344 * Make sure that manager-enable is written through before interface
1345 * device is enabled
1346 */
1347 mb();
1348 writel_relaxed(1, dev->base + INTF_CFG);
1349 /*
1350 * Make sure that interface-enable is written through before enabling
1351 * ported generic device inside MSM manager
1352 */
1353 mb();
Sagar Dharia82e516f2012-03-16 16:01:23 -06001354 writel_relaxed(1, dev->base + CFG_PORT(PGD_CFG, dev->ver));
1355 writel_relaxed(0x3F<<17, dev->base + CFG_PORT(PGD_OWN_EEn, dev->ver) +
1356 (4 * dev->ee));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001357 /*
1358 * Make sure that ported generic device is enabled and port-EE settings
1359 * are written through before finally enabling the component
1360 */
1361 mb();
1362
Sagar Dharia82e516f2012-03-16 16:01:23 -06001363 writel_relaxed(1, dev->base + CFG_PORT(COMP_CFG, dev->ver));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001364 /*
1365 * Make sure that all writes have gone through before exiting this
1366 * function
1367 */
1368 mb();
Sagar Dhariaa6627e02012-08-28 12:20:49 -06001369
1370 /* Add devices registered with board-info now that controller is up */
1371 slim_ctrl_add_boarddevs(&dev->ctrl);
1372
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001373 if (pdev->dev.of_node)
1374 of_register_slim_devices(&dev->ctrl);
1375
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001376 pm_runtime_use_autosuspend(&pdev->dev);
1377 pm_runtime_set_autosuspend_delay(&pdev->dev, MSM_SLIM_AUTOSUSPEND);
1378 pm_runtime_set_active(&pdev->dev);
1379
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001380 dev_dbg(dev->dev, "MSM SB controller is up!\n");
1381 return 0;
1382
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001383err_ctrl_failed:
Sagar Dharia82e516f2012-03-16 16:01:23 -06001384 writel_relaxed(0, dev->base + CFG_PORT(COMP_CFG, dev->ver));
Sagar Dhariab1c0acf2012-02-06 18:16:58 -07001385err_clk_get_failed:
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001386 kfree(dev->satd);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001387err_request_irq_failed:
Sagar Dharia2754ab42012-08-21 18:07:39 -06001388 kthread_stop(dev->rx_msgq_thread);
1389err_thread_create_failed:
Sagar Dharia33beca02012-10-22 16:21:46 -06001390 msm_slim_sps_exit(dev, true);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001391err_sps_init_failed:
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001392 if (dev->hclk) {
1393 clk_disable_unprepare(dev->hclk);
1394 clk_put(dev->hclk);
1395 }
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001396err_of_init_failed:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001397 iounmap(dev->bam.base);
1398err_ioremap_bam_failed:
1399 iounmap(dev->base);
1400err_ioremap_failed:
1401 kfree(dev);
1402err_get_res_failed:
1403 release_mem_region(bam_mem->start, resource_size(bam_mem));
1404err_get_res_bam_failed:
1405 release_mem_region(slim_mem->start, resource_size(slim_mem));
1406 return ret;
1407}
1408
1409static int __devexit msm_slim_remove(struct platform_device *pdev)
1410{
1411 struct msm_slim_ctrl *dev = platform_get_drvdata(pdev);
1412 struct resource *bam_mem;
1413 struct resource *slim_mem;
Sagar Dhariacc969452011-09-19 10:34:30 -06001414 struct resource *slew_mem = dev->slew_mem;
Sagar Dharia790cfd02011-09-25 17:56:24 -06001415 int i;
1416 for (i = 0; i < dev->nsats; i++) {
1417 struct msm_slim_sat *sat = dev->satd[i];
Sagar Dharia0ffdca12011-09-25 18:55:53 -06001418 int j;
1419 for (j = 0; j < sat->nsatch; j++)
1420 slim_dealloc_ch(&sat->satcl, sat->satch[j].chanh);
Sagar Dharia790cfd02011-09-25 17:56:24 -06001421 slim_remove_device(&sat->satcl);
1422 kfree(sat->satch);
1423 destroy_workqueue(sat->wq);
1424 kfree(sat->satcl.name);
1425 kfree(sat);
1426 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001427 pm_runtime_disable(&pdev->dev);
1428 pm_runtime_set_suspended(&pdev->dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001429 free_irq(dev->irq, dev);
1430 slim_del_controller(&dev->ctrl);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001431 clk_put(dev->rclk);
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001432 if (dev->hclk)
1433 clk_put(dev->hclk);
Sagar Dharia33beca02012-10-22 16:21:46 -06001434 msm_slim_sps_exit(dev, true);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001435 kthread_stop(dev->rx_msgq_thread);
1436 iounmap(dev->bam.base);
1437 iounmap(dev->base);
1438 kfree(dev);
1439 bam_mem = platform_get_resource_byname(pdev, IORESOURCE_MEM,
1440 "slimbus_bam_physical");
Sagar Dhariae77961f2011-09-27 14:03:50 -06001441 if (bam_mem)
1442 release_mem_region(bam_mem->start, resource_size(bam_mem));
Sagar Dhariacc969452011-09-19 10:34:30 -06001443 if (slew_mem)
1444 release_mem_region(slew_mem->start, resource_size(slew_mem));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001445 slim_mem = platform_get_resource_byname(pdev, IORESOURCE_MEM,
1446 "slimbus_physical");
Sagar Dhariae77961f2011-09-27 14:03:50 -06001447 if (slim_mem)
1448 release_mem_region(slim_mem->start, resource_size(slim_mem));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001449 return 0;
1450}
1451
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001452#ifdef CONFIG_PM_RUNTIME
1453static int msm_slim_runtime_idle(struct device *device)
1454{
1455 dev_dbg(device, "pm_runtime: idle...\n");
1456 pm_request_autosuspend(device);
1457 return -EAGAIN;
1458}
1459#endif
1460
1461/*
1462 * If PM_RUNTIME is not defined, these 2 functions become helper
1463 * functions to be called from system suspend/resume. So they are not
1464 * inside ifdef CONFIG_PM_RUNTIME
1465 */
Sagar Dharia45e77912012-01-10 09:55:18 -07001466#ifdef CONFIG_PM_SLEEP
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001467static int msm_slim_runtime_suspend(struct device *device)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001468{
1469 struct platform_device *pdev = to_platform_device(device);
1470 struct msm_slim_ctrl *dev = platform_get_drvdata(pdev);
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001471 int ret;
1472 dev_dbg(device, "pm_runtime: suspending...\n");
1473 dev->state = MSM_CTRL_SLEEPING;
1474 ret = slim_ctrl_clk_pause(&dev->ctrl, false, SLIM_CLK_UNSPECIFIED);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -07001475 if (ret) {
1476 dev_err(device, "clk pause not entered:%d", ret);
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001477 dev->state = MSM_CTRL_AWAKE;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -07001478 } else {
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001479 dev->state = MSM_CTRL_ASLEEP;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -07001480 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001481 return ret;
1482}
1483
1484static int msm_slim_runtime_resume(struct device *device)
1485{
1486 struct platform_device *pdev = to_platform_device(device);
1487 struct msm_slim_ctrl *dev = platform_get_drvdata(pdev);
1488 int ret = 0;
1489 dev_dbg(device, "pm_runtime: resuming...\n");
1490 if (dev->state == MSM_CTRL_ASLEEP)
1491 ret = slim_ctrl_clk_pause(&dev->ctrl, true, 0);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -07001492 if (ret) {
1493 dev_err(device, "clk pause not exited:%d", ret);
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001494 dev->state = MSM_CTRL_ASLEEP;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -07001495 } else {
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001496 dev->state = MSM_CTRL_AWAKE;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -07001497 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001498 return ret;
1499}
1500
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001501static int msm_slim_suspend(struct device *dev)
1502{
1503 int ret = 0;
1504 if (!pm_runtime_enabled(dev) || !pm_runtime_suspended(dev)) {
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001505 struct platform_device *pdev = to_platform_device(dev);
1506 struct msm_slim_ctrl *cdev = platform_get_drvdata(pdev);
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001507 dev_dbg(dev, "system suspend");
1508 ret = msm_slim_runtime_suspend(dev);
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001509 if (!ret) {
1510 if (cdev->hclk)
1511 clk_disable_unprepare(cdev->hclk);
1512 }
Sagar Dharia6b559e02011-08-03 17:01:31 -06001513 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001514 if (ret == -EBUSY) {
Sagar Dharia144e5e02011-08-08 17:30:11 -06001515 /*
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001516 * If the clock pause failed due to active channels, there is
1517 * a possibility that some audio stream is active during suspend
1518 * We dont want to return suspend failure in that case so that
1519 * display and relevant components can still go to suspend.
1520 * If there is some other error, then it should be passed-on
1521 * to system level suspend
1522 */
Sagar Dharia144e5e02011-08-08 17:30:11 -06001523 ret = 0;
1524 }
1525 return ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001526}
1527
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001528static int msm_slim_resume(struct device *dev)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001529{
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001530 /* If runtime_pm is enabled, this resume shouldn't do anything */
1531 if (!pm_runtime_enabled(dev) || !pm_runtime_suspended(dev)) {
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001532 struct platform_device *pdev = to_platform_device(dev);
1533 struct msm_slim_ctrl *cdev = platform_get_drvdata(pdev);
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001534 int ret;
1535 dev_dbg(dev, "system resume");
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001536 if (cdev->hclk)
1537 clk_prepare_enable(cdev->hclk);
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001538 ret = msm_slim_runtime_resume(dev);
1539 if (!ret) {
1540 pm_runtime_mark_last_busy(dev);
1541 pm_request_autosuspend(dev);
1542 }
1543 return ret;
1544
Sagar Dharia144e5e02011-08-08 17:30:11 -06001545 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001546 return 0;
1547}
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001548#endif /* CONFIG_PM_SLEEP */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001549
1550static const struct dev_pm_ops msm_slim_dev_pm_ops = {
1551 SET_SYSTEM_SLEEP_PM_OPS(
1552 msm_slim_suspend,
1553 msm_slim_resume
1554 )
1555 SET_RUNTIME_PM_OPS(
1556 msm_slim_runtime_suspend,
1557 msm_slim_runtime_resume,
1558 msm_slim_runtime_idle
1559 )
1560};
1561
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001562static struct of_device_id msm_slim_dt_match[] = {
1563 {
1564 .compatible = "qcom,slim-msm",
1565 },
1566 {}
1567};
1568
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001569static struct platform_driver msm_slim_driver = {
1570 .probe = msm_slim_probe,
1571 .remove = msm_slim_remove,
1572 .driver = {
1573 .name = MSM_SLIM_NAME,
1574 .owner = THIS_MODULE,
1575 .pm = &msm_slim_dev_pm_ops,
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001576 .of_match_table = msm_slim_dt_match,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001577 },
1578};
1579
1580static int msm_slim_init(void)
1581{
1582 return platform_driver_register(&msm_slim_driver);
1583}
1584subsys_initcall(msm_slim_init);
1585
1586static void msm_slim_exit(void)
1587{
1588 platform_driver_unregister(&msm_slim_driver);
1589}
1590module_exit(msm_slim_exit);
1591
1592MODULE_LICENSE("GPL v2");
1593MODULE_VERSION("0.1");
1594MODULE_DESCRIPTION("MSM Slimbus controller");
1595MODULE_ALIAS("platform:msm-slim");