Commit afdf841f authored by Raghu Vatsavayi's avatar Raghu Vatsavayi Committed by David S. Miller

liquidio: RX control commands

Adds support for RX control commands on cn23xx device.
Signed-off-by: default avatarDerek Chickles <derek.chickles@caviumnetworks.com>
Signed-off-by: default avatarSatanand Burla <satananda.burla@caviumnetworks.com>
Signed-off-by: default avatarFelix Manlunas <felix.manlunas@caviumnetworks.com>
Signed-off-by: default avatarRaghu Vatsavayi <raghu.vatsavayi@caviumnetworks.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 7b6b6c95
...@@ -96,6 +96,14 @@ struct liquidio_if_cfg_resp { ...@@ -96,6 +96,14 @@ struct liquidio_if_cfg_resp {
u64 status; u64 status;
}; };
struct liquidio_rx_ctl_context {
int octeon_id;
wait_queue_head_t wc;
int cond;
};
struct oct_link_status_resp { struct oct_link_status_resp {
u64 rh; u64 rh;
struct oct_link_info link_info; struct oct_link_info link_info;
...@@ -1377,6 +1385,34 @@ static void octeon_destroy_resources(struct octeon_device *oct) ...@@ -1377,6 +1385,34 @@ static void octeon_destroy_resources(struct octeon_device *oct)
tasklet_kill(&oct_priv->droq_tasklet); tasklet_kill(&oct_priv->droq_tasklet);
} }
/**
* \brief Callback for rx ctrl
* @param status status of request
* @param buf pointer to resp structure
*/
static void rx_ctl_callback(struct octeon_device *oct,
u32 status,
void *buf)
{
struct octeon_soft_command *sc = (struct octeon_soft_command *)buf;
struct liquidio_rx_ctl_context *ctx;
ctx = (struct liquidio_rx_ctl_context *)sc->ctxptr;
oct = lio_get_device(ctx->octeon_id);
if (status)
dev_err(&oct->pci_dev->dev, "rx ctl instruction failed. Status: %llx\n",
CVM_CAST64(status));
WRITE_ONCE(ctx->cond, 1);
/* This barrier is required to be sure that the response has been
* written fully before waking up the handler
*/
wmb();
wake_up_interruptible(&ctx->wc);
}
/** /**
* \brief Send Rx control command * \brief Send Rx control command
* @param lio per-network private data * @param lio per-network private data
...@@ -1384,17 +1420,55 @@ static void octeon_destroy_resources(struct octeon_device *oct) ...@@ -1384,17 +1420,55 @@ static void octeon_destroy_resources(struct octeon_device *oct)
*/ */
static void send_rx_ctrl_cmd(struct lio *lio, int start_stop) static void send_rx_ctrl_cmd(struct lio *lio, int start_stop)
{ {
struct octnic_ctrl_pkt nctrl; struct octeon_soft_command *sc;
struct liquidio_rx_ctl_context *ctx;
union octnet_cmd *ncmd;
int ctx_size = sizeof(struct liquidio_rx_ctl_context);
struct octeon_device *oct = (struct octeon_device *)lio->oct_dev;
int retval;
memset(&nctrl, 0, sizeof(struct octnic_ctrl_pkt)); if (oct->props[lio->ifidx].rx_on == start_stop)
return;
nctrl.ncmd.s.cmd = OCTNET_CMD_RX_CTL; sc = (struct octeon_soft_command *)
nctrl.ncmd.s.param1 = start_stop; octeon_alloc_soft_command(oct, OCTNET_CMD_SIZE,
nctrl.iq_no = lio->linfo.txpciq[0].s.q_no; 16, ctx_size);
nctrl.netpndev = (u64)lio->netdev;
ncmd = (union octnet_cmd *)sc->virtdptr;
ctx = (struct liquidio_rx_ctl_context *)sc->ctxptr;
WRITE_ONCE(ctx->cond, 0);
ctx->octeon_id = lio_get_device_id(oct);
init_waitqueue_head(&ctx->wc);
ncmd->u64 = 0;
ncmd->s.cmd = OCTNET_CMD_RX_CTL;
ncmd->s.param1 = start_stop;
octeon_swap_8B_data((u64 *)ncmd, (OCTNET_CMD_SIZE >> 3));
sc->iq_no = lio->linfo.txpciq[0].s.q_no;
octeon_prepare_soft_command(oct, sc, OPCODE_NIC,
OPCODE_NIC_CMD, 0, 0, 0);
sc->callback = rx_ctl_callback;
sc->callback_arg = sc;
sc->wait_time = 5000;
if (octnet_send_nic_ctrl_pkt(lio->oct_dev, &nctrl) < 0) retval = octeon_send_soft_command(oct, sc);
if (retval == IQ_SEND_FAILED) {
netif_info(lio, rx_err, lio->netdev, "Failed to send RX Control message\n"); netif_info(lio, rx_err, lio->netdev, "Failed to send RX Control message\n");
} else {
/* Sleep on a wait queue till the cond flag indicates that the
* response arrived or timed-out.
*/
if (sleep_cond(&ctx->wc, &ctx->cond) == -EINTR)
return;
oct->props[lio->ifidx].rx_on = start_stop;
}
octeon_free_soft_command(oct, sc);
} }
/** /**
...@@ -1421,10 +1495,8 @@ static void liquidio_destroy_nic_device(struct octeon_device *oct, int ifidx) ...@@ -1421,10 +1495,8 @@ static void liquidio_destroy_nic_device(struct octeon_device *oct, int ifidx)
dev_dbg(&oct->pci_dev->dev, "NIC device cleanup\n"); dev_dbg(&oct->pci_dev->dev, "NIC device cleanup\n");
send_rx_ctrl_cmd(lio, 0);
if (atomic_read(&lio->ifstate) & LIO_IFSTATE_RUNNING) if (atomic_read(&lio->ifstate) & LIO_IFSTATE_RUNNING)
txqs_stop(netdev); liquidio_stop(netdev);
if (oct->props[lio->ifidx].napi_enabled == 1) { if (oct->props[lio->ifidx].napi_enabled == 1) {
list_for_each_entry_safe(napi, n, &netdev->napi_list, dev_list) list_for_each_entry_safe(napi, n, &netdev->napi_list, dev_list)
...@@ -3567,7 +3639,11 @@ static int setup_nic_devices(struct octeon_device *octeon_dev) ...@@ -3567,7 +3639,11 @@ static int setup_nic_devices(struct octeon_device *octeon_dev)
/* Sleep on a wait queue till the cond flag indicates that the /* Sleep on a wait queue till the cond flag indicates that the
* response arrived or timed-out. * response arrived or timed-out.
*/ */
sleep_cond(&ctx->wc, &ctx->cond); if (sleep_cond(&ctx->wc, &ctx->cond) == -EINTR) {
dev_err(&octeon_dev->pci_dev->dev, "Wait interrupted\n");
goto setup_nic_wait_intr;
}
retval = resp->status; retval = resp->status;
if (retval) { if (retval) {
dev_err(&octeon_dev->pci_dev->dev, "iq/oq config failed\n"); dev_err(&octeon_dev->pci_dev->dev, "iq/oq config failed\n");
...@@ -3768,6 +3844,8 @@ static int setup_nic_devices(struct octeon_device *octeon_dev) ...@@ -3768,6 +3844,8 @@ static int setup_nic_devices(struct octeon_device *octeon_dev)
octeon_free_soft_command(octeon_dev, sc); octeon_free_soft_command(octeon_dev, sc);
setup_nic_wait_intr:
while (i--) { while (i--) {
dev_err(&octeon_dev->pci_dev->dev, dev_err(&octeon_dev->pci_dev->dev,
"NIC ifidx:%d Setup failed\n", i); "NIC ifidx:%d Setup failed\n", i);
......
...@@ -275,6 +275,7 @@ struct octdev_props { ...@@ -275,6 +275,7 @@ struct octdev_props {
/* Each interface in the Octeon device has a network /* Each interface in the Octeon device has a network
* device pointer (used for OS specific calls). * device pointer (used for OS specific calls).
*/ */
int rx_on;
int napi_enabled; int napi_enabled;
int gmxport; int gmxport;
struct net_device *netdev; struct net_device *netdev;
......
...@@ -181,22 +181,26 @@ cnnic_numa_alloc_aligned_dma(u32 size, ...@@ -181,22 +181,26 @@ cnnic_numa_alloc_aligned_dma(u32 size,
#define cnnic_free_aligned_dma(pci_dev, ptr, size, orig_ptr, dma_addr) \ #define cnnic_free_aligned_dma(pci_dev, ptr, size, orig_ptr, dma_addr) \
free_pages(orig_ptr, get_order(size)) free_pages(orig_ptr, get_order(size))
static inline void static inline int
sleep_cond(wait_queue_head_t *wait_queue, int *condition) sleep_cond(wait_queue_head_t *wait_queue, int *condition)
{ {
int errno = 0;
wait_queue_t we; wait_queue_t we;
init_waitqueue_entry(&we, current); init_waitqueue_entry(&we, current);
add_wait_queue(wait_queue, &we); add_wait_queue(wait_queue, &we);
while (!(READ_ONCE(*condition))) { while (!(READ_ONCE(*condition))) {
set_current_state(TASK_INTERRUPTIBLE); set_current_state(TASK_INTERRUPTIBLE);
if (signal_pending(current)) if (signal_pending(current)) {
errno = -EINTR;
goto out; goto out;
}
schedule(); schedule();
} }
out: out:
set_current_state(TASK_RUNNING); set_current_state(TASK_RUNNING);
remove_wait_queue(wait_queue, &we); remove_wait_queue(wait_queue, &we);
return errno;
} }
static inline void static inline void
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment