mirror of
https://github.com/torvalds/linux.git
synced 2026-04-26 10:32:25 -04:00
Implements the mailbox interfaces for wangxun pf drivers ngbe and txgbe. Signed-off-by: Mengyuan Lou <mengyuanlou@net-swift.com> Link: https://patch.msgid.link/70017BD4D67614A4+20250408091556.9640-2-mengyuanlou@net-swift.com Signed-off-by: Jakub Kicinski <kuba@kernel.org>
177 lines
4.1 KiB
C
177 lines
4.1 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/* Copyright (c) 2015 - 2025 Beijing WangXun Technology Co., Ltd. */
|
|
|
|
#include <linux/pci.h>
|
|
#include "wx_type.h"
|
|
#include "wx_mbx.h"
|
|
|
|
/**
|
|
* wx_obtain_mbx_lock_pf - obtain mailbox lock
|
|
* @wx: pointer to the HW structure
|
|
* @vf: the VF index
|
|
*
|
|
* Return: return 0 on success and -EBUSY on failure
|
|
**/
|
|
static int wx_obtain_mbx_lock_pf(struct wx *wx, u16 vf)
|
|
{
|
|
int count = 5;
|
|
u32 mailbox;
|
|
|
|
while (count--) {
|
|
/* Take ownership of the buffer */
|
|
wr32(wx, WX_PXMAILBOX(vf), WX_PXMAILBOX_PFU);
|
|
|
|
/* reserve mailbox for vf use */
|
|
mailbox = rd32(wx, WX_PXMAILBOX(vf));
|
|
if (mailbox & WX_PXMAILBOX_PFU)
|
|
return 0;
|
|
else if (count)
|
|
udelay(10);
|
|
}
|
|
wx_err(wx, "Failed to obtain mailbox lock for PF%d", vf);
|
|
|
|
return -EBUSY;
|
|
}
|
|
|
|
static int wx_check_for_bit_pf(struct wx *wx, u32 mask, int index)
|
|
{
|
|
u32 mbvficr = rd32(wx, WX_MBVFICR(index));
|
|
|
|
if (!(mbvficr & mask))
|
|
return -EBUSY;
|
|
wr32(wx, WX_MBVFICR(index), mask);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* wx_check_for_ack_pf - checks to see if the VF has acked
|
|
* @wx: pointer to the HW structure
|
|
* @vf: the VF index
|
|
*
|
|
* Return: return 0 if the VF has set the status bit or else -EBUSY
|
|
**/
|
|
int wx_check_for_ack_pf(struct wx *wx, u16 vf)
|
|
{
|
|
u32 index = vf / 16, vf_bit = vf % 16;
|
|
|
|
return wx_check_for_bit_pf(wx,
|
|
FIELD_PREP(WX_MBVFICR_VFACK_MASK,
|
|
BIT(vf_bit)),
|
|
index);
|
|
}
|
|
|
|
/**
|
|
* wx_check_for_msg_pf - checks to see if the VF has sent mail
|
|
* @wx: pointer to the HW structure
|
|
* @vf: the VF index
|
|
*
|
|
* Return: return 0 if the VF has got req bit or else -EBUSY
|
|
**/
|
|
int wx_check_for_msg_pf(struct wx *wx, u16 vf)
|
|
{
|
|
u32 index = vf / 16, vf_bit = vf % 16;
|
|
|
|
return wx_check_for_bit_pf(wx,
|
|
FIELD_PREP(WX_MBVFICR_VFREQ_MASK,
|
|
BIT(vf_bit)),
|
|
index);
|
|
}
|
|
|
|
/**
|
|
* wx_write_mbx_pf - Places a message in the mailbox
|
|
* @wx: pointer to the HW structure
|
|
* @msg: The message buffer
|
|
* @size: Length of buffer
|
|
* @vf: the VF index
|
|
*
|
|
* Return: return 0 on success and -EINVAL/-EBUSY on failure
|
|
**/
|
|
int wx_write_mbx_pf(struct wx *wx, u32 *msg, u16 size, u16 vf)
|
|
{
|
|
struct wx_mbx_info *mbx = &wx->mbx;
|
|
int ret, i;
|
|
|
|
/* mbx->size is up to 15 */
|
|
if (size > mbx->size) {
|
|
wx_err(wx, "Invalid mailbox message size %d", size);
|
|
return -EINVAL;
|
|
}
|
|
|
|
/* lock the mailbox to prevent pf/vf race condition */
|
|
ret = wx_obtain_mbx_lock_pf(wx, vf);
|
|
if (ret)
|
|
return ret;
|
|
|
|
/* flush msg and acks as we are overwriting the message buffer */
|
|
wx_check_for_msg_pf(wx, vf);
|
|
wx_check_for_ack_pf(wx, vf);
|
|
|
|
/* copy the caller specified message to the mailbox memory buffer */
|
|
for (i = 0; i < size; i++)
|
|
wr32a(wx, WX_PXMBMEM(vf), i, msg[i]);
|
|
|
|
/* Interrupt VF to tell it a message has been sent and release buffer */
|
|
/* set mirrored mailbox flags */
|
|
wr32a(wx, WX_PXMBMEM(vf), WX_VXMAILBOX_SIZE, WX_PXMAILBOX_STS);
|
|
wr32(wx, WX_PXMAILBOX(vf), WX_PXMAILBOX_STS);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* wx_read_mbx_pf - Read a message from the mailbox
|
|
* @wx: pointer to the HW structure
|
|
* @msg: The message buffer
|
|
* @size: Length of buffer
|
|
* @vf: the VF index
|
|
*
|
|
* Return: return 0 on success and -EBUSY on failure
|
|
**/
|
|
int wx_read_mbx_pf(struct wx *wx, u32 *msg, u16 size, u16 vf)
|
|
{
|
|
struct wx_mbx_info *mbx = &wx->mbx;
|
|
int ret;
|
|
u16 i;
|
|
|
|
/* limit read to size of mailbox and mbx->size is up to 15 */
|
|
if (size > mbx->size)
|
|
size = mbx->size;
|
|
|
|
/* lock the mailbox to prevent pf/vf race condition */
|
|
ret = wx_obtain_mbx_lock_pf(wx, vf);
|
|
if (ret)
|
|
return ret;
|
|
|
|
for (i = 0; i < size; i++)
|
|
msg[i] = rd32a(wx, WX_PXMBMEM(vf), i);
|
|
|
|
/* Acknowledge the message and release buffer */
|
|
/* set mirrored mailbox flags */
|
|
wr32a(wx, WX_PXMBMEM(vf), WX_VXMAILBOX_SIZE, WX_PXMAILBOX_ACK);
|
|
wr32(wx, WX_PXMAILBOX(vf), WX_PXMAILBOX_ACK);
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* wx_check_for_rst_pf - checks to see if the VF has reset
|
|
* @wx: pointer to the HW structure
|
|
* @vf: the VF index
|
|
*
|
|
* Return: return 0 on success and -EBUSY on failure
|
|
**/
|
|
int wx_check_for_rst_pf(struct wx *wx, u16 vf)
|
|
{
|
|
u32 reg_offset = WX_VF_REG_OFFSET(vf);
|
|
u32 vf_shift = WX_VF_IND_SHIFT(vf);
|
|
u32 vflre = 0;
|
|
|
|
vflre = rd32(wx, WX_VFLRE(reg_offset));
|
|
if (!(vflre & BIT(vf_shift)))
|
|
return -EBUSY;
|
|
wr32(wx, WX_VFLREC(reg_offset), BIT(vf_shift));
|
|
|
|
return 0;
|
|
}
|