mirror of
https://github.com/corundum/corundum.git
synced 2025-01-16 08:12:53 +08:00
54d0165f68
Signed-off-by: Alex Forencich <alex@alexforencich.com>
398 lines
13 KiB
C
398 lines
13 KiB
C
// SPDX-License-Identifier: BSD-2-Clause-Views
|
|
/*
|
|
* Copyright (c) 2021-2023 The Regents of the University of California
|
|
*/
|
|
|
|
#include "mqnic.h"
|
|
|
|
struct mqnic_if *mqnic_create_interface(struct mqnic_dev *mdev, int index, u8 __iomem *hw_addr)
|
|
{
|
|
struct device *dev = mdev->dev;
|
|
struct mqnic_if *interface;
|
|
struct mqnic_reg_block *rb;
|
|
int ret = 0;
|
|
int k;
|
|
u32 count, offset, stride;
|
|
u32 desc_block_size;
|
|
u32 val;
|
|
|
|
interface = kzalloc(sizeof(*interface), GFP_KERNEL);
|
|
if (!interface)
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
interface->mdev = mdev;
|
|
interface->dev = dev;
|
|
|
|
interface->index = index;
|
|
|
|
interface->hw_regs_size = mdev->if_stride;
|
|
interface->hw_addr = hw_addr;
|
|
interface->csr_hw_addr = hw_addr + mdev->if_csr_offset;
|
|
|
|
// Enumerate registers
|
|
interface->rb_list = mqnic_enumerate_reg_block_list(interface->hw_addr, mdev->if_csr_offset, interface->hw_regs_size);
|
|
if (!interface->rb_list) {
|
|
ret = -EIO;
|
|
dev_err(dev, "Failed to enumerate blocks");
|
|
goto fail;
|
|
}
|
|
|
|
dev_info(dev, "Interface-level register blocks:");
|
|
for (rb = interface->rb_list; rb->regs; rb++)
|
|
dev_info(dev, " type 0x%08x (v %d.%d.%d.%d)", rb->type, rb->version >> 24,
|
|
(rb->version >> 16) & 0xff, (rb->version >> 8) & 0xff, rb->version & 0xff);
|
|
|
|
interface->if_ctrl_rb = mqnic_find_reg_block(interface->rb_list, MQNIC_RB_IF_CTRL_TYPE, MQNIC_RB_IF_CTRL_VER, 0);
|
|
|
|
if (!interface->if_ctrl_rb) {
|
|
ret = -EIO;
|
|
dev_err(dev, "Interface control block not found");
|
|
goto fail;
|
|
}
|
|
|
|
interface->if_features = ioread32(interface->if_ctrl_rb->regs + MQNIC_RB_IF_CTRL_REG_FEATURES);
|
|
interface->port_count = ioread32(interface->if_ctrl_rb->regs + MQNIC_RB_IF_CTRL_REG_PORT_COUNT);
|
|
interface->sched_block_count = ioread32(interface->if_ctrl_rb->regs + MQNIC_RB_IF_CTRL_REG_SCHED_COUNT);
|
|
interface->max_tx_mtu = ioread32(interface->if_ctrl_rb->regs + MQNIC_RB_IF_CTRL_REG_MAX_TX_MTU);
|
|
interface->max_rx_mtu = ioread32(interface->if_ctrl_rb->regs + MQNIC_RB_IF_CTRL_REG_MAX_RX_MTU);
|
|
interface->tx_fifo_depth = ioread32(interface->if_ctrl_rb->regs + MQNIC_RB_IF_CTRL_REG_TX_FIFO_DEPTH);
|
|
interface->rx_fifo_depth = ioread32(interface->if_ctrl_rb->regs + MQNIC_RB_IF_CTRL_REG_RX_FIFO_DEPTH);
|
|
|
|
dev_info(dev, "IF features: 0x%08x", interface->if_features);
|
|
dev_info(dev, "Port count: %d", interface->port_count);
|
|
dev_info(dev, "Scheduler block count: %d", interface->sched_block_count);
|
|
dev_info(dev, "Max TX MTU: %d", interface->max_tx_mtu);
|
|
dev_info(dev, "Max RX MTU: %d", interface->max_rx_mtu);
|
|
|
|
interface->eq_rb = mqnic_find_reg_block(interface->rb_list, MQNIC_RB_EQM_TYPE, MQNIC_RB_EQM_VER, 0);
|
|
|
|
if (!interface->eq_rb) {
|
|
ret = -EIO;
|
|
dev_err(dev, "EQ block not found");
|
|
goto fail;
|
|
}
|
|
|
|
offset = ioread32(interface->eq_rb->regs + MQNIC_RB_EQM_REG_OFFSET);
|
|
count = ioread32(interface->eq_rb->regs + MQNIC_RB_EQM_REG_COUNT);
|
|
stride = ioread32(interface->eq_rb->regs + MQNIC_RB_EQM_REG_STRIDE);
|
|
|
|
dev_info(dev, "EQ offset: 0x%08x", offset);
|
|
dev_info(dev, "EQ count: %d", count);
|
|
dev_info(dev, "EQ stride: 0x%08x", stride);
|
|
|
|
count = min_t(u32, count, MQNIC_MAX_EQ);
|
|
|
|
interface->eq_res = mqnic_create_res(count, hw_addr + offset, stride);
|
|
|
|
if (IS_ERR_OR_NULL(interface->eq_res)) {
|
|
ret = PTR_ERR(interface->eq_res);
|
|
goto fail;
|
|
}
|
|
|
|
interface->cq_rb = mqnic_find_reg_block(interface->rb_list, MQNIC_RB_CQM_TYPE, MQNIC_RB_CQM_VER, 0);
|
|
|
|
if (!interface->cq_rb) {
|
|
ret = -EIO;
|
|
dev_err(dev, "CQ block not found");
|
|
goto fail;
|
|
}
|
|
|
|
offset = ioread32(interface->cq_rb->regs + MQNIC_RB_CQM_REG_OFFSET);
|
|
count = ioread32(interface->cq_rb->regs + MQNIC_RB_CQM_REG_COUNT);
|
|
stride = ioread32(interface->cq_rb->regs + MQNIC_RB_CQM_REG_STRIDE);
|
|
|
|
dev_info(dev, "CQ offset: 0x%08x", offset);
|
|
dev_info(dev, "CQ count: %d", count);
|
|
dev_info(dev, "CQ stride: 0x%08x", stride);
|
|
|
|
count = min_t(u32, count, MQNIC_MAX_CQ);
|
|
|
|
interface->cq_res = mqnic_create_res(count, hw_addr + offset, stride);
|
|
|
|
if (IS_ERR_OR_NULL(interface->cq_res)) {
|
|
ret = PTR_ERR(interface->cq_res);
|
|
goto fail;
|
|
}
|
|
|
|
interface->txq_rb = mqnic_find_reg_block(interface->rb_list, MQNIC_RB_TX_QM_TYPE, MQNIC_RB_TX_QM_VER, 0);
|
|
|
|
if (!interface->txq_rb) {
|
|
ret = -EIO;
|
|
dev_err(dev, "TXQ block not found");
|
|
goto fail;
|
|
}
|
|
|
|
offset = ioread32(interface->txq_rb->regs + MQNIC_RB_TX_QM_REG_OFFSET);
|
|
count = ioread32(interface->txq_rb->regs + MQNIC_RB_TX_QM_REG_COUNT);
|
|
stride = ioread32(interface->txq_rb->regs + MQNIC_RB_TX_QM_REG_STRIDE);
|
|
|
|
dev_info(dev, "TXQ offset: 0x%08x", offset);
|
|
dev_info(dev, "TXQ count: %d", count);
|
|
dev_info(dev, "TXQ stride: 0x%08x", stride);
|
|
|
|
count = min_t(u32, count, MQNIC_MAX_TXQ);
|
|
|
|
interface->txq_res = mqnic_create_res(count, hw_addr + offset, stride);
|
|
|
|
if (IS_ERR_OR_NULL(interface->txq_res)) {
|
|
ret = PTR_ERR(interface->txq_res);
|
|
goto fail;
|
|
}
|
|
|
|
interface->rxq_rb = mqnic_find_reg_block(interface->rb_list, MQNIC_RB_RX_QM_TYPE, MQNIC_RB_RX_QM_VER, 0);
|
|
|
|
if (!interface->rxq_rb) {
|
|
ret = -EIO;
|
|
dev_err(dev, "RXQ block not found");
|
|
goto fail;
|
|
}
|
|
|
|
offset = ioread32(interface->rxq_rb->regs + MQNIC_RB_RX_QM_REG_OFFSET);
|
|
count = ioread32(interface->rxq_rb->regs + MQNIC_RB_RX_QM_REG_COUNT);
|
|
stride = ioread32(interface->rxq_rb->regs + MQNIC_RB_RX_QM_REG_STRIDE);
|
|
|
|
dev_info(dev, "RXQ offset: 0x%08x", offset);
|
|
dev_info(dev, "RXQ count: %d", count);
|
|
dev_info(dev, "RXQ stride: 0x%08x", stride);
|
|
|
|
count = min_t(u32, count, MQNIC_MAX_RXQ);
|
|
|
|
interface->rxq_res = mqnic_create_res(count, hw_addr + offset, stride);
|
|
|
|
if (IS_ERR_OR_NULL(interface->rxq_res)) {
|
|
ret = PTR_ERR(interface->rxq_res);
|
|
goto fail;
|
|
}
|
|
|
|
interface->rx_queue_map_rb = mqnic_find_reg_block(interface->rb_list, MQNIC_RB_RX_QUEUE_MAP_TYPE, MQNIC_RB_RX_QUEUE_MAP_VER, 0);
|
|
|
|
if (!interface->rx_queue_map_rb) {
|
|
ret = -EIO;
|
|
dev_err(dev, "RX queue map block not found");
|
|
goto fail;
|
|
}
|
|
|
|
val = ioread32(interface->rx_queue_map_rb->regs + MQNIC_RB_RX_QUEUE_MAP_REG_CFG);
|
|
interface->rx_queue_map_indir_table_size = 1 << ((val >> 8) & 0xff);
|
|
|
|
dev_info(dev, "RX queue map indirection table size: %d", interface->rx_queue_map_indir_table_size);
|
|
|
|
for (k = 0; k < interface->port_count; k++) {
|
|
interface->rx_queue_map_indir_table[k] = interface->hw_addr + ioread32(interface->rx_queue_map_rb->regs +
|
|
MQNIC_RB_RX_QUEUE_MAP_CH_OFFSET + MQNIC_RB_RX_QUEUE_MAP_CH_STRIDE*k + MQNIC_RB_RX_QUEUE_MAP_CH_REG_OFFSET);
|
|
|
|
mqnic_interface_set_rx_queue_map_rss_mask(interface, k, 0);
|
|
mqnic_interface_set_rx_queue_map_app_mask(interface, k, 0);
|
|
mqnic_interface_set_rx_queue_map_indir_table(interface, k, 0, 0);
|
|
}
|
|
|
|
// determine desc block size
|
|
iowrite32(MQNIC_QUEUE_CMD_SET_SIZE | 0xff00, mqnic_res_get_addr(interface->txq_res, 0) + MQNIC_QUEUE_CTRL_STATUS_REG);
|
|
interface->max_desc_block_size = 1 << ((ioread32(mqnic_res_get_addr(interface->txq_res, 0) + MQNIC_QUEUE_SIZE_CQN_REG) >> 28) & 0xf);
|
|
iowrite32(MQNIC_QUEUE_CMD_SET_SIZE | 0x0000, mqnic_res_get_addr(interface->txq_res, 0) + MQNIC_QUEUE_CTRL_STATUS_REG);
|
|
|
|
dev_info(dev, "Max desc block size: %d", interface->max_desc_block_size);
|
|
|
|
interface->max_desc_block_size = min_t(u32, interface->max_desc_block_size, MQNIC_MAX_FRAGS);
|
|
|
|
desc_block_size = min_t(u32, interface->max_desc_block_size, 4);
|
|
|
|
// disable queues
|
|
for (k = 0; k < mqnic_res_get_count(interface->eq_res); k++)
|
|
iowrite32(MQNIC_EQ_CMD_SET_ENABLE | 0, mqnic_res_get_addr(interface->eq_res, k) + MQNIC_EQ_CTRL_STATUS_REG);
|
|
|
|
for (k = 0; k < mqnic_res_get_count(interface->cq_res); k++)
|
|
iowrite32(MQNIC_CQ_CMD_SET_ENABLE | 0, mqnic_res_get_addr(interface->cq_res, k) + MQNIC_CQ_CTRL_STATUS_REG);
|
|
|
|
for (k = 0; k < mqnic_res_get_count(interface->txq_res); k++)
|
|
iowrite32(MQNIC_QUEUE_CMD_SET_ENABLE | 0, mqnic_res_get_addr(interface->txq_res, k) + MQNIC_QUEUE_CTRL_STATUS_REG);
|
|
|
|
for (k = 0; k < mqnic_res_get_count(interface->rxq_res); k++)
|
|
iowrite32(MQNIC_QUEUE_CMD_SET_ENABLE | 0, mqnic_res_get_addr(interface->rxq_res, k) + MQNIC_QUEUE_CTRL_STATUS_REG);
|
|
|
|
// create ports
|
|
for (k = 0; k < interface->port_count; k++) {
|
|
struct mqnic_port *port;
|
|
struct mqnic_reg_block *port_rb = mqnic_find_reg_block(interface->rb_list, MQNIC_RB_PORT_TYPE, MQNIC_RB_PORT_VER, k);
|
|
|
|
if (!port_rb) {
|
|
ret = -EIO;
|
|
dev_err(dev, "Port index %d not found", k);
|
|
goto fail;
|
|
}
|
|
|
|
port = mqnic_create_port(interface, k, mdev->phys_port_max++, port_rb);
|
|
if (IS_ERR_OR_NULL(port)) {
|
|
ret = PTR_ERR(port);
|
|
goto fail;
|
|
}
|
|
interface->port[k] = port;
|
|
}
|
|
|
|
// create schedulers
|
|
for (k = 0; k < interface->sched_block_count; k++) {
|
|
struct mqnic_sched_block *sched_block;
|
|
struct mqnic_reg_block *sched_block_rb = mqnic_find_reg_block(interface->rb_list, MQNIC_RB_SCHED_BLOCK_TYPE, MQNIC_RB_SCHED_BLOCK_VER, k);
|
|
|
|
if (!sched_block_rb) {
|
|
ret = -EIO;
|
|
dev_err(dev, "Scheduler block index %d not found", k);
|
|
goto fail;
|
|
}
|
|
|
|
sched_block = mqnic_create_sched_block(interface, k, sched_block_rb);
|
|
if (IS_ERR_OR_NULL(sched_block)) {
|
|
ret = PTR_ERR(sched_block);
|
|
goto fail;
|
|
}
|
|
interface->sched_block[k] = sched_block;
|
|
}
|
|
|
|
// create EQs
|
|
interface->eq_count = mqnic_res_get_count(interface->eq_res);
|
|
for (k = 0; k < interface->eq_count; k++) {
|
|
struct mqnic_eq *eq = mqnic_create_eq(interface);
|
|
if (IS_ERR_OR_NULL(eq)) {
|
|
ret = PTR_ERR(eq);
|
|
goto fail;
|
|
}
|
|
|
|
interface->eq[k] = eq;
|
|
|
|
ret = mqnic_open_eq(eq, mdev->irq[k % mdev->irq_count], mqnic_num_eq_entries);
|
|
if (ret)
|
|
goto fail;
|
|
|
|
mqnic_arm_eq(eq);
|
|
}
|
|
|
|
// create net_devices
|
|
interface->ndev_count = 1;
|
|
for (k = 0; k < interface->ndev_count; k++) {
|
|
struct net_device *ndev = mqnic_create_netdev(interface, k,
|
|
interface->port[k], interface->sched_block[k]);
|
|
if (IS_ERR_OR_NULL(ndev)) {
|
|
ret = PTR_ERR(ndev);
|
|
goto fail;
|
|
}
|
|
interface->ndev[k] = ndev;
|
|
}
|
|
|
|
return interface;
|
|
|
|
fail:
|
|
mqnic_destroy_interface(interface);
|
|
return ERR_PTR(ret);
|
|
}
|
|
|
|
void mqnic_destroy_interface(struct mqnic_if *interface)
|
|
{
|
|
int k;
|
|
|
|
// destroy associated net_devices
|
|
for (k = 0; k < ARRAY_SIZE(interface->ndev); k++) {
|
|
if (interface->ndev[k]) {
|
|
mqnic_destroy_netdev(interface->ndev[k]);
|
|
interface->ndev[k] = NULL;
|
|
}
|
|
}
|
|
|
|
// free EQs
|
|
for (k = 0; k < ARRAY_SIZE(interface->eq); k++) {
|
|
if (interface->eq[k]) {
|
|
mqnic_destroy_eq(interface->eq[k]);
|
|
interface->eq[k] = NULL;
|
|
}
|
|
}
|
|
|
|
// free schedulers
|
|
for (k = 0; k < ARRAY_SIZE(interface->sched_block); k++) {
|
|
if (interface->sched_block[k]) {
|
|
mqnic_destroy_sched_block(interface->sched_block[k]);
|
|
interface->sched_block[k] = NULL;
|
|
}
|
|
}
|
|
|
|
// free ports
|
|
for (k = 0; k < ARRAY_SIZE(interface->port); k++) {
|
|
if (interface->port[k]) {
|
|
mqnic_destroy_port(interface->port[k]);
|
|
interface->port[k] = NULL;
|
|
}
|
|
}
|
|
|
|
mqnic_destroy_res(interface->eq_res);
|
|
mqnic_destroy_res(interface->cq_res);
|
|
mqnic_destroy_res(interface->txq_res);
|
|
mqnic_destroy_res(interface->rxq_res);
|
|
|
|
if (interface->rb_list)
|
|
mqnic_free_reg_block_list(interface->rb_list);
|
|
|
|
kfree(interface);
|
|
}
|
|
|
|
u32 mqnic_interface_get_tx_mtu(struct mqnic_if *interface)
|
|
{
|
|
return ioread32(interface->if_ctrl_rb->regs + MQNIC_RB_IF_CTRL_REG_TX_MTU);
|
|
}
|
|
EXPORT_SYMBOL(mqnic_interface_get_tx_mtu);
|
|
|
|
void mqnic_interface_set_tx_mtu(struct mqnic_if *interface, u32 mtu)
|
|
{
|
|
iowrite32(mtu, interface->if_ctrl_rb->regs + MQNIC_RB_IF_CTRL_REG_TX_MTU);
|
|
}
|
|
EXPORT_SYMBOL(mqnic_interface_set_tx_mtu);
|
|
|
|
u32 mqnic_interface_get_rx_mtu(struct mqnic_if *interface)
|
|
{
|
|
return ioread32(interface->if_ctrl_rb->regs + MQNIC_RB_IF_CTRL_REG_RX_MTU);
|
|
}
|
|
EXPORT_SYMBOL(mqnic_interface_get_rx_mtu);
|
|
|
|
void mqnic_interface_set_rx_mtu(struct mqnic_if *interface, u32 mtu)
|
|
{
|
|
iowrite32(mtu, interface->if_ctrl_rb->regs + MQNIC_RB_IF_CTRL_REG_RX_MTU);
|
|
}
|
|
EXPORT_SYMBOL(mqnic_interface_set_rx_mtu);
|
|
|
|
u32 mqnic_interface_get_rx_queue_map_rss_mask(struct mqnic_if *interface, int port)
|
|
{
|
|
return ioread32(interface->rx_queue_map_rb->regs + MQNIC_RB_RX_QUEUE_MAP_CH_OFFSET +
|
|
MQNIC_RB_RX_QUEUE_MAP_CH_STRIDE*port + MQNIC_RB_RX_QUEUE_MAP_CH_REG_RSS_MASK);
|
|
}
|
|
EXPORT_SYMBOL(mqnic_interface_get_rx_queue_map_rss_mask);
|
|
|
|
void mqnic_interface_set_rx_queue_map_rss_mask(struct mqnic_if *interface, int port, u32 val)
|
|
{
|
|
iowrite32(val, interface->rx_queue_map_rb->regs + MQNIC_RB_RX_QUEUE_MAP_CH_OFFSET +
|
|
MQNIC_RB_RX_QUEUE_MAP_CH_STRIDE*port + MQNIC_RB_RX_QUEUE_MAP_CH_REG_RSS_MASK);
|
|
}
|
|
EXPORT_SYMBOL(mqnic_interface_set_rx_queue_map_rss_mask);
|
|
|
|
u32 mqnic_interface_get_rx_queue_map_app_mask(struct mqnic_if *interface, int port)
|
|
{
|
|
return ioread32(interface->rx_queue_map_rb->regs + MQNIC_RB_RX_QUEUE_MAP_CH_OFFSET +
|
|
MQNIC_RB_RX_QUEUE_MAP_CH_STRIDE*port + MQNIC_RB_RX_QUEUE_MAP_CH_REG_APP_MASK);
|
|
}
|
|
EXPORT_SYMBOL(mqnic_interface_get_rx_queue_map_app_mask);
|
|
|
|
void mqnic_interface_set_rx_queue_map_app_mask(struct mqnic_if *interface, int port, u32 val)
|
|
{
|
|
iowrite32(val, interface->rx_queue_map_rb->regs + MQNIC_RB_RX_QUEUE_MAP_CH_OFFSET +
|
|
MQNIC_RB_RX_QUEUE_MAP_CH_STRIDE*port + MQNIC_RB_RX_QUEUE_MAP_CH_REG_APP_MASK);
|
|
}
|
|
EXPORT_SYMBOL(mqnic_interface_set_rx_queue_map_app_mask);
|
|
|
|
u32 mqnic_interface_get_rx_queue_map_indir_table(struct mqnic_if *interface, int port, int index)
|
|
{
|
|
return ioread32(interface->rx_queue_map_indir_table[port] + index*4);
|
|
}
|
|
EXPORT_SYMBOL(mqnic_interface_get_rx_queue_map_indir_table);
|
|
|
|
void mqnic_interface_set_rx_queue_map_indir_table(struct mqnic_if *interface, int port, int index, u32 val)
|
|
{
|
|
iowrite32(val, interface->rx_queue_map_indir_table[port] + index*4);
|
|
}
|
|
EXPORT_SYMBOL(mqnic_interface_set_rx_queue_map_indir_table);
|