.. Copyright (c) <2018>, Intel Corporation
   All rights reserved.

   Redistribution and use in source and binary forms, with or without
   modification, are permitted provided that the following conditions
   are met:

   - Redistributions of source code must retain the above copyright
     notice, this list of conditions and the following disclaimer.

   - Redistributions in binary form must reproduce the above copyright
     notice, this list of conditions and the following disclaimer in
     the documentation and/or other materials provided with the
     distribution.

   - Neither the name of Intel Corporation nor the names of its
     contributors may be used to endorse or promote products derived
     from this software without specific prior written permission.

   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
   "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
   LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
   FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
   COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
   INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
   (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
   SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
   HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
   STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
   ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
   OF THE POSSIBILITY OF SUCH DAMAGE.

=============
Dynamic queue 
=============
Currently, to configure a DPDK ethdev, the application first specifies 
how many Tx and Rx queues to include in the ethdev. The application then 
sets up each Tx and Rx queue. Finally, once all the queues have been set up, 
the application may then start the device, at this point traffic can flow. 
If device stops, this halts the flow of traffic on all queues in the ethdev, 
not one or part of all queues stop.

According to existing implementation, rte_eth_[rx|tx]_queue_setup will
always return fail if device is already started(rte_eth_dev_start).
This can't satisfy the usage when application wants to defer to setup
part of the queues while keeping traffic running on those queues already
be setup.

Basically this is not a general hardware limitation, because for NIC
like i40e, ixgbe, it is not necessary to stop the whole device before
configure a fresh queue or reconfigure an existing queue with no traffic
on it.

Dynamic queue lets etherdev driver exposes the capability flag through
rte_eth_dev_info_get when it supports deferred queue configuraiton,
then base on this flag, rte_eth_[rx|tx]_queue_setup could decide to
continue to setup the queue or just return fail when device already
started.

Allow ethdevs to setup/reconfigure/tear down queues at runtime without 
stopping the device. Given an ethdev configuration with a specified 
number of Tx and Rx queues, requirements as below:

1.The application should be able to start the device with only some of the 
queues set up.
2.The application should be able to set up additional queues at runtime 
without calling dev_stop().
3.The application should be able to reconfigure existing queues at runtime 
without calling dev_stop().
4.This support should be implemented in such a way that it does not break 
existing PMDs. 

Prerequisites
=============
1. Host PF in DPDK driver::

    ./usertools/dpdk-devbind.py -b igb_uio 81:00.0

2. Start testpmd on host, set chained port topology mode, add txq/rxq to 
   enable multi-queues::
   
    ./testpmd -c 0xf -n 4  -- -i --port-topology=chained --txq=64 --rxq=64


Test Case: Rx queue setup at runtime
====================================
Stop some Rx queues on port 0::

    testpmd> port 0 rxq <id> stop

Set rxonly forward, start testpmd

Send different src or dst IPv4 packets::

    p=Ether()/IP(src="192.168.0.1", dst="192.168.0.1")/Raw("x"*20)

Stop testpmd, find stopped queues can't receive packets, but other queues 
could receive packets
	
Setup these stopped queues on the port::

    testpmd> port 0 rxq <id> setup

Start these stopped queues on the port, start testpmd::

    testpmd> port 0 rxq <id> start

Send different src or dst IPv4 packets

Stop testpmd, check all the setup queues could receive packets


Test Case: Tx queue setup at runtime
====================================
Check txq ring size is 256::
                
    testpmd> show txq info 0 <id>
    Number of TXDs: 256

Stop one Tx queue on port 0::

    testpmd> port 0 txq <id> stop

Set txonly forward, start testpmd

Start testpmd, then stop, check this stopped queue only transmits 255 packets
  
Setup this stopped queue on the port::

    testpmd> port 0 txq <id> setup

Start this stopped queue on the port::

    testpmd> port 0 txq <id> start

Start then stop testpmd, check all queues could transmit lots of packets, 
not only 255 packets

Repeat above steps for 3 times


Test Case: Rx queue configure at runtime
========================================
Stop some Rx queues on port 0::

    testpmd> port 0 rxq <id> stop

Set rxonly forward, start testpmd

Send different src or dst IPv4 packets::

    p=Ether()/IP(src="192.168.0.1", dst="192.168.0.1")/Raw("x"*20)

Stop testpmd, find stopped queues can't receive packets, but other queues
could receive packets

Check rxq ring size is 256::
     
    testpmd> show rxq info 0 <id>
    Number of RXDs: 256

Reconfigure ring size as 512 for the stopped queues on port 0::

    testpmd> port config 0 rxq <id> ring_size 512

Setup these stopped queues on the port::

    testpmd> port 0 rxq <id> setup

Check stopped rxq ring sizes have been changed to 512

Start these stopped queues on the port, start testpmd::

    testpmd> port 0 rxq <id> start

Send different src or dst IPv4 packets

Stop testpmd, check all the setup queues could receive packets


Test Case: Tx queue configure at runtime
========================================
Check txq ring size is 256::

    testpmd> show txq info 0 <id>
    Number of TXDs: 256

Stop one Tx queue on port 0::

    testpmd> port 0 txq <id> stop

Set txonly forward, start testpmd

Start testpmd, then stop, check this stopped queue only transmits 255 packets

Reconfigure ring size as 512 for the stopped queues on port 0::

    testpmd> port config 0 txq <id> ring_size 512

Setup these stopped queues on the port::

    testpmd> port 0 txq <id> setup

Check stopped txq ring sizes have been changed to 512

Start these stopped queues on the port, start testpmd::

    testpmd> port 0 txq <id> start

Stop testpmd, check all queues could transmit lots of packets,
not only 511 packets

Repeat above steps for 3 times