Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit 50c0add5 authored by Prasad Kanneganti's avatar Prasad Kanneganti Committed by David S. Miller
Browse files

liquidio: refactor interrupt moderation code



Refactor interrupt moderation code for flexibility because parameters are
different for 10G and 25G cards.  Currently parameters (for 10G only) come
from macros compiled-in to the PF and VF drivers; fix it so that parameters
suitable for the card (10G or 25G) come from the NIC firmware via response
to a command.

Also bump up driver version to 1.5.1 to match newer NIC firmware version.

Signed-off-by: default avatarPrasad Kanneganti <prasad.kanneganti@cavium.com>
Signed-off-by: default avatarFelix Manlunas <felix.manlunas@cavium.com>
Signed-off-by: default avatarDerek Chickles <derek.chickles@cavium.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 4333d619
Loading
Loading
Loading
Loading
+270 −177
Original line number Original line Diff line number Diff line
@@ -33,6 +33,19 @@


static int octnet_get_link_stats(struct net_device *netdev);
static int octnet_get_link_stats(struct net_device *netdev);


struct oct_intrmod_context {
	int octeon_id;
	wait_queue_head_t wc;
	int cond;
	int status;
};

struct oct_intrmod_resp {
	u64     rh;
	struct oct_intrmod_cfg intrmod;
	u64     status;
};

struct oct_mdio_cmd_context {
struct oct_mdio_cmd_context {
	int octeon_id;
	int octeon_id;
	wait_queue_head_t wc;
	wait_queue_head_t wc;
@@ -1298,95 +1311,103 @@ static int lio_vf_get_sset_count(struct net_device *netdev, int sset)
	}
	}
}
}


static int lio_get_intr_coalesce(struct net_device *netdev,
/* Callback function for intrmod */
				 struct ethtool_coalesce *intr_coal)
