Blame view

drivers/mailbox/omap-mailbox.c 21.6 KB
2ad515765   Suman Anna   mailbox/omap: swi...
1
  // SPDX-License-Identifier: GPL-2.0
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
2
3
4
  /*
   * OMAP mailbox driver
   *
f48cca877   Hiroshi DOYU   omap mailbox: fix...
5
   * Copyright (C) 2006-2009 Nokia Corporation. All rights reserved.
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
6
   * Copyright (C) 2013-2019 Texas Instruments Incorporated - http://www.ti.com
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
7
   *
f48cca877   Hiroshi DOYU   omap mailbox: fix...
8
   * Contact: Hiroshi DOYU <Hiroshi.DOYU@nokia.com>
5040f5343   Suman Anna   mailbox/omap: con...
9
   *          Suman Anna <s-anna@ti.com>
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
10
   */
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
11
  #include <linux/interrupt.h>
b3e69146f   Felipe Contreras   omap: mailbox: re...
12
13
  #include <linux/spinlock.h>
  #include <linux/mutex.h>
5a0e3ad6a   Tejun Heo   include cleanup: ...
14
  #include <linux/slab.h>
b5bebe410   Ohad Ben-Cohen   omap: mailbox: co...
15
16
  #include <linux/kfifo.h>
  #include <linux/err.h>
73017a542   Paul Gortmaker   arm: fix implicit...
17
  #include <linux/module.h>
75288cc66   Suman Anna   mailbox/omap: add...
18
  #include <linux/of_device.h>
5040f5343   Suman Anna   mailbox/omap: con...
19
20
  #include <linux/platform_device.h>
  #include <linux/pm_runtime.h>
5040f5343   Suman Anna   mailbox/omap: con...
21
  #include <linux/omap-mailbox.h>
8841a66aa   Suman Anna   mailbox/omap: ada...
22
23
  #include <linux/mailbox_controller.h>
  #include <linux/mailbox_client.h>
5040f5343   Suman Anna   mailbox/omap: con...
24

8e3c59521   Dave Gerlach   mailbox/omap: Add...
25
  #include "mailbox.h"
5040f5343   Suman Anna   mailbox/omap: con...
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
  #define MAILBOX_REVISION		0x000
  #define MAILBOX_MESSAGE(m)		(0x040 + 4 * (m))
  #define MAILBOX_FIFOSTATUS(m)		(0x080 + 4 * (m))
  #define MAILBOX_MSGSTATUS(m)		(0x0c0 + 4 * (m))
  
  #define OMAP2_MAILBOX_IRQSTATUS(u)	(0x100 + 8 * (u))
  #define OMAP2_MAILBOX_IRQENABLE(u)	(0x104 + 8 * (u))
  
  #define OMAP4_MAILBOX_IRQSTATUS(u)	(0x104 + 0x10 * (u))
  #define OMAP4_MAILBOX_IRQENABLE(u)	(0x108 + 0x10 * (u))
  #define OMAP4_MAILBOX_IRQENABLE_CLR(u)	(0x10c + 0x10 * (u))
  
  #define MAILBOX_IRQSTATUS(type, u)	(type ? OMAP4_MAILBOX_IRQSTATUS(u) : \
  						OMAP2_MAILBOX_IRQSTATUS(u))
  #define MAILBOX_IRQENABLE(type, u)	(type ? OMAP4_MAILBOX_IRQENABLE(u) : \
  						OMAP2_MAILBOX_IRQENABLE(u))
  #define MAILBOX_IRQDISABLE(type, u)	(type ? OMAP4_MAILBOX_IRQENABLE_CLR(u) \
  						: OMAP2_MAILBOX_IRQENABLE(u))
  
  #define MAILBOX_IRQ_NEWMSG(m)		(1 << (2 * (m)))
  #define MAILBOX_IRQ_NOTFULL(m)		(1 << (2 * (m) + 1))
4899f78a3   Suman Anna   mailbox/omap: dro...
47
48
49
  /* Interrupt register configuration types */
  #define MBOX_INTR_CFG_TYPE1		0
  #define MBOX_INTR_CFG_TYPE2		1
5040f5343   Suman Anna   mailbox/omap: con...
50
51
52
53
  struct omap_mbox_fifo {
  	unsigned long msg;
  	unsigned long fifo_stat;
  	unsigned long msg_stat;
5040f5343   Suman Anna   mailbox/omap: con...
54
55
  	unsigned long irqenable;
  	unsigned long irqstatus;
5040f5343   Suman Anna   mailbox/omap: con...
56
  	unsigned long irqdisable;
be3322eb7   Suman Anna   mailbox/omap: rem...
57
  	u32 intr_bit;
5040f5343   Suman Anna   mailbox/omap: con...
58
59
60
61
62
63
  };
  
  struct omap_mbox_queue {
  	spinlock_t		lock;
  	struct kfifo		fifo;
  	struct work_struct	work;
5040f5343   Suman Anna   mailbox/omap: con...
64
65
66
  	struct omap_mbox	*mbox;
  	bool full;
  };
ea2ec1e80   Suman Anna   mailbox/omap: use...
67
68
69
  struct omap_mbox_match_data {
  	u32 intr_type;
  };
72c1c8179   Suman Anna   mailbox/omap: add...
70
71
72
73
  struct omap_mbox_device {
  	struct device *dev;
  	struct mutex cfg_lock;
  	void __iomem *mbox_base;
af1d2f5cb   Suman Anna   mailbox/omap: add...
74
  	u32 *irq_ctx;
72c1c8179   Suman Anna   mailbox/omap: add...
75
76
  	u32 num_users;
  	u32 num_fifos;
2240f8aef   Suman Anna   mailbox/omap: sto...
77
  	u32 intr_type;
72c1c8179   Suman Anna   mailbox/omap: add...
78
  	struct omap_mbox **mboxes;
8841a66aa   Suman Anna   mailbox/omap: ada...
79
  	struct mbox_controller controller;
72c1c8179   Suman Anna   mailbox/omap: add...
80
81
  	struct list_head elem;
  };
75288cc66   Suman Anna   mailbox/omap: add...
82
83
84
85
86
87
88
89
90
91
  struct omap_mbox_fifo_info {
  	int tx_id;
  	int tx_usr;
  	int tx_irq;
  
  	int rx_id;
  	int rx_usr;
  	int rx_irq;
  
  	const char *name;
8e3c59521   Dave Gerlach   mailbox/omap: Add...
92
  	bool send_no_irq;
75288cc66   Suman Anna   mailbox/omap: add...
93
  };
5040f5343   Suman Anna   mailbox/omap: con...
94
95
96
  struct omap_mbox {
  	const char		*name;
  	int			irq;
8841a66aa   Suman Anna   mailbox/omap: ada...
97
  	struct omap_mbox_queue	*rxq;
5040f5343   Suman Anna   mailbox/omap: con...
98
  	struct device		*dev;
72c1c8179   Suman Anna   mailbox/omap: add...
99
  	struct omap_mbox_device *parent;
be3322eb7   Suman Anna   mailbox/omap: rem...
100
101
  	struct omap_mbox_fifo	tx_fifo;
  	struct omap_mbox_fifo	rx_fifo;
be3322eb7   Suman Anna   mailbox/omap: rem...
102
  	u32			intr_type;
8841a66aa   Suman Anna   mailbox/omap: ada...
103
  	struct mbox_chan	*chan;
8e3c59521   Dave Gerlach   mailbox/omap: Add...
104
  	bool			send_no_irq;
5040f5343   Suman Anna   mailbox/omap: con...
105
  };
