From: NorbertX Ciosek <norbertx.cio...@intel.com>

Check if CTS bit is set in the mailbox message before waiting for ACK.
Otherwise ACK will never be received causing the function to timeout. Add
a note for ixgbe_write_mbx that it should be called while holding a lock.

Signed-off-by: NorbertX Ciosek <norbertx.cio...@intel.com>
Signed-off-by: Anatoly Burakov <anatoly.bura...@intel.com>
---
 drivers/net/ixgbe/base/ixgbe_mbx.c | 14 ++++++++++++--
 1 file changed, 12 insertions(+), 2 deletions(-)

diff --git a/drivers/net/ixgbe/base/ixgbe_mbx.c 
b/drivers/net/ixgbe/base/ixgbe_mbx.c
index 23659266d0..fb8ea8ca68 100644
--- a/drivers/net/ixgbe/base/ixgbe_mbx.c
+++ b/drivers/net/ixgbe/base/ixgbe_mbx.c
@@ -82,6 +82,9 @@ s32 ixgbe_poll_mbx(struct ixgbe_hw *hw, u32 *msg, u16 size, 
u16 mbx_id)
  *
  * returns SUCCESS if it successfully copied message into the buffer and
  * received an ACK to that message within specified period
+ *
+ * Note that the caller to this function must lock before calling, since
+ * multiple threads can destroy each other messages.
  **/
 s32 ixgbe_write_mbx(struct ixgbe_hw *hw, u32 *msg, u16 size, u16 mbx_id)
 {
@@ -836,6 +839,11 @@ STATIC s32 ixgbe_obtain_mbx_lock_pf(struct ixgbe_hw *hw, 
u16 vf_id)
        while (countdown--) {
                /* Reserve mailbox for PF use */
                pf_mailbox = IXGBE_READ_REG(hw, IXGBE_PFMAILBOX(vf_id));
+
+               /* Check if other thread holds the PF lock already */
+               if (pf_mailbox & IXGBE_PFMAILBOX_PFU)
+                       goto retry;
+
                pf_mailbox |= IXGBE_PFMAILBOX_PFU;
                IXGBE_WRITE_REG(hw, IXGBE_PFMAILBOX(vf_id), pf_mailbox);
 
@@ -846,6 +854,7 @@ STATIC s32 ixgbe_obtain_mbx_lock_pf(struct ixgbe_hw *hw, 
u16 vf_id)
                        break;
                }
 
+       retry:
                /* Wait a bit before trying again */
                usec_delay(mbx->usec_delay);
        }
@@ -948,13 +957,14 @@ STATIC s32 ixgbe_write_mbx_pf(struct ixgbe_hw *hw, u32 
*msg, u16 size,
        for (i = 0; i < size; i++)
                IXGBE_WRITE_REG_ARRAY(hw, IXGBE_PFMBMEM(vf_id), i, msg[i]);
 
-       /* Interrupt VF to tell it a message has been sent */
+       /* interrupt VF to tell it a message has been sent */
        pf_mailbox = IXGBE_READ_REG(hw, IXGBE_PFMAILBOX(vf_id));
        pf_mailbox |= IXGBE_PFMAILBOX_STS;
        IXGBE_WRITE_REG(hw, IXGBE_PFMAILBOX(vf_id), pf_mailbox);
 
        /* if msg sent wait until we receive an ack */
-       ixgbe_poll_for_ack(hw, vf_id);
+       if (msg[0] & IXGBE_VT_MSGTYPE_CTS)
+               ixgbe_poll_for_ack(hw, vf_id);
 
        /* update stats */
        hw->mbx.stats.msgs_tx++;
-- 
2.43.5

Reply via email to