diff options
-rw-r--r-- | drivers/misc/mei/hw-me.c | 164 | ||||
-rw-r--r-- | drivers/misc/mei/hw-me.h | 3 | ||||
-rw-r--r-- | drivers/misc/mei/interrupt.c | 126 | ||||
-rw-r--r-- | drivers/misc/mei/mei_dev.h | 9 | ||||
-rw-r--r-- | drivers/misc/mei/pci-me.c | 12 |
5 files changed, 164 insertions, 150 deletions
diff --git a/drivers/misc/mei/hw-me.c b/drivers/misc/mei/hw-me.c index 6300943497ae..3bebf8d85ff9 100644 --- a/drivers/misc/mei/hw-me.c +++ b/drivers/misc/mei/hw-me.c | |||
@@ -15,11 +15,16 @@ | |||
15 | */ | 15 | */ |
16 | 16 | ||
17 | #include <linux/pci.h> | 17 | #include <linux/pci.h> |
18 | #include <linux/mei.h> | 18 | |
19 | #include <linux/kthread.h> | ||
20 | #include <linux/interrupt.h> | ||
19 | 21 | ||
20 | #include "mei_dev.h" | 22 | #include "mei_dev.h" |
21 | #include "hw-me.h" | 23 | #include "hw-me.h" |
22 | 24 | ||
25 | #include "hbm.h" | ||
26 | |||
27 | |||
23 | /** | 28 | /** |
24 | * mei_reg_read - Reads 32bit data from the mei device | 29 | * mei_reg_read - Reads 32bit data from the mei device |
25 | * | 30 | * |
@@ -218,29 +223,6 @@ static bool mei_me_hw_is_ready(struct mei_device *dev) | |||
218 | } | 223 | } |
219 | 224 | ||
220 | /** | 225 | /** |
221 | * mei_interrupt_quick_handler - The ISR of the MEI device | ||
222 | * | ||
223 | * @irq: The irq number | ||
224 | * @dev_id: pointer to the device structure | ||
225 | * | ||
226 | * returns irqreturn_t | ||
227 | */ | ||
228 | irqreturn_t mei_interrupt_quick_handler(int irq, void *dev_id) | ||
229 | { | ||
230 | struct mei_device *dev = (struct mei_device *) dev_id; | ||
231 | struct mei_me_hw *hw = to_me_hw(dev); | ||
232 | u32 csr_reg = mei_hcsr_read(hw); | ||
233 | |||
234 | if ((csr_reg & H_IS) != H_IS) | ||
235 | return IRQ_NONE; | ||
236 | |||
237 | /* clear H_IS bit in H_CSR */ | ||
238 | mei_reg_write(hw, H_CSR, csr_reg); | ||
239 | |||
240 | return IRQ_WAKE_THREAD; | ||
241 | } | ||
242 | |||
243 | /** | ||
244 | * mei_hbuf_filled_slots - gets number of device filled buffer slots | 226 | * mei_hbuf_filled_slots - gets number of device filled buffer slots |
245 | * | 227 | * |
246 | * @dev: the device structure | 228 | * @dev: the device structure |
@@ -403,6 +385,139 @@ static int mei_me_read_slots(struct mei_device *dev, unsigned char *buffer, | |||
403 | return 0; | 385 | return 0; |
404 | } | 386 | } |
405 | 387 | ||
388 | /** | ||
389 | * mei_me_irq_quick_handler - The ISR of the MEI device | ||
390 | * | ||
391 | * @irq: The irq number | ||
392 | * @dev_id: pointer to the device structure | ||
393 | * | ||
394 | * returns irqreturn_t | ||
395 | */ | ||
396 | |||
397 | irqreturn_t mei_me_irq_quick_handler(int irq, void *dev_id) | ||
398 | { | ||
399 | struct mei_device *dev = (struct mei_device *) dev_id; | ||
400 | struct mei_me_hw *hw = to_me_hw(dev); | ||
401 | u32 csr_reg = mei_hcsr_read(hw); | ||
402 | |||
403 | if ((csr_reg & H_IS) != H_IS) | ||
404 | return IRQ_NONE; | ||
405 | |||
406 | /* clear H_IS bit in H_CSR */ | ||
407 | mei_reg_write(hw, H_CSR, csr_reg); | ||
408 | |||
409 | return IRQ_WAKE_THREAD; | ||
410 | } | ||
411 | |||
412 | /** | ||
413 | * mei_me_irq_thread_handler - function called after ISR to handle the interrupt | ||
414 | * processing. | ||
415 | * | ||
416 | * @irq: The irq number | ||
417 | * @dev_id: pointer to the device structure | ||
418 | * | ||
419 | * returns irqreturn_t | ||
420 | * | ||
421 | */ | ||
422 | irqreturn_t mei_me_irq_thread_handler(int irq, void *dev_id) | ||
423 | { | ||
424 | struct mei_device *dev = (struct mei_device *) dev_id; | ||
425 | struct mei_cl_cb complete_list; | ||
426 | struct mei_cl_cb *cb_pos = NULL, *cb_next = NULL; | ||
427 | struct mei_cl *cl; | ||
428 | s32 slots; | ||
429 | int rets; | ||
430 | bool bus_message_received; | ||
431 | |||
432 | |||
433 | dev_dbg(&dev->pdev->dev, "function called after ISR to handle the interrupt processing.\n"); | ||
434 | /* initialize our complete list */ | ||
435 | mutex_lock(&dev->device_lock); | ||
436 | mei_io_list_init(&complete_list); | ||
437 | |||
438 | /* Ack the interrupt here | ||
439 | * In case of MSI we don't go through the quick handler */ | ||
440 | if (pci_dev_msi_enabled(dev->pdev)) | ||
441 | mei_clear_interrupts(dev); | ||
442 | |||
443 | /* check if ME wants a reset */ | ||
444 | if (!mei_hw_is_ready(dev) && | ||
445 | dev->dev_state != MEI_DEV_RESETING && | ||
446 | dev->dev_state != MEI_DEV_INITIALIZING) { | ||
447 | dev_dbg(&dev->pdev->dev, "FW not ready.\n"); | ||
448 | mei_reset(dev, 1); | ||
449 | mutex_unlock(&dev->device_lock); | ||
450 | return IRQ_HANDLED; | ||
451 | } | ||
452 | |||
453 | /* check if we need to start the dev */ | ||
454 | if (!mei_host_is_ready(dev)) { | ||
455 | if (mei_hw_is_ready(dev)) { | ||
456 | dev_dbg(&dev->pdev->dev, "we need to start the dev.\n"); | ||
457 | |||
458 | mei_host_set_ready(dev); | ||
459 | |||
460 | dev_dbg(&dev->pdev->dev, "link is established start sending messages.\n"); | ||
461 | /* link is established * start sending messages. */ | ||
462 | |||
463 | dev->dev_state = MEI_DEV_INIT_CLIENTS; | ||
464 | |||
465 | mei_hbm_start_req(dev); | ||
466 | mutex_unlock(&dev->device_lock); | ||
467 | return IRQ_HANDLED; | ||
468 | } else { | ||
469 | dev_dbg(&dev->pdev->dev, "FW not ready.\n"); | ||
470 | mutex_unlock(&dev->device_lock); | ||
471 | return IRQ_HANDLED; | ||
472 | } | ||
473 | } | ||
474 | /* check slots available for reading */ | ||
475 | slots = mei_count_full_read_slots(dev); | ||
476 | while (slots > 0) { | ||
477 | /* we have urgent data to send so break the read */ | ||
478 | if (dev->wr_ext_msg.hdr.length) | ||
479 | break; | ||
480 | dev_dbg(&dev->pdev->dev, "slots =%08x\n", slots); | ||
481 | dev_dbg(&dev->pdev->dev, "call mei_irq_read_handler.\n"); | ||
482 | rets = mei_irq_read_handler(dev, &complete_list, &slots); | ||
483 | if (rets) | ||
484 | goto end; | ||
485 | } | ||
486 | rets = mei_irq_write_handler(dev, &complete_list); | ||
487 | end: | ||
488 | dev_dbg(&dev->pdev->dev, "end of bottom half function.\n"); | ||
489 | dev->mei_host_buffer_is_empty = mei_hbuf_is_ready(dev); | ||
490 | |||
491 | bus_message_received = false; | ||
492 | if (dev->recvd_msg && waitqueue_active(&dev->wait_recvd_msg)) { | ||
493 | dev_dbg(&dev->pdev->dev, "received waiting bus message\n"); | ||
494 | bus_message_received = true; | ||
495 | } | ||
496 | mutex_unlock(&dev->device_lock); | ||
497 | if (bus_message_received) { | ||
498 | dev_dbg(&dev->pdev->dev, "wake up dev->wait_recvd_msg\n"); | ||
499 | wake_up_interruptible(&dev->wait_recvd_msg); | ||
500 | bus_message_received = false; | ||
501 | } | ||
502 | if (list_empty(&complete_list.list)) | ||
503 | return IRQ_HANDLED; | ||
504 | |||
505 | |||
506 | list_for_each_entry_safe(cb_pos, cb_next, &complete_list.list, list) { | ||
507 | cl = cb_pos->cl; | ||
508 | list_del(&cb_pos->list); | ||
509 | if (cl) { | ||
510 | if (cl != &dev->iamthif_cl) { | ||
511 | dev_dbg(&dev->pdev->dev, "completing call back.\n"); | ||
512 | mei_irq_complete_handler(cl, cb_pos); | ||
513 | cb_pos = NULL; | ||
514 | } else if (cl == &dev->iamthif_cl) { | ||
515 | mei_amthif_complete(dev, cb_pos); | ||
516 | } | ||
517 | } | ||
518 | } | ||
519 | return IRQ_HANDLED; | ||
520 | } | ||
406 | static const struct mei_hw_ops mei_me_hw_ops = { | 521 | static const struct mei_hw_ops mei_me_hw_ops = { |
407 | 522 | ||
408 | .host_set_ready = mei_me_host_set_ready, | 523 | .host_set_ready = mei_me_host_set_ready, |
@@ -458,3 +573,4 @@ struct mei_device *mei_me_dev_init(struct pci_dev *pdev) | |||
458 | dev->pdev = pdev; | 573 | dev->pdev = pdev; |
459 | return dev; | 574 | return dev; |
460 | } | 575 | } |
576 | |||
diff --git a/drivers/misc/mei/hw-me.h b/drivers/misc/mei/hw-me.h index 9a3aaab9bcf0..8518d3eeb838 100644 --- a/drivers/misc/mei/hw-me.h +++ b/drivers/misc/mei/hw-me.h | |||
@@ -42,4 +42,7 @@ static inline unsigned char mei_data2slots(size_t length) | |||
42 | return DIV_ROUND_UP(sizeof(struct mei_msg_hdr) + length, 4); | 42 | return DIV_ROUND_UP(sizeof(struct mei_msg_hdr) + length, 4); |
43 | } | 43 | } |
44 | 44 | ||
45 | irqreturn_t mei_me_irq_quick_handler(int irq, void *dev_id); | ||
46 | irqreturn_t mei_me_irq_thread_handler(int irq, void *dev_id); | ||
47 | |||
45 | #endif /* _MEI_INTERFACE_H_ */ | 48 | #endif /* _MEI_INTERFACE_H_ */ |
diff --git a/drivers/misc/mei/interrupt.c b/drivers/misc/mei/interrupt.c index 431aa91fd002..3535b2676c97 100644 --- a/drivers/misc/mei/interrupt.c +++ b/drivers/misc/mei/interrupt.c | |||
@@ -30,12 +30,12 @@ | |||
30 | 30 | ||
31 | 31 | ||
32 | /** | 32 | /** |
33 | * _mei_cmpl - processes completed operation. | 33 | * mei_complete_handler - processes completed operation. |
34 | * | 34 | * |
35 | * @cl: private data of the file object. | 35 | * @cl: private data of the file object. |
36 | * @cb_pos: callback block. | 36 | * @cb_pos: callback block. |
37 | */ | 37 | */ |
38 | static void _mei_cmpl(struct mei_cl *cl, struct mei_cl_cb *cb_pos) | 38 | void mei_irq_complete_handler(struct mei_cl *cl, struct mei_cl_cb *cb_pos) |
39 | { | 39 | { |
40 | if (cb_pos->fop_type == MEI_FOP_WRITE) { | 40 | if (cb_pos->fop_type == MEI_FOP_WRITE) { |
41 | mei_io_cb_free(cb_pos); | 41 | mei_io_cb_free(cb_pos); |
@@ -313,15 +313,14 @@ static int mei_irq_thread_write_complete(struct mei_device *dev, s32 *slots, | |||
313 | * mei_irq_thread_read_handler - bottom half read routine after ISR to | 313 | * mei_irq_thread_read_handler - bottom half read routine after ISR to |
314 | * handle the read processing. | 314 | * handle the read processing. |
315 | * | 315 | * |
316 | * @cmpl_list: An instance of our list structure | ||
317 | * @dev: the device structure | 316 | * @dev: the device structure |
317 | * @cmpl_list: An instance of our list structure | ||
318 | * @slots: slots to read. | 318 | * @slots: slots to read. |
319 | * | 319 | * |
320 | * returns 0 on success, <0 on failure. | 320 | * returns 0 on success, <0 on failure. |
321 | */ | 321 | */ |
322 | static int mei_irq_thread_read_handler(struct mei_cl_cb *cmpl_list, | 322 | int mei_irq_read_handler(struct mei_device *dev, |
323 | struct mei_device *dev, | 323 | struct mei_cl_cb *cmpl_list, s32 *slots) |
324 | s32 *slots) | ||
325 | { | 324 | { |
326 | struct mei_msg_hdr *mei_hdr; | 325 | struct mei_msg_hdr *mei_hdr; |
327 | struct mei_cl *cl_pos = NULL; | 326 | struct mei_cl *cl_pos = NULL; |
@@ -412,15 +411,15 @@ end: | |||
412 | 411 | ||
413 | 412 | ||
414 | /** | 413 | /** |
415 | * mei_irq_thread_write_handler - bottom half write routine after | 414 | * mei_irq_write_handler - dispatch write requests |
416 | * ISR to handle the write processing. | 415 | * after irq received |
417 | * | 416 | * |
418 | * @dev: the device structure | 417 | * @dev: the device structure |
419 | * @cmpl_list: An instance of our list structure | 418 | * @cmpl_list: An instance of our list structure |
420 | * | 419 | * |
421 | * returns 0 on success, <0 on failure. | 420 | * returns 0 on success, <0 on failure. |
422 | */ | 421 | */ |
423 | static int mei_irq_thread_write_handler(struct mei_device *dev, | 422 | int mei_irq_write_handler(struct mei_device *dev, |
424 | struct mei_cl_cb *cmpl_list) | 423 | struct mei_cl_cb *cmpl_list) |
425 | { | 424 | { |
426 | 425 | ||
@@ -666,112 +665,3 @@ out: | |||
666 | mutex_unlock(&dev->device_lock); | 665 | mutex_unlock(&dev->device_lock); |
667 | } | 666 | } |
668 | 667 | ||
669 | /** | ||
670 | * mei_interrupt_thread_handler - function called after ISR to handle the interrupt | ||
671 | * processing. | ||
672 | * | ||
673 | * @irq: The irq number | ||
674 | * @dev_id: pointer to the device structure | ||
675 | * | ||
676 | * returns irqreturn_t | ||
677 | * | ||
678 | */ | ||
679 | irqreturn_t mei_interrupt_thread_handler(int irq, void *dev_id) | ||
680 | { | ||
681 | struct mei_device *dev = (struct mei_device *) dev_id; | ||
682 | struct mei_cl_cb complete_list; | ||
683 | struct mei_cl_cb *cb_pos = NULL, *cb_next = NULL; | ||
684 | struct mei_cl *cl; | ||
685 | s32 slots; | ||
686 | int rets; | ||
687 | bool bus_message_received; | ||
688 | |||
689 | |||
690 | dev_dbg(&dev->pdev->dev, "function called after ISR to handle the interrupt processing.\n"); | ||
691 | /* initialize our complete list */ | ||
692 | mutex_lock(&dev->device_lock); | ||
693 | mei_io_list_init(&complete_list); | ||
694 | |||
695 | /* Ack the interrupt here | ||
696 | * In case of MSI we don't go through the quick handler */ | ||
697 | if (pci_dev_msi_enabled(dev->pdev)) | ||
698 | mei_clear_interrupts(dev); | ||
699 | |||
700 | /* check if ME wants a reset */ | ||
701 | if (!mei_hw_is_ready(dev) && | ||
702 | dev->dev_state != MEI_DEV_RESETING && | ||
703 | dev->dev_state != MEI_DEV_INITIALIZING) { | ||
704 | dev_dbg(&dev->pdev->dev, "FW not ready.\n"); | ||
705 | mei_reset(dev, 1); | ||
706 | mutex_unlock(&dev->device_lock); | ||
707 | return IRQ_HANDLED; | ||
708 | } | ||
709 | |||
710 | /* check if we need to start the dev */ | ||
711 | if (!mei_host_is_ready(dev)) { | ||
712 | if (mei_hw_is_ready(dev)) { | ||
713 | dev_dbg(&dev->pdev->dev, "we need to start the dev.\n"); | ||
714 | |||
715 | mei_host_set_ready(dev); | ||
716 | |||
717 | dev_dbg(&dev->pdev->dev, "link is established start sending messages.\n"); | ||
718 | /* link is established * start sending messages. */ | ||
719 | |||
720 | dev->dev_state = MEI_DEV_INIT_CLIENTS; | ||
721 | |||
722 | mei_hbm_start_req(dev); | ||
723 | mutex_unlock(&dev->device_lock); | ||
724 | return IRQ_HANDLED; | ||
725 | } else { | ||
726 | dev_dbg(&dev->pdev->dev, "FW not ready.\n"); | ||
727 | mutex_unlock(&dev->device_lock); | ||
728 | return IRQ_HANDLED; | ||
729 | } | ||
730 | } | ||
731 | /* check slots available for reading */ | ||
732 | slots = mei_count_full_read_slots(dev); | ||
733 | while (slots > 0) { | ||
734 | /* we have urgent data to send so break the read */ | ||
735 | if (dev->wr_ext_msg.hdr.length) | ||
736 | break; | ||
737 | dev_dbg(&dev->pdev->dev, "slots =%08x\n", slots); | ||
738 | dev_dbg(&dev->pdev->dev, "call mei_irq_thread_read_handler.\n"); | ||
739 | rets = mei_irq_thread_read_handler(&complete_list, dev, &slots); | ||
740 | if (rets) | ||
741 | goto end; | ||
742 | } | ||
743 | rets = mei_irq_thread_write_handler(dev, &complete_list); | ||
744 | end: | ||
745 | dev_dbg(&dev->pdev->dev, "end of bottom half function.\n"); | ||
746 | dev->mei_host_buffer_is_empty = mei_hbuf_is_ready(dev); | ||
747 | |||
748 | bus_message_received = false; | ||
749 | if (dev->recvd_msg && waitqueue_active(&dev->wait_recvd_msg)) { | ||
750 | dev_dbg(&dev->pdev->dev, "received waiting bus message\n"); | ||
751 | bus_message_received = true; | ||
752 | } | ||
753 | mutex_unlock(&dev->device_lock); | ||
754 | if (bus_message_received) { | ||
755 | dev_dbg(&dev->pdev->dev, "wake up dev->wait_recvd_msg\n"); | ||
756 | wake_up_interruptible(&dev->wait_recvd_msg); | ||
757 | bus_message_received = false; | ||
758 | } | ||
759 | if (list_empty(&complete_list.list)) | ||
760 | return IRQ_HANDLED; | ||
761 | |||
762 | |||
763 | list_for_each_entry_safe(cb_pos, cb_next, &complete_list.list, list) { | ||
764 | cl = cb_pos->cl; | ||
765 | list_del(&cb_pos->list); | ||
766 | if (cl) { | ||
767 | if (cl != &dev->iamthif_cl) { | ||
768 | dev_dbg(&dev->pdev->dev, "completing call back.\n"); | ||
769 | _mei_cmpl(cl, cb_pos); | ||
770 | cb_pos = NULL; | ||
771 | } else if (cl == &dev->iamthif_cl) { | ||
772 | mei_amthif_complete(dev, cb_pos); | ||
773 | } | ||
774 | } | ||
775 | } | ||
776 | return IRQ_HANDLED; | ||
777 | } | ||
diff --git a/drivers/misc/mei/mei_dev.h b/drivers/misc/mei/mei_dev.h index c974292f16d6..7d07cef75664 100644 --- a/drivers/misc/mei/mei_dev.h +++ b/drivers/misc/mei/mei_dev.h | |||
@@ -379,9 +379,14 @@ int mei_hw_init(struct mei_device *dev); | |||
379 | /* | 379 | /* |
380 | * MEI interrupt functions prototype | 380 | * MEI interrupt functions prototype |
381 | */ | 381 | */ |
382 | irqreturn_t mei_interrupt_quick_handler(int irq, void *dev_id); | 382 | |
383 | irqreturn_t mei_interrupt_thread_handler(int irq, void *dev_id); | ||
384 | void mei_timer(struct work_struct *work); | 383 | void mei_timer(struct work_struct *work); |
384 | int mei_irq_read_handler(struct mei_device *dev, | ||
385 | struct mei_cl_cb *cmpl_list, s32 *slots); | ||
386 | |||
387 | int mei_irq_write_handler(struct mei_device *dev, struct mei_cl_cb *cmpl_list); | ||
388 | |||
389 | void mei_irq_complete_handler(struct mei_cl *cl, struct mei_cl_cb *cb_pos); | ||
385 | 390 | ||
386 | /* | 391 | /* |
387 | * AMTHIF - AMT Host Interface Functions | 392 | * AMTHIF - AMT Host Interface Functions |
diff --git a/drivers/misc/mei/pci-me.c b/drivers/misc/mei/pci-me.c index 27ac71767981..b40ec0601ab0 100644 --- a/drivers/misc/mei/pci-me.c +++ b/drivers/misc/mei/pci-me.c | |||
@@ -170,12 +170,12 @@ static int mei_probe(struct pci_dev *pdev, const struct pci_device_id *ent) | |||
170 | if (pci_dev_msi_enabled(pdev)) | 170 | if (pci_dev_msi_enabled(pdev)) |
171 | err = request_threaded_irq(pdev->irq, | 171 | err = request_threaded_irq(pdev->irq, |
172 | NULL, | 172 | NULL, |
173 | mei_interrupt_thread_handler, | 173 | mei_me_irq_thread_handler, |
174 | IRQF_ONESHOT, KBUILD_MODNAME, dev); | 174 | IRQF_ONESHOT, KBUILD_MODNAME, dev); |
175 | else | 175 | else |
176 | err = request_threaded_irq(pdev->irq, | 176 | err = request_threaded_irq(pdev->irq, |
177 | mei_interrupt_quick_handler, | 177 | mei_me_irq_quick_handler, |
178 | mei_interrupt_thread_handler, | 178 | mei_me_irq_thread_handler, |
179 | IRQF_SHARED, KBUILD_MODNAME, dev); | 179 | IRQF_SHARED, KBUILD_MODNAME, dev); |
180 | 180 | ||
181 | if (err) { | 181 | if (err) { |
@@ -348,12 +348,12 @@ static int mei_pci_resume(struct device *device) | |||
348 | if (pci_dev_msi_enabled(pdev)) | 348 | if (pci_dev_msi_enabled(pdev)) |
349 | err = request_threaded_irq(pdev->irq, | 349 | err = request_threaded_irq(pdev->irq, |
350 | NULL, | 350 | NULL, |
351 | mei_interrupt_thread_handler, | 351 | mei_me_irq_thread_handler, |
352 | IRQF_ONESHOT, KBUILD_MODNAME, dev); | 352 | IRQF_ONESHOT, KBUILD_MODNAME, dev); |
353 | else | 353 | else |
354 | err = request_threaded_irq(pdev->irq, | 354 | err = request_threaded_irq(pdev->irq, |
355 | mei_interrupt_quick_handler, | 355 | mei_me_irq_quick_handler, |
356 | mei_interrupt_thread_handler, | 356 | mei_me_irq_thread_handler, |
357 | IRQF_SHARED, KBUILD_MODNAME, dev); | 357 | IRQF_SHARED, KBUILD_MODNAME, dev); |
358 | 358 | ||
359 | if (err) { | 359 | if (err) { |