• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Linux network driver for Brocade Converged Network Adapter.
3  *
4  * This program is free software; you can redistribute it and/or modify it
5  * under the terms of the GNU General Public License (GPL) Version 2 as
6  * published by the Free Software Foundation
7  *
8  * This program is distributed in the hope that it will be useful, but
9  * WITHOUT ANY WARRANTY; without even the implied warranty of
10  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
11  * General Public License for more details.
12  */
13 /*
14  * Copyright (c) 2005-2011 Brocade Communications Systems, Inc.
15  * All rights reserved
16  * www.brocade.com
17  */
18 
19 /* MSGQ module source file. */
20 
21 #include "bfi.h"
22 #include "bfa_msgq.h"
23 #include "bfa_ioc.h"
24 
25 #define call_cmdq_ent_cbfn(_cmdq_ent, _status)				\
26 {									\
27 	bfa_msgq_cmdcbfn_t cbfn;					\
28 	void *cbarg;							\
29 	cbfn = (_cmdq_ent)->cbfn;					\
30 	cbarg = (_cmdq_ent)->cbarg;					\
31 	(_cmdq_ent)->cbfn = NULL;					\
32 	(_cmdq_ent)->cbarg = NULL;					\
33 	if (cbfn) {							\
34 		cbfn(cbarg, (_status));					\
35 	}								\
36 }
37 
38 static void bfa_msgq_cmdq_dbell(struct bfa_msgq_cmdq *cmdq);
39 static void bfa_msgq_cmdq_copy_rsp(struct bfa_msgq_cmdq *cmdq);
40 
41 enum cmdq_event {
42 	CMDQ_E_START			= 1,
43 	CMDQ_E_STOP			= 2,
44 	CMDQ_E_FAIL			= 3,
45 	CMDQ_E_POST			= 4,
46 	CMDQ_E_INIT_RESP		= 5,
47 	CMDQ_E_DB_READY			= 6,
48 };
49 
50 bfa_fsm_state_decl(cmdq, stopped, struct bfa_msgq_cmdq, enum cmdq_event);
51 bfa_fsm_state_decl(cmdq, init_wait, struct bfa_msgq_cmdq, enum cmdq_event);
52 bfa_fsm_state_decl(cmdq, ready, struct bfa_msgq_cmdq, enum cmdq_event);
53 bfa_fsm_state_decl(cmdq, dbell_wait, struct bfa_msgq_cmdq,
54 			enum cmdq_event);
55 
56 static void
cmdq_sm_stopped_entry(struct bfa_msgq_cmdq * cmdq)57 cmdq_sm_stopped_entry(struct bfa_msgq_cmdq *cmdq)
58 {
59 	struct bfa_msgq_cmd_entry *cmdq_ent;
60 
61 	cmdq->producer_index = 0;
62 	cmdq->consumer_index = 0;
63 	cmdq->flags = 0;
64 	cmdq->token = 0;
65 	cmdq->offset = 0;
66 	cmdq->bytes_to_copy = 0;
67 	while (!list_empty(&cmdq->pending_q)) {
68 		bfa_q_deq(&cmdq->pending_q, &cmdq_ent);
69 		bfa_q_qe_init(&cmdq_ent->qe);
70 		call_cmdq_ent_cbfn(cmdq_ent, BFA_STATUS_FAILED);
71 	}
72 }
73 
74 static void
cmdq_sm_stopped(struct bfa_msgq_cmdq * cmdq,enum cmdq_event event)75 cmdq_sm_stopped(struct bfa_msgq_cmdq *cmdq, enum cmdq_event event)
76 {
77 	switch (event) {
78 	case CMDQ_E_START:
79 		bfa_fsm_set_state(cmdq, cmdq_sm_init_wait);
80 		break;
81 
82 	case CMDQ_E_STOP:
83 	case CMDQ_E_FAIL:
84 		/* No-op */
85 		break;
86 
87 	case CMDQ_E_POST:
88 		cmdq->flags |= BFA_MSGQ_CMDQ_F_DB_UPDATE;
89 		break;
90 
91 	default:
92 		bfa_sm_fault(event);
93 	}
94 }
95 
96 static void
cmdq_sm_init_wait_entry(struct bfa_msgq_cmdq * cmdq)97 cmdq_sm_init_wait_entry(struct bfa_msgq_cmdq *cmdq)
98 {
99 	bfa_wc_down(&cmdq->msgq->init_wc);
100 }
101 
102 static void
cmdq_sm_init_wait(struct bfa_msgq_cmdq * cmdq,enum cmdq_event event)103 cmdq_sm_init_wait(struct bfa_msgq_cmdq *cmdq, enum cmdq_event event)
104 {
105 	switch (event) {
106 	case CMDQ_E_STOP:
107 	case CMDQ_E_FAIL:
108 		bfa_fsm_set_state(cmdq, cmdq_sm_stopped);
109 		break;
110 
111 	case CMDQ_E_POST:
112 		cmdq->flags |= BFA_MSGQ_CMDQ_F_DB_UPDATE;
113 		break;
114 
115 	case CMDQ_E_INIT_RESP:
116 		if (cmdq->flags & BFA_MSGQ_CMDQ_F_DB_UPDATE) {
117 			cmdq->flags &= ~BFA_MSGQ_CMDQ_F_DB_UPDATE;
118 			bfa_fsm_set_state(cmdq, cmdq_sm_dbell_wait);
119 		} else
120 			bfa_fsm_set_state(cmdq, cmdq_sm_ready);
121 		break;
122 
123 	default:
124 		bfa_sm_fault(event);
125 	}
126 }
127 
128 static void
cmdq_sm_ready_entry(struct bfa_msgq_cmdq * cmdq)129 cmdq_sm_ready_entry(struct bfa_msgq_cmdq *cmdq)
130 {
131 }
132 
133 static void
cmdq_sm_ready(struct bfa_msgq_cmdq * cmdq,enum cmdq_event event)134 cmdq_sm_ready(struct bfa_msgq_cmdq *cmdq, enum cmdq_event event)
135 {
136 	switch (event) {
137 	case CMDQ_E_STOP:
138 	case CMDQ_E_FAIL:
139 		bfa_fsm_set_state(cmdq, cmdq_sm_stopped);
140 		break;
141 
142 	case CMDQ_E_POST:
143 		bfa_fsm_set_state(cmdq, cmdq_sm_dbell_wait);
144 		break;
145 
146 	default:
147 		bfa_sm_fault(event);
148 	}
149 }
150 
151 static void
cmdq_sm_dbell_wait_entry(struct bfa_msgq_cmdq * cmdq)152 cmdq_sm_dbell_wait_entry(struct bfa_msgq_cmdq *cmdq)
153 {
154 	bfa_msgq_cmdq_dbell(cmdq);
155 }
156 
157 static void
cmdq_sm_dbell_wait(struct bfa_msgq_cmdq * cmdq,enum cmdq_event event)158 cmdq_sm_dbell_wait(struct bfa_msgq_cmdq *cmdq, enum cmdq_event event)
159 {
160 	switch (event) {
161 	case CMDQ_E_STOP:
162 	case CMDQ_E_FAIL:
163 		bfa_fsm_set_state(cmdq, cmdq_sm_stopped);
164 		break;
165 
166 	case CMDQ_E_POST:
167 		cmdq->flags |= BFA_MSGQ_CMDQ_F_DB_UPDATE;
168 		break;
169 
170 	case CMDQ_E_DB_READY:
171 		if (cmdq->flags & BFA_MSGQ_CMDQ_F_DB_UPDATE) {
172 			cmdq->flags &= ~BFA_MSGQ_CMDQ_F_DB_UPDATE;
173 			bfa_fsm_set_state(cmdq, cmdq_sm_dbell_wait);
174 		} else
175 			bfa_fsm_set_state(cmdq, cmdq_sm_ready);
176 		break;
177 
178 	default:
179 		bfa_sm_fault(event);
180 	}
181 }
182 
183 static void
bfa_msgq_cmdq_dbell_ready(void * arg)184 bfa_msgq_cmdq_dbell_ready(void *arg)
185 {
186 	struct bfa_msgq_cmdq *cmdq = (struct bfa_msgq_cmdq *)arg;
187 	bfa_fsm_send_event(cmdq, CMDQ_E_DB_READY);
188 }
189 
190 static void
bfa_msgq_cmdq_dbell(struct bfa_msgq_cmdq * cmdq)191 bfa_msgq_cmdq_dbell(struct bfa_msgq_cmdq *cmdq)
192 {
193 	struct bfi_msgq_h2i_db *dbell =
194 		(struct bfi_msgq_h2i_db *)(&cmdq->dbell_mb.msg[0]);
195 
196 	memset(dbell, 0, sizeof(struct bfi_msgq_h2i_db));
197 	bfi_h2i_set(dbell->mh, BFI_MC_MSGQ, BFI_MSGQ_H2I_DOORBELL_PI, 0);
198 	dbell->mh.mtag.i2htok = 0;
199 	dbell->idx.cmdq_pi = htons(cmdq->producer_index);
200 
201 	if (!bfa_nw_ioc_mbox_queue(cmdq->msgq->ioc, &cmdq->dbell_mb,
202 				bfa_msgq_cmdq_dbell_ready, cmdq)) {
203 		bfa_msgq_cmdq_dbell_ready(cmdq);
204 	}
205 }
206 
207 static void
__cmd_copy(struct bfa_msgq_cmdq * cmdq,struct bfa_msgq_cmd_entry * cmd)208 __cmd_copy(struct bfa_msgq_cmdq *cmdq, struct bfa_msgq_cmd_entry *cmd)
209 {
210 	size_t len = cmd->msg_size;
211 	int num_entries = 0;
212 	size_t to_copy;
213 	u8 *src, *dst;
214 
215 	src = (u8 *)cmd->msg_hdr;
216 	dst = (u8 *)cmdq->addr.kva;
217 	dst += (cmdq->producer_index * BFI_MSGQ_CMD_ENTRY_SIZE);
218 
219 	while (len) {
220 		to_copy = (len < BFI_MSGQ_CMD_ENTRY_SIZE) ?
221 				len : BFI_MSGQ_CMD_ENTRY_SIZE;
222 		memcpy(dst, src, to_copy);
223 		len -= to_copy;
224 		src += BFI_MSGQ_CMD_ENTRY_SIZE;
225 		BFA_MSGQ_INDX_ADD(cmdq->producer_index, 1, cmdq->depth);
226 		dst = (u8 *)cmdq->addr.kva;
227 		dst += (cmdq->producer_index * BFI_MSGQ_CMD_ENTRY_SIZE);
228 		num_entries++;
229 	}
230 
231 }
232 
233 static void
bfa_msgq_cmdq_ci_update(struct bfa_msgq_cmdq * cmdq,struct bfi_mbmsg * mb)234 bfa_msgq_cmdq_ci_update(struct bfa_msgq_cmdq *cmdq, struct bfi_mbmsg *mb)
235 {
236 	struct bfi_msgq_i2h_db *dbell = (struct bfi_msgq_i2h_db *)mb;
237 	struct bfa_msgq_cmd_entry *cmd;
238 	int posted = 0;
239 
240 	cmdq->consumer_index = ntohs(dbell->idx.cmdq_ci);
241 
242 	/* Walk through pending list to see if the command can be posted */
243 	while (!list_empty(&cmdq->pending_q)) {
244 		cmd =
245 		(struct bfa_msgq_cmd_entry *)bfa_q_first(&cmdq->pending_q);
246 		if (ntohs(cmd->msg_hdr->num_entries) <=
247 			BFA_MSGQ_FREE_CNT(cmdq)) {
248 			list_del(&cmd->qe);
249 			__cmd_copy(cmdq, cmd);
250 			posted = 1;
251 			call_cmdq_ent_cbfn(cmd, BFA_STATUS_OK);
252 		} else {
253 			break;
254 		}
255 	}
256 
257 	if (posted)
258 		bfa_fsm_send_event(cmdq, CMDQ_E_POST);
259 }
260 
261 static void
bfa_msgq_cmdq_copy_next(void * arg)262 bfa_msgq_cmdq_copy_next(void *arg)
263 {
264 	struct bfa_msgq_cmdq *cmdq = (struct bfa_msgq_cmdq *)arg;
265 
266 	if (cmdq->bytes_to_copy)
267 		bfa_msgq_cmdq_copy_rsp(cmdq);
268 }
269 
270 static void
bfa_msgq_cmdq_copy_req(struct bfa_msgq_cmdq * cmdq,struct bfi_mbmsg * mb)271 bfa_msgq_cmdq_copy_req(struct bfa_msgq_cmdq *cmdq, struct bfi_mbmsg *mb)
272 {
273 	struct bfi_msgq_i2h_cmdq_copy_req *req =
274 		(struct bfi_msgq_i2h_cmdq_copy_req *)mb;
275 
276 	cmdq->token = 0;
277 	cmdq->offset = ntohs(req->offset);
278 	cmdq->bytes_to_copy = ntohs(req->len);
279 	bfa_msgq_cmdq_copy_rsp(cmdq);
280 }
281 
282 static void
bfa_msgq_cmdq_copy_rsp(struct bfa_msgq_cmdq * cmdq)283 bfa_msgq_cmdq_copy_rsp(struct bfa_msgq_cmdq *cmdq)
284 {
285 	struct bfi_msgq_h2i_cmdq_copy_rsp *rsp =
286 		(struct bfi_msgq_h2i_cmdq_copy_rsp *)&cmdq->copy_mb.msg[0];
287 	int copied;
288 	u8 *addr = (u8 *)cmdq->addr.kva;
289 
290 	memset(rsp, 0, sizeof(struct bfi_msgq_h2i_cmdq_copy_rsp));
291 	bfi_h2i_set(rsp->mh, BFI_MC_MSGQ, BFI_MSGQ_H2I_CMDQ_COPY_RSP, 0);
292 	rsp->mh.mtag.i2htok = htons(cmdq->token);
293 	copied = (cmdq->bytes_to_copy >= BFI_CMD_COPY_SZ) ? BFI_CMD_COPY_SZ :
294 		cmdq->bytes_to_copy;
295 	addr += cmdq->offset;
296 	memcpy(rsp->data, addr, copied);
297 
298 	cmdq->token++;
299 	cmdq->offset += copied;
300 	cmdq->bytes_to_copy -= copied;
301 
302 	if (!bfa_nw_ioc_mbox_queue(cmdq->msgq->ioc, &cmdq->copy_mb,
303 				bfa_msgq_cmdq_copy_next, cmdq)) {
304 		bfa_msgq_cmdq_copy_next(cmdq);
305 	}
306 }
307 
308 static void
bfa_msgq_cmdq_attach(struct bfa_msgq_cmdq * cmdq,struct bfa_msgq * msgq)309 bfa_msgq_cmdq_attach(struct bfa_msgq_cmdq *cmdq, struct bfa_msgq *msgq)
310 {
311 	cmdq->depth = BFA_MSGQ_CMDQ_NUM_ENTRY;
312 	INIT_LIST_HEAD(&cmdq->pending_q);
313 	cmdq->msgq = msgq;
314 	bfa_fsm_set_state(cmdq, cmdq_sm_stopped);
315 }
316 
317 static void bfa_msgq_rspq_dbell(struct bfa_msgq_rspq *rspq);
318 
319 enum rspq_event {
320 	RSPQ_E_START			= 1,
321 	RSPQ_E_STOP			= 2,
322 	RSPQ_E_FAIL			= 3,
323 	RSPQ_E_RESP			= 4,
324 	RSPQ_E_INIT_RESP		= 5,
325 	RSPQ_E_DB_READY			= 6,
326 };
327 
328 bfa_fsm_state_decl(rspq, stopped, struct bfa_msgq_rspq, enum rspq_event);
329 bfa_fsm_state_decl(rspq, init_wait, struct bfa_msgq_rspq,
330 			enum rspq_event);
331 bfa_fsm_state_decl(rspq, ready, struct bfa_msgq_rspq, enum rspq_event);
332 bfa_fsm_state_decl(rspq, dbell_wait, struct bfa_msgq_rspq,
333 			enum rspq_event);
334 
335 static void
rspq_sm_stopped_entry(struct bfa_msgq_rspq * rspq)336 rspq_sm_stopped_entry(struct bfa_msgq_rspq *rspq)
337 {
338 	rspq->producer_index = 0;
339 	rspq->consumer_index = 0;
340 	rspq->flags = 0;
341 }
342 
343 static void
rspq_sm_stopped(struct bfa_msgq_rspq * rspq,enum rspq_event event)344 rspq_sm_stopped(struct bfa_msgq_rspq *rspq, enum rspq_event event)
345 {
346 	switch (event) {
347 	case RSPQ_E_START:
348 		bfa_fsm_set_state(rspq, rspq_sm_init_wait);
349 		break;
350 
351 	case RSPQ_E_STOP:
352 	case RSPQ_E_FAIL:
353 		/* No-op */
354 		break;
355 
356 	default:
357 		bfa_sm_fault(event);
358 	}
359 }
360 
361 static void
rspq_sm_init_wait_entry(struct bfa_msgq_rspq * rspq)362 rspq_sm_init_wait_entry(struct bfa_msgq_rspq *rspq)
363 {
364 	bfa_wc_down(&rspq->msgq->init_wc);
365 }
366 
367 static void
rspq_sm_init_wait(struct bfa_msgq_rspq * rspq,enum rspq_event event)368 rspq_sm_init_wait(struct bfa_msgq_rspq *rspq, enum rspq_event event)
369 {
370 	switch (event) {
371 	case RSPQ_E_FAIL:
372 	case RSPQ_E_STOP:
373 		bfa_fsm_set_state(rspq, rspq_sm_stopped);
374 		break;
375 
376 	case RSPQ_E_INIT_RESP:
377 		bfa_fsm_set_state(rspq, rspq_sm_ready);
378 		break;
379 
380 	default:
381 		bfa_sm_fault(event);
382 	}
383 }
384 
385 static void
rspq_sm_ready_entry(struct bfa_msgq_rspq * rspq)386 rspq_sm_ready_entry(struct bfa_msgq_rspq *rspq)
387 {
388 }
389 
390 static void
rspq_sm_ready(struct bfa_msgq_rspq * rspq,enum rspq_event event)391 rspq_sm_ready(struct bfa_msgq_rspq *rspq, enum rspq_event event)
392 {
393 	switch (event) {
394 	case RSPQ_E_STOP:
395 	case RSPQ_E_FAIL:
396 		bfa_fsm_set_state(rspq, rspq_sm_stopped);
397 		break;
398 
399 	case RSPQ_E_RESP:
400 		bfa_fsm_set_state(rspq, rspq_sm_dbell_wait);
401 		break;
402 
403 	default:
404 		bfa_sm_fault(event);
405 	}
406 }
407 
408 static void
rspq_sm_dbell_wait_entry(struct bfa_msgq_rspq * rspq)409 rspq_sm_dbell_wait_entry(struct bfa_msgq_rspq *rspq)
410 {
411 	if (!bfa_nw_ioc_is_disabled(rspq->msgq->ioc))
412 		bfa_msgq_rspq_dbell(rspq);
413 }
414 
415 static void
rspq_sm_dbell_wait(struct bfa_msgq_rspq * rspq,enum rspq_event event)416 rspq_sm_dbell_wait(struct bfa_msgq_rspq *rspq, enum rspq_event event)
417 {
418 	switch (event) {
419 	case RSPQ_E_STOP:
420 	case RSPQ_E_FAIL:
421 		bfa_fsm_set_state(rspq, rspq_sm_stopped);
422 		break;
423 
424 	case RSPQ_E_RESP:
425 		rspq->flags |= BFA_MSGQ_RSPQ_F_DB_UPDATE;
426 		break;
427 
428 	case RSPQ_E_DB_READY:
429 		if (rspq->flags & BFA_MSGQ_RSPQ_F_DB_UPDATE) {
430 			rspq->flags &= ~BFA_MSGQ_RSPQ_F_DB_UPDATE;
431 			bfa_fsm_set_state(rspq, rspq_sm_dbell_wait);
432 		} else
433 			bfa_fsm_set_state(rspq, rspq_sm_ready);
434 		break;
435 
436 	default:
437 		bfa_sm_fault(event);
438 	}
439 }
440 
441 static void
bfa_msgq_rspq_dbell_ready(void * arg)442 bfa_msgq_rspq_dbell_ready(void *arg)
443 {
444 	struct bfa_msgq_rspq *rspq = (struct bfa_msgq_rspq *)arg;
445 	bfa_fsm_send_event(rspq, RSPQ_E_DB_READY);
446 }
447 
448 static void
bfa_msgq_rspq_dbell(struct bfa_msgq_rspq * rspq)449 bfa_msgq_rspq_dbell(struct bfa_msgq_rspq *rspq)
450 {
451 	struct bfi_msgq_h2i_db *dbell =
452 		(struct bfi_msgq_h2i_db *)(&rspq->dbell_mb.msg[0]);
453 
454 	memset(dbell, 0, sizeof(struct bfi_msgq_h2i_db));
455 	bfi_h2i_set(dbell->mh, BFI_MC_MSGQ, BFI_MSGQ_H2I_DOORBELL_CI, 0);
456 	dbell->mh.mtag.i2htok = 0;
457 	dbell->idx.rspq_ci = htons(rspq->consumer_index);
458 
459 	if (!bfa_nw_ioc_mbox_queue(rspq->msgq->ioc, &rspq->dbell_mb,
460 				bfa_msgq_rspq_dbell_ready, rspq)) {
461 		bfa_msgq_rspq_dbell_ready(rspq);
462 	}
463 }
464 
465 static void
bfa_msgq_rspq_pi_update(struct bfa_msgq_rspq * rspq,struct bfi_mbmsg * mb)466 bfa_msgq_rspq_pi_update(struct bfa_msgq_rspq *rspq, struct bfi_mbmsg *mb)
467 {
468 	struct bfi_msgq_i2h_db *dbell = (struct bfi_msgq_i2h_db *)mb;
469 	struct bfi_msgq_mhdr *msghdr;
470 	int num_entries;
471 	int mc;
472 	u8 *rspq_qe;
473 
474 	rspq->producer_index = ntohs(dbell->idx.rspq_pi);
475 
476 	while (rspq->consumer_index != rspq->producer_index) {
477 		rspq_qe = (u8 *)rspq->addr.kva;
478 		rspq_qe += (rspq->consumer_index * BFI_MSGQ_RSP_ENTRY_SIZE);
479 		msghdr = (struct bfi_msgq_mhdr *)rspq_qe;
480 
481 		mc = msghdr->msg_class;
482 		num_entries = ntohs(msghdr->num_entries);
483 
484 		if ((mc >= BFI_MC_MAX) || (rspq->rsphdlr[mc].cbfn == NULL))
485 			break;
486 
487 		(rspq->rsphdlr[mc].cbfn)(rspq->rsphdlr[mc].cbarg, msghdr);
488 
489 		BFA_MSGQ_INDX_ADD(rspq->consumer_index, num_entries,
490 				rspq->depth);
491 	}
492 
493 	bfa_fsm_send_event(rspq, RSPQ_E_RESP);
494 }
495 
496 static void
bfa_msgq_rspq_attach(struct bfa_msgq_rspq * rspq,struct bfa_msgq * msgq)497 bfa_msgq_rspq_attach(struct bfa_msgq_rspq *rspq, struct bfa_msgq *msgq)
498 {
499 	rspq->depth = BFA_MSGQ_RSPQ_NUM_ENTRY;
500 	rspq->msgq = msgq;
501 	bfa_fsm_set_state(rspq, rspq_sm_stopped);
502 }
503 
504 static void
bfa_msgq_init_rsp(struct bfa_msgq * msgq,struct bfi_mbmsg * mb)505 bfa_msgq_init_rsp(struct bfa_msgq *msgq,
506 		 struct bfi_mbmsg *mb)
507 {
508 	bfa_fsm_send_event(&msgq->cmdq, CMDQ_E_INIT_RESP);
509 	bfa_fsm_send_event(&msgq->rspq, RSPQ_E_INIT_RESP);
510 }
511 
512 static void
bfa_msgq_init(void * arg)513 bfa_msgq_init(void *arg)
514 {
515 	struct bfa_msgq *msgq = (struct bfa_msgq *)arg;
516 	struct bfi_msgq_cfg_req *msgq_cfg =
517 		(struct bfi_msgq_cfg_req *)&msgq->init_mb.msg[0];
518 
519 	memset(msgq_cfg, 0, sizeof(struct bfi_msgq_cfg_req));
520 	bfi_h2i_set(msgq_cfg->mh, BFI_MC_MSGQ, BFI_MSGQ_H2I_INIT_REQ, 0);
521 	msgq_cfg->mh.mtag.i2htok = 0;
522 
523 	bfa_dma_be_addr_set(msgq_cfg->cmdq.addr, msgq->cmdq.addr.pa);
524 	msgq_cfg->cmdq.q_depth = htons(msgq->cmdq.depth);
525 	bfa_dma_be_addr_set(msgq_cfg->rspq.addr, msgq->rspq.addr.pa);
526 	msgq_cfg->rspq.q_depth = htons(msgq->rspq.depth);
527 
528 	bfa_nw_ioc_mbox_queue(msgq->ioc, &msgq->init_mb, NULL, NULL);
529 }
530 
531 static void
bfa_msgq_isr(void * cbarg,struct bfi_mbmsg * msg)532 bfa_msgq_isr(void *cbarg, struct bfi_mbmsg *msg)
533 {
534 	struct bfa_msgq *msgq = (struct bfa_msgq *)cbarg;
535 
536 	switch (msg->mh.msg_id) {
537 	case BFI_MSGQ_I2H_INIT_RSP:
538 		bfa_msgq_init_rsp(msgq, msg);
539 		break;
540 
541 	case BFI_MSGQ_I2H_DOORBELL_PI:
542 		bfa_msgq_rspq_pi_update(&msgq->rspq, msg);
543 		break;
544 
545 	case BFI_MSGQ_I2H_DOORBELL_CI:
546 		bfa_msgq_cmdq_ci_update(&msgq->cmdq, msg);
547 		break;
548 
549 	case BFI_MSGQ_I2H_CMDQ_COPY_REQ:
550 		bfa_msgq_cmdq_copy_req(&msgq->cmdq, msg);
551 		break;
552 
553 	default:
554 		BUG_ON(1);
555 	}
556 }
557 
558 static void
bfa_msgq_notify(void * cbarg,enum bfa_ioc_event event)559 bfa_msgq_notify(void *cbarg, enum bfa_ioc_event event)
560 {
561 	struct bfa_msgq *msgq = (struct bfa_msgq *)cbarg;
562 
563 	switch (event) {
564 	case BFA_IOC_E_ENABLED:
565 		bfa_wc_init(&msgq->init_wc, bfa_msgq_init, msgq);
566 		bfa_wc_up(&msgq->init_wc);
567 		bfa_fsm_send_event(&msgq->cmdq, CMDQ_E_START);
568 		bfa_wc_up(&msgq->init_wc);
569 		bfa_fsm_send_event(&msgq->rspq, RSPQ_E_START);
570 		bfa_wc_wait(&msgq->init_wc);
571 		break;
572 
573 	case BFA_IOC_E_DISABLED:
574 		bfa_fsm_send_event(&msgq->cmdq, CMDQ_E_STOP);
575 		bfa_fsm_send_event(&msgq->rspq, RSPQ_E_STOP);
576 		break;
577 
578 	case BFA_IOC_E_FAILED:
579 		bfa_fsm_send_event(&msgq->cmdq, CMDQ_E_FAIL);
580 		bfa_fsm_send_event(&msgq->rspq, RSPQ_E_FAIL);
581 		break;
582 
583 	default:
584 		break;
585 	}
586 }
587 
588 u32
bfa_msgq_meminfo(void)589 bfa_msgq_meminfo(void)
590 {
591 	return roundup(BFA_MSGQ_CMDQ_SIZE, BFA_DMA_ALIGN_SZ) +
592 		roundup(BFA_MSGQ_RSPQ_SIZE, BFA_DMA_ALIGN_SZ);
593 }
594 
595 void
bfa_msgq_memclaim(struct bfa_msgq * msgq,u8 * kva,u64 pa)596 bfa_msgq_memclaim(struct bfa_msgq *msgq, u8 *kva, u64 pa)
597 {
598 	msgq->cmdq.addr.kva = kva;
599 	msgq->cmdq.addr.pa  = pa;
600 
601 	kva += roundup(BFA_MSGQ_CMDQ_SIZE, BFA_DMA_ALIGN_SZ);
602 	pa += roundup(BFA_MSGQ_CMDQ_SIZE, BFA_DMA_ALIGN_SZ);
603 
604 	msgq->rspq.addr.kva = kva;
605 	msgq->rspq.addr.pa = pa;
606 }
607 
608 void
bfa_msgq_attach(struct bfa_msgq * msgq,struct bfa_ioc * ioc)609 bfa_msgq_attach(struct bfa_msgq *msgq, struct bfa_ioc *ioc)
610 {
611 	msgq->ioc    = ioc;
612 
613 	bfa_msgq_cmdq_attach(&msgq->cmdq, msgq);
614 	bfa_msgq_rspq_attach(&msgq->rspq, msgq);
615 
616 	bfa_nw_ioc_mbox_regisr(msgq->ioc, BFI_MC_MSGQ, bfa_msgq_isr, msgq);
617 	bfa_q_qe_init(&msgq->ioc_notify);
618 	bfa_ioc_notify_init(&msgq->ioc_notify, bfa_msgq_notify, msgq);
619 	bfa_nw_ioc_notify_register(msgq->ioc, &msgq->ioc_notify);
620 }
621 
622 void
bfa_msgq_regisr(struct bfa_msgq * msgq,enum bfi_mclass mc,bfa_msgq_mcfunc_t cbfn,void * cbarg)623 bfa_msgq_regisr(struct bfa_msgq *msgq, enum bfi_mclass mc,
624 		bfa_msgq_mcfunc_t cbfn, void *cbarg)
625 {
626 	msgq->rspq.rsphdlr[mc].cbfn	= cbfn;
627 	msgq->rspq.rsphdlr[mc].cbarg	= cbarg;
628 }
629 
630 void
bfa_msgq_cmd_post(struct bfa_msgq * msgq,struct bfa_msgq_cmd_entry * cmd)631 bfa_msgq_cmd_post(struct bfa_msgq *msgq,  struct bfa_msgq_cmd_entry *cmd)
632 {
633 	if (ntohs(cmd->msg_hdr->num_entries) <=
634 		BFA_MSGQ_FREE_CNT(&msgq->cmdq)) {
635 		__cmd_copy(&msgq->cmdq, cmd);
636 		call_cmdq_ent_cbfn(cmd, BFA_STATUS_OK);
637 		bfa_fsm_send_event(&msgq->cmdq, CMDQ_E_POST);
638 	} else {
639 		list_add_tail(&cmd->qe, &msgq->cmdq.pending_q);
640 	}
641 }
642 
643 void
bfa_msgq_rsp_copy(struct bfa_msgq * msgq,u8 * buf,size_t buf_len)644 bfa_msgq_rsp_copy(struct bfa_msgq *msgq, u8 *buf, size_t buf_len)
645 {
646 	struct bfa_msgq_rspq *rspq = &msgq->rspq;
647 	size_t len = buf_len;
648 	size_t to_copy;
649 	int ci;
650 	u8 *src, *dst;
651 
652 	ci = rspq->consumer_index;
653 	src = (u8 *)rspq->addr.kva;
654 	src += (ci * BFI_MSGQ_RSP_ENTRY_SIZE);
655 	dst = buf;
656 
657 	while (len) {
658 		to_copy = (len < BFI_MSGQ_RSP_ENTRY_SIZE) ?
659 				len : BFI_MSGQ_RSP_ENTRY_SIZE;
660 		memcpy(dst, src, to_copy);
661 		len -= to_copy;
662 		dst += BFI_MSGQ_RSP_ENTRY_SIZE;
663 		BFA_MSGQ_INDX_ADD(ci, 1, rspq->depth);
664 		src = (u8 *)rspq->addr.kva;
665 		src += (ci * BFI_MSGQ_RSP_ENTRY_SIZE);
666 	}
667 }
668