72c1c8179   Suman Anna   mailbox/omap: add...
106
107
108
  /* global variables for the mailbox devices */
  static DEFINE_MUTEX(omap_mbox_devices_lock);
  static LIST_HEAD(omap_mbox_devices);
5f00ec64a   C A Subramaniam   omap: mailbox: Ad...
109

b5bebe410   Ohad Ben-Cohen   omap: mailbox: co...
110
111
112
  static unsigned int mbox_kfifo_size = CONFIG_OMAP_MBOX_KFIFO_SIZE;
  module_param(mbox_kfifo_size, uint, S_IRUGO);
  MODULE_PARM_DESC(mbox_kfifo_size, "Size of omap's mailbox kfifo (bytes)");
8841a66aa   Suman Anna   mailbox/omap: ada...
113
114
115
116
117
118
119
  static struct omap_mbox *mbox_chan_to_omap_mbox(struct mbox_chan *chan)
  {
  	if (!chan || !chan->con_priv)
  		return NULL;
  
  	return (struct omap_mbox *)chan->con_priv;
  }
72c1c8179   Suman Anna   mailbox/omap: add...
120
121
  static inline
  unsigned int mbox_read_reg(struct omap_mbox_device *mdev, size_t ofs)
5040f5343   Suman Anna   mailbox/omap: con...
122
  {
72c1c8179   Suman Anna   mailbox/omap: add...
123
  	return __raw_readl(mdev->mbox_base + ofs);
5040f5343   Suman Anna   mailbox/omap: con...
124
  }
72c1c8179   Suman Anna   mailbox/omap: add...
125
126
  static inline
  void mbox_write_reg(struct omap_mbox_device *mdev, u32 val, size_t ofs)
5040f5343   Suman Anna   mailbox/omap: con...
127
  {
72c1c8179   Suman Anna   mailbox/omap: add...
128
  	__raw_writel(val, mdev->mbox_base + ofs);
5040f5343   Suman Anna   mailbox/omap: con...
129
  }
9ae0ee007   Hiroshi DOYU   omap mailbox: mov...
130
  /* Mailbox FIFO handle functions */
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
131
  static u32 mbox_fifo_read(struct omap_mbox *mbox)
9ae0ee007   Hiroshi DOYU   omap mailbox: mov...
132
  {
be3322eb7   Suman Anna   mailbox/omap: rem...
133
  	struct omap_mbox_fifo *fifo = &mbox->rx_fifo;
2665a4c1d   Suman Anna   mailbox/omap: add...
134

9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
135
  	return mbox_read_reg(mbox->parent, fifo->msg);
9ae0ee007   Hiroshi DOYU   omap mailbox: mov...
136
  }
5040f5343   Suman Anna   mailbox/omap: con...
137

9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
138
  static void mbox_fifo_write(struct omap_mbox *mbox, u32 msg)
9ae0ee007   Hiroshi DOYU   omap mailbox: mov...
139
  {
be3322eb7   Suman Anna   mailbox/omap: rem...
140
  	struct omap_mbox_fifo *fifo = &mbox->tx_fifo;
2665a4c1d   Suman Anna   mailbox/omap: add...
141

72c1c8179   Suman Anna   mailbox/omap: add...
142
  	mbox_write_reg(mbox->parent, msg, fifo->msg);
9ae0ee007   Hiroshi DOYU   omap mailbox: mov...
143
  }
5040f5343   Suman Anna   mailbox/omap: con...
144
145
  
  static int mbox_fifo_empty(struct omap_mbox *mbox)
9ae0ee007   Hiroshi DOYU   omap mailbox: mov...
146
  {
be3322eb7   Suman Anna   mailbox/omap: rem...
147
  	struct omap_mbox_fifo *fifo = &mbox->rx_fifo;
2665a4c1d   Suman Anna   mailbox/omap: add...
148

72c1c8179   Suman Anna   mailbox/omap: add...
149
  	return (mbox_read_reg(mbox->parent, fifo->msg_stat) == 0);
9ae0ee007   Hiroshi DOYU   omap mailbox: mov...
150
  }
5040f5343   Suman Anna   mailbox/omap: con...
151
152
  
  static int mbox_fifo_full(struct omap_mbox *mbox)
9ae0ee007   Hiroshi DOYU   omap mailbox: mov...
153
  {
be3322eb7   Suman Anna   mailbox/omap: rem...
154
  	struct omap_mbox_fifo *fifo = &mbox->tx_fifo;
2665a4c1d   Suman Anna   mailbox/omap: add...
155

72c1c8179   Suman Anna   mailbox/omap: add...
156
  	return mbox_read_reg(mbox->parent, fifo->fifo_stat);
9ae0ee007   Hiroshi DOYU   omap mailbox: mov...
157
158
159
  }
  
  /* Mailbox IRQ handle functions */
5040f5343   Suman Anna   mailbox/omap: con...
160
  static void ack_mbox_irq(struct omap_mbox *mbox, omap_mbox_irq_t irq)
9ae0ee007   Hiroshi DOYU   omap mailbox: mov...
161
  {
be3322eb7   Suman Anna   mailbox/omap: rem...
162
163
164
165
  	struct omap_mbox_fifo *fifo = (irq == IRQ_TX) ?
  				&mbox->tx_fifo : &mbox->rx_fifo;
  	u32 bit = fifo->intr_bit;
  	u32 irqstatus = fifo->irqstatus;
5040f5343   Suman Anna   mailbox/omap: con...
166

72c1c8179   Suman Anna   mailbox/omap: add...
167
  	mbox_write_reg(mbox->parent, bit, irqstatus);
5040f5343   Suman Anna   mailbox/omap: con...
168
169
  
  	/* Flush posted write for irq status to avoid spurious interrupts */
72c1c8179   Suman Anna   mailbox/omap: add...
170
  	mbox_read_reg(mbox->parent, irqstatus);
9ae0ee007   Hiroshi DOYU   omap mailbox: mov...
171
  }
5040f5343   Suman Anna   mailbox/omap: con...
172
173
  
  static int is_mbox_irq(struct omap_mbox *mbox, omap_mbox_irq_t irq)
9ae0ee007   Hiroshi DOYU   omap mailbox: mov...
174
  {
be3322eb7   Suman Anna   mailbox/omap: rem...
175
176
177
178
179
  	struct omap_mbox_fifo *fifo = (irq == IRQ_TX) ?
  				&mbox->tx_fifo : &mbox->rx_fifo;
  	u32 bit = fifo->intr_bit;
  	u32 irqenable = fifo->irqenable;
  	u32 irqstatus = fifo->irqstatus;
72c1c8179   Suman Anna   mailbox/omap: add...
180
181
  	u32 enable = mbox_read_reg(mbox->parent, irqenable);
  	u32 status = mbox_read_reg(mbox->parent, irqstatus);
5040f5343   Suman Anna   mailbox/omap: con...
182
183
  
  	return (int)(enable & status & bit);
9ae0ee007   Hiroshi DOYU   omap mailbox: mov...
184
  }
8841a66aa   Suman Anna   mailbox/omap: ada...
185
  static void _omap_mbox_enable_irq(struct omap_mbox *mbox, omap_mbox_irq_t irq)
c869c75c1   Suman Anna   mailbox/omap: mov...
186
  {
be3322eb7   Suman Anna   mailbox/omap: rem...
187
188
189
190
191
  	u32 l;
  	struct omap_mbox_fifo *fifo = (irq == IRQ_TX) ?
  				&mbox->tx_fifo : &mbox->rx_fifo;
  	u32 bit = fifo->intr_bit;
  	u32 irqenable = fifo->irqenable;
5040f5343   Suman Anna   mailbox/omap: con...
192

72c1c8179   Suman Anna   mailbox/omap: add...
193
  	l = mbox_read_reg(mbox->parent, irqenable);
5040f5343   Suman Anna   mailbox/omap: con...
194
  	l |= bit;
72c1c8179   Suman Anna   mailbox/omap: add...
195
  	mbox_write_reg(mbox->parent, l, irqenable);
c869c75c1   Suman Anna   mailbox/omap: mov...
196
  }
c869c75c1   Suman Anna   mailbox/omap: mov...
197

8841a66aa   Suman Anna   mailbox/omap: ada...
198
  static void _omap_mbox_disable_irq(struct omap_mbox *mbox, omap_mbox_irq_t irq)
c869c75c1   Suman Anna   mailbox/omap: mov...
199
  {
be3322eb7   Suman Anna   mailbox/omap: rem...
200
201
202
203
  	struct omap_mbox_fifo *fifo = (irq == IRQ_TX) ?
  				&mbox->tx_fifo : &mbox->rx_fifo;
  	u32 bit = fifo->intr_bit;
  	u32 irqdisable = fifo->irqdisable;
5040f5343   Suman Anna   mailbox/omap: con...
204
205
206
207
208
  
  	/*
  	 * Read and update the interrupt configuration register for pre-OMAP4.
  	 * OMAP4 and later SoCs have a dedicated interrupt disabling register.
  	 */
be3322eb7   Suman Anna   mailbox/omap: rem...
209
  	if (!mbox->intr_type)
72c1c8179   Suman Anna   mailbox/omap: add...
210
  		bit = mbox_read_reg(mbox->parent, irqdisable) & ~bit;
5040f5343   Suman Anna   mailbox/omap: con...
211

72c1c8179   Suman Anna   mailbox/omap: add...
212
  	mbox_write_reg(mbox->parent, bit, irqdisable);
c869c75c1   Suman Anna   mailbox/omap: mov...
213
  }
c869c75c1   Suman Anna   mailbox/omap: mov...
214

8841a66aa   Suman Anna   mailbox/omap: ada...
215
  void omap_mbox_enable_irq(struct mbox_chan *chan, omap_mbox_irq_t irq)
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
216
  {
8841a66aa   Suman Anna   mailbox/omap: ada...
217
  	struct omap_mbox *mbox = mbox_chan_to_omap_mbox(chan);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
218

8841a66aa   Suman Anna   mailbox/omap: ada...
219
220
  	if (WARN_ON(!mbox))
  		return;
b5bebe410   Ohad Ben-Cohen   omap: mailbox: co...
221

8841a66aa   Suman Anna   mailbox/omap: ada...
222
223
224
  	_omap_mbox_enable_irq(mbox, irq);
  }
  EXPORT_SYMBOL(omap_mbox_enable_irq);
b5bebe410   Ohad Ben-Cohen   omap: mailbox: co...
225

8841a66aa   Suman Anna   mailbox/omap: ada...
226
227
228
229
230
231
232
233
  void omap_mbox_disable_irq(struct mbox_chan *chan, omap_mbox_irq_t irq)
  {
  	struct omap_mbox *mbox = mbox_chan_to_omap_mbox(chan);
  
  	if (WARN_ON(!mbox))
  		return;
  
  	_omap_mbox_disable_irq(mbox, irq);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
234
  }
8841a66aa   Suman Anna   mailbox/omap: ada...
235
  EXPORT_SYMBOL(omap_mbox_disable_irq);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
236
237
238
239
240
241
242
243
  
  /*
   * Message receiver(workqueue)
   */
  static void mbox_rx_work(struct work_struct *work)
  {
  	struct omap_mbox_queue *mq =
  			container_of(work, struct omap_mbox_queue, work);
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
244
245
  	mbox_msg_t data;
  	u32 msg;
b5bebe410   Ohad Ben-Cohen   omap: mailbox: co...
246
247
248
249
250
  	int len;
  
  	while (kfifo_len(&mq->fifo) >= sizeof(msg)) {
  		len = kfifo_out(&mq->fifo, (unsigned char *)&msg, sizeof(msg));
  		WARN_ON(len != sizeof(msg));
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
251
  		data = msg;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
252

9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
253
  		mbox_chan_received_data(mq->mbox->chan, (void *)data);
d2295042b   Fernando Guzman Lugo   OMAP: mailbox: ch...
254
255
256
  		spin_lock_irq(&mq->lock);
  		if (mq->full) {
  			mq->full = false;
8841a66aa   Suman Anna   mailbox/omap: ada...
257
  			_omap_mbox_enable_irq(mq->mbox, IRQ_RX);
d2295042b   Fernando Guzman Lugo   OMAP: mailbox: ch...
258
259
  		}
  		spin_unlock_irq(&mq->lock);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
260
261
262
263
264
265
  	}
  }
  
  /*
   * Mailbox interrupt handler
   */
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
266
267
  static void __mbox_tx_interrupt(struct omap_mbox *mbox)
  {
8841a66aa   Suman Anna   mailbox/omap: ada...
268
  	_omap_mbox_disable_irq(mbox, IRQ_TX);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
269
  	ack_mbox_irq(mbox, IRQ_TX);
8841a66aa   Suman Anna   mailbox/omap: ada...
270
  	mbox_chan_txdone(mbox->chan, 0);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
271
272
273
274
  }
  
  static void __mbox_rx_interrupt(struct omap_mbox *mbox)
  {
b5bebe410   Ohad Ben-Cohen   omap: mailbox: co...
275
  	struct omap_mbox_queue *mq = mbox->rxq;
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
276
  	u32 msg;
b5bebe410   Ohad Ben-Cohen   omap: mailbox: co...
277
  	int len;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
278

340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
279
  	while (!mbox_fifo_empty(mbox)) {
b5bebe410   Ohad Ben-Cohen   omap: mailbox: co...
280
  		if (unlikely(kfifo_avail(&mq->fifo) < sizeof(msg))) {
8841a66aa   Suman Anna   mailbox/omap: ada...
281
  			_omap_mbox_disable_irq(mbox, IRQ_RX);
d2295042b   Fernando Guzman Lugo   OMAP: mailbox: ch...
282
  			mq->full = true;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
283
  			goto nomem;
1ea5d6d18   Fernando Guzman Lugo   Mailbox: disable ...
284
  		}
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
285
286
  
  		msg = mbox_fifo_read(mbox);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
287

b5bebe410   Ohad Ben-Cohen   omap: mailbox: co...
288
289
  		len = kfifo_in(&mq->fifo, (unsigned char *)&msg, sizeof(msg));
  		WARN_ON(len != sizeof(msg));
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
290
291
292
293
  	}
  
  	/* no more messages in the fifo. clear IRQ source. */
  	ack_mbox_irq(mbox, IRQ_RX);
f48cca877   Hiroshi DOYU   omap mailbox: fix...
294
  nomem:
c48730056   Tejun Heo   arm/omap: use sys...
295
  	schedule_work(&mbox->rxq->work);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
296
297
298
299
  }
  
  static irqreturn_t mbox_interrupt(int irq, void *p)
  {
2a7057e30   Jeff Garzik   [ARM] Remove poin...
300
  	struct omap_mbox *mbox = p;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
301
302
303
304
305
306
307
308
309
  
  	if (is_mbox_irq(mbox, IRQ_TX))
  		__mbox_tx_interrupt(mbox);
  
  	if (is_mbox_irq(mbox, IRQ_RX))
  		__mbox_rx_interrupt(mbox);
  
  	return IRQ_HANDLED;
  }
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
310
  static struct omap_mbox_queue *mbox_queue_alloc(struct omap_mbox *mbox,
8841a66aa   Suman Anna   mailbox/omap: ada...
311
  					void (*work)(struct work_struct *))
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
312
  {
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
313
  	struct omap_mbox_queue *mq;
8841a66aa   Suman Anna   mailbox/omap: ada...
314
315
  	if (!work)
  		return NULL;
86f6f5e2e   Suman Anna   mailbox/omap: use...
316
  	mq = kzalloc(sizeof(*mq), GFP_KERNEL);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
317
318
319
320
  	if (!mq)
  		return NULL;
  
  	spin_lock_init(&mq->lock);
b5bebe410   Ohad Ben-Cohen   omap: mailbox: co...
321
  	if (kfifo_alloc(&mq->fifo, mbox_kfifo_size, GFP_KERNEL))
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
322
  		goto error;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
323

8841a66aa   Suman Anna   mailbox/omap: ada...
324
  	INIT_WORK(&mq->work, work);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
325
  	return mq;
8841a66aa   Suman Anna   mailbox/omap: ada...
326

340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
327
328
329
330
331
332
333
  error:
  	kfree(mq);
  	return NULL;
  }
  
  static void mbox_queue_free(struct omap_mbox_queue *q)
  {
b5bebe410   Ohad Ben-Cohen   omap: mailbox: co...
334
  	kfifo_free(&q->fifo);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
335
336
  	kfree(q);
  }
