Merge tag 'printk-for-6.6' of git://git.kernel.org/pub/scm/linux/kernel/git/printk...
[linux-2.6-microblaze.git] / drivers / cdx / controller / mcdi.c
1 // SPDX-License-Identifier: GPL-2.0
2 /*
3  * Management-Controller-to-Driver Interface
4  *
5  * Copyright 2008-2013 Solarflare Communications Inc.
6  * Copyright (C) 2022-2023, Advanced Micro Devices, Inc.
7  */
8 #include <linux/delay.h>
9 #include <linux/slab.h>
10 #include <linux/io.h>
11 #include <linux/spinlock.h>
12 #include <linux/netdevice.h>
13 #include <linux/etherdevice.h>
14 #include <linux/ethtool.h>
15 #include <linux/if_vlan.h>
16 #include <linux/timer.h>
17 #include <linux/list.h>
18 #include <linux/pci.h>
19 #include <linux/device.h>
20 #include <linux/rwsem.h>
21 #include <linux/vmalloc.h>
22 #include <net/netevent.h>
23 #include <linux/log2.h>
24 #include <linux/net_tstamp.h>
25 #include <linux/wait.h>
26
27 #include "bitfield.h"
28 #include "mcdi.h"
29
30 struct cdx_mcdi_copy_buffer {
31         struct cdx_dword buffer[DIV_ROUND_UP(MCDI_CTL_SDU_LEN_MAX, 4)];
32 };
33
34 static void cdx_mcdi_cancel_cmd(struct cdx_mcdi *cdx, struct cdx_mcdi_cmd *cmd);
35 static void cdx_mcdi_wait_for_cleanup(struct cdx_mcdi *cdx);
36 static int cdx_mcdi_rpc_async_internal(struct cdx_mcdi *cdx,
37                                        struct cdx_mcdi_cmd *cmd,
38                                        unsigned int *handle);
39 static void cdx_mcdi_start_or_queue(struct cdx_mcdi_iface *mcdi,
40                                     bool allow_retry);
41 static void cdx_mcdi_cmd_start_or_queue(struct cdx_mcdi_iface *mcdi,
42                                         struct cdx_mcdi_cmd *cmd);
43 static bool cdx_mcdi_complete_cmd(struct cdx_mcdi_iface *mcdi,
44                                   struct cdx_mcdi_cmd *cmd,
45                                   struct cdx_dword *outbuf,
46                                   int len,
47                                   struct list_head *cleanup_list);
48 static void cdx_mcdi_timeout_cmd(struct cdx_mcdi_iface *mcdi,
49                                  struct cdx_mcdi_cmd *cmd,
50                                  struct list_head *cleanup_list);
51 static void cdx_mcdi_cmd_work(struct work_struct *context);
52 static void cdx_mcdi_mode_fail(struct cdx_mcdi *cdx, struct list_head *cleanup_list);
53 static void _cdx_mcdi_display_error(struct cdx_mcdi *cdx, unsigned int cmd,
54                                     size_t inlen, int raw, int arg, int err_no);
55
56 static bool cdx_cmd_cancelled(struct cdx_mcdi_cmd *cmd)
57 {
58         return cmd->state == MCDI_STATE_RUNNING_CANCELLED;
59 }
60
61 static void cdx_mcdi_cmd_release(struct kref *ref)
62 {
63         kfree(container_of(ref, struct cdx_mcdi_cmd, ref));
64 }
65
66 static unsigned int cdx_mcdi_cmd_handle(struct cdx_mcdi_cmd *cmd)
67 {
68         return cmd->handle;
69 }
70
71 static void _cdx_mcdi_remove_cmd(struct cdx_mcdi_iface *mcdi,
72                                  struct cdx_mcdi_cmd *cmd,
73                                  struct list_head *cleanup_list)
74 {
75         /* if cancelled, the completers have already been called */
76         if (cdx_cmd_cancelled(cmd))
77                 return;
78
79         if (cmd->completer) {
80                 list_add_tail(&cmd->cleanup_list, cleanup_list);
81                 ++mcdi->outstanding_cleanups;
82                 kref_get(&cmd->ref);
83         }
84 }
85
86 static void cdx_mcdi_remove_cmd(struct cdx_mcdi_iface *mcdi,
87                                 struct cdx_mcdi_cmd *cmd,
88                                 struct list_head *cleanup_list)
89 {
90         list_del(&cmd->list);
91         _cdx_mcdi_remove_cmd(mcdi, cmd, cleanup_list);
92         cmd->state = MCDI_STATE_FINISHED;
93         kref_put(&cmd->ref, cdx_mcdi_cmd_release);
94         if (list_empty(&mcdi->cmd_list))
95                 wake_up(&mcdi->cmd_complete_wq);
96 }
97
98 static unsigned long cdx_mcdi_rpc_timeout(struct cdx_mcdi *cdx, unsigned int cmd)
99 {
100         if (!cdx->mcdi_ops->mcdi_rpc_timeout)
101                 return MCDI_RPC_TIMEOUT;
102         else
103                 return cdx->mcdi_ops->mcdi_rpc_timeout(cdx, cmd);
104 }
105
106 int cdx_mcdi_init(struct cdx_mcdi *cdx)
107 {
108         struct cdx_mcdi_iface *mcdi;
109         int rc = -ENOMEM;
110
111         cdx->mcdi = kzalloc(sizeof(*cdx->mcdi), GFP_KERNEL);
112         if (!cdx->mcdi)
113                 goto fail;
114
115         mcdi = cdx_mcdi_if(cdx);
116         mcdi->cdx = cdx;
117
118         mcdi->workqueue = alloc_ordered_workqueue("mcdi_wq", 0);
119         if (!mcdi->workqueue)
120                 goto fail2;
121         mutex_init(&mcdi->iface_lock);
122         mcdi->mode = MCDI_MODE_EVENTS;
123         INIT_LIST_HEAD(&mcdi->cmd_list);
124         init_waitqueue_head(&mcdi->cmd_complete_wq);
125
126         mcdi->new_epoch = true;
127
128         return 0;
129 fail2:
130         kfree(cdx->mcdi);
131         cdx->mcdi = NULL;
132 fail:
133         return rc;
134 }
135
136 void cdx_mcdi_finish(struct cdx_mcdi *cdx)
137 {
138         struct cdx_mcdi_iface *mcdi;
139
140         mcdi = cdx_mcdi_if(cdx);
141         if (!mcdi)
142                 return;
143
144         cdx_mcdi_wait_for_cleanup(cdx);
145
146         destroy_workqueue(mcdi->workqueue);
147         kfree(cdx->mcdi);
148         cdx->mcdi = NULL;
149 }
150
151 static bool cdx_mcdi_flushed(struct cdx_mcdi_iface *mcdi, bool ignore_cleanups)
152 {
153         bool flushed;
154
155         mutex_lock(&mcdi->iface_lock);
156         flushed = list_empty(&mcdi->cmd_list) &&
157                   (ignore_cleanups || !mcdi->outstanding_cleanups);
158         mutex_unlock(&mcdi->iface_lock);
159         return flushed;
160 }
161
162 /* Wait for outstanding MCDI commands to complete. */
163 static void cdx_mcdi_wait_for_cleanup(struct cdx_mcdi *cdx)
164 {
165         struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
166
167         if (!mcdi)
168                 return;
169
170         wait_event(mcdi->cmd_complete_wq,
171                    cdx_mcdi_flushed(mcdi, false));
172 }
173
174 int cdx_mcdi_wait_for_quiescence(struct cdx_mcdi *cdx,
175                                  unsigned int timeout_jiffies)
176 {
177         struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
178         DEFINE_WAIT_FUNC(wait, woken_wake_function);
179         int rc = 0;
180
181         if (!mcdi)
182                 return -EINVAL;
183
184         flush_workqueue(mcdi->workqueue);
185
186         add_wait_queue(&mcdi->cmd_complete_wq, &wait);
187
188         while (!cdx_mcdi_flushed(mcdi, true)) {
189                 rc = wait_woken(&wait, TASK_IDLE, timeout_jiffies);
190                 if (rc)
191                         continue;
192                 break;
193         }
194
195         remove_wait_queue(&mcdi->cmd_complete_wq, &wait);
196
197         if (rc > 0)
198                 rc = 0;
199         else if (rc == 0)
200                 rc = -ETIMEDOUT;
201
202         return rc;
203 }
204
205 static u8 cdx_mcdi_payload_csum(const struct cdx_dword *hdr, size_t hdr_len,
206                                 const struct cdx_dword *sdu, size_t sdu_len)
207 {
208         u8 *p = (u8 *)hdr;
209         u8 csum = 0;
210         int i;
211
212         for (i = 0; i < hdr_len; i++)
213                 csum += p[i];
214
215         p = (u8 *)sdu;
216         for (i = 0; i < sdu_len; i++)
217                 csum += p[i];
218
219         return ~csum & 0xff;
220 }
221
222 static void cdx_mcdi_send_request(struct cdx_mcdi *cdx,
223                                   struct cdx_mcdi_cmd *cmd)
224 {
225         struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
226         const struct cdx_dword *inbuf = cmd->inbuf;
227         size_t inlen = cmd->inlen;
228         struct cdx_dword hdr[2];
229         size_t hdr_len;
230         bool not_epoch;
231         u32 xflags;
232
233         if (!mcdi)
234                 return;
235
236         mcdi->prev_seq = cmd->seq;
237         mcdi->seq_held_by[cmd->seq] = cmd;
238         mcdi->db_held_by = cmd;
239         cmd->started = jiffies;
240
241         not_epoch = !mcdi->new_epoch;
242         xflags = 0;
243
244         /* MCDI v2 */
245         WARN_ON(inlen > MCDI_CTL_SDU_LEN_MAX_V2);
246         CDX_POPULATE_DWORD_7(hdr[0],
247                              MCDI_HEADER_RESPONSE, 0,
248                              MCDI_HEADER_RESYNC, 1,
249                              MCDI_HEADER_CODE, MC_CMD_V2_EXTN,
250                              MCDI_HEADER_DATALEN, 0,
251                              MCDI_HEADER_SEQ, cmd->seq,
252                              MCDI_HEADER_XFLAGS, xflags,
253                              MCDI_HEADER_NOT_EPOCH, not_epoch);
254         CDX_POPULATE_DWORD_3(hdr[1],
255                              MC_CMD_V2_EXTN_IN_EXTENDED_CMD, cmd->cmd,
256                              MC_CMD_V2_EXTN_IN_ACTUAL_LEN, inlen,
257                              MC_CMD_V2_EXTN_IN_MESSAGE_TYPE,
258                              MC_CMD_V2_EXTN_IN_MCDI_MESSAGE_TYPE_PLATFORM);
259         hdr_len = 8;
260
261         hdr[0].cdx_u32 |= (__force __le32)(cdx_mcdi_payload_csum(hdr, hdr_len, inbuf, inlen) <<
262                          MCDI_HEADER_XFLAGS_LBN);
263
264         print_hex_dump_debug("MCDI REQ HEADER: ", DUMP_PREFIX_NONE, 32, 4, hdr, hdr_len, false);
265         print_hex_dump_debug("MCDI REQ PAYLOAD: ", DUMP_PREFIX_NONE, 32, 4, inbuf, inlen, false);
266
267         cdx->mcdi_ops->mcdi_request(cdx, hdr, hdr_len, inbuf, inlen);
268
269         mcdi->new_epoch = false;
270 }
271
272 static int cdx_mcdi_errno(struct cdx_mcdi *cdx, unsigned int mcdi_err)
273 {
274         switch (mcdi_err) {
275         case 0:
276         case MC_CMD_ERR_QUEUE_FULL:
277                 return mcdi_err;
278         case MC_CMD_ERR_EPERM:
279                 return -EPERM;
280         case MC_CMD_ERR_ENOENT:
281                 return -ENOENT;
282         case MC_CMD_ERR_EINTR:
283                 return -EINTR;
284         case MC_CMD_ERR_EAGAIN:
285                 return -EAGAIN;
286         case MC_CMD_ERR_EACCES:
287                 return -EACCES;
288         case MC_CMD_ERR_EBUSY:
289                 return -EBUSY;
290         case MC_CMD_ERR_EINVAL:
291                 return -EINVAL;
292         case MC_CMD_ERR_ERANGE:
293                 return -ERANGE;
294         case MC_CMD_ERR_EDEADLK:
295                 return -EDEADLK;
296         case MC_CMD_ERR_ENOSYS:
297                 return -EOPNOTSUPP;
298         case MC_CMD_ERR_ETIME:
299                 return -ETIME;
300         case MC_CMD_ERR_EALREADY:
301                 return -EALREADY;
302         case MC_CMD_ERR_ENOSPC:
303                 return -ENOSPC;
304         case MC_CMD_ERR_ENOMEM:
305                 return -ENOMEM;
306         case MC_CMD_ERR_ENOTSUP:
307                 return -EOPNOTSUPP;
308         case MC_CMD_ERR_ALLOC_FAIL:
309                 return -ENOBUFS;
310         case MC_CMD_ERR_MAC_EXIST:
311                 return -EADDRINUSE;
312         case MC_CMD_ERR_NO_EVB_PORT:
313                 return -EAGAIN;
314         default:
315                 return -EPROTO;
316         }
317 }
318
319 static void cdx_mcdi_process_cleanup_list(struct cdx_mcdi *cdx,
320                                           struct list_head *cleanup_list)
321 {
322         struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
323         unsigned int cleanups = 0;
324
325         if (!mcdi)
326                 return;
327
328         while (!list_empty(cleanup_list)) {
329                 struct cdx_mcdi_cmd *cmd =
330                         list_first_entry(cleanup_list,
331                                          struct cdx_mcdi_cmd, cleanup_list);
332                 cmd->completer(cdx, cmd->cookie, cmd->rc,
333                                cmd->outbuf, cmd->outlen);
334                 list_del(&cmd->cleanup_list);
335                 kref_put(&cmd->ref, cdx_mcdi_cmd_release);
336                 ++cleanups;
337         }
338
339         if (cleanups) {
340                 bool all_done;
341
342                 mutex_lock(&mcdi->iface_lock);
343                 CDX_WARN_ON_PARANOID(cleanups > mcdi->outstanding_cleanups);
344                 all_done = (mcdi->outstanding_cleanups -= cleanups) == 0;
345                 mutex_unlock(&mcdi->iface_lock);
346                 if (all_done)
347                         wake_up(&mcdi->cmd_complete_wq);
348         }
349 }
350
351 static void _cdx_mcdi_cancel_cmd(struct cdx_mcdi_iface *mcdi,
352                                  unsigned int handle,
353                                  struct list_head *cleanup_list)
354 {
355         struct cdx_mcdi_cmd *cmd;
356
357         list_for_each_entry(cmd, &mcdi->cmd_list, list)
358                 if (cdx_mcdi_cmd_handle(cmd) == handle) {
359                         switch (cmd->state) {
360                         case MCDI_STATE_QUEUED:
361                         case MCDI_STATE_RETRY:
362                                 pr_debug("command %#x inlen %zu cancelled in queue\n",
363                                          cmd->cmd, cmd->inlen);
364                                 /* if not yet running, properly cancel it */
365                                 cmd->rc = -EPIPE;
366                                 cdx_mcdi_remove_cmd(mcdi, cmd, cleanup_list);
367                                 break;
368                         case MCDI_STATE_RUNNING:
369                         case MCDI_STATE_RUNNING_CANCELLED:
370                         case MCDI_STATE_FINISHED:
371                         default:
372                                 /* invalid state? */
373                                 WARN_ON(1);
374                         }
375                         break;
376                 }
377 }
378
379 static void cdx_mcdi_cancel_cmd(struct cdx_mcdi *cdx, struct cdx_mcdi_cmd *cmd)
380 {
381         struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
382         LIST_HEAD(cleanup_list);
383
384         if (!mcdi)
385                 return;
386
387         mutex_lock(&mcdi->iface_lock);
388         cdx_mcdi_timeout_cmd(mcdi, cmd, &cleanup_list);
389         mutex_unlock(&mcdi->iface_lock);
390         cdx_mcdi_process_cleanup_list(cdx, &cleanup_list);
391 }
392
393 struct cdx_mcdi_blocking_data {
394         struct kref ref;
395         bool done;
396         wait_queue_head_t wq;
397         int rc;
398         struct cdx_dword *outbuf;
399         size_t outlen;
400         size_t outlen_actual;
401 };
402
403 static void cdx_mcdi_blocking_data_release(struct kref *ref)
404 {
405         kfree(container_of(ref, struct cdx_mcdi_blocking_data, ref));
406 }
407
408 static void cdx_mcdi_rpc_completer(struct cdx_mcdi *cdx, unsigned long cookie,
409                                    int rc, struct cdx_dword *outbuf,
410                                    size_t outlen_actual)
411 {
412         struct cdx_mcdi_blocking_data *wait_data =
413                 (struct cdx_mcdi_blocking_data *)cookie;
414
415         wait_data->rc = rc;
416         memcpy(wait_data->outbuf, outbuf,
417                min(outlen_actual, wait_data->outlen));
418         wait_data->outlen_actual = outlen_actual;
419         /* memory barrier */
420         smp_wmb();
421         wait_data->done = true;
422         wake_up(&wait_data->wq);
423         kref_put(&wait_data->ref, cdx_mcdi_blocking_data_release);
424 }
425
426 static int cdx_mcdi_rpc_sync(struct cdx_mcdi *cdx, unsigned int cmd,
427                              const struct cdx_dword *inbuf, size_t inlen,
428                              struct cdx_dword *outbuf, size_t outlen,
429                              size_t *outlen_actual, bool quiet)
430 {
431         struct cdx_mcdi_blocking_data *wait_data;
432         struct cdx_mcdi_cmd *cmd_item;
433         unsigned int handle;
434         int rc;
435
436         if (outlen_actual)
437                 *outlen_actual = 0;
438
439         wait_data = kmalloc(sizeof(*wait_data), GFP_KERNEL);
440         if (!wait_data)
441                 return -ENOMEM;
442
443         cmd_item = kmalloc(sizeof(*cmd_item), GFP_KERNEL);
444         if (!cmd_item) {
445                 kfree(wait_data);
446                 return -ENOMEM;
447         }
448
449         kref_init(&wait_data->ref);
450         wait_data->done = false;
451         init_waitqueue_head(&wait_data->wq);
452         wait_data->outbuf = outbuf;
453         wait_data->outlen = outlen;
454
455         kref_init(&cmd_item->ref);
456         cmd_item->quiet = quiet;
457         cmd_item->cookie = (unsigned long)wait_data;
458         cmd_item->completer = &cdx_mcdi_rpc_completer;
459         cmd_item->cmd = cmd;
460         cmd_item->inlen = inlen;
461         cmd_item->inbuf = inbuf;
462
463         /* Claim an extra reference for the completer to put. */
464         kref_get(&wait_data->ref);
465         rc = cdx_mcdi_rpc_async_internal(cdx, cmd_item, &handle);
466         if (rc) {
467                 kref_put(&wait_data->ref, cdx_mcdi_blocking_data_release);
468                 goto out;
469         }
470
471         if (!wait_event_timeout(wait_data->wq, wait_data->done,
472                                 cdx_mcdi_rpc_timeout(cdx, cmd)) &&
473             !wait_data->done) {
474                 pr_err("MC command 0x%x inlen %zu timed out (sync)\n",
475                        cmd, inlen);
476
477                 cdx_mcdi_cancel_cmd(cdx, cmd_item);
478
479                 wait_data->rc = -ETIMEDOUT;
480                 wait_data->outlen_actual = 0;
481         }
482
483         if (outlen_actual)
484                 *outlen_actual = wait_data->outlen_actual;
485         rc = wait_data->rc;
486
487 out:
488         kref_put(&wait_data->ref, cdx_mcdi_blocking_data_release);
489
490         return rc;
491 }
492
493 static bool cdx_mcdi_get_seq(struct cdx_mcdi_iface *mcdi, unsigned char *seq)
494 {
495         *seq = mcdi->prev_seq;
496         do {
497                 *seq = (*seq + 1) % ARRAY_SIZE(mcdi->seq_held_by);
498         } while (mcdi->seq_held_by[*seq] && *seq != mcdi->prev_seq);
499         return !mcdi->seq_held_by[*seq];
500 }
501
502 static int cdx_mcdi_rpc_async_internal(struct cdx_mcdi *cdx,
503                                        struct cdx_mcdi_cmd *cmd,
504                                        unsigned int *handle)
505 {
506         struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
507         LIST_HEAD(cleanup_list);
508
509         if (!mcdi) {
510                 kref_put(&cmd->ref, cdx_mcdi_cmd_release);
511                 return -ENETDOWN;
512         }
513
514         if (mcdi->mode == MCDI_MODE_FAIL) {
515                 kref_put(&cmd->ref, cdx_mcdi_cmd_release);
516                 return -ENETDOWN;
517         }
518
519         cmd->mcdi = mcdi;
520         INIT_WORK(&cmd->work, cdx_mcdi_cmd_work);
521         INIT_LIST_HEAD(&cmd->list);
522         INIT_LIST_HEAD(&cmd->cleanup_list);
523         cmd->rc = 0;
524         cmd->outbuf = NULL;
525         cmd->outlen = 0;
526
527         queue_work(mcdi->workqueue, &cmd->work);
528         return 0;
529 }
530
531 static void cdx_mcdi_cmd_start_or_queue(struct cdx_mcdi_iface *mcdi,
532                                         struct cdx_mcdi_cmd *cmd)
533 {
534         struct cdx_mcdi *cdx = mcdi->cdx;
535         u8 seq;
536
537         if (!mcdi->db_held_by &&
538             cdx_mcdi_get_seq(mcdi, &seq)) {
539                 cmd->seq = seq;
540                 cmd->reboot_seen = false;
541                 cdx_mcdi_send_request(cdx, cmd);
542                 cmd->state = MCDI_STATE_RUNNING;
543         } else {
544                 cmd->state = MCDI_STATE_QUEUED;
545         }
546 }
547
548 /* try to advance other commands */
549 static void cdx_mcdi_start_or_queue(struct cdx_mcdi_iface *mcdi,
550                                     bool allow_retry)
551 {
552         struct cdx_mcdi_cmd *cmd, *tmp;
553
554         list_for_each_entry_safe(cmd, tmp, &mcdi->cmd_list, list)
555                 if (cmd->state == MCDI_STATE_QUEUED ||
556                     (cmd->state == MCDI_STATE_RETRY && allow_retry))
557                         cdx_mcdi_cmd_start_or_queue(mcdi, cmd);
558 }
559
560 void cdx_mcdi_process_cmd(struct cdx_mcdi *cdx, struct cdx_dword *outbuf, int len)
561 {
562         struct cdx_mcdi_iface *mcdi;
563         struct cdx_mcdi_cmd *cmd;
564         LIST_HEAD(cleanup_list);
565         unsigned int respseq;
566
567         if (!len || !outbuf) {
568                 pr_err("Got empty MC response\n");
569                 return;
570         }
571
572         mcdi = cdx_mcdi_if(cdx);
573         if (!mcdi)
574                 return;
575
576         respseq = CDX_DWORD_FIELD(outbuf[0], MCDI_HEADER_SEQ);
577
578         mutex_lock(&mcdi->iface_lock);
579         cmd = mcdi->seq_held_by[respseq];
580
581         if (cmd) {
582                 if (cmd->state == MCDI_STATE_FINISHED) {
583                         mutex_unlock(&mcdi->iface_lock);
584                         kref_put(&cmd->ref, cdx_mcdi_cmd_release);
585                         return;
586                 }
587
588                 cdx_mcdi_complete_cmd(mcdi, cmd, outbuf, len, &cleanup_list);
589         } else {
590                 pr_err("MC response unexpected for seq : %0X\n", respseq);
591         }
592
593         mutex_unlock(&mcdi->iface_lock);
594
595         cdx_mcdi_process_cleanup_list(mcdi->cdx, &cleanup_list);
596 }
597
598 static void cdx_mcdi_cmd_work(struct work_struct *context)
599 {
600         struct cdx_mcdi_cmd *cmd =
601                 container_of(context, struct cdx_mcdi_cmd, work);
602         struct cdx_mcdi_iface *mcdi = cmd->mcdi;
603
604         mutex_lock(&mcdi->iface_lock);
605
606         cmd->handle = mcdi->prev_handle++;
607         list_add_tail(&cmd->list, &mcdi->cmd_list);
608         cdx_mcdi_cmd_start_or_queue(mcdi, cmd);
609
610         mutex_unlock(&mcdi->iface_lock);
611 }
612
613 /*
614  * Returns true if the MCDI module is finished with the command.
615  * (examples of false would be if the command was proxied, or it was
616  * rejected by the MC due to lack of resources and requeued).
617  */
618 static bool cdx_mcdi_complete_cmd(struct cdx_mcdi_iface *mcdi,
619                                   struct cdx_mcdi_cmd *cmd,
620                                   struct cdx_dword *outbuf,
621                                   int len,
622                                   struct list_head *cleanup_list)
623 {
624         size_t resp_hdr_len, resp_data_len;
625         struct cdx_mcdi *cdx = mcdi->cdx;
626         unsigned int respcmd, error;
627         bool completed = false;
628         int rc;
629
630         /* ensure the command can't go away before this function returns */
631         kref_get(&cmd->ref);
632
633         respcmd = CDX_DWORD_FIELD(outbuf[0], MCDI_HEADER_CODE);
634         error = CDX_DWORD_FIELD(outbuf[0], MCDI_HEADER_ERROR);
635
636         if (respcmd != MC_CMD_V2_EXTN) {
637                 resp_hdr_len = 4;
638                 resp_data_len = CDX_DWORD_FIELD(outbuf[0], MCDI_HEADER_DATALEN);
639         } else {
640                 resp_data_len = 0;
641                 resp_hdr_len = 8;
642                 if (len >= 8)
643                         resp_data_len =
644                                 CDX_DWORD_FIELD(outbuf[1], MC_CMD_V2_EXTN_IN_ACTUAL_LEN);
645         }
646
647         if ((resp_hdr_len + resp_data_len) > len) {
648                 pr_warn("Incomplete MCDI response received %d. Expected %zu\n",
649                         len, (resp_hdr_len + resp_data_len));
650                 resp_data_len = 0;
651         }
652
653         print_hex_dump_debug("MCDI RESP HEADER: ", DUMP_PREFIX_NONE, 32, 4,
654                              outbuf, resp_hdr_len, false);
655         print_hex_dump_debug("MCDI RESP PAYLOAD: ", DUMP_PREFIX_NONE, 32, 4,
656                              outbuf + (resp_hdr_len / 4), resp_data_len, false);
657
658         if (error && resp_data_len == 0) {
659                 /* MC rebooted during command */
660                 rc = -EIO;
661         } else {
662                 if (WARN_ON_ONCE(error && resp_data_len < 4))
663                         resp_data_len = 4;
664                 if (error) {
665                         rc = CDX_DWORD_FIELD(outbuf[resp_hdr_len / 4], CDX_DWORD);
666                         if (!cmd->quiet) {
667                                 int err_arg = 0;
668
669                                 if (resp_data_len >= MC_CMD_ERR_ARG_OFST + 4) {
670                                         int offset = (resp_hdr_len + MC_CMD_ERR_ARG_OFST) / 4;
671
672                                         err_arg = CDX_DWORD_VAL(outbuf[offset]);
673                                 }
674
675                                 _cdx_mcdi_display_error(cdx, cmd->cmd,
676                                                         cmd->inlen, rc, err_arg,
677                                                         cdx_mcdi_errno(cdx, rc));
678                         }
679                         rc = cdx_mcdi_errno(cdx, rc);
680                 } else {
681                         rc = 0;
682                 }
683         }
684
685         /* free doorbell */
686         if (mcdi->db_held_by == cmd)
687                 mcdi->db_held_by = NULL;
688
689         if (cdx_cmd_cancelled(cmd)) {
690                 list_del(&cmd->list);
691                 kref_put(&cmd->ref, cdx_mcdi_cmd_release);
692                 completed = true;
693         } else if (rc == MC_CMD_ERR_QUEUE_FULL) {
694                 cmd->state = MCDI_STATE_RETRY;
695         } else {
696                 cmd->rc = rc;
697                 cmd->outbuf = outbuf + DIV_ROUND_UP(resp_hdr_len, 4);
698                 cmd->outlen = resp_data_len;
699                 cdx_mcdi_remove_cmd(mcdi, cmd, cleanup_list);
700                 completed = true;
701         }
702
703         /* free sequence number and buffer */
704         mcdi->seq_held_by[cmd->seq] = NULL;
705
706         cdx_mcdi_start_or_queue(mcdi, rc != MC_CMD_ERR_QUEUE_FULL);
707
708         /* wake up anyone waiting for flush */
709         wake_up(&mcdi->cmd_complete_wq);
710
711         kref_put(&cmd->ref, cdx_mcdi_cmd_release);
712
713         return completed;
714 }
715
716 static void cdx_mcdi_timeout_cmd(struct cdx_mcdi_iface *mcdi,
717                                  struct cdx_mcdi_cmd *cmd,
718                                  struct list_head *cleanup_list)
719 {
720         struct cdx_mcdi *cdx = mcdi->cdx;
721
722         pr_err("MC command 0x%x inlen %zu state %d timed out after %u ms\n",
723                cmd->cmd, cmd->inlen, cmd->state,
724                jiffies_to_msecs(jiffies - cmd->started));
725
726         cmd->rc = -ETIMEDOUT;
727         cdx_mcdi_remove_cmd(mcdi, cmd, cleanup_list);
728
729         cdx_mcdi_mode_fail(cdx, cleanup_list);
730 }
731
732 /**
733  * cdx_mcdi_rpc - Issue an MCDI command and wait for completion
734  * @cdx: NIC through which to issue the command
735  * @cmd: Command type number
736  * @inbuf: Command parameters
737  * @inlen: Length of command parameters, in bytes. Must be a multiple
738  *      of 4 and no greater than %MCDI_CTL_SDU_LEN_MAX_V1.
739  * @outbuf: Response buffer. May be %NULL if @outlen is 0.
740  * @outlen: Length of response buffer, in bytes. If the actual
741  *      response is longer than @outlen & ~3, it will be truncated
742  *      to that length.
743  * @outlen_actual: Pointer through which to return the actual response
744  *      length. May be %NULL if this is not needed.
745  *
746  * This function may sleep and therefore must be called in process
747  * context.
748  *
749  * Return: A negative error code, or zero if successful. The error
750  *      code may come from the MCDI response or may indicate a failure
751  *      to communicate with the MC. In the former case, the response
752  *      will still be copied to @outbuf and *@outlen_actual will be
753  *      set accordingly. In the latter case, *@outlen_actual will be
754  *      set to zero.
755  */
756 int cdx_mcdi_rpc(struct cdx_mcdi *cdx, unsigned int cmd,
757                  const struct cdx_dword *inbuf, size_t inlen,
758                  struct cdx_dword *outbuf, size_t outlen,
759                  size_t *outlen_actual)
760 {
761         return cdx_mcdi_rpc_sync(cdx, cmd, inbuf, inlen, outbuf, outlen,
762                                  outlen_actual, false);
763 }
764
765 /**
766  * cdx_mcdi_rpc_async - Schedule an MCDI command to run asynchronously
767  * @cdx: NIC through which to issue the command
768  * @cmd: Command type number
769  * @inbuf: Command parameters
770  * @inlen: Length of command parameters, in bytes
771  * @complete: Function to be called on completion or cancellation.
772  * @cookie: Arbitrary value to be passed to @complete.
773  *
774  * This function does not sleep and therefore may be called in atomic
775  * context.  It will fail if event queues are disabled or if MCDI
776  * event completions have been disabled due to an error.
777  *
778  * If it succeeds, the @complete function will be called exactly once
779  * in process context, when one of the following occurs:
780  * (a) the completion event is received (in process context)
781  * (b) event queues are disabled (in the process that disables them)
782  */
783 int
784 cdx_mcdi_rpc_async(struct cdx_mcdi *cdx, unsigned int cmd,
785                    const struct cdx_dword *inbuf, size_t inlen,
786                    cdx_mcdi_async_completer *complete, unsigned long cookie)
787 {
788         struct cdx_mcdi_cmd *cmd_item =
789                 kmalloc(sizeof(struct cdx_mcdi_cmd) + inlen, GFP_ATOMIC);
790
791         if (!cmd_item)
792                 return -ENOMEM;
793
794         kref_init(&cmd_item->ref);
795         cmd_item->quiet = true;
796         cmd_item->cookie = cookie;
797         cmd_item->completer = complete;
798         cmd_item->cmd = cmd;
799         cmd_item->inlen = inlen;
800         /* inbuf is probably not valid after return, so take a copy */
801         cmd_item->inbuf = (struct cdx_dword *)(cmd_item + 1);
802         memcpy(cmd_item + 1, inbuf, inlen);
803
804         return cdx_mcdi_rpc_async_internal(cdx, cmd_item, NULL);
805 }
806
807 static void _cdx_mcdi_display_error(struct cdx_mcdi *cdx, unsigned int cmd,
808                                     size_t inlen, int raw, int arg, int err_no)
809 {
810         pr_err("MC command 0x%x inlen %d failed err_no=%d (raw=%d) arg=%d\n",
811                cmd, (int)inlen, err_no, raw, arg);
812 }
813
814 /*
815  * Set MCDI mode to fail to prevent any new commands, then cancel any
816  * outstanding commands.
817  * Caller must hold the mcdi iface_lock.
818  */
819 static void cdx_mcdi_mode_fail(struct cdx_mcdi *cdx, struct list_head *cleanup_list)
820 {
821         struct cdx_mcdi_iface *mcdi = cdx_mcdi_if(cdx);
822
823         if (!mcdi)
824                 return;
825
826         mcdi->mode = MCDI_MODE_FAIL;
827
828         while (!list_empty(&mcdi->cmd_list)) {
829                 struct cdx_mcdi_cmd *cmd;
830
831                 cmd = list_first_entry(&mcdi->cmd_list, struct cdx_mcdi_cmd,
832                                        list);
833                 _cdx_mcdi_cancel_cmd(mcdi, cdx_mcdi_cmd_handle(cmd), cleanup_list);
834         }
835 }