blob: 9a864aa80a98be5a43defff843cc917400fa134f [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
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070033#define QC_MFGID_LSB 0x2
34#define QC_MFGID_MSB 0x17
35#define QC_CHIPID_SL 0x10
36#define QC_DEVID_SAT1 0x3
37#define QC_DEVID_SAT2 0x4
38#define QC_DEVID_PGD 0x5
Sagar Dharia45ee38a2011-08-03 17:01:31 -060039#define QC_MSM_DEVS 5
Sagar Dhariaac913452012-09-04 11:27:26 -060040#define INIT_MX_RETRIES 10
41#define DEF_RETRY_MS 10
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070042
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070043/* Manager registers */
44enum mgr_reg {
45 MGR_CFG = 0x200,
46 MGR_STATUS = 0x204,
47 MGR_RX_MSGQ_CFG = 0x208,
48 MGR_INT_EN = 0x210,
49 MGR_INT_STAT = 0x214,
50 MGR_INT_CLR = 0x218,
51 MGR_TX_MSG = 0x230,
52 MGR_RX_MSG = 0x270,
Sagar Dhariaac913452012-09-04 11:27:26 -060053 MGR_IE_STAT = 0x2F0,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070054 MGR_VE_STAT = 0x300,
55};
56
57enum msg_cfg {
58 MGR_CFG_ENABLE = 1,
59 MGR_CFG_RX_MSGQ_EN = 1 << 1,
60 MGR_CFG_TX_MSGQ_EN_HIGH = 1 << 2,
61 MGR_CFG_TX_MSGQ_EN_LOW = 1 << 3,
62};
63/* Message queue types */
64enum msm_slim_msgq_type {
65 MSGQ_RX = 0,
66 MSGQ_TX_LOW = 1,
67 MSGQ_TX_HIGH = 2,
68};
69/* Framer registers */
70enum frm_reg {
71 FRM_CFG = 0x400,
72 FRM_STAT = 0x404,
73 FRM_INT_EN = 0x410,
74 FRM_INT_STAT = 0x414,
75 FRM_INT_CLR = 0x418,
76 FRM_WAKEUP = 0x41C,
77 FRM_CLKCTL_DONE = 0x420,
78 FRM_IE_STAT = 0x430,
79 FRM_VE_STAT = 0x440,
80};
81
82/* Interface registers */
83enum intf_reg {
84 INTF_CFG = 0x600,
85 INTF_STAT = 0x604,
86 INTF_INT_EN = 0x610,
87 INTF_INT_STAT = 0x614,
88 INTF_INT_CLR = 0x618,
89 INTF_IE_STAT = 0x630,
90 INTF_VE_STAT = 0x640,
91};
92
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070093enum mgr_intr {
94 MGR_INT_RECFG_DONE = 1 << 24,
95 MGR_INT_TX_NACKED_2 = 1 << 25,
96 MGR_INT_MSG_BUF_CONTE = 1 << 26,
97 MGR_INT_RX_MSG_RCVD = 1 << 30,
98 MGR_INT_TX_MSG_SENT = 1 << 31,
99};
100
101enum frm_cfg {
102 FRM_ACTIVE = 1,
103 CLK_GEAR = 7,
104 ROOT_FREQ = 11,
105 REF_CLK_GEAR = 15,
Sagar Dhariadebc8b72012-08-11 15:02:12 -0600106 INTR_WAKE = 19,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700107};
108
Sagar Dharia790cfd02011-09-25 17:56:24 -0600109static struct msm_slim_sat *msm_slim_alloc_sat(struct msm_slim_ctrl *dev);
110
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700111static int msm_sat_enqueue(struct msm_slim_sat *sat, u32 *buf, u8 len)
112{
113 struct msm_slim_ctrl *dev = sat->dev;
Stephen Boyd3cdfc712012-10-31 17:41:43 -0700114 unsigned long flags;
115 spin_lock_irqsave(&sat->lock, flags);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700116 if ((sat->stail + 1) % SAT_CONCUR_MSG == sat->shead) {
Stephen Boyd3cdfc712012-10-31 17:41:43 -0700117 spin_unlock_irqrestore(&sat->lock, flags);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700118 dev_err(dev->dev, "SAT QUEUE full!");
119 return -EXFULL;
120 }
121 memcpy(sat->sat_msgs[sat->stail], (u8 *)buf, len);
122 sat->stail = (sat->stail + 1) % SAT_CONCUR_MSG;
Stephen Boyd3cdfc712012-10-31 17:41:43 -0700123 spin_unlock_irqrestore(&sat->lock, flags);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700124 return 0;
125}
126
127static int msm_sat_dequeue(struct msm_slim_sat *sat, u8 *buf)
128{
129 unsigned long flags;
130 spin_lock_irqsave(&sat->lock, flags);
131 if (sat->stail == sat->shead) {
132 spin_unlock_irqrestore(&sat->lock, flags);
133 return -ENODATA;
134 }
135 memcpy(buf, sat->sat_msgs[sat->shead], 40);
136 sat->shead = (sat->shead + 1) % SAT_CONCUR_MSG;
137 spin_unlock_irqrestore(&sat->lock, flags);
138 return 0;
139}
140
141static void msm_get_eaddr(u8 *e_addr, u32 *buffer)
142{
143 e_addr[0] = (buffer[1] >> 24) & 0xff;
144 e_addr[1] = (buffer[1] >> 16) & 0xff;
145 e_addr[2] = (buffer[1] >> 8) & 0xff;
146 e_addr[3] = buffer[1] & 0xff;
147 e_addr[4] = (buffer[0] >> 24) & 0xff;
148 e_addr[5] = (buffer[0] >> 16) & 0xff;
149}
150
151static bool msm_is_sat_dev(u8 *e_addr)
152{
153 if (e_addr[5] == QC_MFGID_LSB && e_addr[4] == QC_MFGID_MSB &&
154 e_addr[2] != QC_CHIPID_SL &&
155 (e_addr[1] == QC_DEVID_SAT1 || e_addr[1] == QC_DEVID_SAT2))
156 return true;
157 return false;
158}
159
Sagar Dharia790cfd02011-09-25 17:56:24 -0600160static struct msm_slim_sat *addr_to_sat(struct msm_slim_ctrl *dev, u8 laddr)
161{
162 struct msm_slim_sat *sat = NULL;
163 int i = 0;
164 while (!sat && i < dev->nsats) {
165 if (laddr == dev->satd[i]->satcl.laddr)
166 sat = dev->satd[i];
167 i++;
168 }
169 return sat;
170}
171
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700172static irqreturn_t msm_slim_interrupt(int irq, void *d)
173{
174 struct msm_slim_ctrl *dev = d;
175 u32 pstat;
176 u32 stat = readl_relaxed(dev->base + MGR_INT_STAT);
177
178 if (stat & MGR_INT_TX_MSG_SENT || stat & MGR_INT_TX_NACKED_2) {
179 if (stat & MGR_INT_TX_MSG_SENT)
180 writel_relaxed(MGR_INT_TX_MSG_SENT,
181 dev->base + MGR_INT_CLR);
182 else {
Sagar Dhariaac913452012-09-04 11:27:26 -0600183 u32 mgr_stat = readl_relaxed(dev->base + MGR_STATUS);
184 u32 mgr_ie_stat = readl_relaxed(dev->base +
185 MGR_IE_STAT);
186 u32 frm_stat = readl_relaxed(dev->base + FRM_STAT);
187 u32 frm_cfg = readl_relaxed(dev->base + FRM_CFG);
188 u32 frm_intr_stat = readl_relaxed(dev->base +
189 FRM_INT_STAT);
190 u32 frm_ie_stat = readl_relaxed(dev->base +
191 FRM_IE_STAT);
192 u32 intf_stat = readl_relaxed(dev->base + INTF_STAT);
193 u32 intf_intr_stat = readl_relaxed(dev->base +
194 INTF_INT_STAT);
195 u32 intf_ie_stat = readl_relaxed(dev->base +
196 INTF_IE_STAT);
197
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700198 writel_relaxed(MGR_INT_TX_NACKED_2,
199 dev->base + MGR_INT_CLR);
Sagar Dhariaac913452012-09-04 11:27:26 -0600200 pr_err("TX Nack MGR dump:int_stat:0x%x, mgr_stat:0x%x",
201 stat, mgr_stat);
202 pr_err("TX Nack MGR dump:ie_stat:0x%x", mgr_ie_stat);
203 pr_err("TX Nack FRM dump:int_stat:0x%x, frm_stat:0x%x",
204 frm_intr_stat, frm_stat);
205 pr_err("TX Nack FRM dump:frm_cfg:0x%x, ie_stat:0x%x",
206 frm_cfg, frm_ie_stat);
207 pr_err("TX Nack INTF dump:intr_st:0x%x, intf_stat:0x%x",
208 intf_intr_stat, intf_stat);
209 pr_err("TX Nack INTF dump:ie_stat:0x%x", intf_ie_stat);
210
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700211 dev->err = -EIO;
212 }
213 /*
214 * Guarantee that interrupt clear bit write goes through before
215 * signalling completion/exiting ISR
216 */
217 mb();
218 if (dev->wr_comp)
219 complete(dev->wr_comp);
220 }
221 if (stat & MGR_INT_RX_MSG_RCVD) {
222 u32 rx_buf[10];
223 u32 mc, mt;
224 u8 len, i;
225 rx_buf[0] = readl_relaxed(dev->base + MGR_RX_MSG);
226 len = rx_buf[0] & 0x1F;
227 for (i = 1; i < ((len + 3) >> 2); i++) {
228 rx_buf[i] = readl_relaxed(dev->base + MGR_RX_MSG +
229 (4 * i));
230 dev_dbg(dev->dev, "reading data: %x\n", rx_buf[i]);
231 }
232 mt = (rx_buf[0] >> 5) & 0x7;
233 mc = (rx_buf[0] >> 8) & 0xff;
234 dev_dbg(dev->dev, "MC: %x, MT: %x\n", mc, mt);
235 if (mt == SLIM_MSG_MT_DEST_REFERRED_USER ||
236 mt == SLIM_MSG_MT_SRC_REFERRED_USER) {
Sagar Dharia790cfd02011-09-25 17:56:24 -0600237 u8 laddr = (u8)((rx_buf[0] >> 16) & 0xFF);
238 struct msm_slim_sat *sat = addr_to_sat(dev, laddr);
239 if (sat)
240 msm_sat_enqueue(sat, rx_buf, len);
241 else
242 dev_err(dev->dev, "unknown sat:%d message",
243 laddr);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700244 writel_relaxed(MGR_INT_RX_MSG_RCVD,
245 dev->base + MGR_INT_CLR);
246 /*
247 * Guarantee that CLR bit write goes through before
248 * queuing work
249 */
250 mb();
Sagar Dharia790cfd02011-09-25 17:56:24 -0600251 if (sat)
252 queue_work(sat->wq, &sat->wd);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700253 } else if (mt == SLIM_MSG_MT_CORE &&
254 mc == SLIM_MSG_MC_REPORT_PRESENT) {
255 u8 e_addr[6];
256 msm_get_eaddr(e_addr, rx_buf);
Sagar Dharia790cfd02011-09-25 17:56:24 -0600257 msm_slim_rx_enqueue(dev, rx_buf, len);
258 writel_relaxed(MGR_INT_RX_MSG_RCVD, dev->base +
259 MGR_INT_CLR);
260 /*
261 * Guarantee that CLR bit write goes through
262 * before signalling completion
263 */
264 mb();
265 complete(&dev->rx_msgq_notify);
Kiran Gunda33a23a92013-01-10 17:03:33 +0530266 } else if (mt == SLIM_MSG_MT_CORE &&
267 mc == SLIM_MSG_MC_REPORT_ABSENT) {
268 writel_relaxed(MGR_INT_RX_MSG_RCVD, dev->base +
269 MGR_INT_CLR);
270 /*
271 * Guarantee that CLR bit write goes through
272 * before signalling completion
273 */
274 mb();
275 complete(&dev->rx_msgq_notify);
276
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700277 } else if (mc == SLIM_MSG_MC_REPLY_INFORMATION ||
278 mc == SLIM_MSG_MC_REPLY_VALUE) {
279 msm_slim_rx_enqueue(dev, rx_buf, len);
280 writel_relaxed(MGR_INT_RX_MSG_RCVD, dev->base +
281 MGR_INT_CLR);
282 /*
283 * Guarantee that CLR bit write goes through
284 * before signalling completion
285 */
286 mb();
287 complete(&dev->rx_msgq_notify);
Sagar Dharia144e5e02011-08-08 17:30:11 -0600288 } else if (mc == SLIM_MSG_MC_REPORT_INFORMATION) {
289 u8 *buf = (u8 *)rx_buf;
290 u8 l_addr = buf[2];
291 u16 ele = (u16)buf[4] << 4;
292 ele |= ((buf[3] & 0xf0) >> 4);
293 dev_err(dev->dev, "Slim-dev:%d report inf element:0x%x",
294 l_addr, ele);
295 for (i = 0; i < len - 5; i++)
296 dev_err(dev->dev, "offset:0x%x:bit mask:%x",
297 i, buf[i+5]);
298 writel_relaxed(MGR_INT_RX_MSG_RCVD, dev->base +
299 MGR_INT_CLR);
300 /*
301 * Guarantee that CLR bit write goes through
302 * before exiting
303 */
304 mb();
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700305 } else {
306 dev_err(dev->dev, "Unexpected MC,%x MT:%x, len:%d",
307 mc, mt, len);
308 for (i = 0; i < ((len + 3) >> 2); i++)
309 dev_err(dev->dev, "error msg: %x", rx_buf[i]);
310 writel_relaxed(MGR_INT_RX_MSG_RCVD, dev->base +
311 MGR_INT_CLR);
312 /*
313 * Guarantee that CLR bit write goes through
314 * before exiting
315 */
316 mb();
317 }
318 }
319 if (stat & MGR_INT_RECFG_DONE) {
320 writel_relaxed(MGR_INT_RECFG_DONE, dev->base + MGR_INT_CLR);
321 /*
322 * Guarantee that CLR bit write goes through
323 * before exiting ISR
324 */
325 mb();
326 complete(&dev->reconf);
327 }
Sagar Dharia82e516f2012-03-16 16:01:23 -0600328 pstat = readl_relaxed(PGD_THIS_EE(PGD_PORT_INT_ST_EEn, dev->ver));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700329 if (pstat != 0) {
330 int i = 0;
331 for (i = dev->pipe_b; i < MSM_SLIM_NPORTS; i++) {
332 if (pstat & 1 << i) {
Sagar Dharia82e516f2012-03-16 16:01:23 -0600333 u32 val = readl_relaxed(PGD_PORT(PGD_PORT_STATn,
334 i, dev->ver));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700335 if (val & (1 << 19)) {
336 dev->ctrl.ports[i].err =
337 SLIM_P_DISCONNECT;
338 dev->pipes[i-dev->pipe_b].connected =
339 false;
340 /*
341 * SPS will call completion since
342 * ERROR flags are registered
343 */
344 } else if (val & (1 << 2))
345 dev->ctrl.ports[i].err =
346 SLIM_P_OVERFLOW;
347 else if (val & (1 << 3))
348 dev->ctrl.ports[i].err =
349 SLIM_P_UNDERFLOW;
350 }
Sagar Dharia82e516f2012-03-16 16:01:23 -0600351 writel_relaxed(1, PGD_THIS_EE(PGD_PORT_INT_CL_EEn,
352 dev->ver));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700353 }
354 /*
355 * Guarantee that port interrupt bit(s) clearing writes go
356 * through before exiting ISR
357 */
358 mb();
359 }
360
361 return IRQ_HANDLED;
362}
363
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700364static int msm_xfer_msg(struct slim_controller *ctrl, struct slim_msg_txn *txn)
365{
366 DECLARE_COMPLETION_ONSTACK(done);
367 struct msm_slim_ctrl *dev = slim_get_ctrldata(ctrl);
368 u32 *pbuf;
369 u8 *puc;
370 int timeout;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700371 int msgv = -1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700372 u8 la = txn->la;
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600373 u8 mc = (u8)(txn->mc & 0xFF);
374 /*
375 * Voting for runtime PM: Slimbus has 2 possible use cases:
376 * 1. messaging
377 * 2. Data channels
378 * Messaging case goes through messaging slots and data channels
379 * use their own slots
380 * This "get" votes for messaging bandwidth
381 */
382 if (!(txn->mc & SLIM_MSG_CLK_PAUSE_SEQ_FLG))
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700383 msgv = msm_slim_get_ctrl(dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700384 mutex_lock(&dev->tx_lock);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700385 if (dev->state == MSM_CTRL_ASLEEP ||
386 ((!(txn->mc & SLIM_MSG_CLK_PAUSE_SEQ_FLG)) &&
387 dev->state == MSM_CTRL_SLEEPING)) {
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600388 dev_err(dev->dev, "runtime or system PM suspended state");
389 mutex_unlock(&dev->tx_lock);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700390 if (msgv >= 0)
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600391 msm_slim_put_ctrl(dev);
392 return -EBUSY;
393 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700394 if (txn->mt == SLIM_MSG_MT_CORE &&
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600395 mc == SLIM_MSG_MC_BEGIN_RECONFIGURATION) {
396 if (dev->reconf_busy) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700397 wait_for_completion(&dev->reconf);
398 dev->reconf_busy = false;
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600399 }
400 /* This "get" votes for data channels */
401 if (dev->ctrl.sched.usedslots != 0 &&
402 !dev->chan_active) {
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700403 int chv = msm_slim_get_ctrl(dev);
404 if (chv >= 0)
405 dev->chan_active = true;
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600406 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700407 }
408 txn->rl--;
Sagar Dharia2754ab42012-08-21 18:07:39 -0600409 pbuf = msm_get_msg_buf(dev, txn->rl);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700410 dev->wr_comp = NULL;
411 dev->err = 0;
412
413 if (txn->dt == SLIM_MSG_DEST_ENUMADDR) {
414 mutex_unlock(&dev->tx_lock);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700415 if (msgv >= 0)
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600416 msm_slim_put_ctrl(dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700417 return -EPROTONOSUPPORT;
418 }
419 if (txn->mt == SLIM_MSG_MT_CORE && txn->la == 0xFF &&
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600420 (mc == SLIM_MSG_MC_CONNECT_SOURCE ||
421 mc == SLIM_MSG_MC_CONNECT_SINK ||
422 mc == SLIM_MSG_MC_DISCONNECT_PORT))
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700423 la = dev->pgdla;
424 if (txn->dt == SLIM_MSG_DEST_LOGICALADDR)
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600425 *pbuf = SLIM_MSG_ASM_FIRST_WORD(txn->rl, txn->mt, mc, 0, la);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700426 else
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600427 *pbuf = SLIM_MSG_ASM_FIRST_WORD(txn->rl, txn->mt, mc, 1, la);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700428 if (txn->dt == SLIM_MSG_DEST_LOGICALADDR)
429 puc = ((u8 *)pbuf) + 3;
430 else
431 puc = ((u8 *)pbuf) + 2;
432 if (txn->rbuf)
433 *(puc++) = txn->tid;
434 if ((txn->mt == SLIM_MSG_MT_CORE) &&
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600435 ((mc >= SLIM_MSG_MC_REQUEST_INFORMATION &&
436 mc <= SLIM_MSG_MC_REPORT_INFORMATION) ||
437 (mc >= SLIM_MSG_MC_REQUEST_VALUE &&
438 mc <= SLIM_MSG_MC_CHANGE_VALUE))) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700439 *(puc++) = (txn->ec & 0xFF);
440 *(puc++) = (txn->ec >> 8)&0xFF;
441 }
442 if (txn->wbuf)
443 memcpy(puc, txn->wbuf, txn->len);
444 if (txn->mt == SLIM_MSG_MT_CORE && txn->la == 0xFF &&
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600445 (mc == SLIM_MSG_MC_CONNECT_SOURCE ||
446 mc == SLIM_MSG_MC_CONNECT_SINK ||
447 mc == SLIM_MSG_MC_DISCONNECT_PORT)) {
448 if (mc != SLIM_MSG_MC_DISCONNECT_PORT)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700449 dev->err = msm_slim_connect_pipe_port(dev, *puc);
450 else {
451 struct msm_slim_endp *endpoint = &dev->pipes[*puc];
452 struct sps_register_event sps_event;
453 memset(&sps_event, 0, sizeof(sps_event));
454 sps_register_event(endpoint->sps, &sps_event);
455 sps_disconnect(endpoint->sps);
456 /*
457 * Remove channel disconnects master-side ports from
458 * channel. No need to send that again on the bus
459 */
460 dev->pipes[*puc].connected = false;
461 mutex_unlock(&dev->tx_lock);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700462 if (msgv >= 0)
463 msm_slim_put_ctrl(dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700464 return 0;
465 }
466 if (dev->err) {
467 dev_err(dev->dev, "pipe-port connect err:%d", dev->err);
468 mutex_unlock(&dev->tx_lock);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700469 if (msgv >= 0)
470 msm_slim_put_ctrl(dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700471 return dev->err;
472 }
473 *(puc) = *(puc) + dev->pipe_b;
474 }
475 if (txn->mt == SLIM_MSG_MT_CORE &&
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600476 mc == SLIM_MSG_MC_BEGIN_RECONFIGURATION)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700477 dev->reconf_busy = true;
478 dev->wr_comp = &done;
Sagar Dharia2754ab42012-08-21 18:07:39 -0600479 msm_send_msg_buf(dev, pbuf, txn->rl, MGR_TX_MSG);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700480 timeout = wait_for_completion_timeout(&done, HZ);
Sagar Dhariaac913452012-09-04 11:27:26 -0600481 if (!timeout)
482 dev->wr_comp = NULL;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700483 if (mc == SLIM_MSG_MC_RECONFIGURE_NOW) {
484 if ((txn->mc == (SLIM_MSG_MC_RECONFIGURE_NOW |
485 SLIM_MSG_CLK_PAUSE_SEQ_FLG)) &&
486 timeout) {
487 timeout = wait_for_completion_timeout(&dev->reconf, HZ);
488 dev->reconf_busy = false;
489 if (timeout) {
Sagar Dharia9acf7f42012-03-08 09:45:30 -0700490 clk_disable_unprepare(dev->rclk);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700491 disable_irq(dev->irq);
492 }
493 }
494 if ((txn->mc == (SLIM_MSG_MC_RECONFIGURE_NOW |
495 SLIM_MSG_CLK_PAUSE_SEQ_FLG)) &&
496 !timeout) {
497 dev->reconf_busy = false;
498 dev_err(dev->dev, "clock pause failed");
499 mutex_unlock(&dev->tx_lock);
500 return -ETIMEDOUT;
501 }
502 if (txn->mt == SLIM_MSG_MT_CORE &&
503 txn->mc == SLIM_MSG_MC_RECONFIGURE_NOW) {
504 if (dev->ctrl.sched.usedslots == 0 &&
505 dev->chan_active) {
506 dev->chan_active = false;
507 msm_slim_put_ctrl(dev);
508 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600509 }
510 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600511 mutex_unlock(&dev->tx_lock);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700512 if (msgv >= 0)
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600513 msm_slim_put_ctrl(dev);
514
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700515 if (!timeout)
516 dev_err(dev->dev, "TX timed out:MC:0x%x,mt:0x%x", txn->mc,
517 txn->mt);
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600518
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700519 return timeout ? dev->err : -ETIMEDOUT;
520}
521
Sagar Dhariaac913452012-09-04 11:27:26 -0600522static void msm_slim_wait_retry(struct msm_slim_ctrl *dev)
523{
524 int msec_per_frm = 0;
525 int sfr_per_sec;
526 /* Wait for 1 superframe, or default time and then retry */
527 sfr_per_sec = dev->framer.superfreq /
528 (1 << (SLIM_MAX_CLK_GEAR - dev->ctrl.clkgear));
529 if (sfr_per_sec)
530 msec_per_frm = MSEC_PER_SEC / sfr_per_sec;
531 if (msec_per_frm < DEF_RETRY_MS)
532 msec_per_frm = DEF_RETRY_MS;
533 msleep(msec_per_frm);
534}
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700535static int msm_set_laddr(struct slim_controller *ctrl, const u8 *ea,
536 u8 elen, u8 laddr)
537{
538 struct msm_slim_ctrl *dev = slim_get_ctrldata(ctrl);
Sagar Dhariaac913452012-09-04 11:27:26 -0600539 struct completion done;
540 int timeout, ret, retries = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700541 u32 *buf;
Sagar Dhariaac913452012-09-04 11:27:26 -0600542retry_laddr:
543 init_completion(&done);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700544 mutex_lock(&dev->tx_lock);
Sagar Dharia2754ab42012-08-21 18:07:39 -0600545 buf = msm_get_msg_buf(dev, 9);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700546 buf[0] = SLIM_MSG_ASM_FIRST_WORD(9, SLIM_MSG_MT_CORE,
547 SLIM_MSG_MC_ASSIGN_LOGICAL_ADDRESS,
548 SLIM_MSG_DEST_LOGICALADDR,
549 ea[5] | ea[4] << 8);
550 buf[1] = ea[3] | (ea[2] << 8) | (ea[1] << 16) | (ea[0] << 24);
551 buf[2] = laddr;
552
553 dev->wr_comp = &done;
Sagar Dharia2754ab42012-08-21 18:07:39 -0600554 ret = msm_send_msg_buf(dev, buf, 9, MGR_TX_MSG);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700555 timeout = wait_for_completion_timeout(&done, HZ);
Sagar Dhariaac913452012-09-04 11:27:26 -0600556 if (!timeout)
557 dev->err = -ETIMEDOUT;
558 if (dev->err) {
559 ret = dev->err;
560 dev->err = 0;
561 dev->wr_comp = NULL;
562 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700563 mutex_unlock(&dev->tx_lock);
Sagar Dhariaac913452012-09-04 11:27:26 -0600564 if (ret) {
565 pr_err("set LADDR:0x%x failed:ret:%d, retrying", laddr, ret);
566 if (retries < INIT_MX_RETRIES) {
567 msm_slim_wait_retry(dev);
568 retries++;
569 goto retry_laddr;
570 } else {
571 pr_err("set LADDR failed after retrying:ret:%d", ret);
572 }
573 }
574 return ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700575}
576
Sagar Dharia144e5e02011-08-08 17:30:11 -0600577static int msm_clk_pause_wakeup(struct slim_controller *ctrl)
578{
579 struct msm_slim_ctrl *dev = slim_get_ctrldata(ctrl);
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600580 enable_irq(dev->irq);
Sagar Dharia9acf7f42012-03-08 09:45:30 -0700581 clk_prepare_enable(dev->rclk);
Sagar Dharia144e5e02011-08-08 17:30:11 -0600582 writel_relaxed(1, dev->base + FRM_WAKEUP);
583 /* Make sure framer wakeup write goes through before exiting function */
584 mb();
585 /*
586 * Workaround: Currently, slave is reporting lost-sync messages
587 * after slimbus comes out of clock pause.
588 * Transaction with slave fail before slave reports that message
589 * Give some time for that report to come
590 * Slimbus wakes up in clock gear 10 at 24.576MHz. With each superframe
591 * being 250 usecs, we wait for 20 superframes here to ensure
592 * we get the message
593 */
594 usleep_range(5000, 5000);
595 return 0;
596}
597
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700598static int msm_sat_define_ch(struct msm_slim_sat *sat, u8 *buf, u8 len, u8 mc)
599{
600 struct msm_slim_ctrl *dev = sat->dev;
601 enum slim_ch_control oper;
602 int i;
603 int ret = 0;
604 if (mc == SLIM_USR_MC_CHAN_CTRL) {
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600605 for (i = 0; i < sat->nsatch; i++) {
606 if (buf[5] == sat->satch[i].chan)
607 break;
608 }
609 if (i >= sat->nsatch)
610 return -ENOTCONN;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700611 oper = ((buf[3] & 0xC0) >> 6);
612 /* part of grp. activating/removing 1 will take care of rest */
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600613 ret = slim_control_ch(&sat->satcl, sat->satch[i].chanh, oper,
614 false);
615 if (!ret) {
616 for (i = 5; i < len; i++) {
617 int j;
618 for (j = 0; j < sat->nsatch; j++) {
619 if (buf[i] == sat->satch[j].chan) {
620 if (oper == SLIM_CH_REMOVE)
621 sat->satch[j].req_rem++;
622 else
623 sat->satch[j].req_def++;
624 break;
625 }
626 }
627 }
628 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700629 } else {
630 u16 chh[40];
631 struct slim_ch prop;
632 u32 exp;
Sagar Dhariab886e042012-10-17 22:41:57 -0600633 u16 *grph = NULL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700634 u8 coeff, cc;
635 u8 prrate = buf[6];
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600636 if (len <= 8)
637 return -EINVAL;
638 for (i = 8; i < len; i++) {
639 int j = 0;
640 for (j = 0; j < sat->nsatch; j++) {
641 if (sat->satch[j].chan == buf[i]) {
642 chh[i - 8] = sat->satch[j].chanh;
643 break;
644 }
645 }
646 if (j < sat->nsatch) {
647 u16 dummy;
648 ret = slim_query_ch(&sat->satcl, buf[i],
649 &dummy);
650 if (ret)
651 return ret;
652 if (mc == SLIM_USR_MC_DEF_ACT_CHAN)
653 sat->satch[j].req_def++;
Sagar Dhariab886e042012-10-17 22:41:57 -0600654 /* First channel in group from satellite */
655 if (i == 8)
656 grph = &sat->satch[j].chanh;
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600657 continue;
658 }
659 if (sat->nsatch >= MSM_MAX_SATCH)
660 return -EXFULL;
661 ret = slim_query_ch(&sat->satcl, buf[i], &chh[i - 8]);
662 if (ret)
663 return ret;
664 sat->satch[j].chan = buf[i];
665 sat->satch[j].chanh = chh[i - 8];
666 if (mc == SLIM_USR_MC_DEF_ACT_CHAN)
667 sat->satch[j].req_def++;
Sagar Dhariab886e042012-10-17 22:41:57 -0600668 if (i == 8)
669 grph = &sat->satch[j].chanh;
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600670 sat->nsatch++;
671 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700672 prop.dataf = (enum slim_ch_dataf)((buf[3] & 0xE0) >> 5);
673 prop.auxf = (enum slim_ch_auxf)((buf[4] & 0xC0) >> 5);
674 prop.baser = SLIM_RATE_4000HZ;
675 if (prrate & 0x8)
676 prop.baser = SLIM_RATE_11025HZ;
677 else
678 prop.baser = SLIM_RATE_4000HZ;
679 prop.prot = (enum slim_ch_proto)(buf[5] & 0x0F);
680 prop.sampleszbits = (buf[4] & 0x1F)*SLIM_CL_PER_SL;
681 exp = (u32)((buf[5] & 0xF0) >> 4);
682 coeff = (buf[4] & 0x20) >> 5;
683 cc = (coeff ? 3 : 1);
684 prop.ratem = cc * (1 << exp);
685 if (i > 9)
686 ret = slim_define_ch(&sat->satcl, &prop, chh, len - 8,
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600687 true, &chh[0]);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700688 else
689 ret = slim_define_ch(&sat->satcl, &prop,
Sagar Dhariab886e042012-10-17 22:41:57 -0600690 chh, 1, true, &chh[0]);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700691 dev_dbg(dev->dev, "define sat grp returned:%d", ret);
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600692 if (ret)
693 return ret;
Sagar Dhariab886e042012-10-17 22:41:57 -0600694 else if (grph)
695 *grph = chh[0];
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700696
697 /* part of group so activating 1 will take care of rest */
698 if (mc == SLIM_USR_MC_DEF_ACT_CHAN)
699 ret = slim_control_ch(&sat->satcl,
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600700 chh[0],
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700701 SLIM_CH_ACTIVATE, false);
702 }
703 return ret;
704}
705
706static void msm_slim_rxwq(struct msm_slim_ctrl *dev)
707{
708 u8 buf[40];
709 u8 mc, mt, len;
710 int i, ret;
711 if ((msm_slim_rx_dequeue(dev, (u8 *)buf)) != -ENODATA) {
712 len = buf[0] & 0x1F;
713 mt = (buf[0] >> 5) & 0x7;
714 mc = buf[1];
715 if (mt == SLIM_MSG_MT_CORE &&
716 mc == SLIM_MSG_MC_REPORT_PRESENT) {
717 u8 laddr;
718 u8 e_addr[6];
719 for (i = 0; i < 6; i++)
720 e_addr[i] = buf[7-i];
721
Sagar Dhariaf0b9c752012-09-09 17:32:46 -0600722 ret = slim_assign_laddr(&dev->ctrl, e_addr, 6, &laddr,
723 false);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700724 /* Is this Qualcomm ported generic device? */
725 if (!ret && e_addr[5] == QC_MFGID_LSB &&
726 e_addr[4] == QC_MFGID_MSB &&
727 e_addr[1] == QC_DEVID_PGD &&
728 e_addr[2] != QC_CHIPID_SL)
729 dev->pgdla = laddr;
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600730 if (!ret && !pm_runtime_enabled(dev->dev) &&
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700731 laddr == (QC_MSM_DEVS - 1))
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600732 pm_runtime_enable(dev->dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700733
Sagar Dharia790cfd02011-09-25 17:56:24 -0600734 if (!ret && msm_is_sat_dev(e_addr)) {
735 struct msm_slim_sat *sat = addr_to_sat(dev,
736 laddr);
737 if (!sat)
738 sat = msm_slim_alloc_sat(dev);
739 if (!sat)
740 return;
741
742 sat->satcl.laddr = laddr;
743 msm_sat_enqueue(sat, (u32 *)buf, len);
744 queue_work(sat->wq, &sat->wd);
745 }
Sagar Dhariaac913452012-09-04 11:27:26 -0600746 if (ret)
747 pr_err("assign laddr failed, error:%d", ret);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700748 } else if (mc == SLIM_MSG_MC_REPLY_INFORMATION ||
749 mc == SLIM_MSG_MC_REPLY_VALUE) {
750 u8 tid = buf[3];
751 dev_dbg(dev->dev, "tid:%d, len:%d\n", tid, len - 4);
752 slim_msg_response(&dev->ctrl, &buf[4], tid,
753 len - 4);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700754 pm_runtime_mark_last_busy(dev->dev);
Sagar Dharia144e5e02011-08-08 17:30:11 -0600755 } else if (mc == SLIM_MSG_MC_REPORT_INFORMATION) {
756 u8 l_addr = buf[2];
757 u16 ele = (u16)buf[4] << 4;
758 ele |= ((buf[3] & 0xf0) >> 4);
759 dev_err(dev->dev, "Slim-dev:%d report inf element:0x%x",
760 l_addr, ele);
761 for (i = 0; i < len - 5; i++)
762 dev_err(dev->dev, "offset:0x%x:bit mask:%x",
763 i, buf[i+5]);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700764 } else {
765 dev_err(dev->dev, "unexpected message:mc:%x, mt:%x",
766 mc, mt);
767 for (i = 0; i < len; i++)
768 dev_err(dev->dev, "error msg: %x", buf[i]);
769
770 }
771 } else
772 dev_err(dev->dev, "rxwq called and no dequeue");
773}
774
775static void slim_sat_rxprocess(struct work_struct *work)
776{
777 struct msm_slim_sat *sat = container_of(work, struct msm_slim_sat, wd);
778 struct msm_slim_ctrl *dev = sat->dev;
779 u8 buf[40];
780
781 while ((msm_sat_dequeue(sat, buf)) != -ENODATA) {
782 struct slim_msg_txn txn;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700783 u8 len, mc, mt;
784 u32 bw_sl;
785 int ret = 0;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700786 int satv = -1;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700787 bool gen_ack = false;
788 u8 tid;
789 u8 wbuf[8];
Sagar Dhariaac913452012-09-04 11:27:26 -0600790 int i, retries = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700791 txn.mt = SLIM_MSG_MT_SRC_REFERRED_USER;
792 txn.dt = SLIM_MSG_DEST_LOGICALADDR;
793 txn.ec = 0;
794 txn.rbuf = NULL;
795 txn.la = sat->satcl.laddr;
796 /* satellite handling */
797 len = buf[0] & 0x1F;
798 mc = buf[1];
799 mt = (buf[0] >> 5) & 0x7;
800
801 if (mt == SLIM_MSG_MT_CORE &&
802 mc == SLIM_MSG_MC_REPORT_PRESENT) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700803 u8 e_addr[6];
804 for (i = 0; i < 6; i++)
805 e_addr[i] = buf[7-i];
806
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600807 if (pm_runtime_enabled(dev->dev)) {
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700808 satv = msm_slim_get_ctrl(dev);
809 if (satv >= 0)
810 sat->pending_capability = true;
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600811 }
Sagar Dharia69bf5572012-02-21 14:45:35 -0700812 /*
813 * Since capability message is already sent, present
814 * message will indicate subsystem hosting this
815 * satellite has restarted.
816 * Remove all active channels of this satellite
817 * when this is detected
818 */
819 if (sat->sent_capability) {
820 for (i = 0; i < sat->nsatch; i++) {
Sagar Dharia1944a132012-08-15 00:01:57 -0600821 if (sat->satch[i].reconf) {
822 pr_err("SSR, sat:%d, rm ch:%d",
Sagar Dhariaac913452012-09-04 11:27:26 -0600823 sat->satcl.laddr,
Sagar Dharia69bf5572012-02-21 14:45:35 -0700824 sat->satch[i].chan);
Sagar Dharia69bf5572012-02-21 14:45:35 -0700825 slim_control_ch(&sat->satcl,
826 sat->satch[i].chanh,
827 SLIM_CH_REMOVE, true);
Sagar Dhariab886e042012-10-17 22:41:57 -0600828 slim_dealloc_ch(&sat->satcl,
829 sat->satch[i].chanh);
Sagar Dharia1944a132012-08-15 00:01:57 -0600830 sat->satch[i].reconf = false;
831 }
Sagar Dharia69bf5572012-02-21 14:45:35 -0700832 }
833 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600834 } else if (mt != SLIM_MSG_MT_CORE &&
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700835 mc != SLIM_MSG_MC_REPORT_PRESENT) {
836 satv = msm_slim_get_ctrl(dev);
837 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700838 switch (mc) {
839 case SLIM_MSG_MC_REPORT_PRESENT:
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600840 /* Remove runtime_pm vote once satellite acks */
841 if (mt != SLIM_MSG_MT_CORE) {
842 if (pm_runtime_enabled(dev->dev) &&
843 sat->pending_capability) {
844 msm_slim_put_ctrl(dev);
845 sat->pending_capability = false;
846 }
847 continue;
848 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700849 /* send a Manager capability msg */
Sagar Dharia790cfd02011-09-25 17:56:24 -0600850 if (sat->sent_capability) {
851 if (mt == SLIM_MSG_MT_CORE)
852 goto send_capability;
853 else
854 continue;
855 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700856 ret = slim_add_device(&dev->ctrl, &sat->satcl);
857 if (ret) {
858 dev_err(dev->dev,
859 "Satellite-init failed");
860 continue;
861 }
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600862 /* Satellite-channels */
863 sat->satch = kzalloc(MSM_MAX_SATCH *
864 sizeof(struct msm_sat_chan),
865 GFP_KERNEL);
Sagar Dharia790cfd02011-09-25 17:56:24 -0600866send_capability:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700867 txn.mc = SLIM_USR_MC_MASTER_CAPABILITY;
868 txn.mt = SLIM_MSG_MT_SRC_REFERRED_USER;
869 txn.la = sat->satcl.laddr;
870 txn.rl = 8;
871 wbuf[0] = SAT_MAGIC_LSB;
872 wbuf[1] = SAT_MAGIC_MSB;
873 wbuf[2] = SAT_MSG_VER;
874 wbuf[3] = SAT_MSG_PROT;
875 txn.wbuf = wbuf;
876 txn.len = 4;
Sagar Dhariaac913452012-09-04 11:27:26 -0600877 ret = msm_xfer_msg(&dev->ctrl, &txn);
878 if (ret) {
879 pr_err("capability for:0x%x fail:%d, retry:%d",
880 sat->satcl.laddr, ret, retries);
881 if (retries < INIT_MX_RETRIES) {
882 msm_slim_wait_retry(dev);
883 retries++;
884 goto send_capability;
885 } else {
886 pr_err("failed after all retries:%d",
887 ret);
888 }
889 } else {
890 sat->sent_capability = true;
891 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700892 break;
893 case SLIM_USR_MC_ADDR_QUERY:
894 memcpy(&wbuf[1], &buf[4], 6);
895 ret = slim_get_logical_addr(&sat->satcl,
896 &wbuf[1], 6, &wbuf[7]);
897 if (ret)
898 memset(&wbuf[1], 0, 6);
899 wbuf[0] = buf[3];
900 txn.mc = SLIM_USR_MC_ADDR_REPLY;
901 txn.rl = 12;
902 txn.len = 8;
903 txn.wbuf = wbuf;
904 msm_xfer_msg(&dev->ctrl, &txn);
905 break;
906 case SLIM_USR_MC_DEFINE_CHAN:
907 case SLIM_USR_MC_DEF_ACT_CHAN:
908 case SLIM_USR_MC_CHAN_CTRL:
909 if (mc != SLIM_USR_MC_CHAN_CTRL)
910 tid = buf[7];
911 else
912 tid = buf[4];
913 gen_ack = true;
914 ret = msm_sat_define_ch(sat, buf, len, mc);
915 if (ret) {
916 dev_err(dev->dev,
917 "SAT define_ch returned:%d",
918 ret);
919 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600920 if (!sat->pending_reconf) {
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700921 int chv = msm_slim_get_ctrl(dev);
922 if (chv >= 0)
923 sat->pending_reconf = true;
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600924 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700925 break;
926 case SLIM_USR_MC_RECONFIG_NOW:
927 tid = buf[3];
928 gen_ack = true;
929 ret = slim_reconfigure_now(&sat->satcl);
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600930 for (i = 0; i < sat->nsatch; i++) {
931 struct msm_sat_chan *sch = &sat->satch[i];
Sagar Dhariaef444892012-09-05 12:19:24 -0600932 if (sch->req_rem && sch->reconf) {
Sagar Dharia1944a132012-08-15 00:01:57 -0600933 if (!ret) {
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600934 slim_dealloc_ch(&sat->satcl,
935 sch->chanh);
Sagar Dharia1944a132012-08-15 00:01:57 -0600936 sch->reconf = false;
937 }
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600938 sch->req_rem--;
939 } else if (sch->req_def) {
940 if (ret)
941 slim_dealloc_ch(&sat->satcl,
942 sch->chanh);
Sagar Dharia1944a132012-08-15 00:01:57 -0600943 else
944 sch->reconf = true;
Sagar Dharia0ffdca12011-09-25 18:55:53 -0600945 sch->req_def--;
946 }
947 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600948 if (sat->pending_reconf) {
949 msm_slim_put_ctrl(dev);
950 sat->pending_reconf = false;
951 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700952 break;
953 case SLIM_USR_MC_REQ_BW:
954 /* what we get is in SLOTS */
955 bw_sl = (u32)buf[4] << 3 |
956 ((buf[3] & 0xE0) >> 5);
957 sat->satcl.pending_msgsl = bw_sl;
958 tid = buf[5];
959 gen_ack = true;
960 break;
961 case SLIM_USR_MC_CONNECT_SRC:
962 case SLIM_USR_MC_CONNECT_SINK:
963 if (mc == SLIM_USR_MC_CONNECT_SRC)
964 txn.mc = SLIM_MSG_MC_CONNECT_SOURCE;
965 else
966 txn.mc = SLIM_MSG_MC_CONNECT_SINK;
967 wbuf[0] = buf[4] & 0x1F;
968 wbuf[1] = buf[5];
969 tid = buf[6];
970 txn.la = buf[3];
971 txn.mt = SLIM_MSG_MT_CORE;
972 txn.rl = 6;
973 txn.len = 2;
974 txn.wbuf = wbuf;
975 gen_ack = true;
976 ret = msm_xfer_msg(&dev->ctrl, &txn);
977 break;
978 case SLIM_USR_MC_DISCONNECT_PORT:
979 txn.mc = SLIM_MSG_MC_DISCONNECT_PORT;
980 wbuf[0] = buf[4] & 0x1F;
981 tid = buf[5];
982 txn.la = buf[3];
983 txn.rl = 5;
984 txn.len = 1;
985 txn.mt = SLIM_MSG_MT_CORE;
986 txn.wbuf = wbuf;
987 gen_ack = true;
988 ret = msm_xfer_msg(&dev->ctrl, &txn);
Kiran Gunda33a23a92013-01-10 17:03:33 +0530989 break;
990 case SLIM_MSG_MC_REPORT_ABSENT:
991 dev_info(dev->dev, "Received Report Absent Message\n");
992 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700993 default:
994 break;
995 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600996 if (!gen_ack) {
Sagar Dhariad3ef30a2011-12-09 14:30:45 -0700997 if (mc != SLIM_MSG_MC_REPORT_PRESENT && satv >= 0)
Sagar Dharia45ee38a2011-08-03 17:01:31 -0600998 msm_slim_put_ctrl(dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700999 continue;
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001000 }
1001
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001002 wbuf[0] = tid;
1003 if (!ret)
1004 wbuf[1] = MSM_SAT_SUCCSS;
1005 else
1006 wbuf[1] = 0;
1007 txn.mc = SLIM_USR_MC_GENERIC_ACK;
1008 txn.la = sat->satcl.laddr;
1009 txn.rl = 6;
1010 txn.len = 2;
1011 txn.wbuf = wbuf;
1012 txn.mt = SLIM_MSG_MT_SRC_REFERRED_USER;
1013 msm_xfer_msg(&dev->ctrl, &txn);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -07001014 if (satv >= 0)
1015 msm_slim_put_ctrl(dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001016 }
1017}
1018
Sagar Dharia790cfd02011-09-25 17:56:24 -06001019static struct msm_slim_sat *msm_slim_alloc_sat(struct msm_slim_ctrl *dev)
1020{
1021 struct msm_slim_sat *sat;
1022 char *name;
1023 if (dev->nsats >= MSM_MAX_NSATS)
1024 return NULL;
1025
1026 sat = kzalloc(sizeof(struct msm_slim_sat), GFP_KERNEL);
1027 if (!sat) {
1028 dev_err(dev->dev, "no memory for satellite");
1029 return NULL;
1030 }
1031 name = kzalloc(SLIMBUS_NAME_SIZE, GFP_KERNEL);
1032 if (!name) {
1033 dev_err(dev->dev, "no memory for satellite name");
1034 kfree(sat);
1035 return NULL;
1036 }
1037 dev->satd[dev->nsats] = sat;
1038 sat->dev = dev;
1039 snprintf(name, SLIMBUS_NAME_SIZE, "msm_sat%d", dev->nsats);
1040 sat->satcl.name = name;
1041 spin_lock_init(&sat->lock);
1042 INIT_WORK(&sat->wd, slim_sat_rxprocess);
1043 sat->wq = create_singlethread_workqueue(sat->satcl.name);
1044 if (!sat->wq) {
1045 kfree(name);
1046 kfree(sat);
1047 return NULL;
1048 }
1049 /*
1050 * Both sats will be allocated from RX thread and RX thread will
1051 * process messages sequentially. No synchronization necessary
1052 */
1053 dev->nsats++;
1054 return sat;
1055}
1056
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001057static int msm_slim_rx_msgq_thread(void *data)
1058{
1059 struct msm_slim_ctrl *dev = (struct msm_slim_ctrl *)data;
1060 struct completion *notify = &dev->rx_msgq_notify;
1061 struct msm_slim_sat *sat = NULL;
1062 u32 mc = 0;
1063 u32 mt = 0;
1064 u32 buffer[10];
1065 int index = 0;
1066 u8 msg_len = 0;
1067 int ret;
1068
1069 dev_dbg(dev->dev, "rx thread started");
1070
1071 while (!kthread_should_stop()) {
1072 set_current_state(TASK_INTERRUPTIBLE);
1073 ret = wait_for_completion_interruptible(notify);
1074
1075 if (ret)
1076 dev_err(dev->dev, "rx thread wait error:%d", ret);
1077
1078 /* 1 irq notification per message */
Sagar Dharia24419e32013-01-14 17:56:32 -07001079 if (dev->use_rx_msgqs != MSM_MSGQ_ENABLED) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001080 msm_slim_rxwq(dev);
1081 continue;
1082 }
1083
1084 ret = msm_slim_rx_msgq_get(dev, buffer, index);
1085 if (ret) {
1086 dev_err(dev->dev, "rx_msgq_get() failed 0x%x\n", ret);
1087 continue;
1088 }
1089
1090 pr_debug("message[%d] = 0x%x\n", index, *buffer);
1091
1092 /* Decide if we use generic RX or satellite RX */
1093 if (index++ == 0) {
1094 msg_len = *buffer & 0x1F;
1095 pr_debug("Start of new message, len = %d\n", msg_len);
1096 mt = (buffer[0] >> 5) & 0x7;
1097 mc = (buffer[0] >> 8) & 0xff;
1098 dev_dbg(dev->dev, "MC: %x, MT: %x\n", mc, mt);
1099 if (mt == SLIM_MSG_MT_DEST_REFERRED_USER ||
Sagar Dharia790cfd02011-09-25 17:56:24 -06001100 mt == SLIM_MSG_MT_SRC_REFERRED_USER) {
1101 u8 laddr;
1102 laddr = (u8)((buffer[0] >> 16) & 0xff);
1103 sat = addr_to_sat(dev, laddr);
1104 }
Kiran Gunda33a23a92013-01-10 17:03:33 +05301105 }
1106 if ((index * 4) >= msg_len) {
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001107 index = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001108 if (sat) {
1109 msm_sat_enqueue(sat, buffer, msg_len);
1110 queue_work(sat->wq, &sat->wd);
1111 sat = NULL;
1112 } else {
1113 msm_slim_rx_enqueue(dev, buffer, msg_len);
1114 msm_slim_rxwq(dev);
1115 }
1116 }
1117 }
1118
1119 return 0;
1120}
1121
Sagar Dhariacc969452011-09-19 10:34:30 -06001122static void msm_slim_prg_slew(struct platform_device *pdev,
1123 struct msm_slim_ctrl *dev)
1124{
1125 struct resource *slew_io;
1126 void __iomem *slew_reg;
1127 /* SLEW RATE register for this slimbus */
1128 dev->slew_mem = platform_get_resource_byname(pdev, IORESOURCE_MEM,
1129 "slimbus_slew_reg");
1130 if (!dev->slew_mem) {
1131 dev_dbg(&pdev->dev, "no slimbus slew resource\n");
1132 return;
1133 }
1134 slew_io = request_mem_region(dev->slew_mem->start,
1135 resource_size(dev->slew_mem), pdev->name);
1136 if (!slew_io) {
1137 dev_dbg(&pdev->dev, "slimbus-slew mem claimed\n");
1138 dev->slew_mem = NULL;
1139 return;
1140 }
1141
1142 slew_reg = ioremap(dev->slew_mem->start, resource_size(dev->slew_mem));
1143 if (!slew_reg) {
1144 dev_dbg(dev->dev, "slew register mapping failed");
1145 release_mem_region(dev->slew_mem->start,
1146 resource_size(dev->slew_mem));
1147 dev->slew_mem = NULL;
1148 return;
1149 }
1150 writel_relaxed(1, slew_reg);
1151 /* Make sure slimbus-slew rate enabling goes through */
1152 wmb();
1153 iounmap(slew_reg);
1154}
1155
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001156static int __devinit msm_slim_probe(struct platform_device *pdev)
1157{
1158 struct msm_slim_ctrl *dev;
1159 int ret;
Joonwoo Parkf69f77a2012-08-28 15:26:11 -07001160 enum apr_subsys_state q6_state;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001161 struct resource *bam_mem, *bam_io;
1162 struct resource *slim_mem, *slim_io;
1163 struct resource *irq, *bam_irq;
Sagar Dharia1beb2202012-07-31 19:06:21 -06001164 bool rxreg_access = false;
Joonwoo Parkf69f77a2012-08-28 15:26:11 -07001165
1166 q6_state = apr_get_q6_state();
1167 if (q6_state == APR_SUBSYS_DOWN) {
1168 dev_dbg(&pdev->dev, "defering %s, adsp_state %d\n", __func__,
1169 q6_state);
1170 return -EPROBE_DEFER;
1171 } else
1172 dev_dbg(&pdev->dev, "adsp is ready\n");
1173
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001174 slim_mem = platform_get_resource_byname(pdev, IORESOURCE_MEM,
1175 "slimbus_physical");
1176 if (!slim_mem) {
1177 dev_err(&pdev->dev, "no slimbus physical memory resource\n");
1178 return -ENODEV;
1179 }
1180 slim_io = request_mem_region(slim_mem->start, resource_size(slim_mem),
1181 pdev->name);
1182 if (!slim_io) {
1183 dev_err(&pdev->dev, "slimbus memory already claimed\n");
1184 return -EBUSY;
1185 }
1186
1187 bam_mem = platform_get_resource_byname(pdev, IORESOURCE_MEM,
1188 "slimbus_bam_physical");
1189 if (!bam_mem) {
1190 dev_err(&pdev->dev, "no slimbus BAM memory resource\n");
1191 ret = -ENODEV;
1192 goto err_get_res_bam_failed;
1193 }
1194 bam_io = request_mem_region(bam_mem->start, resource_size(bam_mem),
1195 pdev->name);
1196 if (!bam_io) {
1197 release_mem_region(slim_mem->start, resource_size(slim_mem));
1198 dev_err(&pdev->dev, "slimbus BAM memory already claimed\n");
1199 ret = -EBUSY;
1200 goto err_get_res_bam_failed;
1201 }
1202 irq = platform_get_resource_byname(pdev, IORESOURCE_IRQ,
1203 "slimbus_irq");
1204 if (!irq) {
1205 dev_err(&pdev->dev, "no slimbus IRQ resource\n");
1206 ret = -ENODEV;
1207 goto err_get_res_failed;
1208 }
1209 bam_irq = platform_get_resource_byname(pdev, IORESOURCE_IRQ,
1210 "slimbus_bam_irq");
1211 if (!bam_irq) {
1212 dev_err(&pdev->dev, "no slimbus BAM IRQ resource\n");
1213 ret = -ENODEV;
1214 goto err_get_res_failed;
1215 }
1216
1217 dev = kzalloc(sizeof(struct msm_slim_ctrl), GFP_KERNEL);
1218 if (!dev) {
1219 dev_err(&pdev->dev, "no memory for MSM slimbus controller\n");
1220 ret = -ENOMEM;
1221 goto err_get_res_failed;
1222 }
1223 dev->dev = &pdev->dev;
1224 platform_set_drvdata(pdev, dev);
1225 slim_set_ctrldata(&dev->ctrl, dev);
1226 dev->base = ioremap(slim_mem->start, resource_size(slim_mem));
1227 if (!dev->base) {
1228 dev_err(&pdev->dev, "IOremap failed\n");
1229 ret = -ENOMEM;
1230 goto err_ioremap_failed;
1231 }
1232 dev->bam.base = ioremap(bam_mem->start, resource_size(bam_mem));
1233 if (!dev->bam.base) {
1234 dev_err(&pdev->dev, "BAM IOremap failed\n");
1235 ret = -ENOMEM;
1236 goto err_ioremap_bam_failed;
1237 }
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001238 if (pdev->dev.of_node) {
1239
1240 ret = of_property_read_u32(pdev->dev.of_node, "cell-index",
1241 &dev->ctrl.nr);
1242 if (ret) {
1243 dev_err(&pdev->dev, "Cell index not specified:%d", ret);
1244 goto err_of_init_failed;
1245 }
Sagar Dharia1beb2202012-07-31 19:06:21 -06001246 rxreg_access = of_property_read_bool(pdev->dev.of_node,
1247 "qcom,rxreg-access");
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001248 /* Optional properties */
1249 ret = of_property_read_u32(pdev->dev.of_node,
1250 "qcom,min-clk-gear", &dev->ctrl.min_cg);
1251 ret = of_property_read_u32(pdev->dev.of_node,
1252 "qcom,max-clk-gear", &dev->ctrl.max_cg);
Sagar Dharia1beb2202012-07-31 19:06:21 -06001253 pr_debug("min_cg:%d, max_cg:%d, rxreg: %d", dev->ctrl.min_cg,
1254 dev->ctrl.max_cg, rxreg_access);
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001255 } else {
1256 dev->ctrl.nr = pdev->id;
1257 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001258 dev->ctrl.nchans = MSM_SLIM_NCHANS;
1259 dev->ctrl.nports = MSM_SLIM_NPORTS;
1260 dev->ctrl.set_laddr = msm_set_laddr;
1261 dev->ctrl.xfer_msg = msm_xfer_msg;
Sagar Dharia144e5e02011-08-08 17:30:11 -06001262 dev->ctrl.wakeup = msm_clk_pause_wakeup;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001263 dev->ctrl.config_port = msm_config_port;
1264 dev->ctrl.port_xfer = msm_slim_port_xfer;
1265 dev->ctrl.port_xfer_status = msm_slim_port_xfer_status;
1266 /* Reserve some messaging BW for satellite-apps driver communication */
1267 dev->ctrl.sched.pending_msgsl = 30;
1268
1269 init_completion(&dev->reconf);
1270 mutex_init(&dev->tx_lock);
1271 spin_lock_init(&dev->rx_lock);
1272 dev->ee = 1;
Sagar Dharia1beb2202012-07-31 19:06:21 -06001273 if (rxreg_access)
Sagar Dharia24419e32013-01-14 17:56:32 -07001274 dev->use_rx_msgqs = MSM_MSGQ_DISABLED;
Sagar Dharia1beb2202012-07-31 19:06:21 -06001275 else
Sagar Dharia24419e32013-01-14 17:56:32 -07001276 dev->use_rx_msgqs = MSM_MSGQ_RESET;
Sagar Dharia1beb2202012-07-31 19:06:21 -06001277
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001278 dev->irq = irq->start;
1279 dev->bam.irq = bam_irq->start;
1280
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001281 dev->hclk = clk_get(dev->dev, "iface_clk");
1282 if (IS_ERR(dev->hclk))
1283 dev->hclk = NULL;
1284 else
1285 clk_prepare_enable(dev->hclk);
1286
Sagar Dharia60f59a72012-10-17 12:42:03 -06001287 ret = msm_slim_sps_init(dev, bam_mem, MGR_STATUS, false);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001288 if (ret != 0) {
1289 dev_err(dev->dev, "error SPS init\n");
1290 goto err_sps_init_failed;
1291 }
1292
Sagar Dharia2754ab42012-08-21 18:07:39 -06001293 /* Fire up the Rx message queue thread */
1294 dev->rx_msgq_thread = kthread_run(msm_slim_rx_msgq_thread, dev,
1295 MSM_SLIM_NAME "_rx_msgq_thread");
1296 if (IS_ERR(dev->rx_msgq_thread)) {
1297 ret = PTR_ERR(dev->rx_msgq_thread);
1298 dev_err(dev->dev, "Failed to start Rx message queue thread\n");
1299 goto err_thread_create_failed;
1300 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001301
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001302 dev->framer.rootfreq = SLIM_ROOT_FREQ >> 3;
1303 dev->framer.superfreq =
1304 dev->framer.rootfreq / SLIM_CL_PER_SUPERFRAME_DIV8;
1305 dev->ctrl.a_framer = &dev->framer;
1306 dev->ctrl.clkgear = SLIM_MAX_CLK_GEAR;
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001307 dev->ctrl.dev.parent = &pdev->dev;
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001308 dev->ctrl.dev.of_node = pdev->dev.of_node;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001309
1310 ret = request_irq(dev->irq, msm_slim_interrupt, IRQF_TRIGGER_HIGH,
1311 "msm_slim_irq", dev);
1312 if (ret) {
1313 dev_err(&pdev->dev, "request IRQ failed\n");
1314 goto err_request_irq_failed;
1315 }
1316
Sagar Dhariacc969452011-09-19 10:34:30 -06001317 msm_slim_prg_slew(pdev, dev);
Sagar Dhariab1c0acf2012-02-06 18:16:58 -07001318
1319 /* Register with framework before enabling frame, clock */
1320 ret = slim_add_numbered_controller(&dev->ctrl);
1321 if (ret) {
1322 dev_err(dev->dev, "error adding controller\n");
1323 goto err_ctrl_failed;
1324 }
1325
1326
Tianyi Gou44a81b02012-02-06 17:49:07 -08001327 dev->rclk = clk_get(dev->dev, "core_clk");
Sagar Dhariab1c0acf2012-02-06 18:16:58 -07001328 if (!dev->rclk) {
1329 dev_err(dev->dev, "slimbus clock not found");
1330 goto err_clk_get_failed;
1331 }
Sagar Dhariacc969452011-09-19 10:34:30 -06001332 clk_set_rate(dev->rclk, SLIM_ROOT_FREQ);
Sagar Dharia9acf7f42012-03-08 09:45:30 -07001333 clk_prepare_enable(dev->rclk);
Sagar Dhariacc969452011-09-19 10:34:30 -06001334
Sagar Dharia82e516f2012-03-16 16:01:23 -06001335 dev->ver = readl_relaxed(dev->base);
1336 /* Version info in 16 MSbits */
1337 dev->ver >>= 16;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001338 /* Component register initialization */
Sagar Dharia82e516f2012-03-16 16:01:23 -06001339 writel_relaxed(1, dev->base + CFG_PORT(COMP_CFG, dev->ver));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001340 writel_relaxed((EE_MGR_RSC_GRP | EE_NGD_2 | EE_NGD_1),
Sagar Dharia82e516f2012-03-16 16:01:23 -06001341 dev->base + CFG_PORT(COMP_TRUST_CFG, dev->ver));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001342
1343 /*
1344 * Manager register initialization
1345 * If RX msg Q is used, disable RX_MSG_RCVD interrupt
1346 */
Sagar Dharia24419e32013-01-14 17:56:32 -07001347 if (dev->use_rx_msgqs == MSM_MSGQ_ENABLED)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001348 writel_relaxed((MGR_INT_RECFG_DONE | MGR_INT_TX_NACKED_2 |
1349 MGR_INT_MSG_BUF_CONTE | /* MGR_INT_RX_MSG_RCVD | */
1350 MGR_INT_TX_MSG_SENT), dev->base + MGR_INT_EN);
1351 else
1352 writel_relaxed((MGR_INT_RECFG_DONE | MGR_INT_TX_NACKED_2 |
1353 MGR_INT_MSG_BUF_CONTE | MGR_INT_RX_MSG_RCVD |
1354 MGR_INT_TX_MSG_SENT), dev->base + MGR_INT_EN);
1355 writel_relaxed(1, dev->base + MGR_CFG);
1356 /*
1357 * Framer registers are beyond 1K memory region after Manager and/or
1358 * component registers. Make sure those writes are ordered
1359 * before framer register writes
1360 */
1361 wmb();
1362
1363 /* Framer register initialization */
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001364 writel_relaxed((1 << INTR_WAKE) | (0xA << REF_CLK_GEAR) |
1365 (0xA << CLK_GEAR) | (1 << ROOT_FREQ) | (1 << FRM_ACTIVE) | 1,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001366 dev->base + FRM_CFG);
1367 /*
1368 * Make sure that framer wake-up and enabling writes go through
1369 * before any other component is enabled. Framer is responsible for
1370 * clocking the bus and enabling framer first will ensure that other
1371 * devices can report presence when they are enabled
1372 */
1373 mb();
1374
1375 /* Enable RX msg Q */
Sagar Dharia24419e32013-01-14 17:56:32 -07001376 if (dev->use_rx_msgqs == MSM_MSGQ_ENABLED)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001377 writel_relaxed(MGR_CFG_ENABLE | MGR_CFG_RX_MSGQ_EN,
1378 dev->base + MGR_CFG);
1379 else
1380 writel_relaxed(MGR_CFG_ENABLE, dev->base + MGR_CFG);
1381 /*
1382 * Make sure that manager-enable is written through before interface
1383 * device is enabled
1384 */
1385 mb();
1386 writel_relaxed(1, dev->base + INTF_CFG);
1387 /*
1388 * Make sure that interface-enable is written through before enabling
1389 * ported generic device inside MSM manager
1390 */
1391 mb();
Sagar Dharia82e516f2012-03-16 16:01:23 -06001392 writel_relaxed(1, dev->base + CFG_PORT(PGD_CFG, dev->ver));
1393 writel_relaxed(0x3F<<17, dev->base + CFG_PORT(PGD_OWN_EEn, dev->ver) +
1394 (4 * dev->ee));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001395 /*
1396 * Make sure that ported generic device is enabled and port-EE settings
1397 * are written through before finally enabling the component
1398 */
1399 mb();
1400
Sagar Dharia82e516f2012-03-16 16:01:23 -06001401 writel_relaxed(1, dev->base + CFG_PORT(COMP_CFG, dev->ver));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001402 /*
1403 * Make sure that all writes have gone through before exiting this
1404 * function
1405 */
1406 mb();
Sagar Dhariaa6627e02012-08-28 12:20:49 -06001407
1408 /* Add devices registered with board-info now that controller is up */
1409 slim_ctrl_add_boarddevs(&dev->ctrl);
1410
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001411 if (pdev->dev.of_node)
1412 of_register_slim_devices(&dev->ctrl);
1413
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001414 pm_runtime_use_autosuspend(&pdev->dev);
1415 pm_runtime_set_autosuspend_delay(&pdev->dev, MSM_SLIM_AUTOSUSPEND);
1416 pm_runtime_set_active(&pdev->dev);
1417
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001418 dev_dbg(dev->dev, "MSM SB controller is up!\n");
1419 return 0;
1420
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001421err_ctrl_failed:
Sagar Dharia82e516f2012-03-16 16:01:23 -06001422 writel_relaxed(0, dev->base + CFG_PORT(COMP_CFG, dev->ver));
Sagar Dhariab1c0acf2012-02-06 18:16:58 -07001423err_clk_get_failed:
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001424 kfree(dev->satd);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001425err_request_irq_failed:
Sagar Dharia2754ab42012-08-21 18:07:39 -06001426 kthread_stop(dev->rx_msgq_thread);
1427err_thread_create_failed:
Sagar Dharia33beca02012-10-22 16:21:46 -06001428 msm_slim_sps_exit(dev, true);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001429err_sps_init_failed:
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001430 if (dev->hclk) {
1431 clk_disable_unprepare(dev->hclk);
1432 clk_put(dev->hclk);
1433 }
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001434err_of_init_failed:
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001435 iounmap(dev->bam.base);
1436err_ioremap_bam_failed:
1437 iounmap(dev->base);
1438err_ioremap_failed:
1439 kfree(dev);
1440err_get_res_failed:
1441 release_mem_region(bam_mem->start, resource_size(bam_mem));
1442err_get_res_bam_failed:
1443 release_mem_region(slim_mem->start, resource_size(slim_mem));
1444 return ret;
1445}
1446
1447static int __devexit msm_slim_remove(struct platform_device *pdev)
1448{
1449 struct msm_slim_ctrl *dev = platform_get_drvdata(pdev);
1450 struct resource *bam_mem;
1451 struct resource *slim_mem;
Sagar Dhariacc969452011-09-19 10:34:30 -06001452 struct resource *slew_mem = dev->slew_mem;
Sagar Dharia790cfd02011-09-25 17:56:24 -06001453 int i;
1454 for (i = 0; i < dev->nsats; i++) {
1455 struct msm_slim_sat *sat = dev->satd[i];
Sagar Dharia0ffdca12011-09-25 18:55:53 -06001456 int j;
1457 for (j = 0; j < sat->nsatch; j++)
1458 slim_dealloc_ch(&sat->satcl, sat->satch[j].chanh);
Sagar Dharia790cfd02011-09-25 17:56:24 -06001459 slim_remove_device(&sat->satcl);
1460 kfree(sat->satch);
1461 destroy_workqueue(sat->wq);
1462 kfree(sat->satcl.name);
1463 kfree(sat);
1464 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001465 pm_runtime_disable(&pdev->dev);
1466 pm_runtime_set_suspended(&pdev->dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001467 free_irq(dev->irq, dev);
1468 slim_del_controller(&dev->ctrl);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001469 clk_put(dev->rclk);
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001470 if (dev->hclk)
1471 clk_put(dev->hclk);
Sagar Dharia33beca02012-10-22 16:21:46 -06001472 msm_slim_sps_exit(dev, true);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001473 kthread_stop(dev->rx_msgq_thread);
1474 iounmap(dev->bam.base);
1475 iounmap(dev->base);
1476 kfree(dev);
1477 bam_mem = platform_get_resource_byname(pdev, IORESOURCE_MEM,
1478 "slimbus_bam_physical");
Sagar Dhariae77961f2011-09-27 14:03:50 -06001479 if (bam_mem)
1480 release_mem_region(bam_mem->start, resource_size(bam_mem));
Sagar Dhariacc969452011-09-19 10:34:30 -06001481 if (slew_mem)
1482 release_mem_region(slew_mem->start, resource_size(slew_mem));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001483 slim_mem = platform_get_resource_byname(pdev, IORESOURCE_MEM,
1484 "slimbus_physical");
Sagar Dhariae77961f2011-09-27 14:03:50 -06001485 if (slim_mem)
1486 release_mem_region(slim_mem->start, resource_size(slim_mem));
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001487 return 0;
1488}
1489
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001490#ifdef CONFIG_PM_RUNTIME
1491static int msm_slim_runtime_idle(struct device *device)
1492{
1493 dev_dbg(device, "pm_runtime: idle...\n");
1494 pm_request_autosuspend(device);
1495 return -EAGAIN;
1496}
1497#endif
1498
1499/*
1500 * If PM_RUNTIME is not defined, these 2 functions become helper
1501 * functions to be called from system suspend/resume. So they are not
1502 * inside ifdef CONFIG_PM_RUNTIME
1503 */
Sagar Dharia45e77912012-01-10 09:55:18 -07001504#ifdef CONFIG_PM_SLEEP
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001505static int msm_slim_runtime_suspend(struct device *device)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001506{
1507 struct platform_device *pdev = to_platform_device(device);
1508 struct msm_slim_ctrl *dev = platform_get_drvdata(pdev);
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001509 int ret;
1510 dev_dbg(device, "pm_runtime: suspending...\n");
1511 dev->state = MSM_CTRL_SLEEPING;
1512 ret = slim_ctrl_clk_pause(&dev->ctrl, false, SLIM_CLK_UNSPECIFIED);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -07001513 if (ret) {
1514 dev_err(device, "clk pause not entered:%d", ret);
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001515 dev->state = MSM_CTRL_AWAKE;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -07001516 } else {
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001517 dev->state = MSM_CTRL_ASLEEP;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -07001518 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001519 return ret;
1520}
1521
1522static int msm_slim_runtime_resume(struct device *device)
1523{
1524 struct platform_device *pdev = to_platform_device(device);
1525 struct msm_slim_ctrl *dev = platform_get_drvdata(pdev);
1526 int ret = 0;
1527 dev_dbg(device, "pm_runtime: resuming...\n");
1528 if (dev->state == MSM_CTRL_ASLEEP)
1529 ret = slim_ctrl_clk_pause(&dev->ctrl, true, 0);
Sagar Dhariad3ef30a2011-12-09 14:30:45 -07001530 if (ret) {
1531 dev_err(device, "clk pause not exited:%d", ret);
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001532 dev->state = MSM_CTRL_ASLEEP;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -07001533 } else {
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001534 dev->state = MSM_CTRL_AWAKE;
Sagar Dhariad3ef30a2011-12-09 14:30:45 -07001535 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001536 return ret;
1537}
1538
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001539static int msm_slim_suspend(struct device *dev)
1540{
1541 int ret = 0;
1542 if (!pm_runtime_enabled(dev) || !pm_runtime_suspended(dev)) {
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001543 struct platform_device *pdev = to_platform_device(dev);
1544 struct msm_slim_ctrl *cdev = platform_get_drvdata(pdev);
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001545 dev_dbg(dev, "system suspend");
1546 ret = msm_slim_runtime_suspend(dev);
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001547 if (!ret) {
1548 if (cdev->hclk)
1549 clk_disable_unprepare(cdev->hclk);
1550 }
Sagar Dharia6b559e02011-08-03 17:01:31 -06001551 }
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001552 if (ret == -EBUSY) {
Sagar Dharia144e5e02011-08-08 17:30:11 -06001553 /*
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001554 * If the clock pause failed due to active channels, there is
1555 * a possibility that some audio stream is active during suspend
1556 * We dont want to return suspend failure in that case so that
1557 * display and relevant components can still go to suspend.
1558 * If there is some other error, then it should be passed-on
1559 * to system level suspend
1560 */
Sagar Dharia144e5e02011-08-08 17:30:11 -06001561 ret = 0;
1562 }
1563 return ret;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001564}
1565
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001566static int msm_slim_resume(struct device *dev)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001567{
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001568 /* If runtime_pm is enabled, this resume shouldn't do anything */
1569 if (!pm_runtime_enabled(dev) || !pm_runtime_suspended(dev)) {
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001570 struct platform_device *pdev = to_platform_device(dev);
1571 struct msm_slim_ctrl *cdev = platform_get_drvdata(pdev);
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001572 int ret;
1573 dev_dbg(dev, "system resume");
Sagar Dhariadebc8b72012-08-11 15:02:12 -06001574 if (cdev->hclk)
1575 clk_prepare_enable(cdev->hclk);
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001576 ret = msm_slim_runtime_resume(dev);
1577 if (!ret) {
1578 pm_runtime_mark_last_busy(dev);
1579 pm_request_autosuspend(dev);
1580 }
1581 return ret;
1582
Sagar Dharia144e5e02011-08-08 17:30:11 -06001583 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001584 return 0;
1585}
Sagar Dharia45ee38a2011-08-03 17:01:31 -06001586#endif /* CONFIG_PM_SLEEP */
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001587
1588static const struct dev_pm_ops msm_slim_dev_pm_ops = {
1589 SET_SYSTEM_SLEEP_PM_OPS(
1590 msm_slim_suspend,
1591 msm_slim_resume
1592 )
1593 SET_RUNTIME_PM_OPS(
1594 msm_slim_runtime_suspend,
1595 msm_slim_runtime_resume,
1596 msm_slim_runtime_idle
1597 )
1598};
1599
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001600static struct of_device_id msm_slim_dt_match[] = {
1601 {
1602 .compatible = "qcom,slim-msm",
1603 },
1604 {}
1605};
1606
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001607static struct platform_driver msm_slim_driver = {
1608 .probe = msm_slim_probe,
1609 .remove = msm_slim_remove,
1610 .driver = {
1611 .name = MSM_SLIM_NAME,
1612 .owner = THIS_MODULE,
1613 .pm = &msm_slim_dev_pm_ops,
Sagar Dhariaf8f603b2012-03-21 15:25:17 -06001614 .of_match_table = msm_slim_dt_match,
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001615 },
1616};
1617
1618static int msm_slim_init(void)
1619{
1620 return platform_driver_register(&msm_slim_driver);
1621}
1622subsys_initcall(msm_slim_init);
1623
1624static void msm_slim_exit(void)
1625{
1626 platform_driver_unregister(&msm_slim_driver);
1627}
1628module_exit(msm_slim_exit);
1629
1630MODULE_LICENSE("GPL v2");
1631MODULE_VERSION("0.1");
1632MODULE_DESCRIPTION("MSM Slimbus controller");
1633MODULE_ALIAS("platform:msm-slim");