c7c158e57   Hiroshi DOYU   omap: mailbox: re...
337
  static int omap_mbox_startup(struct omap_mbox *mbox)
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
338
  {
5f00ec64a   C A Subramaniam   omap: mailbox: Ad...
339
  	int ret = 0;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
340
  	struct omap_mbox_queue *mq;
8841a66aa   Suman Anna   mailbox/omap: ada...
341
342
343
344
345
346
347
348
349
350
351
352
353
  	mq = mbox_queue_alloc(mbox, mbox_rx_work);
  	if (!mq)
  		return -ENOMEM;
  	mbox->rxq = mq;
  	mq->mbox = mbox;
  
  	ret = request_irq(mbox->irq, mbox_interrupt, IRQF_SHARED,
  			  mbox->name, mbox);
  	if (unlikely(ret)) {
  		pr_err("failed to register mailbox interrupt:%d
  ", ret);
  		goto fail_request_irq;
  	}
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
354

8e3c59521   Dave Gerlach   mailbox/omap: Add...
355
356
  	if (mbox->send_no_irq)
  		mbox->chan->txdone_method = TXDONE_BY_ACK;
8841a66aa   Suman Anna   mailbox/omap: ada...
357
  	_omap_mbox_enable_irq(mbox, IRQ_RX);
1d8a0e963   Juan Gutierrez   ARM: OMAP: enable...
358

340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
359
  	return 0;
ecf305cf8   Suman Anna   omap: mailbox: ca...
360
361
  fail_request_irq:
  	mbox_queue_free(mbox->rxq);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
362
363
364
365
366
  	return ret;
  }
  
  static void omap_mbox_fini(struct omap_mbox *mbox)
  {
8841a66aa   Suman Anna   mailbox/omap: ada...
367
368
369
370
  	_omap_mbox_disable_irq(mbox, IRQ_RX);
  	free_irq(mbox->irq, mbox);
  	flush_work(&mbox->rxq->work);
  	mbox_queue_free(mbox->rxq);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
371
  }
72c1c8179   Suman Anna   mailbox/omap: add...
372
373
  static struct omap_mbox *omap_mbox_device_find(struct omap_mbox_device *mdev,
  					       const char *mbox_name)
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
374
  {
c03773206   Kevin Hilman   OMAP2+: mailbox: ...
375
  	struct omap_mbox *_mbox, *mbox = NULL;
72c1c8179   Suman Anna   mailbox/omap: add...
376
377
  	struct omap_mbox **mboxes = mdev->mboxes;
  	int i;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
378

9c80c8cd7   Felipe Contreras   omap: mailbox: si...
379
  	if (!mboxes)
72c1c8179   Suman Anna   mailbox/omap: add...
380
  		return NULL;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
381

c03773206   Kevin Hilman   OMAP2+: mailbox: ...
382
  	for (i = 0; (_mbox = mboxes[i]); i++) {
72c1c8179   Suman Anna   mailbox/omap: add...
383
  		if (!strcmp(_mbox->name, mbox_name)) {
c03773206   Kevin Hilman   OMAP2+: mailbox: ...
384
  			mbox = _mbox;
9c80c8cd7   Felipe Contreras   omap: mailbox: si...
385
  			break;
c03773206   Kevin Hilman   OMAP2+: mailbox: ...
386
387
  		}
  	}
72c1c8179   Suman Anna   mailbox/omap: add...
388
389
  	return mbox;
  }
8841a66aa   Suman Anna   mailbox/omap: ada...
390
391
  struct mbox_chan *omap_mbox_request_channel(struct mbox_client *cl,
  					    const char *chan_name)
72c1c8179   Suman Anna   mailbox/omap: add...
392
  {
8841a66aa   Suman Anna   mailbox/omap: ada...
393
  	struct device *dev = cl->dev;
72c1c8179   Suman Anna   mailbox/omap: add...
394
395
  	struct omap_mbox *mbox = NULL;
  	struct omap_mbox_device *mdev;
8841a66aa   Suman Anna   mailbox/omap: ada...
396
397
  	struct mbox_chan *chan;
  	unsigned long flags;
72c1c8179   Suman Anna   mailbox/omap: add...
398
  	int ret;
8841a66aa   Suman Anna   mailbox/omap: ada...
399
400
401
402
403
404
405
406
407
  	if (!dev)
  		return ERR_PTR(-ENODEV);
  
  	if (dev->of_node) {
  		pr_err("%s: please use mbox_request_channel(), this API is supported only for OMAP non-DT usage
  ",
  		       __func__);
  		return ERR_PTR(-ENODEV);
  	}
72c1c8179   Suman Anna   mailbox/omap: add...
408
409
  	mutex_lock(&omap_mbox_devices_lock);
  	list_for_each_entry(mdev, &omap_mbox_devices, elem) {
8841a66aa   Suman Anna   mailbox/omap: ada...
410
  		mbox = omap_mbox_device_find(mdev, chan_name);
72c1c8179   Suman Anna   mailbox/omap: add...
411
412
413
414
  		if (mbox)
  			break;
  	}
  	mutex_unlock(&omap_mbox_devices_lock);
9c80c8cd7   Felipe Contreras   omap: mailbox: si...
415

8841a66aa   Suman Anna   mailbox/omap: ada...
416
  	if (!mbox || !mbox->chan)
9c80c8cd7   Felipe Contreras   omap: mailbox: si...
417
  		return ERR_PTR(-ENOENT);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
418

8841a66aa   Suman Anna   mailbox/omap: ada...
419
420
421
422
423
424
425
426
  	chan = mbox->chan;
  	spin_lock_irqsave(&chan->lock, flags);
  	chan->msg_free = 0;
  	chan->msg_count = 0;
  	chan->active_req = NULL;
  	chan->cl = cl;
  	init_completion(&chan->tx_complete);
  	spin_unlock_irqrestore(&chan->lock, flags);
582563074   Kanigeri, Hari   OMAP: mailbox: ad...
427

8841a66aa   Suman Anna   mailbox/omap: ada...
428
  	ret = chan->mbox->ops->startup(chan);
1d8a0e963   Juan Gutierrez   ARM: OMAP: enable...
429
  	if (ret) {
8841a66aa   Suman Anna   mailbox/omap: ada...
430
431
432
433
  		pr_err("Unable to startup the chan (%d)
  ", ret);
  		mbox_free_channel(chan);
  		chan = ERR_PTR(ret);
1d8a0e963   Juan Gutierrez   ARM: OMAP: enable...
434
  	}
8841a66aa   Suman Anna   mailbox/omap: ada...
435
  	return chan;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
436
  }
8841a66aa   Suman Anna   mailbox/omap: ada...
437
  EXPORT_SYMBOL(omap_mbox_request_channel);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
438

6b2339859   Hiroshi DOYU   omap mailbox: Set...
439
  static struct class omap_mbox_class = { .name = "mbox", };
72c1c8179   Suman Anna   mailbox/omap: add...
440
  static int omap_mbox_register(struct omap_mbox_device *mdev)
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
441
  {
9c80c8cd7   Felipe Contreras   omap: mailbox: si...
442
443
  	int ret;
  	int i;
72c1c8179   Suman Anna   mailbox/omap: add...
444
  	struct omap_mbox **mboxes;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
445

72c1c8179   Suman Anna   mailbox/omap: add...
446
  	if (!mdev || !mdev->mboxes)
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
447
  		return -EINVAL;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
448

72c1c8179   Suman Anna   mailbox/omap: add...
449
  	mboxes = mdev->mboxes;
9c80c8cd7   Felipe Contreras   omap: mailbox: si...
450
451
  	for (i = 0; mboxes[i]; i++) {
  		struct omap_mbox *mbox = mboxes[i];
2665a4c1d   Suman Anna   mailbox/omap: add...
452

8841a66aa   Suman Anna   mailbox/omap: ada...
453
454
  		mbox->dev = device_create(&omap_mbox_class, mdev->dev,
  					0, mbox, "%s", mbox->name);
9c80c8cd7   Felipe Contreras   omap: mailbox: si...
455
456
457
458
459
  		if (IS_ERR(mbox->dev)) {
  			ret = PTR_ERR(mbox->dev);
  			goto err_out;
  		}
  	}
72c1c8179   Suman Anna   mailbox/omap: add...
460
461
462
463
  
  	mutex_lock(&omap_mbox_devices_lock);
  	list_add(&mdev->elem, &omap_mbox_devices);
  	mutex_unlock(&omap_mbox_devices_lock);
a3abf4363   Thierry Reding   mailbox: omap: Us...
464
  	ret = devm_mbox_controller_register(mdev->dev, &mdev->controller);
f48cca877   Hiroshi DOYU   omap mailbox: fix...
465

9c80c8cd7   Felipe Contreras   omap: mailbox: si...
466
  err_out:
8841a66aa   Suman Anna   mailbox/omap: ada...
467
468
469
470
  	if (ret) {
  		while (i--)
  			device_unregister(mboxes[i]->dev);
  	}
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
471
472
  	return ret;
  }
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
473

72c1c8179   Suman Anna   mailbox/omap: add...
474
  static int omap_mbox_unregister(struct omap_mbox_device *mdev)
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
475
  {
9c80c8cd7   Felipe Contreras   omap: mailbox: si...
476
  	int i;
72c1c8179   Suman Anna   mailbox/omap: add...
477
  	struct omap_mbox **mboxes;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
478

72c1c8179   Suman Anna   mailbox/omap: add...
479
  	if (!mdev || !mdev->mboxes)
9c80c8cd7   Felipe Contreras   omap: mailbox: si...
480
  		return -EINVAL;
72c1c8179   Suman Anna   mailbox/omap: add...
481
482
483
484
485
  	mutex_lock(&omap_mbox_devices_lock);
  	list_del(&mdev->elem);
  	mutex_unlock(&omap_mbox_devices_lock);
  
  	mboxes = mdev->mboxes;
9c80c8cd7   Felipe Contreras   omap: mailbox: si...
486
487
  	for (i = 0; mboxes[i]; i++)
  		device_unregister(mboxes[i]->dev);
9c80c8cd7   Felipe Contreras   omap: mailbox: si...
488
  	return 0;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
489
  }
5040f5343   Suman Anna   mailbox/omap: con...
490

8841a66aa   Suman Anna   mailbox/omap: ada...
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
  static int omap_mbox_chan_startup(struct mbox_chan *chan)
  {
  	struct omap_mbox *mbox = mbox_chan_to_omap_mbox(chan);
  	struct omap_mbox_device *mdev = mbox->parent;
  	int ret = 0;
  
  	mutex_lock(&mdev->cfg_lock);
  	pm_runtime_get_sync(mdev->dev);
  	ret = omap_mbox_startup(mbox);
  	if (ret)
  		pm_runtime_put_sync(mdev->dev);
  	mutex_unlock(&mdev->cfg_lock);
  	return ret;
  }
  
  static void omap_mbox_chan_shutdown(struct mbox_chan *chan)
  {
  	struct omap_mbox *mbox = mbox_chan_to_omap_mbox(chan);
  	struct omap_mbox_device *mdev = mbox->parent;
  
  	mutex_lock(&mdev->cfg_lock);
  	omap_mbox_fini(mbox);
  	pm_runtime_put_sync(mdev->dev);
  	mutex_unlock(&mdev->cfg_lock);
  }
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
516
  static int omap_mbox_chan_send_noirq(struct omap_mbox *mbox, u32 msg)
8841a66aa   Suman Anna   mailbox/omap: ada...
517
  {
8841a66aa   Suman Anna   mailbox/omap: ada...
518
  	int ret = -EBUSY;
8e3c59521   Dave Gerlach   mailbox/omap: Add...
519
520
  	if (!mbox_fifo_full(mbox)) {
  		_omap_mbox_enable_irq(mbox, IRQ_RX);
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
521
  		mbox_fifo_write(mbox, msg);
8e3c59521   Dave Gerlach   mailbox/omap: Add...
522
523
524
525
526
527
528
529
530
531
  		ret = 0;
  		_omap_mbox_disable_irq(mbox, IRQ_RX);
  
  		/* we must read and ack the interrupt directly from here */
  		mbox_fifo_read(mbox);
  		ack_mbox_irq(mbox, IRQ_RX);
  	}
  
  	return ret;
  }
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
532
  static int omap_mbox_chan_send(struct omap_mbox *mbox, u32 msg)
8e3c59521   Dave Gerlach   mailbox/omap: Add...
533
534
  {
  	int ret = -EBUSY;
8841a66aa   Suman Anna   mailbox/omap: ada...
535
536
  
  	if (!mbox_fifo_full(mbox)) {
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
537
  		mbox_fifo_write(mbox, msg);
8841a66aa   Suman Anna   mailbox/omap: ada...
538
539
540
541
542
543
544
  		ret = 0;
  	}
  
  	/* always enable the interrupt */
  	_omap_mbox_enable_irq(mbox, IRQ_TX);
  	return ret;
  }
8e3c59521   Dave Gerlach   mailbox/omap: Add...
545
546
547
548
  static int omap_mbox_chan_send_data(struct mbox_chan *chan, void *data)
  {
  	struct omap_mbox *mbox = mbox_chan_to_omap_mbox(chan);
  	int ret;
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
549
  	u32 msg = omap_mbox_message(data);
8e3c59521   Dave Gerlach   mailbox/omap: Add...
550
551
552
553
554
  
  	if (!mbox)
  		return -EINVAL;
  
  	if (mbox->send_no_irq)
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
555
  		ret = omap_mbox_chan_send_noirq(mbox, msg);
8e3c59521   Dave Gerlach   mailbox/omap: Add...
556
  	else
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
557
  		ret = omap_mbox_chan_send(mbox, msg);
8e3c59521   Dave Gerlach   mailbox/omap: Add...
558
559
560
  
  	return ret;
  }
05ae79756   Andrew Bresticker   mailbox: Make mbo...
561
  static const struct mbox_chan_ops omap_mbox_chan_ops = {
8841a66aa   Suman Anna   mailbox/omap: ada...
562
563
564
565
  	.startup        = omap_mbox_chan_startup,
  	.send_data      = omap_mbox_chan_send_data,
  	.shutdown       = omap_mbox_chan_shutdown,
  };
af1d2f5cb   Suman Anna   mailbox/omap: add...
566
567
568
569
  #ifdef CONFIG_PM_SLEEP
  static int omap_mbox_suspend(struct device *dev)
  {
  	struct omap_mbox_device *mdev = dev_get_drvdata(dev);
9f0cee984   Suman Anna   mailbox/omap: che...
570
  	u32 usr, fifo, reg;
af1d2f5cb   Suman Anna   mailbox/omap: add...
571
572
573
  
  	if (pm_runtime_status_suspended(dev))
  		return 0;
9f0cee984   Suman Anna   mailbox/omap: che...
574
575
576
577
578
579
580
581
  	for (fifo = 0; fifo < mdev->num_fifos; fifo++) {
  		if (mbox_read_reg(mdev, MAILBOX_MSGSTATUS(fifo))) {
  			dev_err(mdev->dev, "fifo %d has unexpected unread messages
  ",
  				fifo);
  			return -EBUSY;
  		}
  	}
af1d2f5cb   Suman Anna   mailbox/omap: add...
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
  	for (usr = 0; usr < mdev->num_users; usr++) {
  		reg = MAILBOX_IRQENABLE(mdev->intr_type, usr);
  		mdev->irq_ctx[usr] = mbox_read_reg(mdev, reg);
  	}
  
  	return 0;
  }
  
  static int omap_mbox_resume(struct device *dev)
  {
  	struct omap_mbox_device *mdev = dev_get_drvdata(dev);
  	u32 usr, reg;
  
  	if (pm_runtime_status_suspended(dev))
  		return 0;
  
  	for (usr = 0; usr < mdev->num_users; usr++) {
  		reg = MAILBOX_IRQENABLE(mdev->intr_type, usr);
  		mbox_write_reg(mdev, mdev->irq_ctx[usr], reg);
  	}
  
  	return 0;
  }
  #endif
  
  static const struct dev_pm_ops omap_mbox_pm_ops = {
  	SET_SYSTEM_SLEEP_PM_OPS(omap_mbox_suspend, omap_mbox_resume)
  };
ea2ec1e80   Suman Anna   mailbox/omap: use...
610
611
  static const struct omap_mbox_match_data omap2_data = { MBOX_INTR_CFG_TYPE1 };
  static const struct omap_mbox_match_data omap4_data = { MBOX_INTR_CFG_TYPE2 };
75288cc66   Suman Anna   mailbox/omap: add...
612
613
614
  static const struct of_device_id omap_mailbox_of_match[] = {
  	{
  		.compatible	= "ti,omap2-mailbox",
ea2ec1e80   Suman Anna   mailbox/omap: use...
615
  		.data		= &omap2_data,
75288cc66   Suman Anna   mailbox/omap: add...
616
617
618
  	},
  	{
  		.compatible	= "ti,omap3-mailbox",
ea2ec1e80   Suman Anna   mailbox/omap: use...
619
  		.data		= &omap2_data,
75288cc66   Suman Anna   mailbox/omap: add...
620
621
622
  	},
  	{
  		.compatible	= "ti,omap4-mailbox",
ea2ec1e80   Suman Anna   mailbox/omap: use...
623
  		.data		= &omap4_data,
75288cc66   Suman Anna   mailbox/omap: add...
624
625
  	},
  	{
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
626
627
628
629
  		.compatible	= "ti,am654-mailbox",
  		.data		= &omap4_data,
  	},
  	{
75288cc66   Suman Anna   mailbox/omap: add...
630
631
632
633
  		/* end */
  	},
  };
  MODULE_DEVICE_TABLE(of, omap_mailbox_of_match);
8841a66aa   Suman Anna   mailbox/omap: ada...
634
635
636
637
638
639
640
641
642
643
  static struct mbox_chan *omap_mbox_of_xlate(struct mbox_controller *controller,
  					    const struct of_phandle_args *sp)
  {
  	phandle phandle = sp->args[0];
  	struct device_node *node;
  	struct omap_mbox_device *mdev;
  	struct omap_mbox *mbox;
  
  	mdev = container_of(controller, struct omap_mbox_device, controller);
  	if (WARN_ON(!mdev))
2d805fc1c   Benson Leung   mailbox: Fix up e...
644
  		return ERR_PTR(-EINVAL);
8841a66aa   Suman Anna   mailbox/omap: ada...
645
646
647
648
649
650
  
  	node = of_find_node_by_phandle(phandle);
  	if (!node) {
  		pr_err("%s: could not find node phandle 0x%x
  ",
  		       __func__, phandle);
2d805fc1c   Benson Leung   mailbox: Fix up e...
651
  		return ERR_PTR(-ENODEV);
8841a66aa   Suman Anna   mailbox/omap: ada...
652
653
654
655
  	}
  
  	mbox = omap_mbox_device_find(mdev, node->name);
  	of_node_put(node);
2d805fc1c   Benson Leung   mailbox: Fix up e...
656
  	return mbox ? mbox->chan : ERR_PTR(-ENOENT);
8841a66aa   Suman Anna   mailbox/omap: ada...
657
  }
5040f5343   Suman Anna   mailbox/omap: con...
658
659
660
661
  static int omap_mbox_probe(struct platform_device *pdev)
  {
  	struct resource *mem;
  	int ret;
8841a66aa   Suman Anna   mailbox/omap: ada...
662
  	struct mbox_chan *chnls;
5040f5343   Suman Anna   mailbox/omap: con...
663
  	struct omap_mbox **list, *mbox, *mboxblk;
75288cc66   Suman Anna   mailbox/omap: add...
664
  	struct omap_mbox_fifo_info *finfo, *finfoblk;
72c1c8179   Suman Anna   mailbox/omap: add...
665
  	struct omap_mbox_device *mdev;
be3322eb7   Suman Anna   mailbox/omap: rem...
666
  	struct omap_mbox_fifo *fifo;
75288cc66   Suman Anna   mailbox/omap: add...
667
668
  	struct device_node *node = pdev->dev.of_node;
  	struct device_node *child;
ea2ec1e80   Suman Anna   mailbox/omap: use...
669
  	const struct omap_mbox_match_data *match_data;
75288cc66   Suman Anna   mailbox/omap: add...
670
671
672
  	u32 intr_type, info_count;
  	u32 num_users, num_fifos;
  	u32 tmp[3];
5040f5343   Suman Anna   mailbox/omap: con...
673
674
  	u32 l;
  	int i;
4899f78a3   Suman Anna   mailbox/omap: dro...
675
676
677
  	if (!node) {
  		pr_err("%s: only DT-based devices are supported
  ", __func__);
5040f5343   Suman Anna   mailbox/omap: con...
678
679
  		return -ENODEV;
  	}
ea2ec1e80   Suman Anna   mailbox/omap: use...
680
681
  	match_data = of_device_get_match_data(&pdev->dev);
  	if (!match_data)
4899f78a3   Suman Anna   mailbox/omap: dro...
682
  		return -ENODEV;
ea2ec1e80   Suman Anna   mailbox/omap: use...
683
  	intr_type = match_data->intr_type;
75288cc66   Suman Anna   mailbox/omap: add...
684

4899f78a3   Suman Anna   mailbox/omap: dro...
685
686
  	if (of_property_read_u32(node, "ti,mbox-num-users", &num_users))
  		return -ENODEV;
75288cc66   Suman Anna   mailbox/omap: add...
687

4899f78a3   Suman Anna   mailbox/omap: dro...
688
689
  	if (of_property_read_u32(node, "ti,mbox-num-fifos", &num_fifos))
  		return -ENODEV;
75288cc66   Suman Anna   mailbox/omap: add...
690

4899f78a3   Suman Anna   mailbox/omap: dro...
691
692
693
694
695
  	info_count = of_get_available_child_count(node);
  	if (!info_count) {
  		dev_err(&pdev->dev, "no available mbox devices found
  ");
  		return -ENODEV;
75288cc66   Suman Anna   mailbox/omap: add...
696
  	}
a86854d0c   Kees Cook   treewide: devm_kz...
697
  	finfoblk = devm_kcalloc(&pdev->dev, info_count, sizeof(*finfoblk),
75288cc66   Suman Anna   mailbox/omap: add...
698
699
700
701
702
703
704
  				GFP_KERNEL);
  	if (!finfoblk)
  		return -ENOMEM;
  
  	finfo = finfoblk;
  	child = NULL;
  	for (i = 0; i < info_count; i++, finfo++) {
4899f78a3   Suman Anna   mailbox/omap: dro...
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
  		child = of_get_next_available_child(node, child);
  		ret = of_property_read_u32_array(child, "ti,mbox-tx", tmp,
  						 ARRAY_SIZE(tmp));
  		if (ret)
  			return ret;
  		finfo->tx_id = tmp[0];
  		finfo->tx_irq = tmp[1];
  		finfo->tx_usr = tmp[2];
  
  		ret = of_property_read_u32_array(child, "ti,mbox-rx", tmp,
  						 ARRAY_SIZE(tmp));
  		if (ret)
  			return ret;
  		finfo->rx_id = tmp[0];
  		finfo->rx_irq = tmp[1];
  		finfo->rx_usr = tmp[2];
  
  		finfo->name = child->name;
  
  		if (of_find_property(child, "ti,mbox-send-noirq", NULL))
  			finfo->send_no_irq = true;
75288cc66   Suman Anna   mailbox/omap: add...
726
727
728
729
  		if (finfo->tx_id >= num_fifos || finfo->rx_id >= num_fifos ||
  		    finfo->tx_usr >= num_users || finfo->rx_usr >= num_users)
  			return -EINVAL;
  	}
72c1c8179   Suman Anna   mailbox/omap: add...
730
731
732
733
734
735
736
737
  	mdev = devm_kzalloc(&pdev->dev, sizeof(*mdev), GFP_KERNEL);
  	if (!mdev)
  		return -ENOMEM;
  
  	mem = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  	mdev->mbox_base = devm_ioremap_resource(&pdev->dev, mem);
  	if (IS_ERR(mdev->mbox_base))
  		return PTR_ERR(mdev->mbox_base);
a86854d0c   Kees Cook   treewide: devm_kz...
738
  	mdev->irq_ctx = devm_kcalloc(&pdev->dev, num_users, sizeof(u32),
af1d2f5cb   Suman Anna   mailbox/omap: add...
739
740
741
  				     GFP_KERNEL);
  	if (!mdev->irq_ctx)
  		return -ENOMEM;
5040f5343   Suman Anna   mailbox/omap: con...
742
  	/* allocate one extra for marking end of list */
a86854d0c   Kees Cook   treewide: devm_kz...
743
  	list = devm_kcalloc(&pdev->dev, info_count + 1, sizeof(*list),
5040f5343   Suman Anna   mailbox/omap: con...
744
745
746
  			    GFP_KERNEL);
  	if (!list)
  		return -ENOMEM;
a86854d0c   Kees Cook   treewide: devm_kz...
747
  	chnls = devm_kcalloc(&pdev->dev, info_count + 1, sizeof(*chnls),
8841a66aa   Suman Anna   mailbox/omap: ada...
748
749
750
  			     GFP_KERNEL);
  	if (!chnls)
  		return -ENOMEM;
a86854d0c   Kees Cook   treewide: devm_kz...
751
  	mboxblk = devm_kcalloc(&pdev->dev, info_count, sizeof(*mbox),
5040f5343   Suman Anna   mailbox/omap: con...
752
753
754
  			       GFP_KERNEL);
  	if (!mboxblk)
  		return -ENOMEM;
5040f5343   Suman Anna   mailbox/omap: con...
755
  	mbox = mboxblk;
75288cc66   Suman Anna   mailbox/omap: add...
756
757
  	finfo = finfoblk;
  	for (i = 0; i < info_count; i++, finfo++) {
be3322eb7   Suman Anna   mailbox/omap: rem...
758
  		fifo = &mbox->tx_fifo;
75288cc66   Suman Anna   mailbox/omap: add...
759
760
761
762
763
764
  		fifo->msg = MAILBOX_MESSAGE(finfo->tx_id);
  		fifo->fifo_stat = MAILBOX_FIFOSTATUS(finfo->tx_id);
  		fifo->intr_bit = MAILBOX_IRQ_NOTFULL(finfo->tx_id);
  		fifo->irqenable = MAILBOX_IRQENABLE(intr_type, finfo->tx_usr);
  		fifo->irqstatus = MAILBOX_IRQSTATUS(intr_type, finfo->tx_usr);
  		fifo->irqdisable = MAILBOX_IRQDISABLE(intr_type, finfo->tx_usr);
be3322eb7   Suman Anna   mailbox/omap: rem...
765
766
  
  		fifo = &mbox->rx_fifo;
75288cc66   Suman Anna   mailbox/omap: add...
767
768
769
770
771
772
  		fifo->msg = MAILBOX_MESSAGE(finfo->rx_id);
  		fifo->msg_stat =  MAILBOX_MSGSTATUS(finfo->rx_id);
  		fifo->intr_bit = MAILBOX_IRQ_NEWMSG(finfo->rx_id);
  		fifo->irqenable = MAILBOX_IRQENABLE(intr_type, finfo->rx_usr);
  		fifo->irqstatus = MAILBOX_IRQSTATUS(intr_type, finfo->rx_usr);
  		fifo->irqdisable = MAILBOX_IRQDISABLE(intr_type, finfo->rx_usr);
be3322eb7   Suman Anna   mailbox/omap: rem...
773

8e3c59521   Dave Gerlach   mailbox/omap: Add...
774
  		mbox->send_no_irq = finfo->send_no_irq;
be3322eb7   Suman Anna   mailbox/omap: rem...
775
  		mbox->intr_type = intr_type;
72c1c8179   Suman Anna   mailbox/omap: add...
776
  		mbox->parent = mdev;
75288cc66   Suman Anna   mailbox/omap: add...
777
778
  		mbox->name = finfo->name;
  		mbox->irq = platform_get_irq(pdev, finfo->tx_irq);
5040f5343   Suman Anna   mailbox/omap: con...
779
780
  		if (mbox->irq < 0)
  			return mbox->irq;
8841a66aa   Suman Anna   mailbox/omap: ada...
781
782
  		mbox->chan = &chnls[i];
  		chnls[i].con_priv = mbox;
5040f5343   Suman Anna   mailbox/omap: con...
783
784
  		list[i] = mbox++;
  	}
72c1c8179   Suman Anna   mailbox/omap: add...
785
786
  	mutex_init(&mdev->cfg_lock);
  	mdev->dev = &pdev->dev;
75288cc66   Suman Anna   mailbox/omap: add...
787
788
  	mdev->num_users = num_users;
  	mdev->num_fifos = num_fifos;
2240f8aef   Suman Anna   mailbox/omap: sto...
789
  	mdev->intr_type = intr_type;
72c1c8179   Suman Anna   mailbox/omap: add...
790
  	mdev->mboxes = list;
8841a66aa   Suman Anna   mailbox/omap: ada...
791

9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
792
793
794
795
  	/*
  	 * OMAP/K3 Mailbox IP does not have a Tx-Done IRQ, but rather a Tx-Ready
  	 * IRQ and is needed to run the Tx state machine
  	 */
8841a66aa   Suman Anna   mailbox/omap: ada...
796
797
798
799
800
801
  	mdev->controller.txdone_irq = true;
  	mdev->controller.dev = mdev->dev;
  	mdev->controller.ops = &omap_mbox_chan_ops;
  	mdev->controller.chans = chnls;
  	mdev->controller.num_chans = info_count;
  	mdev->controller.of_xlate = omap_mbox_of_xlate;
72c1c8179   Suman Anna   mailbox/omap: add...
802
  	ret = omap_mbox_register(mdev);
5040f5343   Suman Anna   mailbox/omap: con...
803
804
  	if (ret)
  		return ret;
72c1c8179   Suman Anna   mailbox/omap: add...
805
806
  	platform_set_drvdata(pdev, mdev);
  	pm_runtime_enable(mdev->dev);
5040f5343   Suman Anna   mailbox/omap: con...
807

72c1c8179   Suman Anna   mailbox/omap: add...
808
  	ret = pm_runtime_get_sync(mdev->dev);
5040f5343   Suman Anna   mailbox/omap: con...
809
  	if (ret < 0) {
72c1c8179   Suman Anna   mailbox/omap: add...
810
  		pm_runtime_put_noidle(mdev->dev);
5040f5343   Suman Anna   mailbox/omap: con...
811
812
813
814
815
816
817
  		goto unregister;
  	}
  
  	/*
  	 * just print the raw revision register, the format is not
  	 * uniform across all SoCs
  	 */
72c1c8179   Suman Anna   mailbox/omap: add...
818
819
820
  	l = mbox_read_reg(mdev, MAILBOX_REVISION);
  	dev_info(mdev->dev, "omap mailbox rev 0x%x
  ", l);
5040f5343   Suman Anna   mailbox/omap: con...
821

72c1c8179   Suman Anna   mailbox/omap: add...
822
  	ret = pm_runtime_put_sync(mdev->dev);
5040f5343   Suman Anna   mailbox/omap: con...
823
824
  	if (ret < 0)
  		goto unregister;
75288cc66   Suman Anna   mailbox/omap: add...
825
  	devm_kfree(&pdev->dev, finfoblk);
5040f5343   Suman Anna   mailbox/omap: con...
826
827
828
  	return 0;
  
  unregister:
72c1c8179   Suman Anna   mailbox/omap: add...
829
830
  	pm_runtime_disable(mdev->dev);
  	omap_mbox_unregister(mdev);
5040f5343   Suman Anna   mailbox/omap: con...
831
832
833
834
835
  	return ret;
  }
  
  static int omap_mbox_remove(struct platform_device *pdev)
  {
72c1c8179   Suman Anna   mailbox/omap: add...
836
837
838
839
  	struct omap_mbox_device *mdev = platform_get_drvdata(pdev);
  
  	pm_runtime_disable(mdev->dev);
  	omap_mbox_unregister(mdev);
5040f5343   Suman Anna   mailbox/omap: con...
840
841
842
843
844
845
846
847
848
  
  	return 0;
  }
  
  static struct platform_driver omap_mbox_driver = {
  	.probe	= omap_mbox_probe,
  	.remove	= omap_mbox_remove,
  	.driver	= {
  		.name = "omap-mailbox",
af1d2f5cb   Suman Anna   mailbox/omap: add...
849
  		.pm = &omap_mbox_pm_ops,
75288cc66   Suman Anna   mailbox/omap: add...
850
  		.of_match_table = of_match_ptr(omap_mailbox_of_match),
5040f5343   Suman Anna   mailbox/omap: con...
851
852
  	},
  };
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
853

c7c158e57   Hiroshi DOYU   omap: mailbox: re...
854
  static int __init omap_mbox_init(void)
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
855
  {
6b2339859   Hiroshi DOYU   omap mailbox: Set...
856
857
858
859
860
  	int err;
  
  	err = class_register(&omap_mbox_class);
  	if (err)
  		return err;
b5bebe410   Ohad Ben-Cohen   omap: mailbox: co...
861
  	/* kfifo size sanity check: alignment and minimal size */
9c1f2a5dc   Suman Anna   mailbox: omap: Ad...
862
863
  	mbox_kfifo_size = ALIGN(mbox_kfifo_size, sizeof(u32));
  	mbox_kfifo_size = max_t(unsigned int, mbox_kfifo_size, sizeof(u32));
b5bebe410   Ohad Ben-Cohen   omap: mailbox: co...
864

1f90a2162   Arvind Yadav   mailbox/omap: unr...
865
866
867
868
869
  	err = platform_driver_register(&omap_mbox_driver);
  	if (err)
  		class_unregister(&omap_mbox_class);
  
  	return err;
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
870
  }
6b2339859   Hiroshi DOYU   omap mailbox: Set...
871
  subsys_initcall(omap_mbox_init);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
872

c7c158e57   Hiroshi DOYU   omap: mailbox: re...
873
  static void __exit omap_mbox_exit(void)
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
874
  {
5040f5343   Suman Anna   mailbox/omap: con...
875
  	platform_driver_unregister(&omap_mbox_driver);
6b2339859   Hiroshi DOYU   omap mailbox: Set...
876
  	class_unregister(&omap_mbox_class);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
877
  }
c7c158e57   Hiroshi DOYU   omap: mailbox: re...
878
  module_exit(omap_mbox_exit);
340a614ac   Hiroshi DOYU   ARM: OMAP: Add ma...
879

f48cca877   Hiroshi DOYU   omap mailbox: fix...
880
881
  MODULE_LICENSE("GPL v2");
  MODULE_DESCRIPTION("omap mailbox: interrupt driven messaging");
f375325a0   Ohad Ben-Cohen   omap: mailbox cle...
882
883
  MODULE_AUTHOR("Toshihiro Kobayashi");
  MODULE_AUTHOR("Hiroshi DOYU");