static void octnet_intrmod_callback(struct octeon_device *oct_dev,
				    u32 status,
				    void *ptr)
{
{
	struct lio *lio = GET_LIO(netdev);
	struct octeon_soft_command *sc = (struct octeon_soft_command *)ptr;
	struct octeon_device *oct = lio->oct_dev;
	struct oct_intrmod_context *ctx;
	struct octeon_instr_queue *iq;
	struct oct_intrmod_cfg *intrmod_cfg;


	intrmod_cfg = &oct->intrmod;
	ctx  = (struct oct_intrmod_context *)sc->ctxptr;


	switch (oct->chip_id) {
	ctx->status = status;
	case OCTEON_CN23XX_PF_VID:
	case OCTEON_CN23XX_VF_VID:
		if (!intrmod_cfg->rx_enable) {
			intr_coal->rx_coalesce_usecs = intrmod_cfg->rx_usecs;
			intr_coal->rx_max_coalesced_frames =
				intrmod_cfg->rx_frames;
		}
		if (!intrmod_cfg->tx_enable)
			intr_coal->tx_max_coalesced_frames =
				intrmod_cfg->tx_frames;
		break;
	case OCTEON_CN68XX:
	case OCTEON_CN66XX: {
		struct octeon_cn6xxx *cn6xxx =
			(struct octeon_cn6xxx *)oct->chip;


		if (!intrmod_cfg->rx_enable) {
	oct_dev = lio_get_device(ctx->octeon_id);
			intr_coal->rx_coalesce_usecs =

				CFG_GET_OQ_INTR_TIME(cn6xxx->conf);
	WRITE_ONCE(ctx->cond, 1);
			intr_coal->rx_max_coalesced_frames =

				CFG_GET_OQ_INTR_PKT(cn6xxx->conf);
	/* This barrier is required to be sure that the response has been
		}
	 * written fully before waking up the handler
		iq = oct->instr_queue[lio->linfo.txpciq[0].s.q_no];
	 */
		intr_coal->tx_max_coalesced_frames = iq->fill_threshold;
	wmb();
		break;

	wake_up_interruptible(&ctx->wc);
}
}
	default:

		netif_info(lio, drv, lio->netdev, "Unknown Chip !!\n");
/*  get interrupt moderation parameters */
static int octnet_get_intrmod_cfg(struct lio *lio,
				  struct oct_intrmod_cfg *intr_cfg)
{
	struct octeon_soft_command *sc;
	struct oct_intrmod_context *ctx;
	struct oct_intrmod_resp *resp;
	int retval;
	struct octeon_device *oct_dev = lio->oct_dev;

	/* Alloc soft command */
	sc = (struct octeon_soft_command *)
		octeon_alloc_soft_command(oct_dev,
					  0,
					  sizeof(struct oct_intrmod_resp),
					  sizeof(struct oct_intrmod_context));

	if (!sc)
		return -ENOMEM;

	resp = (struct oct_intrmod_resp *)sc->virtrptr;
	memset(resp, 0, sizeof(struct oct_intrmod_resp));

	ctx = (struct oct_intrmod_context *)sc->ctxptr;
	memset(resp, 0, sizeof(struct oct_intrmod_context));
	WRITE_ONCE(ctx->cond, 0);
	ctx->octeon_id = lio_get_device_id(oct_dev);
	init_waitqueue_head(&ctx->wc);

	sc->iq_no = lio->linfo.txpciq[0].s.q_no;

	octeon_prepare_soft_command(oct_dev, sc, OPCODE_NIC,
				    OPCODE_NIC_INTRMOD_PARAMS, 0, 0, 0);

	sc->callback = octnet_intrmod_callback;
	sc->callback_arg = sc;
	sc->wait_time = 1000;

	retval = octeon_send_soft_command(oct_dev, sc);
	if (retval == IQ_SEND_FAILED) {
		octeon_free_soft_command(oct_dev, sc);
		return -EINVAL;
		return -EINVAL;
	}
	}
	if (intrmod_cfg->rx_enable) {

		intr_coal->use_adaptive_rx_coalesce =
	/* Sleep on a wait queue till the cond flag indicates that the
			intrmod_cfg->rx_enable;
	 * response arrived or timed-out.
		intr_coal->rate_sample_interval =
	 */
			intrmod_cfg->check_intrvl;
	if (sleep_cond(&ctx->wc, &ctx->cond) == -EINTR) {
		intr_coal->pkt_rate_high =
		dev_err(&oct_dev->pci_dev->dev, "Wait interrupted\n");
			intrmod_cfg->maxpkt_ratethr;
		goto intrmod_info_wait_intr;
		intr_coal->pkt_rate_low =
			intrmod_cfg->minpkt_ratethr;
		intr_coal->rx_max_coalesced_frames_high =
			intrmod_cfg->rx_maxcnt_trigger;
		intr_coal->rx_coalesce_usecs_high =
			intrmod_cfg->rx_maxtmr_trigger;
		intr_coal->rx_coalesce_usecs_low =
			intrmod_cfg->rx_mintmr_trigger;
		intr_coal->rx_max_coalesced_frames_low =
		    intrmod_cfg->rx_mincnt_trigger;
	}
	if ((OCTEON_CN23XX_PF(oct) || OCTEON_CN23XX_VF(oct)) &&
	    (intrmod_cfg->tx_enable)) {
		intr_coal->use_adaptive_tx_coalesce = intrmod_cfg->tx_enable;
		intr_coal->tx_max_coalesced_frames_high =
		    intrmod_cfg->tx_maxcnt_trigger;
		intr_coal->tx_max_coalesced_frames_low =
		    intrmod_cfg->tx_mincnt_trigger;
	}
	}
	return 0;

	retval = ctx->status || resp->status;
	if (retval) {
		dev_err(&oct_dev->pci_dev->dev,
			"Get interrupt moderation parameters failed\n");
		goto intrmod_info_wait_fail;
	}
	}


/* Callback function for intrmod */
	octeon_swap_8B_data((u64 *)&resp->intrmod,
static void octnet_intrmod_callback(struct octeon_device *oct_dev,
			    (sizeof(struct oct_intrmod_cfg)) / 8);
				    u32 status,
	memcpy(intr_cfg, &resp->intrmod, sizeof(struct oct_intrmod_cfg));
				    void *ptr)
	octeon_free_soft_command(oct_dev, sc);
{
	struct oct_intrmod_cmd *cmd = ptr;
	struct octeon_soft_command *sc = cmd->sc;


	oct_dev = cmd->oct_dev;
	return 0;


	if (status)
intrmod_info_wait_fail:
		dev_err(&oct_dev->pci_dev->dev, "intrmod config failed. Status: %llx\n",
			CVM_CAST64(status));
	else
		dev_info(&oct_dev->pci_dev->dev,
			 "Rx-Adaptive Interrupt moderation enabled:%llx\n",
			 oct_dev->intrmod.rx_enable);


	octeon_free_soft_command(oct_dev, sc);
	octeon_free_soft_command(oct_dev, sc);

intrmod_info_wait_intr:

	return -ENODEV;
}
}


/*  Configure interrupt moderation parameters */
/*  Configure interrupt moderation parameters */
@@ -1394,7 +1415,7 @@ static int octnet_set_intrmod_cfg(struct lio *lio,
				  struct oct_intrmod_cfg *intr_cfg)
				  struct oct_intrmod_cfg *intr_cfg)
{
{
	struct octeon_soft_command *sc;
	struct octeon_soft_command *sc;
	struct oct_intrmod_cmd *cmd;
	struct oct_intrmod_context *ctx;
	struct oct_intrmod_cfg *cfg;
	struct oct_intrmod_cfg *cfg;
	int retval;
	int retval;
	struct octeon_device *oct_dev = lio->oct_dev;
	struct octeon_device *oct_dev = lio->oct_dev;
@@ -1404,19 +1425,21 @@ static int octnet_set_intrmod_cfg(struct lio *lio,
		octeon_alloc_soft_command(oct_dev,
		octeon_alloc_soft_command(oct_dev,
					  sizeof(struct oct_intrmod_cfg),
					  sizeof(struct oct_intrmod_cfg),
					  0,
					  0,
					  sizeof(struct oct_intrmod_cmd));
					  sizeof(struct oct_intrmod_context));


	if (!sc)
	if (!sc)
		return -ENOMEM;
		return -ENOMEM;


	cmd = (struct oct_intrmod_cmd *)sc->ctxptr;
	ctx = (struct oct_intrmod_context *)sc->ctxptr;

	WRITE_ONCE(ctx->cond, 0);
	ctx->octeon_id = lio_get_device_id(oct_dev);
	init_waitqueue_head(&ctx->wc);

	cfg = (struct oct_intrmod_cfg *)sc->virtdptr;
	cfg = (struct oct_intrmod_cfg *)sc->virtdptr;


	memcpy(cfg, intr_cfg, sizeof(struct oct_intrmod_cfg));
	memcpy(cfg, intr_cfg, sizeof(struct oct_intrmod_cfg));
	octeon_swap_8B_data((u64 *)cfg, (sizeof(struct oct_intrmod_cfg)) / 8);
	octeon_swap_8B_data((u64 *)cfg, (sizeof(struct oct_intrmod_cfg)) / 8);
	cmd->sc = sc;
	cmd->cfg = cfg;
	cmd->oct_dev = oct_dev;


	sc->iq_no = lio->linfo.txpciq[0].s.q_no;
	sc->iq_no = lio->linfo.txpciq[0].s.q_no;


@@ -1424,7 +1447,7 @@ static int octnet_set_intrmod_cfg(struct lio *lio,
				    OPCODE_NIC_INTRMOD_CFG, 0, 0, 0);
				    OPCODE_NIC_INTRMOD_CFG, 0, 0, 0);


	sc->callback = octnet_intrmod_callback;
	sc->callback = octnet_intrmod_callback;
	sc->callback_arg = cmd;
	sc->callback_arg = sc;
	sc->wait_time = 1000;
	sc->wait_time = 1000;


	retval = octeon_send_soft_command(oct_dev, sc);
	retval = octeon_send_soft_command(oct_dev, sc);
@@ -1433,7 +1456,29 @@ static int octnet_set_intrmod_cfg(struct lio *lio,
		return -EINVAL;
		return -EINVAL;
	}
	}


