fm10k: don't handle mailbox events in iov_event path and always process mailbox
authorJeff Kirsher <jeffrey.t.kirsher@intel.com>
Sat, 11 Apr 2015 00:20:17 +0000 (17:20 -0700)
committerJeff Kirsher <jeffrey.t.kirsher@intel.com>
Tue, 14 Apr 2015 22:47:28 +0000 (15:47 -0700)
Since we already schedule the service task, we can just wait for this
task to handle the mailbox events from the VF. This reduces some complex
code flow, and makes it so we have a single path for handling the VF
messages. There is a possibility that we have a slight delay in handling
VF messages, but it should be minimal.

The result of tx_complete and !rx_ready is insufficient to determine
whether we need to process the mailbox. There is a possible race
condition whereby the VF fills up the mbmem for us, but we have already
recently processed the mailboxes in the interrupt. During this time,
the interrupt is disabled. Thus, our Rx FIFO is empty, but the mbmem now
has data in it. Since we continually check whether Rx FIFO is empty, we
then never call process. This results in the possibility to prevent PF
from handling the VF mailbox messages.

Instead, just call process every time, despite the fact that we may or
may not have anything to process for the VF. There should be minimal
overhead for doing this, and it resolves an issue where the VF never
comes up due to never getting response for its SET_LPORT_STATE message.

Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
Signed-off-by: Jacob Keller <jacob.e.keller@intel.com>
Acked-by: Matthew Vick <matthew.vick@intel.com>
Tested-by: Krishneil Singh <krishneil.k.singh@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
drivers/net/ethernet/intel/fm10k/fm10k_iov.c
drivers/net/ethernet/intel/fm10k/fm10k_pci.c

index 69cbfde7fc09e039cd5e92295b029eeb5cf8a321..0b37e197e3005c09cdcb3322b3dad62c998efb25 100644 (file)
@@ -47,7 +47,7 @@ s32 fm10k_iov_event(struct fm10k_intfc *interface)
 {
        struct fm10k_hw *hw = &interface->hw;
        struct fm10k_iov_data *iov_data;
-       s64 mbicr, vflre;
+       s64 vflre;
        int i;
 
        /* if there is no iov_data then there is no mailboxes to process */
@@ -63,7 +63,7 @@ s32 fm10k_iov_event(struct fm10k_intfc *interface)
                goto read_unlock;
 
        if (!(fm10k_read_reg(hw, FM10K_EICR) & FM10K_EICR_VFLR))
-               goto process_mbx;
+               goto read_unlock;
 
        /* read VFLRE to determine if any VFs have been reset */
        do {
@@ -86,32 +86,6 @@ s32 fm10k_iov_event(struct fm10k_intfc *interface)
                }
        } while (i != iov_data->num_vfs);
 
-process_mbx:
-       /* read MBICR to determine which VFs require attention */
-       mbicr = fm10k_read_reg(hw, FM10K_MBICR(1));
-       mbicr <<= 32;
-       mbicr |= fm10k_read_reg(hw, FM10K_MBICR(0));
-
-       i = iov_data->next_vf_mbx ? : iov_data->num_vfs;
-
-       for (mbicr <<= 64 - i; i--; mbicr += mbicr) {
-               struct fm10k_mbx_info *mbx = &iov_data->vf_info[i].mbx;
-
-               if (mbicr >= 0)
-                       continue;
-
-               if (!hw->mbx.ops.tx_ready(&hw->mbx, FM10K_VFMBX_MSG_MTU))
-                       break;
-
-               mbx->ops.process(hw, mbx);
-       }
-
-       if (i >= 0) {
-               iov_data->next_vf_mbx = i + 1;
-       } else if (iov_data->next_vf_mbx) {
-               iov_data->next_vf_mbx = 0;
-               goto process_mbx;
-       }
 read_unlock:
        rcu_read_unlock();
 
@@ -155,10 +129,6 @@ process_mbx:
                        mbx->ops.connect(hw, mbx);
                }
 
-               /* no work pending, then just continue */
-               if (mbx->ops.tx_complete(mbx) && !mbx->ops.rx_ready(mbx))
-                       continue;
-
                /* guarantee we have free space in the SM mailbox */
                if (!hw->mbx.ops.tx_ready(&hw->mbx, FM10K_VFMBX_MSG_MTU))
                        break;
index 514f35c31d1d2a872a653ecf5992b980687b220f..32e99ebdae662c4d504f10d9b94dafbcd4a3d9da 100644 (file)
@@ -1006,6 +1006,7 @@ static irqreturn_t fm10k_msix_mbx_pf(int __always_unused irq, void *data)
        /* service mailboxes */
        if (fm10k_mbx_trylock(interface)) {
                mbx->ops.process(hw, mbx);
+               /* handle VFLRE events */
                fm10k_iov_event(interface);
                fm10k_mbx_unlock(interface);
        }
@@ -1022,6 +1023,8 @@ static irqreturn_t fm10k_msix_mbx_pf(int __always_unused irq, void *data)
 
        /* we should validate host state after interrupt event */
        hw->mac.get_host_state = 1;
+
+       /* validate host state, and handle VF mailboxes in the service task */
        fm10k_service_event_schedule(interface);
 
        /* re-enable mailbox interrupt and indicate 20us delay */