	return 0;
	/* 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) {
		retval = ctx->status;
		if (retval)
			dev_err(&oct_dev->pci_dev->dev,
				"intrmod config failed. Status: %llx\n",
				CVM_CAST64(retval));
		else
			dev_info(&oct_dev->pci_dev->dev,
				 "Rx-Adaptive Interrupt moderation %s\n",
				 (intr_cfg->rx_enable) ?
				 "enabled" : "disabled");

		octeon_free_soft_command(oct_dev, sc);

		return ((retval) ? -ENODEV : 0);
	}

	dev_err(&oct_dev->pci_dev->dev, "iq/oq config failed\n");

	return -EINTR;
}
}


static void
static void
@@ -1590,80 +1635,106 @@ static int octnet_get_link_stats(struct net_device *netdev)
	return 0;
	return 0;
}
}


/* Enable/Disable auto interrupt Moderation */
static int lio_get_intr_coalesce(struct net_device *netdev,
static int oct_cfg_adaptive_intr(struct lio *lio, struct ethtool_coalesce
				 struct ethtool_coalesce *intr_coal)
				 *intr_coal)
{
{
	int ret = 0;
	struct lio *lio = GET_LIO(netdev);
	struct octeon_device *oct = lio->oct_dev;
	struct octeon_device *oct = lio->oct_dev;
	struct oct_intrmod_cfg *intrmod_cfg;
	struct octeon_instr_queue *iq;
	struct oct_intrmod_cfg intrmod_cfg;


	intrmod_cfg = &oct->intrmod;
	if (octnet_get_intrmod_cfg(lio, &intrmod_cfg))
		return -ENODEV;


	if (oct->intrmod.rx_enable || oct->intrmod.tx_enable) {
	switch (oct->chip_id) {
		if (intr_coal->rate_sample_interval)
	case OCTEON_CN23XX_PF_VID:
			intrmod_cfg->check_intrvl =
	case OCTEON_CN23XX_VF_VID: {
				intr_coal->rate_sample_interval;
		if (!intrmod_cfg.rx_enable) {
		else
			intr_coal->rx_coalesce_usecs = oct->rx_coalesce_usecs;
			intrmod_cfg->check_intrvl =
			intr_coal->rx_max_coalesced_frames =
				LIO_INTRMOD_CHECK_INTERVAL;
				oct->rx_max_coalesced_frames;
		}
		if (!intrmod_cfg.tx_enable)
			intr_coal->tx_max_coalesced_frames =
				oct->tx_max_coalesced_frames;
		break;
	}
	case OCTEON_CN68XX:
	case OCTEON_CN66XX: {
		struct octeon_cn6xxx *cn6xxx =
			(struct octeon_cn6xxx *)oct->chip;


		if (intr_coal->pkt_rate_high)
		if (!intrmod_cfg.rx_enable) {
			intrmod_cfg->maxpkt_ratethr =
			intr_coal->rx_coalesce_usecs =
				intr_coal->pkt_rate_high;
				CFG_GET_OQ_INTR_TIME(cn6xxx->conf);
		else
			intr_coal->rx_max_coalesced_frames =
			intrmod_cfg->maxpkt_ratethr =
				CFG_GET_OQ_INTR_PKT(cn6xxx->conf);
				LIO_INTRMOD_MAXPKT_RATETHR;
		}
		iq = oct->instr_queue[lio->linfo.txpciq[0].s.q_no];
		intr_coal->tx_max_coalesced_frames = iq->fill_threshold;
		break;
	}
	default:
		netif_info(lio, drv, lio->netdev, "Unknown Chip !!\n");
		return -EINVAL;
	}
	if (intrmod_cfg.rx_enable) {
		intr_coal->use_adaptive_rx_coalesce =
			intrmod_cfg.rx_enable;
		intr_coal->rate_sample_interval =
			intrmod_cfg.check_intrvl;
		intr_coal->pkt_rate_high =
			intrmod_cfg.maxpkt_ratethr;
		intr_coal->pkt_rate_low =
			intrmod_cfg.minpkt_ratethr;
		intr_coal->rx_max_coalesced_frames_high =
			intrmod_cfg.rx_maxcnt_trigger;
		intr_coal->rx_coalesce_usecs_high =
			intrmod_cfg.rx_maxtmr_trigger;
		intr_coal->rx_coalesce_usecs_low =
			intrmod_cfg.rx_mintmr_trigger;
		intr_coal->rx_max_coalesced_frames_low =
			intrmod_cfg.rx_mincnt_trigger;
	}
	if ((OCTEON_CN23XX_PF(oct) || OCTEON_CN23XX_VF(oct)) &&
	    (intrmod_cfg.tx_enable)) {
		intr_coal->use_adaptive_tx_coalesce =
			intrmod_cfg.tx_enable;
		intr_coal->tx_max_coalesced_frames_high =
			intrmod_cfg.tx_maxcnt_trigger;
		intr_coal->tx_max_coalesced_frames_low =
			intrmod_cfg.tx_mincnt_trigger;
	}
	return 0;
}


		if (intr_coal->pkt_rate_low)
/* Enable/Disable auto interrupt Moderation */
			intrmod_cfg->minpkt_ratethr =
static int oct_cfg_adaptive_intr(struct lio *lio,
				intr_coal->pkt_rate_low;
				 struct oct_intrmod_cfg *intrmod_cfg,
		else
				 struct ethtool_coalesce *intr_coal)
			intrmod_cfg->minpkt_ratethr =
{
				LIO_INTRMOD_MINPKT_RATETHR;
	int ret = 0;

	if (intrmod_cfg->rx_enable || intrmod_cfg->tx_enable) {
		intrmod_cfg->check_intrvl = intr_coal->rate_sample_interval;
		intrmod_cfg->maxpkt_ratethr = intr_coal->pkt_rate_high;
		intrmod_cfg->minpkt_ratethr = intr_coal->pkt_rate_low;
	}
	}
	if (oct->intrmod.rx_enable) {
	if (intrmod_cfg->rx_enable) {
		if (intr_coal->rx_max_coalesced_frames_high)
		intrmod_cfg->rx_maxcnt_trigger =
		intrmod_cfg->rx_maxcnt_trigger =
			intr_coal->rx_max_coalesced_frames_high;
			intr_coal->rx_max_coalesced_frames_high;
		else
			intrmod_cfg->rx_maxcnt_trigger =
				LIO_INTRMOD_RXMAXCNT_TRIGGER;

		if (intr_coal->rx_coalesce_usecs_high)
		intrmod_cfg->rx_maxtmr_trigger =
		intrmod_cfg->rx_maxtmr_trigger =
			intr_coal->rx_coalesce_usecs_high;
			intr_coal->rx_coalesce_usecs_high;
		else
			intrmod_cfg->rx_maxtmr_trigger =
				LIO_INTRMOD_RXMAXTMR_TRIGGER;

		if (intr_coal->rx_coalesce_usecs_low)
		intrmod_cfg->rx_mintmr_trigger =
		intrmod_cfg->rx_mintmr_trigger =
			intr_coal->rx_coalesce_usecs_low;
			intr_coal->rx_coalesce_usecs_low;
		else
			intrmod_cfg->rx_mintmr_trigger =
				LIO_INTRMOD_RXMINTMR_TRIGGER;

		if (intr_coal->rx_max_coalesced_frames_low)
		intrmod_cfg->rx_mincnt_trigger =
		intrmod_cfg->rx_mincnt_trigger =
			intr_coal->rx_max_coalesced_frames_low;
			intr_coal->rx_max_coalesced_frames_low;
		else
			intrmod_cfg->rx_mincnt_trigger =
				LIO_INTRMOD_RXMINCNT_TRIGGER;
	}
	}
	if (oct->intrmod.tx_enable) {
	if (intrmod_cfg->tx_enable) {
		if (intr_coal->tx_max_coalesced_frames_high)
		intrmod_cfg->tx_maxcnt_trigger =
		intrmod_cfg->tx_maxcnt_trigger =
			intr_coal->tx_max_coalesced_frames_high;
			intr_coal->tx_max_coalesced_frames_high;
		else
			intrmod_cfg->tx_maxcnt_trigger =
				LIO_INTRMOD_TXMAXCNT_TRIGGER;
		if (intr_coal->tx_max_coalesced_frames_low)
		intrmod_cfg->tx_mincnt_trigger =
		intrmod_cfg->tx_mincnt_trigger =
			intr_coal->tx_max_coalesced_frames_low;
			intr_coal->tx_max_coalesced_frames_low;
		else
			intrmod_cfg->tx_mincnt_trigger =
				LIO_INTRMOD_TXMINCNT_TRIGGER;
	}
	}


	ret = octnet_set_intrmod_cfg(lio, intrmod_cfg);
	ret = octnet_set_intrmod_cfg(lio, intrmod_cfg);
@@ -1672,7 +1743,9 @@ static int oct_cfg_adaptive_intr(struct lio *lio, struct ethtool_coalesce
}
}


static int
static int
oct_cfg_rx_intrcnt(struct lio *lio, struct ethtool_coalesce *intr_coal)
oct_cfg_rx_intrcnt(struct lio *lio,
		   struct oct_intrmod_cfg *intrmod,
		   struct ethtool_coalesce *intr_coal)
{
{
	struct octeon_device *oct = lio->oct_dev;
	struct octeon_device *oct = lio->oct_dev;
	u32 rx_max_coalesced_frames;
	u32 rx_max_coalesced_frames;
@@ -1698,7 +1771,7 @@ oct_cfg_rx_intrcnt(struct lio *lio, struct ethtool_coalesce *intr_coal)
		int q_no;
		int q_no;


		if (!intr_coal->rx_max_coalesced_frames)
		if (!intr_coal->rx_max_coalesced_frames)
			rx_max_coalesced_frames = oct->intrmod.rx_frames;
			rx_max_coalesced_frames = intrmod->rx_frames;
		else
		else
			rx_max_coalesced_frames =
			rx_max_coalesced_frames =
			    intr_coal->rx_max_coalesced_frames;
			    intr_coal->rx_max_coalesced_frames;
@@ -1709,17 +1782,18 @@ oct_cfg_rx_intrcnt(struct lio *lio, struct ethtool_coalesce *intr_coal)
			    (octeon_read_csr64(
			    (octeon_read_csr64(
				 oct, CN23XX_SLI_OQ_PKT_INT_LEVELS(q_no)) &
				 oct, CN23XX_SLI_OQ_PKT_INT_LEVELS(q_no)) &
			     (0x3fffff00000000UL)) |
			     (0x3fffff00000000UL)) |
				rx_max_coalesced_frames);
				(rx_max_coalesced_frames - 1));
			/*consider setting resend bit*/
			/*consider setting resend bit*/
		}
		}
		oct->intrmod.rx_frames = rx_max_coalesced_frames;
		intrmod->rx_frames = rx_max_coalesced_frames;
		oct->rx_max_coalesced_frames = rx_max_coalesced_frames;
		break;
		break;
	}
	}
	case OCTEON_CN23XX_VF_VID: {
	case OCTEON_CN23XX_VF_VID: {
		int q_no;
		int q_no;


		if (!intr_coal->rx_max_coalesced_frames)
		if (!intr_coal->rx_max_coalesced_frames)
			rx_max_coalesced_frames = oct->intrmod.rx_frames;
			rx_max_coalesced_frames = intrmod->rx_frames;
		else
		else
			rx_max_coalesced_frames =
			rx_max_coalesced_frames =
			    intr_coal->rx_max_coalesced_frames;
			    intr_coal->rx_max_coalesced_frames;
@@ -1732,7 +1806,8 @@ oct_cfg_rx_intrcnt(struct lio *lio, struct ethtool_coalesce *intr_coal)
				rx_max_coalesced_frames);
				rx_max_coalesced_frames);
			/*consider writing to resend bit here*/
			/*consider writing to resend bit here*/
		}
		}
		oct->intrmod.rx_frames = rx_max_coalesced_frames;
		intrmod->rx_frames = rx_max_coalesced_frames;
		oct->rx_max_coalesced_frames = rx_max_coalesced_frames;
		break;
		break;
	}
	}
	default:
	default:
@@ -1742,6 +1817,7 @@ oct_cfg_rx_intrcnt(struct lio *lio, struct ethtool_coalesce *intr_coal)
}
}


static int oct_cfg_rx_intrtime(struct lio *lio,
static int oct_cfg_rx_intrtime(struct lio *lio,
			       struct oct_intrmod_cfg *intrmod,
			       struct ethtool_coalesce *intr_coal)
			       struct ethtool_coalesce *intr_coal)
{
{
	struct octeon_device *oct = lio->oct_dev;
	struct octeon_device *oct = lio->oct_dev;
@@ -1772,7 +1848,7 @@ static int oct_cfg_rx_intrtime(struct lio *lio,
		int q_no;
		int q_no;


		if (!intr_coal->rx_coalesce_usecs)
		if (!intr_coal->rx_coalesce_usecs)
			rx_coalesce_usecs = oct->intrmod.rx_usecs;
			rx_coalesce_usecs = intrmod->rx_usecs;
		else
		else
			rx_coalesce_usecs = intr_coal->rx_coalesce_usecs;
			rx_coalesce_usecs = intr_coal->rx_coalesce_usecs;
		time_threshold =
		time_threshold =
@@ -1781,11 +1857,12 @@ static int oct_cfg_rx_intrtime(struct lio *lio,
			q_no += oct->sriov_info.pf_srn;
			q_no += oct->sriov_info.pf_srn;
			octeon_write_csr64(oct,
			octeon_write_csr64(oct,
					   CN23XX_SLI_OQ_PKT_INT_LEVELS(q_no),
					   CN23XX_SLI_OQ_PKT_INT_LEVELS(q_no),
					   (oct->intrmod.rx_frames |
					   (intrmod->rx_frames |
					    (time_threshold << 32)));
					    ((u64)time_threshold << 32)));
			/*consider writing to resend bit here*/
			/*consider writing to resend bit here*/
		}
		}
		oct->intrmod.rx_usecs = rx_coalesce_usecs;
		intrmod->rx_usecs = rx_coalesce_usecs;
		oct->rx_coalesce_usecs = rx_coalesce_usecs;
		break;
		break;
	}
	}
	case OCTEON_CN23XX_VF_VID: {
	case OCTEON_CN23XX_VF_VID: {
@@ -1793,7 +1870,7 @@ static int oct_cfg_rx_intrtime(struct lio *lio,
		int q_no;
		int q_no;


		if (!intr_coal->rx_coalesce_usecs)
		if (!intr_coal->rx_coalesce_usecs)
			rx_coalesce_usecs = oct->intrmod.rx_usecs;
			rx_coalesce_usecs = intrmod->rx_usecs;
		else
		else
			rx_coalesce_usecs = intr_coal->rx_coalesce_usecs;
			rx_coalesce_usecs = intr_coal->rx_coalesce_usecs;


@@ -1802,11 +1879,12 @@ static int oct_cfg_rx_intrtime(struct lio *lio,
		for (q_no = 0; q_no < oct->num_oqs; q_no++) {
		for (q_no = 0; q_no < oct->num_oqs; q_no++) {
			octeon_write_csr64(
			octeon_write_csr64(
				oct, CN23XX_VF_SLI_OQ_PKT_INT_LEVELS(q_no),
				oct, CN23XX_VF_SLI_OQ_PKT_INT_LEVELS(q_no),
				(oct->intrmod.rx_frames |
				(intrmod->rx_frames |
				 (time_threshold << 32)));
				 ((u64)time_threshold << 32)));
			/*consider setting resend bit*/
			/*consider setting resend bit*/
		}
		}
		oct->intrmod.rx_usecs = rx_coalesce_usecs;
		intrmod->rx_usecs = rx_coalesce_usecs;
		oct->rx_coalesce_usecs = rx_coalesce_usecs;
		break;
		break;
	}
	}
	default:
	default:
@@ -1817,8 +1895,9 @@ static int oct_cfg_rx_intrtime(struct lio *lio,
}
}


static int
static int
oct_cfg_tx_intrcnt(struct lio *lio, struct ethtool_coalesce *intr_coal
oct_cfg_tx_intrcnt(struct lio *lio,
		   __attribute__((unused)))
		   struct oct_intrmod_cfg *intrmod,
		   struct ethtool_coalesce *intr_coal)
{
{
	struct octeon_device *oct = lio->oct_dev;
	struct octeon_device *oct = lio->oct_dev;
	u32 iq_intr_pkt;
	u32 iq_intr_pkt;
@@ -1845,12 +1924,13 @@ oct_cfg_tx_intrcnt(struct lio *lio, struct ethtool_coalesce *intr_coal
			val = readq(inst_cnt_reg);
			val = readq(inst_cnt_reg);
			/*clear wmark and count.dont want to write count back*/
			/*clear wmark and count.dont want to write count back*/
			val = (val & 0xFFFF000000000000ULL) |
			val = (val & 0xFFFF000000000000ULL) |
			      ((u64)iq_intr_pkt
			      ((u64)(iq_intr_pkt - 1)
			       << CN23XX_PKT_IN_DONE_WMARK_BIT_POS);
			       << CN23XX_PKT_IN_DONE_WMARK_BIT_POS);
			writeq(val, inst_cnt_reg);
			writeq(val, inst_cnt_reg);
			/*consider setting resend bit*/
			/*consider setting resend bit*/
		}
		}
		oct->intrmod.tx_frames = iq_intr_pkt;
		intrmod->tx_frames = iq_intr_pkt;
		oct->tx_max_coalesced_frames = iq_intr_pkt;
		break;
		break;
	}
	}
	default:
	default:
@@ -1865,6 +1945,7 @@ static int lio_set_intr_coalesce(struct net_device *netdev,
	struct lio *lio = GET_LIO(netdev);
	struct lio *lio = GET_LIO(netdev);
	int ret;
	int ret;
	struct octeon_device *oct = lio->oct_dev;
	struct octeon_device *oct = lio->oct_dev;
	struct oct_intrmod_cfg intrmod = {0};
	u32 j, q_no;
	u32 j, q_no;
	int db_max, db_min;
	int db_max, db_min;


@@ -1883,8 +1964,8 @@ static int lio_set_intr_coalesce(struct net_device *netdev,
		} else {
		} else {
			dev_err(&oct->pci_dev->dev,
			dev_err(&oct->pci_dev->dev,
				"LIQUIDIO: Invalid tx-frames:%d. Range is min:%d max:%d\n",
				"LIQUIDIO: Invalid tx-frames:%d. Range is min:%d max:%d\n",
				intr_coal->tx_max_coalesced_frames, db_min,
				intr_coal->tx_max_coalesced_frames,
				db_max);
				db_min, db_max);
			return -EINVAL;
			return -EINVAL;
		}
		}
		break;
		break;
@@ -1895,24 +1976,36 @@ static int lio_set_intr_coalesce(struct net_device *netdev,
		return -EINVAL;
		return -EINVAL;
	}
	}


	oct->intrmod.rx_enable = intr_coal->use_adaptive_rx_coalesce ? 1 : 0;
	intrmod.rx_enable = intr_coal->use_adaptive_rx_coalesce ? 1 : 0;
	oct->intrmod.tx_enable = intr_coal->use_adaptive_tx_coalesce ? 1 : 0;
	intrmod.tx_enable = intr_coal->use_adaptive_tx_coalesce ? 1 : 0;
	intrmod.rx_frames = CFG_GET_OQ_INTR_PKT(octeon_get_conf(oct));
	intrmod.rx_usecs = CFG_GET_OQ_INTR_TIME(octeon_get_conf(oct));
	intrmod.tx_frames = CFG_GET_IQ_INTR_PKT(octeon_get_conf(oct));


	ret = oct_cfg_adaptive_intr(lio, intr_coal);
	ret = oct_cfg_adaptive_intr(lio, &intrmod, intr_coal);


	if (!intr_coal->use_adaptive_rx_coalesce) {
	if (!intr_coal->use_adaptive_rx_coalesce) {
		ret = oct_cfg_rx_intrtime(lio, intr_coal);
		ret = oct_cfg_rx_intrtime(lio, &intrmod, intr_coal);
		if (ret)
		if (ret)
			goto ret_intrmod;
			goto ret_intrmod;


		ret = oct_cfg_rx_intrcnt(lio, intr_coal);
		ret = oct_cfg_rx_intrcnt(lio, &intrmod, intr_coal);
		if (ret)
		if (ret)
			goto ret_intrmod;
			goto ret_intrmod;
	} else {
		oct->rx_coalesce_usecs =
			CFG_GET_OQ_INTR_TIME(octeon_get_conf(oct));
		oct->rx_max_coalesced_frames =
			CFG_GET_OQ_INTR_PKT(octeon_get_conf(oct));
	}
	}

	if (!intr_coal->use_adaptive_tx_coalesce) {
	if (!intr_coal->use_adaptive_tx_coalesce) {
		ret = oct_cfg_tx_intrcnt(lio, intr_coal);
		ret = oct_cfg_tx_intrcnt(lio, &intrmod, intr_coal);
		if (ret)
		if (ret)
			goto ret_intrmod;
			goto ret_intrmod;
	} else {
		oct->tx_max_coalesced_frames =
			CFG_GET_IQ_INTR_PKT(octeon_get_conf(oct));
	}
	}


	return 0;
	return 0;
+0 −17
Original line number Original line Diff line number Diff line
@@ -4320,7 +4320,6 @@ static int liquidio_enable_sriov(struct pci_dev *dev, int num_vfs)
 */
 */
static int liquidio_init_nic_module(struct octeon_device *oct)
static int liquidio_init_nic_module(struct octeon_device *oct)
{
{
	struct oct_intrmod_cfg *intrmod_cfg;
	int i, retval = 0;
	int i, retval = 0;
	int num_nic_ports = CFG_GET_NUM_NIC_PORTS(octeon_get_conf(oct));
	int num_nic_ports = CFG_GET_NUM_NIC_PORTS(octeon_get_conf(oct));


@@ -4345,22 +4344,6 @@ static int liquidio_init_nic_module(struct octeon_device *oct)


	liquidio_ptp_init(oct);
	liquidio_ptp_init(oct);


	/* Initialize interrupt moderation params */
	intrmod_cfg = &((struct octeon_device *)oct)->intrmod;
	intrmod_cfg->rx_enable = 1;
	intrmod_cfg->check_intrvl = LIO_INTRMOD_CHECK_INTERVAL;
	intrmod_cfg->maxpkt_ratethr = LIO_INTRMOD_MAXPKT_RATETHR;
	intrmod_cfg->minpkt_ratethr = LIO_INTRMOD_MINPKT_RATETHR;
	intrmod_cfg->rx_maxcnt_trigger = LIO_INTRMOD_RXMAXCNT_TRIGGER;
	intrmod_cfg->rx_maxtmr_trigger = LIO_INTRMOD_RXMAXTMR_TRIGGER;
	intrmod_cfg->rx_mintmr_trigger = LIO_INTRMOD_RXMINTMR_TRIGGER;
	intrmod_cfg->rx_mincnt_trigger = LIO_INTRMOD_RXMINCNT_TRIGGER;
	intrmod_cfg->tx_enable = 1;
	intrmod_cfg->tx_maxcnt_trigger = LIO_INTRMOD_TXMAXCNT_TRIGGER;
	intrmod_cfg->tx_mincnt_trigger = LIO_INTRMOD_TXMINCNT_TRIGGER;
	intrmod_cfg->rx_frames = CFG_GET_OQ_INTR_PKT(octeon_get_conf(oct));
	intrmod_cfg->rx_usecs = CFG_GET_OQ_INTR_TIME(octeon_get_conf(oct));
	intrmod_cfg->tx_frames = CFG_GET_IQ_INTR_PKT(octeon_get_conf(oct));
	dev_dbg(&oct->pci_dev->dev, "Network interfaces ready\n");
	dev_dbg(&oct->pci_dev->dev, "Network interfaces ready\n");


	return retval;
	return retval;
+0 −17
Original line number Original line Diff line number Diff line
@@ -3057,7 +3057,6 @@ static int setup_nic_devices(struct octeon_device *octeon_dev)
 */
 */
static int liquidio_init_nic_module(struct octeon_device *oct)
static int liquidio_init_nic_module(struct octeon_device *oct)
{
{
	struct oct_intrmod_cfg *intrmod_cfg;
	int num_nic_ports = 1;
	int num_nic_ports = 1;
	int i, retval = 0;
	int i, retval = 0;


@@ -3079,22 +3078,6 @@ static int liquidio_init_nic_module(struct octeon_device *oct)
		goto octnet_init_failure;
		goto octnet_init_failure;
	}
	}


	/* Initialize interrupt moderation params */
	intrmod_cfg = &((struct octeon_device *)oct)->intrmod;
	intrmod_cfg->rx_enable = 1;
	intrmod_cfg->check_intrvl = LIO_INTRMOD_CHECK_INTERVAL;
	intrmod_cfg->maxpkt_ratethr = LIO_INTRMOD_MAXPKT_RATETHR;
	intrmod_cfg->minpkt_ratethr = LIO_INTRMOD_MINPKT_RATETHR;
	intrmod_cfg->rx_maxcnt_trigger = LIO_INTRMOD_RXMAXCNT_TRIGGER;
	intrmod_cfg->rx_maxtmr_trigger = LIO_INTRMOD_RXMAXTMR_TRIGGER;
	intrmod_cfg->rx_mintmr_trigger = LIO_INTRMOD_RXMINTMR_TRIGGER;
	intrmod_cfg->rx_mincnt_trigger = LIO_INTRMOD_RXMINCNT_TRIGGER;
	intrmod_cfg->tx_enable = 1;
	intrmod_cfg->tx_maxcnt_trigger = LIO_INTRMOD_TXMAXCNT_TRIGGER;
	intrmod_cfg->tx_mincnt_trigger = LIO_INTRMOD_TXMINCNT_TRIGGER;
	intrmod_cfg->rx_frames = CFG_GET_OQ_INTR_PKT(octeon_get_conf(oct));
	intrmod_cfg->rx_usecs = CFG_GET_OQ_INTR_TIME(octeon_get_conf(oct));
	intrmod_cfg->tx_frames = CFG_GET_IQ_INTR_PKT(octeon_get_conf(oct));
	dev_dbg(&oct->pci_dev->dev, "Network interfaces ready\n");
	dev_dbg(&oct->pci_dev->dev, "Network interfaces ready\n");


	return retval;
	return retval;
+2 −24
Original line number Original line Diff line number Diff line
@@ -27,7 +27,7 @@


#define LIQUIDIO_PACKAGE ""
#define LIQUIDIO_PACKAGE ""
#define LIQUIDIO_BASE_MAJOR_VERSION 1
#define LIQUIDIO_BASE_MAJOR_VERSION 1
#define LIQUIDIO_BASE_MINOR_VERSION 4
#define LIQUIDIO_BASE_MINOR_VERSION 5
#define LIQUIDIO_BASE_MICRO_VERSION 1
#define LIQUIDIO_BASE_MICRO_VERSION 1
#define LIQUIDIO_BASE_VERSION   __stringify(LIQUIDIO_BASE_MAJOR_VERSION) "." \
#define LIQUIDIO_BASE_VERSION   __stringify(LIQUIDIO_BASE_MAJOR_VERSION) "." \
				__stringify(LIQUIDIO_BASE_MINOR_VERSION)
				__stringify(LIQUIDIO_BASE_MINOR_VERSION)
@@ -83,6 +83,7 @@ enum octeon_tag_type {
#define OPCODE_NIC_INTRMOD_CFG         0x08
#define OPCODE_NIC_INTRMOD_CFG         0x08
#define OPCODE_NIC_IF_CFG              0x09
#define OPCODE_NIC_IF_CFG              0x09
#define OPCODE_NIC_VF_DRV_NOTICE       0x0A
#define OPCODE_NIC_VF_DRV_NOTICE       0x0A
#define OPCODE_NIC_INTRMOD_PARAMS      0x0B
#define VF_DRV_LOADED                  1
#define VF_DRV_LOADED                  1
#define VF_DRV_REMOVED                -1
#define VF_DRV_REMOVED                -1
#define VF_DRV_MACADDR_CHANGED         2
#define VF_DRV_MACADDR_CHANGED         2
@@ -851,29 +852,6 @@ struct oct_mdio_cmd {


#define OCT_LINK_STATS_SIZE   (sizeof(struct oct_link_stats))
#define OCT_LINK_STATS_SIZE   (sizeof(struct oct_link_stats))


/* intrmod: max. packet rate threshold */
#define LIO_INTRMOD_MAXPKT_RATETHR	196608
/* intrmod: min. packet rate threshold */
#define LIO_INTRMOD_MINPKT_RATETHR	9216
/* intrmod: max. packets to trigger interrupt */
#define LIO_INTRMOD_RXMAXCNT_TRIGGER	384
/* intrmod: min. packets to trigger interrupt */
#define LIO_INTRMOD_RXMINCNT_TRIGGER	0
/* intrmod: max. time to trigger interrupt */
#define LIO_INTRMOD_RXMAXTMR_TRIGGER	128
/* 66xx:intrmod: min. time to trigger interrupt
 * (value of 1 is optimum for TCP_RR)
 */
#define LIO_INTRMOD_RXMINTMR_TRIGGER	1

/* intrmod: max. packets to trigger interrupt */
#define LIO_INTRMOD_TXMAXCNT_TRIGGER	64
/* intrmod: min. packets to trigger interrupt */
#define LIO_INTRMOD_TXMINCNT_TRIGGER	0

/* intrmod: poll interval in seconds */
#define LIO_INTRMOD_CHECK_INTERVAL  1

struct oct_intrmod_cfg {
struct oct_intrmod_cfg {
	u64 rx_enable;
	u64 rx_enable;
	u64 tx_enable;
	u64 tx_enable;
+4 −9
Original line number Original line Diff line number Diff line
@@ -453,9 +453,6 @@ struct octeon_device {
	/** List of dispatch functions */
	/** List of dispatch functions */
	struct octeon_dispatch_list dispatch;
	struct octeon_dispatch_list dispatch;


	/* Interrupt Moderation */
	struct oct_intrmod_cfg intrmod;

	u32 int_status;
	u32 int_status;


	u64 droq_intr;
	u64 droq_intr;
@@ -541,6 +538,10 @@ struct octeon_device {
	u32 priv_flags;
	u32 priv_flags;


	void *watchdog_task;
	void *watchdog_task;

	u32 rx_coalesce_usecs;
	u32 rx_max_coalesced_frames;
	u32 tx_max_coalesced_frames;
};
};


#define  OCT_DRV_ONLINE 1
#define  OCT_DRV_ONLINE 1
@@ -554,12 +555,6 @@ struct octeon_device {
#define CHIP_CONF(oct, TYPE)             \
#define CHIP_CONF(oct, TYPE)             \
	(((struct octeon_ ## TYPE  *)((oct)->chip))->conf)
	(((struct octeon_ ## TYPE  *)((oct)->chip))->conf)


struct oct_intrmod_cmd {
	struct octeon_device *oct_dev;
	struct octeon_soft_command *sc;
	struct oct_intrmod_cfg *cfg;
};

/*------------------ Function Prototypes ----------------------*/
/*------------------ Function Prototypes ----------------------*/


/** Initialize device list memory */
/** Initialize device list memory */