2013-09-11 16:40:07 +08:00
/*******************************************************************************
*
* Intel Ethernet Controller XL710 Family Linux Driver
2016-01-14 08:51:45 +08:00
* Copyright ( c ) 2013 - 2016 Intel Corporation .
2013-09-11 16:40:07 +08:00
*
* This program is free software ; you can redistribute it and / or modify it
* under the terms and conditions of the GNU General Public License ,
* version 2 , as published by the Free Software Foundation .
*
* This program is distributed in the hope it will be useful , but WITHOUT
* ANY WARRANTY ; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE . See the GNU General Public License for
* more details .
*
2013-12-18 21:45:51 +08:00
* You should have received a copy of the GNU General Public License along
* with this program . If not , see < http : //www.gnu.org/licenses/>.
2013-09-11 16:40:07 +08:00
*
* The full GNU General Public License is included in this distribution in
* the file called " COPYING " .
*
* Contact Information :
* e1000 - devel Mailing List < e1000 - devel @ lists . sourceforge . net >
* Intel Corporation , 5200 N . E . Elam Young Parkway , Hillsboro , OR 97124 - 6497
*
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# include "i40e.h"
2015-04-08 07:45:34 +08:00
/*********************notification routines***********************/
/**
* i40e_vc_vf_broadcast
* @ pf : pointer to the PF structure
* @ opcode : operation code
* @ retval : return value
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
* send a message to all VFs on a given PF
* */
static void i40e_vc_vf_broadcast ( struct i40e_pf * pf ,
2017-05-12 02:23:11 +08:00
enum virtchnl_ops v_opcode ,
2015-04-08 07:45:34 +08:00
i40e_status v_retval , u8 * msg ,
u16 msglen )
{
struct i40e_hw * hw = & pf - > hw ;
struct i40e_vf * vf = pf - > vf ;
int i ;
for ( i = 0 ; i < pf - > num_alloc_vfs ; i + + , vf + + ) {
2016-04-13 18:08:29 +08:00
int abs_vf_id = vf - > vf_id + ( int ) hw - > func_caps . vf_base_id ;
2015-04-08 07:45:34 +08:00
/* Not all vfs are enabled so skip the ones that are not */
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_INIT , & vf - > vf_states ) & &
! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) )
2015-04-08 07:45:34 +08:00
continue ;
/* Ignore return value on purpose - a given VF may fail, but
* we need to keep going and send to all of them
*/
i40e_aq_send_msg_to_vf ( hw , abs_vf_id , v_opcode , v_retval ,
msg , msglen , NULL ) ;
}
}
/**
2016-03-11 06:59:50 +08:00
* i40e_vc_notify_vf_link_state
2015-04-08 07:45:34 +08:00
* @ vf : pointer to the VF structure
*
* send a link status message to a single VF
* */
static void i40e_vc_notify_vf_link_state ( struct i40e_vf * vf )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_pf_event pfe ;
2015-04-08 07:45:34 +08:00
struct i40e_pf * pf = vf - > pf ;
struct i40e_hw * hw = & pf - > hw ;
struct i40e_link_status * ls = & pf - > hw . phy . link_info ;
2016-04-13 18:08:29 +08:00
int abs_vf_id = vf - > vf_id + ( int ) hw - > func_caps . vf_base_id ;
2015-04-08 07:45:34 +08:00
2017-05-12 02:23:11 +08:00
pfe . event = VIRTCHNL_EVENT_LINK_CHANGE ;
2015-04-08 07:45:34 +08:00
pfe . severity = I40E_PF_EVENT_SEVERITY_INFO ;
if ( vf - > link_forced ) {
pfe . event_data . link_event . link_status = vf - > link_up ;
pfe . event_data . link_event . link_speed =
( vf - > link_up ? I40E_LINK_SPEED_40GB : 0 ) ;
} else {
pfe . event_data . link_event . link_status =
ls - > link_info & I40E_AQ_LINK_UP ;
pfe . event_data . link_event . link_speed = ls - > link_speed ;
}
2017-05-12 02:23:11 +08:00
i40e_aq_send_msg_to_vf ( hw , abs_vf_id , VIRTCHNL_OP_EVENT ,
2015-04-08 07:45:34 +08:00
0 , ( u8 * ) & pfe , sizeof ( pfe ) , NULL ) ;
}
/**
* i40e_vc_notify_link_state
* @ pf : pointer to the PF structure
*
* send a link status message to all VFs on a given PF
* */
void i40e_vc_notify_link_state ( struct i40e_pf * pf )
{
int i ;
for ( i = 0 ; i < pf - > num_alloc_vfs ; i + + )
i40e_vc_notify_vf_link_state ( & pf - > vf [ i ] ) ;
}
/**
* i40e_vc_notify_reset
* @ pf : pointer to the PF structure
*
* indicate a pending reset to all VFs on a given PF
* */
void i40e_vc_notify_reset ( struct i40e_pf * pf )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_pf_event pfe ;
2015-04-08 07:45:34 +08:00
2017-05-12 02:23:11 +08:00
pfe . event = VIRTCHNL_EVENT_RESET_IMPENDING ;
2015-04-08 07:45:34 +08:00
pfe . severity = I40E_PF_EVENT_SEVERITY_CERTAIN_DOOM ;
2017-05-12 02:23:11 +08:00
i40e_vc_vf_broadcast ( pf , VIRTCHNL_OP_EVENT , 0 ,
( u8 * ) & pfe , sizeof ( struct virtchnl_pf_event ) ) ;
2015-04-08 07:45:34 +08:00
}
/**
* i40e_vc_notify_vf_reset
* @ vf : pointer to the VF structure
*
* indicate a pending reset to the given VF
* */
void i40e_vc_notify_vf_reset ( struct i40e_vf * vf )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_pf_event pfe ;
2015-04-08 07:45:34 +08:00
int abs_vf_id ;
/* validate the request */
if ( ! vf | | vf - > vf_id > = vf - > pf - > num_alloc_vfs )
return ;
/* verify if the VF is in either init or active before proceeding */
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_INIT , & vf - > vf_states ) & &
! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) )
2015-04-08 07:45:34 +08:00
return ;
2016-04-13 18:08:29 +08:00
abs_vf_id = vf - > vf_id + ( int ) vf - > pf - > hw . func_caps . vf_base_id ;
2015-04-08 07:45:34 +08:00
2017-05-12 02:23:11 +08:00
pfe . event = VIRTCHNL_EVENT_RESET_IMPENDING ;
2015-04-08 07:45:34 +08:00
pfe . severity = I40E_PF_EVENT_SEVERITY_CERTAIN_DOOM ;
2017-05-12 02:23:11 +08:00
i40e_aq_send_msg_to_vf ( & vf - > pf - > hw , abs_vf_id , VIRTCHNL_OP_EVENT ,
2015-04-08 07:45:34 +08:00
0 , ( u8 * ) & pfe ,
2017-05-12 02:23:11 +08:00
sizeof ( struct virtchnl_pf_event ) , NULL ) ;
2015-04-08 07:45:34 +08:00
}
2013-09-11 16:40:07 +08:00
/***********************misc routines*****************************/
2014-03-06 17:02:28 +08:00
/**
* i40e_vc_disable_vf
2015-02-27 17:18:34 +08:00
* @ pf : pointer to the PF info
* @ vf : pointer to the VF info
2014-03-06 17:02:28 +08:00
*
* Disable the VF through a SW reset
* */
static inline void i40e_vc_disable_vf ( struct i40e_pf * pf , struct i40e_vf * vf )
{
2015-04-28 02:57:14 +08:00
i40e_vc_notify_vf_reset ( vf ) ;
i40e_reset_vf ( vf , false ) ;
2014-03-06 17:02:28 +08:00
}
2013-09-11 16:40:07 +08:00
/**
* i40e_vc_isvalid_vsi_id
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
* @ vsi_id : VF relative VSI id
2013-09-11 16:40:07 +08:00
*
2015-02-27 17:18:34 +08:00
* check for the valid VSI id
2013-09-11 16:40:07 +08:00
* */
2015-03-31 15:45:05 +08:00
static inline bool i40e_vc_isvalid_vsi_id ( struct i40e_vf * vf , u16 vsi_id )
2013-09-11 16:40:07 +08:00
{
struct i40e_pf * pf = vf - > pf ;
2015-03-31 15:45:05 +08:00
struct i40e_vsi * vsi = i40e_find_vsi_from_id ( pf , vsi_id ) ;
2013-09-11 16:40:07 +08:00
2015-03-31 15:45:05 +08:00
return ( vsi & & ( vsi - > vf_id = = vf - > vf_id ) ) ;
2013-09-11 16:40:07 +08:00
}
/**
* i40e_vc_isvalid_queue_id
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ vsi_id : vsi id
* @ qid : vsi relative queue id
*
* check for the valid queue id
* */
2015-03-31 15:45:05 +08:00
static inline bool i40e_vc_isvalid_queue_id ( struct i40e_vf * vf , u16 vsi_id ,
2013-09-11 16:40:07 +08:00
u8 qid )
{
struct i40e_pf * pf = vf - > pf ;
2015-03-31 15:45:05 +08:00
struct i40e_vsi * vsi = i40e_find_vsi_from_id ( pf , vsi_id ) ;
2013-09-11 16:40:07 +08:00
2015-03-31 15:45:05 +08:00
return ( vsi & & ( qid < vsi - > alloc_queue_pairs ) ) ;
2013-09-11 16:40:07 +08:00
}
/**
* i40e_vc_isvalid_vector_id
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
* @ vector_id : VF relative vector id
2013-09-11 16:40:07 +08:00
*
* check for the valid vector id
* */
static inline bool i40e_vc_isvalid_vector_id ( struct i40e_vf * vf , u8 vector_id )
{
struct i40e_pf * pf = vf - > pf ;
2014-02-11 16:26:32 +08:00
return vector_id < pf - > hw . func_caps . num_msix_vectors_vf ;
2013-09-11 16:40:07 +08:00
}
/***********************vf resource mgmt routines*****************/
/**
* i40e_vc_get_pf_queue_id
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2015-03-31 15:45:05 +08:00
* @ vsi_id : id of VSI as provided by the FW
2013-09-11 16:40:07 +08:00
* @ vsi_queue_id : vsi relative queue id
*
2015-02-27 17:18:34 +08:00
* return PF relative queue id
2013-09-11 16:40:07 +08:00
* */
2015-03-31 15:45:05 +08:00
static u16 i40e_vc_get_pf_queue_id ( struct i40e_vf * vf , u16 vsi_id ,
2013-09-11 16:40:07 +08:00
u8 vsi_queue_id )
{
struct i40e_pf * pf = vf - > pf ;
2015-03-31 15:45:05 +08:00
struct i40e_vsi * vsi = i40e_find_vsi_from_id ( pf , vsi_id ) ;
2013-09-11 16:40:07 +08:00
u16 pf_queue_id = I40E_QUEUE_END_OF_LIST ;
2015-03-31 15:45:05 +08:00
if ( ! vsi )
return pf_queue_id ;
2013-09-11 16:40:07 +08:00
if ( le16_to_cpu ( vsi - > info . mapping_flags ) &
I40E_AQ_VSI_QUE_MAP_NONCONTIG )
pf_queue_id =
le16_to_cpu ( vsi - > info . queue_mapping [ vsi_queue_id ] ) ;
else
pf_queue_id = le16_to_cpu ( vsi - > info . queue_mapping [ 0 ] ) +
vsi_queue_id ;
return pf_queue_id ;
}
/**
* i40e_config_irq_link_list
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2015-03-31 15:45:05 +08:00
* @ vsi_id : id of VSI as given by the FW
2013-09-11 16:40:07 +08:00
* @ vecmap : irq map info
*
* configure irq link list from the map
* */
2015-03-31 15:45:05 +08:00
static void i40e_config_irq_link_list ( struct i40e_vf * vf , u16 vsi_id ,
2017-05-12 02:23:11 +08:00
struct virtchnl_vector_map * vecmap )
2013-09-11 16:40:07 +08:00
{
unsigned long linklistmap = 0 , tempmap ;
struct i40e_pf * pf = vf - > pf ;
struct i40e_hw * hw = & pf - > hw ;
u16 vsi_queue_id , pf_queue_id ;
enum i40e_queue_type qtype ;
u16 next_q , vector_id ;
u32 reg , reg_idx ;
u16 itr_idx = 0 ;
vector_id = vecmap - > vector_id ;
/* setup the head */
if ( 0 = = vector_id )
reg_idx = I40E_VPINT_LNKLST0 ( vf - > vf_id ) ;
else
reg_idx = I40E_VPINT_LNKLSTN (
2014-02-11 16:26:32 +08:00
( ( pf - > hw . func_caps . num_msix_vectors_vf - 1 ) * vf - > vf_id ) +
( vector_id - 1 ) ) ;
2013-09-11 16:40:07 +08:00
if ( vecmap - > rxq_map = = 0 & & vecmap - > txq_map = = 0 ) {
/* Special case - No queues mapped on this vector */
wr32 ( hw , reg_idx , I40E_VPINT_LNKLST0_FIRSTQ_INDX_MASK ) ;
goto irq_list_done ;
}
tempmap = vecmap - > rxq_map ;
2013-09-24 13:17:36 +08:00
for_each_set_bit ( vsi_queue_id , & tempmap , I40E_MAX_VSI_QP ) {
2015-06-05 04:24:02 +08:00
linklistmap | = ( BIT ( I40E_VIRTCHNL_SUPPORTED_QTYPES *
vsi_queue_id ) ) ;
2013-09-11 16:40:07 +08:00
}
tempmap = vecmap - > txq_map ;
2013-09-24 13:17:36 +08:00
for_each_set_bit ( vsi_queue_id , & tempmap , I40E_MAX_VSI_QP ) {
2015-06-05 04:24:02 +08:00
linklistmap | = ( BIT ( I40E_VIRTCHNL_SUPPORTED_QTYPES *
vsi_queue_id + 1 ) ) ;
2013-09-11 16:40:07 +08:00
}
next_q = find_first_bit ( & linklistmap ,
( I40E_MAX_VSI_QP *
I40E_VIRTCHNL_SUPPORTED_QTYPES ) ) ;
2015-11-07 07:26:10 +08:00
vsi_queue_id = next_q / I40E_VIRTCHNL_SUPPORTED_QTYPES ;
qtype = next_q % I40E_VIRTCHNL_SUPPORTED_QTYPES ;
2015-03-31 15:45:05 +08:00
pf_queue_id = i40e_vc_get_pf_queue_id ( vf , vsi_id , vsi_queue_id ) ;
2013-09-11 16:40:07 +08:00
reg = ( ( qtype < < I40E_VPINT_LNKLSTN_FIRSTQ_TYPE_SHIFT ) | pf_queue_id ) ;
wr32 ( hw , reg_idx , reg ) ;
while ( next_q < ( I40E_MAX_VSI_QP * I40E_VIRTCHNL_SUPPORTED_QTYPES ) ) {
switch ( qtype ) {
case I40E_QUEUE_TYPE_RX :
reg_idx = I40E_QINT_RQCTL ( pf_queue_id ) ;
itr_idx = vecmap - > rxitr_idx ;
break ;
case I40E_QUEUE_TYPE_TX :
reg_idx = I40E_QINT_TQCTL ( pf_queue_id ) ;
itr_idx = vecmap - > txitr_idx ;
break ;
default :
break ;
}
next_q = find_next_bit ( & linklistmap ,
( I40E_MAX_VSI_QP *
I40E_VIRTCHNL_SUPPORTED_QTYPES ) ,
next_q + 1 ) ;
2013-12-18 21:46:00 +08:00
if ( next_q <
( I40E_MAX_VSI_QP * I40E_VIRTCHNL_SUPPORTED_QTYPES ) ) {
2013-09-11 16:40:07 +08:00
vsi_queue_id = next_q / I40E_VIRTCHNL_SUPPORTED_QTYPES ;
qtype = next_q % I40E_VIRTCHNL_SUPPORTED_QTYPES ;
2015-03-31 15:45:05 +08:00
pf_queue_id = i40e_vc_get_pf_queue_id ( vf , vsi_id ,
2013-09-11 16:40:07 +08:00
vsi_queue_id ) ;
} else {
pf_queue_id = I40E_QUEUE_END_OF_LIST ;
qtype = 0 ;
}
/* format for the RQCTL & TQCTL regs is same */
reg = ( vector_id ) |
( qtype < < I40E_QINT_RQCTL_NEXTQ_TYPE_SHIFT ) |
( pf_queue_id < < I40E_QINT_RQCTL_NEXTQ_INDX_SHIFT ) |
2015-06-05 04:24:02 +08:00
BIT ( I40E_QINT_RQCTL_CAUSE_ENA_SHIFT ) |
2013-09-11 16:40:07 +08:00
( itr_idx < < I40E_QINT_RQCTL_ITR_INDX_SHIFT ) ;
wr32 ( hw , reg_idx , reg ) ;
}
2015-07-11 07:36:08 +08:00
/* if the vf is running in polling mode and using interrupt zero,
* need to disable auto - mask on enabling zero interrupt for VFs .
*/
2017-05-12 02:23:11 +08:00
if ( ( vf - > driver_caps & VIRTCHNL_VF_OFFLOAD_RX_POLLING ) & &
2015-07-11 07:36:08 +08:00
( vector_id = = 0 ) ) {
reg = rd32 ( hw , I40E_GLINT_CTL ) ;
if ( ! ( reg & I40E_GLINT_CTL_DIS_AUTOMASK_VF0_MASK ) ) {
reg | = I40E_GLINT_CTL_DIS_AUTOMASK_VF0_MASK ;
wr32 ( hw , I40E_GLINT_CTL , reg ) ;
}
}
2013-09-11 16:40:07 +08:00
irq_list_done :
i40e_flush ( hw ) ;
}
2016-01-21 03:40:01 +08:00
/**
* i40e_release_iwarp_qvlist
* @ vf : pointer to the VF .
*
* */
static void i40e_release_iwarp_qvlist ( struct i40e_vf * vf )
{
struct i40e_pf * pf = vf - > pf ;
2017-05-12 02:23:11 +08:00
struct virtchnl_iwarp_qvlist_info * qvlist_info = vf - > qvlist_info ;
2016-01-21 03:40:01 +08:00
u32 msix_vf ;
u32 i ;
if ( ! vf - > qvlist_info )
return ;
msix_vf = pf - > hw . func_caps . num_msix_vectors_vf ;
for ( i = 0 ; i < qvlist_info - > num_vectors ; i + + ) {
2017-05-12 02:23:11 +08:00
struct virtchnl_iwarp_qv_info * qv_info ;
2016-01-21 03:40:01 +08:00
u32 next_q_index , next_q_type ;
struct i40e_hw * hw = & pf - > hw ;
u32 v_idx , reg_idx , reg ;
qv_info = & qvlist_info - > qv_info [ i ] ;
if ( ! qv_info )
continue ;
v_idx = qv_info - > v_idx ;
if ( qv_info - > ceq_idx ! = I40E_QUEUE_INVALID_IDX ) {
/* Figure out the queue after CEQ and make that the
* first queue .
*/
reg_idx = ( msix_vf - 1 ) * vf - > vf_id + qv_info - > ceq_idx ;
reg = rd32 ( hw , I40E_VPINT_CEQCTL ( reg_idx ) ) ;
next_q_index = ( reg & I40E_VPINT_CEQCTL_NEXTQ_INDX_MASK )
> > I40E_VPINT_CEQCTL_NEXTQ_INDX_SHIFT ;
next_q_type = ( reg & I40E_VPINT_CEQCTL_NEXTQ_TYPE_MASK )
> > I40E_VPINT_CEQCTL_NEXTQ_TYPE_SHIFT ;
reg_idx = ( ( msix_vf - 1 ) * vf - > vf_id ) + ( v_idx - 1 ) ;
reg = ( next_q_index &
I40E_VPINT_LNKLSTN_FIRSTQ_INDX_MASK ) |
( next_q_type < <
I40E_VPINT_LNKLSTN_FIRSTQ_TYPE_SHIFT ) ;
wr32 ( hw , I40E_VPINT_LNKLSTN ( reg_idx ) , reg ) ;
}
}
kfree ( vf - > qvlist_info ) ;
vf - > qvlist_info = NULL ;
}
/**
* i40e_config_iwarp_qvlist
* @ vf : pointer to the VF info
* @ qvlist_info : queue and vector list
*
* Return 0 on success or < 0 on error
* */
static int i40e_config_iwarp_qvlist ( struct i40e_vf * vf ,
2017-05-12 02:23:11 +08:00
struct virtchnl_iwarp_qvlist_info * qvlist_info )
2016-01-21 03:40:01 +08:00
{
struct i40e_pf * pf = vf - > pf ;
struct i40e_hw * hw = & pf - > hw ;
2017-05-12 02:23:11 +08:00
struct virtchnl_iwarp_qv_info * qv_info ;
2016-01-21 03:40:01 +08:00
u32 v_idx , i , reg_idx , reg ;
u32 next_q_idx , next_q_type ;
u32 msix_vf , size ;
2017-05-12 02:23:11 +08:00
size = sizeof ( struct virtchnl_iwarp_qvlist_info ) +
( sizeof ( struct virtchnl_iwarp_qv_info ) *
2016-01-21 03:40:01 +08:00
( qvlist_info - > num_vectors - 1 ) ) ;
vf - > qvlist_info = kzalloc ( size , GFP_KERNEL ) ;
vf - > qvlist_info - > num_vectors = qvlist_info - > num_vectors ;
msix_vf = pf - > hw . func_caps . num_msix_vectors_vf ;
for ( i = 0 ; i < qvlist_info - > num_vectors ; i + + ) {
qv_info = & qvlist_info - > qv_info [ i ] ;
if ( ! qv_info )
continue ;
v_idx = qv_info - > v_idx ;
/* Validate vector id belongs to this vf */
if ( ! i40e_vc_isvalid_vector_id ( vf , v_idx ) )
goto err ;
vf - > qvlist_info - > qv_info [ i ] = * qv_info ;
reg_idx = ( ( msix_vf - 1 ) * vf - > vf_id ) + ( v_idx - 1 ) ;
/* We might be sharing the interrupt, so get the first queue
* index and type , push it down the list by adding the new
* queue on top . Also link it with the new queue in CEQCTL .
*/
reg = rd32 ( hw , I40E_VPINT_LNKLSTN ( reg_idx ) ) ;
next_q_idx = ( ( reg & I40E_VPINT_LNKLSTN_FIRSTQ_INDX_MASK ) > >
I40E_VPINT_LNKLSTN_FIRSTQ_INDX_SHIFT ) ;
next_q_type = ( ( reg & I40E_VPINT_LNKLSTN_FIRSTQ_TYPE_MASK ) > >
I40E_VPINT_LNKLSTN_FIRSTQ_TYPE_SHIFT ) ;
if ( qv_info - > ceq_idx ! = I40E_QUEUE_INVALID_IDX ) {
reg_idx = ( msix_vf - 1 ) * vf - > vf_id + qv_info - > ceq_idx ;
reg = ( I40E_VPINT_CEQCTL_CAUSE_ENA_MASK |
( v_idx < < I40E_VPINT_CEQCTL_MSIX_INDX_SHIFT ) |
( qv_info - > itr_idx < < I40E_VPINT_CEQCTL_ITR_INDX_SHIFT ) |
( next_q_type < < I40E_VPINT_CEQCTL_NEXTQ_TYPE_SHIFT ) |
( next_q_idx < < I40E_VPINT_CEQCTL_NEXTQ_INDX_SHIFT ) ) ;
wr32 ( hw , I40E_VPINT_CEQCTL ( reg_idx ) , reg ) ;
reg_idx = ( ( msix_vf - 1 ) * vf - > vf_id ) + ( v_idx - 1 ) ;
reg = ( qv_info - > ceq_idx &
I40E_VPINT_LNKLSTN_FIRSTQ_INDX_MASK ) |
( I40E_QUEUE_TYPE_PE_CEQ < <
I40E_VPINT_LNKLSTN_FIRSTQ_TYPE_SHIFT ) ;
wr32 ( hw , I40E_VPINT_LNKLSTN ( reg_idx ) , reg ) ;
}
if ( qv_info - > aeq_idx ! = I40E_QUEUE_INVALID_IDX ) {
reg = ( I40E_VPINT_AEQCTL_CAUSE_ENA_MASK |
( v_idx < < I40E_VPINT_AEQCTL_MSIX_INDX_SHIFT ) |
( qv_info - > itr_idx < < I40E_VPINT_AEQCTL_ITR_INDX_SHIFT ) ) ;
wr32 ( hw , I40E_VPINT_AEQCTL ( vf - > vf_id ) , reg ) ;
}
}
return 0 ;
err :
kfree ( vf - > qvlist_info ) ;
vf - > qvlist_info = NULL ;
return - EINVAL ;
}
2013-09-11 16:40:07 +08:00
/**
* i40e_config_vsi_tx_queue
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2015-03-31 15:45:05 +08:00
* @ vsi_id : id of VSI as provided by the FW
2013-09-11 16:40:07 +08:00
* @ vsi_queue_id : vsi relative queue index
* @ info : config . info
*
* configure tx queue
* */
2015-03-31 15:45:05 +08:00
static int i40e_config_vsi_tx_queue ( struct i40e_vf * vf , u16 vsi_id ,
2013-09-11 16:40:07 +08:00
u16 vsi_queue_id ,
2017-05-12 02:23:11 +08:00
struct virtchnl_txq_info * info )
2013-09-11 16:40:07 +08:00
{
struct i40e_pf * pf = vf - > pf ;
struct i40e_hw * hw = & pf - > hw ;
struct i40e_hmc_obj_txq tx_ctx ;
2015-03-31 15:45:05 +08:00
struct i40e_vsi * vsi ;
2013-09-11 16:40:07 +08:00
u16 pf_queue_id ;
u32 qtx_ctl ;
int ret = 0 ;
2016-08-25 02:33:50 +08:00
if ( ! i40e_vc_isvalid_vsi_id ( vf , info - > vsi_id ) ) {
ret = - ENOENT ;
goto error_context ;
}
2015-03-31 15:45:05 +08:00
pf_queue_id = i40e_vc_get_pf_queue_id ( vf , vsi_id , vsi_queue_id ) ;
vsi = i40e_find_vsi_from_id ( pf , vsi_id ) ;
2016-08-25 02:33:50 +08:00
if ( ! vsi ) {
ret = - ENOENT ;
goto error_context ;
}
2013-09-11 16:40:07 +08:00
/* clear the context structure first */
memset ( & tx_ctx , 0 , sizeof ( struct i40e_hmc_obj_txq ) ) ;
/* only set the required fields */
tx_ctx . base = info - > dma_ring_addr / 128 ;
tx_ctx . qlen = info - > ring_len ;
2015-03-31 15:45:05 +08:00
tx_ctx . rdylist = le16_to_cpu ( vsi - > info . qs_handle [ 0 ] ) ;
2013-09-11 16:40:07 +08:00
tx_ctx . rdylist_act = 0 ;
2014-05-22 14:31:25 +08:00
tx_ctx . head_wb_ena = info - > headwb_enabled ;
tx_ctx . head_wb_addr = info - > dma_headwb_addr ;
2013-09-11 16:40:07 +08:00
/* clear the context in the HMC */
ret = i40e_clear_lan_tx_queue_context ( hw , pf_queue_id ) ;
if ( ret ) {
dev_err ( & pf - > pdev - > dev ,
" Failed to clear VF LAN Tx queue context %d, error: %d \n " ,
pf_queue_id , ret ) ;
ret = - ENOENT ;
goto error_context ;
}
/* set the context in the HMC */
ret = i40e_set_lan_tx_queue_context ( hw , pf_queue_id , & tx_ctx ) ;
if ( ret ) {
dev_err ( & pf - > pdev - > dev ,
" Failed to set VF LAN Tx queue context %d error: %d \n " ,
pf_queue_id , ret ) ;
ret = - ENOENT ;
goto error_context ;
}
/* associate this queue with the PCI VF function */
qtx_ctl = I40E_QTX_CTL_VF_QUEUE ;
2013-09-28 15:14:19 +08:00
qtx_ctl | = ( ( hw - > pf_id < < I40E_QTX_CTL_PF_INDX_SHIFT )
2013-09-11 16:40:07 +08:00
& I40E_QTX_CTL_PF_INDX_MASK ) ;
qtx_ctl | = ( ( ( vf - > vf_id + hw - > func_caps . vf_base_id )
< < I40E_QTX_CTL_VFVM_INDX_SHIFT )
& I40E_QTX_CTL_VFVM_INDX_MASK ) ;
wr32 ( hw , I40E_QTX_CTL ( pf_queue_id ) , qtx_ctl ) ;
i40e_flush ( hw ) ;
error_context :
return ret ;
}
/**
* i40e_config_vsi_rx_queue
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2015-03-31 15:45:05 +08:00
* @ vsi_id : id of VSI as provided by the FW
2013-09-11 16:40:07 +08:00
* @ vsi_queue_id : vsi relative queue index
* @ info : config . info
*
* configure rx queue
* */
2015-03-31 15:45:05 +08:00
static int i40e_config_vsi_rx_queue ( struct i40e_vf * vf , u16 vsi_id ,
2013-09-11 16:40:07 +08:00
u16 vsi_queue_id ,
2017-05-12 02:23:11 +08:00
struct virtchnl_rxq_info * info )
2013-09-11 16:40:07 +08:00
{
struct i40e_pf * pf = vf - > pf ;
struct i40e_hw * hw = & pf - > hw ;
struct i40e_hmc_obj_rxq rx_ctx ;
u16 pf_queue_id ;
int ret = 0 ;
2015-03-31 15:45:05 +08:00
pf_queue_id = i40e_vc_get_pf_queue_id ( vf , vsi_id , vsi_queue_id ) ;
2013-09-11 16:40:07 +08:00
/* clear the context structure first */
memset ( & rx_ctx , 0 , sizeof ( struct i40e_hmc_obj_rxq ) ) ;
/* only set the required fields */
rx_ctx . base = info - > dma_ring_addr / 128 ;
rx_ctx . qlen = info - > ring_len ;
if ( info - > splithdr_enabled ) {
rx_ctx . hsplit_0 = I40E_RX_SPLIT_L2 |
I40E_RX_SPLIT_IP |
I40E_RX_SPLIT_TCP_UDP |
I40E_RX_SPLIT_SCTP ;
/* header length validation */
if ( info - > hdr_size > ( ( 2 * 1024 ) - 64 ) ) {
ret = - EINVAL ;
goto error_param ;
}
rx_ctx . hbuff = info - > hdr_size > > I40E_RXQ_CTX_HBUFF_SHIFT ;
2016-04-19 02:33:45 +08:00
/* set split mode 10b */
2016-01-16 06:33:08 +08:00
rx_ctx . dtype = I40E_RX_DTYPE_HEADER_SPLIT ;
2013-09-11 16:40:07 +08:00
}
/* databuffer length validation */
if ( info - > databuffer_size > ( ( 16 * 1024 ) - 128 ) ) {
ret = - EINVAL ;
goto error_param ;
}
rx_ctx . dbuff = info - > databuffer_size > > I40E_RXQ_CTX_DBUFF_SHIFT ;
/* max pkt. length validation */
if ( info - > max_pkt_size > = ( 16 * 1024 ) | | info - > max_pkt_size < 64 ) {
ret = - EINVAL ;
goto error_param ;
}
rx_ctx . rxmax = info - > max_pkt_size ;
/* enable 32bytes desc always */
rx_ctx . dsize = 1 ;
/* default values */
rx_ctx . lrxqthresh = 2 ;
rx_ctx . crcstrip = 1 ;
2014-04-09 13:58:55 +08:00
rx_ctx . prefena = 1 ;
2014-07-29 12:01:03 +08:00
rx_ctx . l2tsel = 1 ;
2013-09-11 16:40:07 +08:00
/* clear the context in the HMC */
ret = i40e_clear_lan_rx_queue_context ( hw , pf_queue_id ) ;
if ( ret ) {
dev_err ( & pf - > pdev - > dev ,
" Failed to clear VF LAN Rx queue context %d, error: %d \n " ,
pf_queue_id , ret ) ;
ret = - ENOENT ;
goto error_param ;
}
/* set the context in the HMC */
ret = i40e_set_lan_rx_queue_context ( hw , pf_queue_id , & rx_ctx ) ;
if ( ret ) {
dev_err ( & pf - > pdev - > dev ,
" Failed to set VF LAN Rx queue context %d error: %d \n " ,
pf_queue_id , ret ) ;
ret = - ENOENT ;
goto error_param ;
}
error_param :
return ret ;
}
/**
* i40e_alloc_vsi_res
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ type : type of VSI to allocate
*
2015-02-27 17:18:34 +08:00
* alloc VF vsi context & resources
2013-09-11 16:40:07 +08:00
* */
static int i40e_alloc_vsi_res ( struct i40e_vf * vf , enum i40e_vsi_type type )
{
struct i40e_mac_filter * f = NULL ;
struct i40e_pf * pf = vf - > pf ;
struct i40e_vsi * vsi ;
int ret = 0 ;
vsi = i40e_vsi_setup ( pf , type , pf - > vsi [ pf - > lan_vsi ] - > seid , vf - > vf_id ) ;
if ( ! vsi ) {
dev_err ( & pf - > pdev - > dev ,
2015-02-27 17:18:34 +08:00
" add vsi failed for VF %d, aq_err %d \n " ,
2013-09-11 16:40:07 +08:00
vf - > vf_id , pf - > hw . aq . asq_last_status ) ;
ret = - ENOENT ;
goto error_alloc_vsi_res ;
}
if ( type = = I40E_VSI_SRIOV ) {
2016-05-17 01:26:33 +08:00
u64 hena = i40e_pf_get_default_rss_hena ( pf ) ;
2016-11-09 05:05:10 +08:00
u8 broadcast [ ETH_ALEN ] ;
2016-05-17 01:26:33 +08:00
2015-03-31 15:45:05 +08:00
vf - > lan_vsi_idx = vsi - > idx ;
2013-09-11 16:40:07 +08:00
vf - > lan_vsi_id = vsi - > id ;
2013-11-28 14:39:34 +08:00
/* If the port VLAN has been configured and then the
* VF driver was removed then the VSI port VLAN
* configuration was destroyed . Check if there is
* a port VLAN and restore the VSI configuration if
* needed .
*/
if ( vf - > port_vlan_id )
i40e_vsi_add_pvid ( vsi , vf - > port_vlan_id ) ;
2015-10-01 02:09:03 +08:00
2016-10-06 00:30:37 +08:00
spin_lock_bh ( & vsi - > mac_filter_hash_lock ) ;
2015-11-20 03:34:17 +08:00
if ( is_valid_ether_addr ( vf - > default_lan_addr . addr ) ) {
2016-11-12 04:39:37 +08:00
f = i40e_add_mac_filter ( vsi ,
vf - > default_lan_addr . addr ) ;
2015-11-20 03:34:17 +08:00
if ( ! f )
dev_info ( & pf - > pdev - > dev ,
" Could not add MAC filter %pM for VF %d \n " ,
vf - > default_lan_addr . addr , vf - > vf_id ) ;
}
2016-11-09 05:05:10 +08:00
eth_broadcast_addr ( broadcast ) ;
2016-11-12 04:39:37 +08:00
f = i40e_add_mac_filter ( vsi , broadcast ) ;
2016-11-09 05:05:10 +08:00
if ( ! f )
dev_info ( & pf - > pdev - > dev ,
" Could not allocate VF broadcast filter \n " ) ;
2016-10-06 00:30:37 +08:00
spin_unlock_bh ( & vsi - > mac_filter_hash_lock ) ;
2017-01-31 04:29:33 +08:00
wr32 ( & pf - > hw , I40E_VFQF_HENA1 ( 0 , vf - > vf_id ) , ( u32 ) hena ) ;
wr32 ( & pf - > hw , I40E_VFQF_HENA1 ( 1 , vf - > vf_id ) , ( u32 ) ( hena > > 32 ) ) ;
2013-09-11 16:40:07 +08:00
}
2013-11-26 18:49:24 +08:00
2013-09-11 16:40:07 +08:00
/* program mac filter */
2015-11-06 09:01:02 +08:00
ret = i40e_sync_vsi_filters ( vsi ) ;
2014-02-13 19:48:44 +08:00
if ( ret )
2013-09-11 16:40:07 +08:00
dev_err ( & pf - > pdev - > dev , " Unable to program ucast filters \n " ) ;
2014-03-06 17:02:29 +08:00
/* Set VF bandwidth if specified */
if ( vf - > tx_rate ) {
ret = i40e_aq_config_vsi_bw_limit ( & pf - > hw , vsi - > seid ,
vf - > tx_rate / 50 , 0 , NULL ) ;
if ( ret )
dev_err ( & pf - > pdev - > dev , " Unable to set tx rate, VF %d, error code %d. \n " ,
vf - > vf_id , ret ) ;
}
2013-09-11 16:40:07 +08:00
error_alloc_vsi_res :
return ret ;
}
2013-11-28 14:39:26 +08:00
/**
* i40e_enable_vf_mappings
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-11-28 14:39:26 +08:00
*
2015-02-27 17:18:34 +08:00
* enable VF mappings
2013-11-28 14:39:26 +08:00
* */
static void i40e_enable_vf_mappings ( struct i40e_vf * vf )
{
struct i40e_pf * pf = vf - > pf ;
struct i40e_hw * hw = & pf - > hw ;
u32 reg , total_queue_pairs = 0 ;
int j ;
/* Tell the hardware we're using noncontiguous mapping. HW requires
* that VF queues be mapped using this method , even when they are
* contiguous in real life
*/
2016-02-18 08:12:21 +08:00
i40e_write_rx_ctl ( hw , I40E_VSILAN_QBASE ( vf - > lan_vsi_id ) ,
I40E_VSILAN_QBASE_VSIQTABLE_ENA_MASK ) ;
2013-11-28 14:39:26 +08:00
/* enable VF vplan_qtable mappings */
reg = I40E_VPLAN_MAPENA_TXRX_ENA_MASK ;
wr32 ( hw , I40E_VPLAN_MAPENA ( vf - > vf_id ) , reg ) ;
/* map PF queues to VF queues */
2015-03-31 15:45:05 +08:00
for ( j = 0 ; j < pf - > vsi [ vf - > lan_vsi_idx ] - > alloc_queue_pairs ; j + + ) {
u16 qid = i40e_vc_get_pf_queue_id ( vf , vf - > lan_vsi_id , j ) ;
2015-08-29 05:55:54 +08:00
2013-11-28 14:39:26 +08:00
reg = ( qid & I40E_VPLAN_QTABLE_QINDEX_MASK ) ;
wr32 ( hw , I40E_VPLAN_QTABLE ( total_queue_pairs , vf - > vf_id ) , reg ) ;
total_queue_pairs + + ;
}
/* map PF queues to VSI */
for ( j = 0 ; j < 7 ; j + + ) {
2015-03-31 15:45:05 +08:00
if ( j * 2 > = pf - > vsi [ vf - > lan_vsi_idx ] - > alloc_queue_pairs ) {
2013-11-28 14:39:26 +08:00
reg = 0x07FF07FF ; /* unused */
} else {
2015-03-31 15:45:05 +08:00
u16 qid = i40e_vc_get_pf_queue_id ( vf , vf - > lan_vsi_id ,
2013-11-28 14:39:26 +08:00
j * 2 ) ;
reg = qid ;
2015-03-31 15:45:05 +08:00
qid = i40e_vc_get_pf_queue_id ( vf , vf - > lan_vsi_id ,
2013-11-28 14:39:26 +08:00
( j * 2 ) + 1 ) ;
reg | = qid < < 16 ;
}
2016-02-18 08:12:21 +08:00
i40e_write_rx_ctl ( hw , I40E_VSILAN_QTABLE ( j , vf - > lan_vsi_id ) ,
reg ) ;
2013-11-28 14:39:26 +08:00
}
i40e_flush ( hw ) ;
}
/**
* i40e_disable_vf_mappings
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-11-28 14:39:26 +08:00
*
2015-02-27 17:18:34 +08:00
* disable VF mappings
2013-11-28 14:39:26 +08:00
* */
static void i40e_disable_vf_mappings ( struct i40e_vf * vf )
{
struct i40e_pf * pf = vf - > pf ;
struct i40e_hw * hw = & pf - > hw ;
int i ;
/* disable qp mappings */
wr32 ( hw , I40E_VPLAN_MAPENA ( vf - > vf_id ) , 0 ) ;
for ( i = 0 ; i < I40E_MAX_VSI_QP ; i + + )
wr32 ( hw , I40E_VPLAN_QTABLE ( i , vf - > vf_id ) ,
I40E_QUEUE_END_OF_LIST ) ;
i40e_flush ( hw ) ;
}
/**
* i40e_free_vf_res
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-11-28 14:39:26 +08:00
*
2015-02-27 17:18:34 +08:00
* free VF resources
2013-11-28 14:39:26 +08:00
* */
static void i40e_free_vf_res ( struct i40e_vf * vf )
{
struct i40e_pf * pf = vf - > pf ;
2013-11-28 14:39:27 +08:00
struct i40e_hw * hw = & pf - > hw ;
u32 reg_idx , reg ;
int i , msix_vf ;
2013-11-28 14:39:26 +08:00
2017-02-22 07:55:42 +08:00
/* Start by disabling VF's configuration API to prevent the OS from
* accessing the VF ' s VSI after it ' s freed / invalidated .
*/
2017-04-13 16:45:54 +08:00
clear_bit ( I40E_VF_STATE_INIT , & vf - > vf_states ) ;
2017-02-22 07:55:42 +08:00
2013-11-28 14:39:26 +08:00
/* free vsi & disconnect it from the parent uplink */
2015-03-31 15:45:05 +08:00
if ( vf - > lan_vsi_idx ) {
i40e_vsi_release ( pf - > vsi [ vf - > lan_vsi_idx ] ) ;
vf - > lan_vsi_idx = 0 ;
2013-11-28 14:39:26 +08:00
vf - > lan_vsi_id = 0 ;
2016-10-12 06:26:55 +08:00
vf - > num_mac = 0 ;
2013-11-28 14:39:26 +08:00
}
2014-02-11 16:26:32 +08:00
msix_vf = pf - > hw . func_caps . num_msix_vectors_vf ;
2013-11-28 14:39:27 +08:00
/* disable interrupts so the VF starts in a known state */
for ( i = 0 ; i < msix_vf ; i + + ) {
/* format is same for both registers */
if ( 0 = = i )
reg_idx = I40E_VFINT_DYN_CTL0 ( vf - > vf_id ) ;
else
reg_idx = I40E_VFINT_DYN_CTLN ( ( ( msix_vf - 1 ) *
( vf - > vf_id ) )
+ ( i - 1 ) ) ;
wr32 ( hw , reg_idx , I40E_VFINT_DYN_CTLN_CLEARPBA_MASK ) ;
i40e_flush ( hw ) ;
}
2013-11-28 14:39:26 +08:00
2013-11-28 14:39:27 +08:00
/* clear the irq settings */
for ( i = 0 ; i < msix_vf ; i + + ) {
/* format is same for both registers */
if ( 0 = = i )
reg_idx = I40E_VPINT_LNKLST0 ( vf - > vf_id ) ;
else
reg_idx = I40E_VPINT_LNKLSTN ( ( ( msix_vf - 1 ) *
( vf - > vf_id ) )
+ ( i - 1 ) ) ;
reg = ( I40E_VPINT_LNKLSTN_FIRSTQ_TYPE_MASK |
I40E_VPINT_LNKLSTN_FIRSTQ_INDX_MASK ) ;
wr32 ( hw , reg_idx , reg ) ;
i40e_flush ( hw ) ;
}
2017-02-28 06:29:06 +08:00
/* reset some of the state variables keeping track of the resources */
2013-11-28 14:39:26 +08:00
vf - > num_queue_pairs = 0 ;
vf - > vf_states = 0 ;
}
/**
* i40e_alloc_vf_res
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-11-28 14:39:26 +08:00
*
2015-02-27 17:18:34 +08:00
* allocate VF resources
2013-11-28 14:39:26 +08:00
* */
static int i40e_alloc_vf_res ( struct i40e_vf * vf )
{
struct i40e_pf * pf = vf - > pf ;
int total_queue_pairs = 0 ;
int ret ;
/* allocate hw vsi context & associated resources */
ret = i40e_alloc_vsi_res ( vf , I40E_VSI_SRIOV ) ;
if ( ret )
goto error_alloc ;
2015-03-31 15:45:05 +08:00
total_queue_pairs + = pf - > vsi [ vf - > lan_vsi_idx ] - > alloc_queue_pairs ;
2016-04-13 18:08:21 +08:00
if ( vf - > trusted )
set_bit ( I40E_VIRTCHNL_VF_CAP_PRIVILEGE , & vf - > vf_caps ) ;
else
clear_bit ( I40E_VIRTCHNL_VF_CAP_PRIVILEGE , & vf - > vf_caps ) ;
2013-11-28 14:39:26 +08:00
/* store the total qps number for the runtime
2015-02-27 17:18:34 +08:00
* VF req validation
2013-11-28 14:39:26 +08:00
*/
vf - > num_queue_pairs = total_queue_pairs ;
2015-02-27 17:18:34 +08:00
/* VF is now completely initialized */
2017-04-13 16:45:54 +08:00
set_bit ( I40E_VF_STATE_INIT , & vf - > vf_states ) ;
2013-11-28 14:39:26 +08:00
error_alloc :
if ( ret )
i40e_free_vf_res ( vf ) ;
return ret ;
}
2013-11-28 14:39:27 +08:00
# define VF_DEVICE_STATUS 0xAA
# define VF_TRANS_PENDING_MASK 0x20
/**
* i40e_quiesce_vf_pci
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF structure
2013-11-28 14:39:27 +08:00
*
* Wait for VF PCI transactions to be cleared after reset . Returns - EIO
* if the transactions never clear .
* */
static int i40e_quiesce_vf_pci ( struct i40e_vf * vf )
{
struct i40e_pf * pf = vf - > pf ;
struct i40e_hw * hw = & pf - > hw ;
int vf_abs_id , i ;
u32 reg ;
2013-11-28 14:39:36 +08:00
vf_abs_id = vf - > vf_id + hw - > func_caps . vf_base_id ;
2013-11-28 14:39:27 +08:00
wr32 ( hw , I40E_PF_PCI_CIAA ,
VF_DEVICE_STATUS | ( vf_abs_id < < I40E_PF_PCI_CIAA_VF_NUM_SHIFT ) ) ;
for ( i = 0 ; i < 100 ; i + + ) {
reg = rd32 ( hw , I40E_PF_PCI_CIAD ) ;
if ( ( reg & VF_TRANS_PENDING_MASK ) = = 0 )
return 0 ;
udelay ( 1 ) ;
}
return - EIO ;
}
2013-09-11 16:40:07 +08:00
/**
2017-04-13 16:45:51 +08:00
* i40e_trigger_vf_reset
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF structure
2013-09-11 16:40:07 +08:00
* @ flr : VFLR was issued or not
*
2017-04-13 16:45:51 +08:00
* Trigger hardware to start a reset for a particular VF . Expects the caller
* to wait the proper amount of time to allow hardware to reset the VF before
* it cleans up and restores VF functionality .
2013-09-11 16:40:07 +08:00
* */
2017-04-13 16:45:51 +08:00
static void i40e_trigger_vf_reset ( struct i40e_vf * vf , bool flr )
2013-09-11 16:40:07 +08:00
{
struct i40e_pf * pf = vf - > pf ;
struct i40e_hw * hw = & pf - > hw ;
2016-03-11 06:59:47 +08:00
u32 reg , reg_idx , bit_idx ;
i40e: add locking around VF reset
During VF deallocation, we need to lock out the VF reset code. However,
we cannot depend on simply masking the interrupt, as this does not lock
out the service task, which can still call the reset routine. Instead,
leave the interrupt enabled, but add locking around the VF disable and
reset routines.
For the disable code, we wait to get the lock, as the reset code will
take a finite amount of time to run. For the reset code, we just return
if we fail to get the lock. Since we know that the VFs are being
disabled, we don't need to handle the reset.
This fixes a panic when disabling SR-IOV.
Change-ID: Iea0a6cdef35c331f48c6d5b2f8e6f0e86322e7d8
Signed-off-by: Mitch Williams <mitch.a.williams@intel.com>
Tested-by: Jim Young <james.m.young@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2015-01-09 19:18:15 +08:00
2013-09-11 16:40:07 +08:00
/* warn the VF */
2017-04-13 16:45:54 +08:00
clear_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) ;
2013-09-11 16:40:07 +08:00
2017-02-22 07:55:42 +08:00
/* Disable VF's configuration API during reset. The flag is re-enabled
* in i40e_alloc_vf_res ( ) , when it ' s safe again to access VF ' s VSI .
* It ' s normally disabled in i40e_free_vf_res ( ) , but it ' s safer
* to do it earlier to give some time to finish to any VF config
* functions that may still be running at this point .
*/
2017-04-13 16:45:54 +08:00
clear_bit ( I40E_VF_STATE_INIT , & vf - > vf_states ) ;
2017-02-22 07:55:42 +08:00
2013-11-28 14:39:27 +08:00
/* In the case of a VFLR, the HW has already reset the VF and we
* just need to clean up , so don ' t hit the VFRTRIG register .
2013-09-11 16:40:07 +08:00
*/
if ( ! flr ) {
2015-02-27 17:18:34 +08:00
/* reset VF using VPGEN_VFRTRIG reg */
2013-11-28 14:39:27 +08:00
reg = rd32 ( hw , I40E_VPGEN_VFRTRIG ( vf - > vf_id ) ) ;
reg | = I40E_VPGEN_VFRTRIG_VFSWR_MASK ;
2013-09-11 16:40:07 +08:00
wr32 ( hw , I40E_VPGEN_VFRTRIG ( vf - > vf_id ) , reg ) ;
i40e_flush ( hw ) ;
}
2016-03-19 03:18:09 +08:00
/* clear the VFLR bit in GLGEN_VFLRSTAT */
reg_idx = ( hw - > func_caps . vf_base_id + vf - > vf_id ) / 32 ;
bit_idx = ( hw - > func_caps . vf_base_id + vf - > vf_id ) % 32 ;
wr32 ( hw , I40E_GLGEN_VFLRSTAT ( reg_idx ) , BIT ( bit_idx ) ) ;
2016-04-01 18:56:03 +08:00
i40e_flush ( hw ) ;
2013-09-11 16:40:07 +08:00
2013-11-28 14:39:27 +08:00
if ( i40e_quiesce_vf_pci ( vf ) )
dev_err ( & pf - > pdev - > dev , " VF %d PCI transactions stuck \n " ,
vf - > vf_id ) ;
2017-04-13 16:45:51 +08:00
}
2013-11-28 14:39:27 +08:00
2017-04-13 16:45:51 +08:00
/**
* i40e_cleanup_reset_vf
* @ vf : pointer to the VF structure
*
* Cleanup a VF after the hardware reset is finished . Expects the caller to
* have verified whether the reset is finished properly , and ensure the
* minimum amount of wait time has passed .
* */
static void i40e_cleanup_reset_vf ( struct i40e_vf * vf )
{
struct i40e_pf * pf = vf - > pf ;
struct i40e_hw * hw = & pf - > hw ;
u32 reg ;
2013-11-28 14:39:27 +08:00
2017-02-22 07:55:42 +08:00
/* free VF resources to begin resetting the VSI state */
2013-11-28 14:39:27 +08:00
i40e_free_vf_res ( vf ) ;
2017-02-22 07:55:42 +08:00
/* Enable hardware by clearing the reset bit in the VPGEN_VFRTRIG reg.
* By doing this we allow HW to access VF memory at any point . If we
* did it any sooner , HW could access memory while it was being freed
* in i40e_free_vf_res ( ) , causing an IOMMU fault .
*
* On the other hand , this needs to be done ASAP , because the VF driver
* is waiting for this to happen and may report a timeout . It ' s
* harmless , but it gets logged into Guest OS kernel log , so best avoid
* it .
*/
reg = rd32 ( hw , I40E_VPGEN_VFRTRIG ( vf - > vf_id ) ) ;
reg & = ~ I40E_VPGEN_VFRTRIG_VFSWR_MASK ;
wr32 ( hw , I40E_VPGEN_VFRTRIG ( vf - > vf_id ) , reg ) ;
/* reallocate VF resources to finish resetting the VSI state */
2015-09-01 07:54:48 +08:00
if ( ! i40e_alloc_vf_res ( vf ) ) {
2016-01-21 03:40:01 +08:00
int abs_vf_id = vf - > vf_id + hw - > func_caps . vf_base_id ;
2015-09-01 07:54:48 +08:00
i40e_enable_vf_mappings ( vf ) ;
2017-04-13 16:45:54 +08:00
set_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) ;
clear_bit ( I40E_VF_STATE_DISABLED , & vf - > vf_states ) ;
2016-07-28 03:02:37 +08:00
/* Do not notify the client during VF init */
2017-04-13 16:45:55 +08:00
if ( test_and_clear_bit ( I40E_VF_STATE_PRE_ENABLE ,
& vf - > vf_states ) )
2016-07-28 03:02:37 +08:00
i40e_notify_client_of_vf_reset ( pf , abs_vf_id ) ;
2016-07-28 03:02:31 +08:00
vf - > num_vlan = 0 ;
2015-09-01 07:54:48 +08:00
}
2017-02-22 07:55:42 +08:00
/* Tell the VF driver the reset is done. This needs to be done only
* after VF has been fully initialized , because the VF driver may
* request resources immediately after setting this flag .
*/
2017-05-12 02:23:11 +08:00
wr32 ( hw , I40E_VFGEN_RSTAT1 ( vf - > vf_id ) , VIRTCHNL_VFR_VFACTIVE ) ;
2017-04-13 16:45:51 +08:00
}
/**
* i40e_reset_vf
* @ vf : pointer to the VF structure
* @ flr : VFLR was issued or not
*
* reset the VF
* */
void i40e_reset_vf ( struct i40e_vf * vf , bool flr )
{
struct i40e_pf * pf = vf - > pf ;
struct i40e_hw * hw = & pf - > hw ;
bool rsd = false ;
u32 reg ;
int i ;
/* If VFs have been disabled, there is no need to reset */
2017-04-19 21:25:55 +08:00
if ( test_and_set_bit ( __I40E_VF_DISABLE , pf - > state ) )
2017-04-13 16:45:51 +08:00
return ;
i40e_trigger_vf_reset ( vf , flr ) ;
/* poll VPGEN_VFRSTAT reg to make sure
* that reset is complete
*/
for ( i = 0 ; i < 10 ; i + + ) {
/* VF reset requires driver to first reset the VF and then
* poll the status register to make sure that the reset
* completed successfully . Due to internal HW FIFO flushes ,
* we must wait 10 ms before the register will be valid .
*/
usleep_range ( 10000 , 20000 ) ;
reg = rd32 ( hw , I40E_VPGEN_VFRSTAT ( vf - > vf_id ) ) ;
if ( reg & I40E_VPGEN_VFRSTAT_VFRD_MASK ) {
rsd = true ;
break ;
}
}
if ( flr )
usleep_range ( 10000 , 20000 ) ;
if ( ! rsd )
dev_err ( & pf - > pdev - > dev , " VF reset check timeout on VF %d \n " ,
vf - > vf_id ) ;
usleep_range ( 10000 , 20000 ) ;
/* On initial reset, we don't have any queues to disable */
if ( vf - > lan_vsi_idx ! = 0 )
i40e_vsi_stop_rings ( pf - > vsi [ vf - > lan_vsi_idx ] ) ;
i40e_cleanup_reset_vf ( vf ) ;
2016-03-11 06:59:47 +08:00
2013-09-11 16:40:07 +08:00
i40e_flush ( hw ) ;
2017-04-19 21:25:55 +08:00
clear_bit ( __I40E_VF_DISABLE , pf - > state ) ;
2013-09-11 16:40:07 +08:00
}
2013-12-13 16:38:38 +08:00
i40e: reset all VFs in parallel when rebuilding PF
When there are a lot of active VFs, it can take multiple seconds to
finish resetting all of them during certain flows., which can cause some
VFs to fail to wait long enough for the reset to occur. The user might
see messages like "Never saw reset" or "Reset never finished" and the VF
driver will stop functioning properly.
The naive solution would be to simply increase the wait timer. We can
get much more clever. Notice that i40e_reset_vf is run in a serialized
fashion, and includes lots of delays.
There are two prominent delays which take most of the time. First, when
we begin resetting VFs, we have multiple 10ms delays which accrue
because we reset each VF in a serial fashion. These delays accumulate to
almost 4 seconds when handling the maximum number of VFs (128).
Secondly, there is a massive 50ms delay for each time we disable queues
on a VSI. This delay is necessary to allow HW to finish disabling queues
before we restore functionality. However, just like with the first case,
we are paying the cost for each VF, rather than disabling all VFs and
waiting once.
Both of these can be fixed, but required some previous refactoring to
handle the special case. First, we will need the
i40e_vsi_wait_queues_disabled function which was previously DCB
specific. Second, we will need to implement our own
i40e_vsi_stop_rings_no_wait function which will handle the stopping of
rings without the delays.
Finally, implement an i40e_reset_all_vfs function, which will first
start the reset of all VFs, and pay the wait cost all at once, rather
than serially waiting for each VF before we start processing then next
one. After the VF has been reset, we'll disable all the VF queues, and
then wait for them to disable. Again, we'll organize the flow such that
we pay the wait cost only once.
Finally, after we've disabled queues we'll go ahead and begin restoring
VF functionality. The result is reducing the wait time by a large factor
and ensuring that VFs do not timeout when waiting in the VF driver.
Change-ID: Ia6e8cf8d98131b78aec89db78afb8d905c9b12be
Signed-off-by: Jacob Keller <jacob.e.keller@intel.com>
Tested-by: Andrew Bowers <andrewx.bowers@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2017-04-13 16:45:52 +08:00
/**
* i40e_reset_all_vfs
* @ pf : pointer to the PF structure
* @ flr : VFLR was issued or not
*
* Reset all allocated VFs in one go . First , tell the hardware to reset each
* VF , then do all the waiting in one chunk , and finally finish restoring each
* VF after the wait . This is useful during PF routines which need to reset
* all VFs , as otherwise it must perform these resets in a serialized fashion .
* */
void i40e_reset_all_vfs ( struct i40e_pf * pf , bool flr )
{
struct i40e_hw * hw = & pf - > hw ;
struct i40e_vf * vf ;
int i , v ;
u32 reg ;
/* If we don't have any VFs, then there is nothing to reset */
if ( ! pf - > num_alloc_vfs )
return ;
/* If VFs have been disabled, there is no need to reset */
2017-04-19 21:25:55 +08:00
if ( test_and_set_bit ( __I40E_VF_DISABLE , pf - > state ) )
i40e: reset all VFs in parallel when rebuilding PF
When there are a lot of active VFs, it can take multiple seconds to
finish resetting all of them during certain flows., which can cause some
VFs to fail to wait long enough for the reset to occur. The user might
see messages like "Never saw reset" or "Reset never finished" and the VF
driver will stop functioning properly.
The naive solution would be to simply increase the wait timer. We can
get much more clever. Notice that i40e_reset_vf is run in a serialized
fashion, and includes lots of delays.
There are two prominent delays which take most of the time. First, when
we begin resetting VFs, we have multiple 10ms delays which accrue
because we reset each VF in a serial fashion. These delays accumulate to
almost 4 seconds when handling the maximum number of VFs (128).
Secondly, there is a massive 50ms delay for each time we disable queues
on a VSI. This delay is necessary to allow HW to finish disabling queues
before we restore functionality. However, just like with the first case,
we are paying the cost for each VF, rather than disabling all VFs and
waiting once.
Both of these can be fixed, but required some previous refactoring to
handle the special case. First, we will need the
i40e_vsi_wait_queues_disabled function which was previously DCB
specific. Second, we will need to implement our own
i40e_vsi_stop_rings_no_wait function which will handle the stopping of
rings without the delays.
Finally, implement an i40e_reset_all_vfs function, which will first
start the reset of all VFs, and pay the wait cost all at once, rather
than serially waiting for each VF before we start processing then next
one. After the VF has been reset, we'll disable all the VF queues, and
then wait for them to disable. Again, we'll organize the flow such that
we pay the wait cost only once.
Finally, after we've disabled queues we'll go ahead and begin restoring
VF functionality. The result is reducing the wait time by a large factor
and ensuring that VFs do not timeout when waiting in the VF driver.
Change-ID: Ia6e8cf8d98131b78aec89db78afb8d905c9b12be
Signed-off-by: Jacob Keller <jacob.e.keller@intel.com>
Tested-by: Andrew Bowers <andrewx.bowers@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2017-04-13 16:45:52 +08:00
return ;
/* Begin reset on all VFs at once */
for ( v = 0 ; v < pf - > num_alloc_vfs ; v + + )
i40e_trigger_vf_reset ( & pf - > vf [ v ] , flr ) ;
/* HW requires some time to make sure it can flush the FIFO for a VF
* when it resets it . Poll the VPGEN_VFRSTAT register for each VF in
* sequence to make sure that it has completed . We ' ll keep track of
* the VFs using a simple iterator that increments once that VF has
* finished resetting .
*/
for ( i = 0 , v = 0 ; i < 10 & & v < pf - > num_alloc_vfs ; i + + ) {
usleep_range ( 10000 , 20000 ) ;
/* Check each VF in sequence, beginning with the VF to fail
* the previous check .
*/
while ( v < pf - > num_alloc_vfs ) {
vf = & pf - > vf [ v ] ;
reg = rd32 ( hw , I40E_VPGEN_VFRSTAT ( vf - > vf_id ) ) ;
if ( ! ( reg & I40E_VPGEN_VFRSTAT_VFRD_MASK ) )
break ;
/* If the current VF has finished resetting, move on
* to the next VF in sequence .
*/
v + + ;
}
}
if ( flr )
usleep_range ( 10000 , 20000 ) ;
/* Display a warning if at least one VF didn't manage to reset in
* time , but continue on with the operation .
*/
if ( v < pf - > num_alloc_vfs )
dev_err ( & pf - > pdev - > dev , " VF reset check timeout on VF %d \n " ,
pf - > vf [ v ] . vf_id ) ;
usleep_range ( 10000 , 20000 ) ;
/* Begin disabling all the rings associated with VFs, but do not wait
* between each VF .
*/
for ( v = 0 ; v < pf - > num_alloc_vfs ; v + + ) {
/* On initial reset, we don't have any queues to disable */
if ( pf - > vf [ v ] . lan_vsi_idx = = 0 )
continue ;
i40e_vsi_stop_rings_no_wait ( pf - > vsi [ pf - > vf [ v ] . lan_vsi_idx ] ) ;
}
/* Now that we've notified HW to disable all of the VF rings, wait
* until they finish .
*/
for ( v = 0 ; v < pf - > num_alloc_vfs ; v + + ) {
/* On initial reset, we don't have any queues to disable */
if ( pf - > vf [ v ] . lan_vsi_idx = = 0 )
continue ;
i40e_vsi_wait_queues_disabled ( pf - > vsi [ pf - > vf [ v ] . lan_vsi_idx ] ) ;
}
/* Hw may need up to 50ms to finish disabling the RX queues. We
* minimize the wait by delaying only once for all VFs .
*/
mdelay ( 50 ) ;
/* Finish the reset on each VF */
for ( v = 0 ; v < pf - > num_alloc_vfs ; v + + )
i40e_cleanup_reset_vf ( & pf - > vf [ v ] ) ;
i40e_flush ( hw ) ;
2017-04-19 21:25:55 +08:00
clear_bit ( __I40E_VF_DISABLE , pf - > state ) ;
i40e: reset all VFs in parallel when rebuilding PF
When there are a lot of active VFs, it can take multiple seconds to
finish resetting all of them during certain flows., which can cause some
VFs to fail to wait long enough for the reset to occur. The user might
see messages like "Never saw reset" or "Reset never finished" and the VF
driver will stop functioning properly.
The naive solution would be to simply increase the wait timer. We can
get much more clever. Notice that i40e_reset_vf is run in a serialized
fashion, and includes lots of delays.
There are two prominent delays which take most of the time. First, when
we begin resetting VFs, we have multiple 10ms delays which accrue
because we reset each VF in a serial fashion. These delays accumulate to
almost 4 seconds when handling the maximum number of VFs (128).
Secondly, there is a massive 50ms delay for each time we disable queues
on a VSI. This delay is necessary to allow HW to finish disabling queues
before we restore functionality. However, just like with the first case,
we are paying the cost for each VF, rather than disabling all VFs and
waiting once.
Both of these can be fixed, but required some previous refactoring to
handle the special case. First, we will need the
i40e_vsi_wait_queues_disabled function which was previously DCB
specific. Second, we will need to implement our own
i40e_vsi_stop_rings_no_wait function which will handle the stopping of
rings without the delays.
Finally, implement an i40e_reset_all_vfs function, which will first
start the reset of all VFs, and pay the wait cost all at once, rather
than serially waiting for each VF before we start processing then next
one. After the VF has been reset, we'll disable all the VF queues, and
then wait for them to disable. Again, we'll organize the flow such that
we pay the wait cost only once.
Finally, after we've disabled queues we'll go ahead and begin restoring
VF functionality. The result is reducing the wait time by a large factor
and ensuring that VFs do not timeout when waiting in the VF driver.
Change-ID: Ia6e8cf8d98131b78aec89db78afb8d905c9b12be
Signed-off-by: Jacob Keller <jacob.e.keller@intel.com>
Tested-by: Andrew Bowers <andrewx.bowers@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2017-04-13 16:45:52 +08:00
}
2013-09-11 16:40:07 +08:00
/**
* i40e_free_vfs
2015-02-27 17:18:34 +08:00
* @ pf : pointer to the PF structure
2013-09-11 16:40:07 +08:00
*
2015-02-27 17:18:34 +08:00
* free VF resources
2013-09-11 16:40:07 +08:00
* */
void i40e_free_vfs ( struct i40e_pf * pf )
{
2013-11-28 14:39:40 +08:00
struct i40e_hw * hw = & pf - > hw ;
u32 reg_idx , bit_idx ;
int i , tmp , vf_id ;
2013-09-11 16:40:07 +08:00
if ( ! pf - > vf )
return ;
2017-04-19 21:25:55 +08:00
while ( test_and_set_bit ( __I40E_VF_DISABLE , pf - > state ) )
i40e: add locking around VF reset
During VF deallocation, we need to lock out the VF reset code. However,
we cannot depend on simply masking the interrupt, as this does not lock
out the service task, which can still call the reset routine. Instead,
leave the interrupt enabled, but add locking around the VF disable and
reset routines.
For the disable code, we wait to get the lock, as the reset code will
take a finite amount of time to run. For the reset code, we just return
if we fail to get the lock. Since we know that the VFs are being
disabled, we don't need to handle the reset.
This fixes a panic when disabling SR-IOV.
Change-ID: Iea0a6cdef35c331f48c6d5b2f8e6f0e86322e7d8
Signed-off-by: Mitch Williams <mitch.a.williams@intel.com>
Tested-by: Jim Young <james.m.young@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2015-01-09 19:18:15 +08:00
usleep_range ( 1000 , 2000 ) ;
2013-09-11 16:40:07 +08:00
2016-01-21 03:40:01 +08:00
i40e_notify_client_of_vf_enable ( pf , 0 ) ;
2017-04-19 21:25:52 +08:00
/* Amortize wait time by stopping all VFs at the same time */
for ( i = 0 ; i < pf - > num_alloc_vfs ; i + + ) {
if ( test_bit ( I40E_VF_STATE_INIT , & pf - > vf [ i ] . vf_states ) )
continue ;
i40e_vsi_stop_rings_no_wait ( pf - > vsi [ pf - > vf [ i ] . lan_vsi_idx ] ) ;
}
for ( i = 0 ; i < pf - > num_alloc_vfs ; i + + ) {
2017-04-13 16:45:54 +08:00
if ( test_bit ( I40E_VF_STATE_INIT , & pf - > vf [ i ] . vf_states ) )
2017-04-19 21:25:52 +08:00
continue ;
i40e_vsi_wait_queues_disabled ( pf - > vsi [ pf - > vf [ i ] . lan_vsi_idx ] ) ;
}
2015-04-08 02:32:55 +08:00
2014-12-09 16:53:01 +08:00
/* Disable IOV before freeing resources. This lets any VF drivers
* running in the host get themselves cleaned up before we yank
* the carpet out from underneath their feet .
*/
if ( ! pci_vfs_assigned ( pf - > pdev ) )
pci_disable_sriov ( pf - > pdev ) ;
2015-03-31 15:45:02 +08:00
else
dev_warn ( & pf - > pdev - > dev , " VFs are assigned - not disabling SR-IOV \n " ) ;
2014-12-09 16:53:01 +08:00
2015-02-27 17:18:34 +08:00
/* free up VF resources */
2013-11-28 14:39:30 +08:00
tmp = pf - > num_alloc_vfs ;
pf - > num_alloc_vfs = 0 ;
for ( i = 0 ; i < tmp ; i + + ) {
2017-04-13 16:45:54 +08:00
if ( test_bit ( I40E_VF_STATE_INIT , & pf - > vf [ i ] . vf_states ) )
2013-09-11 16:40:07 +08:00
i40e_free_vf_res ( & pf - > vf [ i ] ) ;
/* disable qp mappings */
i40e_disable_vf_mappings ( & pf - > vf [ i ] ) ;
}
kfree ( pf - > vf ) ;
pf - > vf = NULL ;
2014-04-04 12:43:14 +08:00
/* This check is for when the driver is unloaded while VFs are
* assigned . Setting the number of VFs to 0 through sysfs is caught
* before this function ever gets called .
*/
2014-07-23 02:36:43 +08:00
if ( ! pci_vfs_assigned ( pf - > pdev ) ) {
2013-11-28 14:39:40 +08:00
/* Acknowledge VFLR for all VFS. Without this, VFs will fail to
* work correctly when SR - IOV gets re - enabled .
*/
for ( vf_id = 0 ; vf_id < tmp ; vf_id + + ) {
reg_idx = ( hw - > func_caps . vf_base_id + vf_id ) / 32 ;
bit_idx = ( hw - > func_caps . vf_base_id + vf_id ) % 32 ;
2015-06-05 04:24:02 +08:00
wr32 ( hw , I40E_GLGEN_VFLRSTAT ( reg_idx ) , BIT ( bit_idx ) ) ;
2013-11-28 14:39:40 +08:00
}
2013-12-13 16:38:38 +08:00
}
2017-04-19 21:25:55 +08:00
clear_bit ( __I40E_VF_DISABLE , pf - > state ) ;
2013-09-11 16:40:07 +08:00
}
# ifdef CONFIG_PCI_IOV
/**
* i40e_alloc_vfs
2015-02-27 17:18:34 +08:00
* @ pf : pointer to the PF structure
* @ num_alloc_vfs : number of VFs to allocate
2013-09-11 16:40:07 +08:00
*
2015-02-27 17:18:34 +08:00
* allocate VF resources
2013-09-11 16:40:07 +08:00
* */
2014-02-13 19:48:47 +08:00
int i40e_alloc_vfs ( struct i40e_pf * pf , u16 num_alloc_vfs )
2013-09-11 16:40:07 +08:00
{
struct i40e_vf * vfs ;
int i , ret = 0 ;
2013-11-28 14:39:30 +08:00
/* Disable interrupt 0 so we don't try to handle the VFLR. */
2013-11-28 14:39:32 +08:00
i40e_irq_dynamic_disable_icr0 ( pf ) ;
2014-02-13 19:48:47 +08:00
/* Check to see if we're just allocating resources for extant VFs */
if ( pci_num_vf ( pf - > pdev ) ! = num_alloc_vfs ) {
ret = pci_enable_sriov ( pf - > pdev , num_alloc_vfs ) ;
if ( ret ) {
2015-10-02 02:37:40 +08:00
pf - > flags & = ~ I40E_FLAG_VEB_MODE_ENABLED ;
2014-02-13 19:48:47 +08:00
pf - > num_alloc_vfs = 0 ;
goto err_iov ;
}
2013-09-11 16:40:07 +08:00
}
/* allocate memory */
2014-02-06 13:51:02 +08:00
vfs = kcalloc ( num_alloc_vfs , sizeof ( struct i40e_vf ) , GFP_KERNEL ) ;
2013-09-11 16:40:07 +08:00
if ( ! vfs ) {
ret = - ENOMEM ;
goto err_alloc ;
}
2014-05-20 16:01:40 +08:00
pf - > vf = vfs ;
2013-09-11 16:40:07 +08:00
/* apply default profile */
for ( i = 0 ; i < num_alloc_vfs ; i + + ) {
vfs [ i ] . pf = pf ;
vfs [ i ] . parent_type = I40E_SWITCH_ELEMENT_TYPE_VEB ;
vfs [ i ] . vf_id = i ;
/* assign default capabilities */
set_bit ( I40E_VIRTCHNL_VF_CAP_L2 , & vfs [ i ] . vf_caps ) ;
2014-05-20 16:01:40 +08:00
vfs [ i ] . spoofchk = true ;
2017-04-13 16:45:55 +08:00
set_bit ( I40E_VF_STATE_PRE_ENABLE , & vfs [ i ] . vf_states ) ;
2013-09-11 16:40:07 +08:00
}
pf - > num_alloc_vfs = num_alloc_vfs ;
2017-04-13 16:45:55 +08:00
/* VF resources get allocated during reset */
i40e_reset_all_vfs ( pf , false ) ;
2016-07-28 03:02:37 +08:00
i40e_notify_client_of_vf_enable ( pf , num_alloc_vfs ) ;
2013-09-11 16:40:07 +08:00
err_alloc :
if ( ret )
i40e_free_vfs ( pf ) ;
err_iov :
2013-11-28 14:39:30 +08:00
/* Re-enable interrupt 0. */
2016-01-14 08:51:45 +08:00
i40e_irq_dynamic_enable_icr0 ( pf , false ) ;
2013-09-11 16:40:07 +08:00
return ret ;
}
# endif
/**
* i40e_pci_sriov_enable
* @ pdev : pointer to a pci_dev structure
2015-02-27 17:18:34 +08:00
* @ num_vfs : number of VFs to allocate
2013-09-11 16:40:07 +08:00
*
* Enable or change the number of VFs
* */
static int i40e_pci_sriov_enable ( struct pci_dev * pdev , int num_vfs )
{
# ifdef CONFIG_PCI_IOV
struct i40e_pf * pf = pci_get_drvdata ( pdev ) ;
int pre_existing_vfs = pci_num_vf ( pdev ) ;
int err = 0 ;
2017-04-19 21:25:55 +08:00
if ( test_bit ( __I40E_TESTING , pf - > state ) ) {
2015-04-17 08:05:59 +08:00
dev_warn ( & pdev - > dev ,
" Cannot enable SR-IOV virtual functions while the device is undergoing diagnostic testing \n " ) ;
err = - EPERM ;
goto err_out ;
}
2013-09-11 16:40:07 +08:00
if ( pre_existing_vfs & & pre_existing_vfs ! = num_vfs )
i40e_free_vfs ( pf ) ;
else if ( pre_existing_vfs & & pre_existing_vfs = = num_vfs )
goto out ;
if ( num_vfs > pf - > num_req_vfs ) {
2015-08-27 23:42:35 +08:00
dev_warn ( & pdev - > dev , " Unable to enable %d VFs. Limited to %d VFs due to device resource constraints. \n " ,
num_vfs , pf - > num_req_vfs ) ;
2013-09-11 16:40:07 +08:00
err = - EPERM ;
goto err_out ;
}
2015-08-27 23:42:35 +08:00
dev_info ( & pdev - > dev , " Allocating %d VFs. \n " , num_vfs ) ;
2013-09-11 16:40:07 +08:00
err = i40e_alloc_vfs ( pf , num_vfs ) ;
if ( err ) {
dev_warn ( & pdev - > dev , " Failed to enable SR-IOV: %d \n " , err ) ;
goto err_out ;
}
out :
return num_vfs ;
err_out :
return err ;
# endif
return 0 ;
}
/**
* i40e_pci_sriov_configure
* @ pdev : pointer to a pci_dev structure
2015-02-27 17:18:34 +08:00
* @ num_vfs : number of VFs to allocate
2013-09-11 16:40:07 +08:00
*
* Enable or change the number of VFs . Called when the user updates the number
* of VFs in sysfs .
* */
int i40e_pci_sriov_configure ( struct pci_dev * pdev , int num_vfs )
{
struct i40e_pf * pf = pci_get_drvdata ( pdev ) ;
2015-05-09 06:35:57 +08:00
if ( num_vfs ) {
if ( ! ( pf - > flags & I40E_FLAG_VEB_MODE_ENABLED ) ) {
pf - > flags | = I40E_FLAG_VEB_MODE_ENABLED ;
i40e_do_reset_safe ( pf ,
BIT_ULL ( __I40E_PF_RESET_REQUESTED ) ) ;
}
2013-09-11 16:40:07 +08:00
return i40e_pci_sriov_enable ( pdev , num_vfs ) ;
2015-05-09 06:35:57 +08:00
}
2013-09-11 16:40:07 +08:00
2014-07-23 02:36:43 +08:00
if ( ! pci_vfs_assigned ( pf - > pdev ) ) {
2014-04-04 12:43:14 +08:00
i40e_free_vfs ( pf ) ;
2015-05-09 06:35:57 +08:00
pf - > flags & = ~ I40E_FLAG_VEB_MODE_ENABLED ;
i40e_do_reset_safe ( pf , BIT_ULL ( __I40E_PF_RESET_REQUESTED ) ) ;
2014-04-04 12:43:14 +08:00
} else {
dev_warn ( & pdev - > dev , " Unable to free VFs because some are assigned to VMs. \n " ) ;
return - EINVAL ;
}
2013-09-11 16:40:07 +08:00
return 0 ;
}
/***********************virtual channel routines******************/
/**
* i40e_vc_send_msg_to_vf
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ v_opcode : virtual channel opcode
* @ v_retval : virtual channel return value
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
2015-02-27 17:18:34 +08:00
* send msg to VF
2013-09-11 16:40:07 +08:00
* */
static int i40e_vc_send_msg_to_vf ( struct i40e_vf * vf , u32 v_opcode ,
u32 v_retval , u8 * msg , u16 msglen )
{
2014-07-10 15:58:21 +08:00
struct i40e_pf * pf ;
struct i40e_hw * hw ;
int abs_vf_id ;
2013-09-11 16:40:07 +08:00
i40e_status aq_ret ;
2014-07-10 15:58:21 +08:00
/* validate the request */
if ( ! vf | | vf - > vf_id > = vf - > pf - > num_alloc_vfs )
return - EINVAL ;
pf = vf - > pf ;
hw = & pf - > hw ;
abs_vf_id = vf - > vf_id + hw - > func_caps . vf_base_id ;
2013-09-11 16:40:07 +08:00
/* single place to detect unsuccessful return values */
if ( v_retval ) {
vf - > num_invalid_msgs + + ;
2016-03-19 03:18:14 +08:00
dev_info ( & pf - > pdev - > dev , " VF %d failed opcode %d, retval: %d \n " ,
vf - > vf_id , v_opcode , v_retval ) ;
2013-09-11 16:40:07 +08:00
if ( vf - > num_invalid_msgs >
I40E_DEFAULT_NUM_INVALID_MSGS_ALLOWED ) {
dev_err ( & pf - > pdev - > dev ,
" Number of invalid messages exceeded for VF %d \n " ,
vf - > vf_id ) ;
dev_err ( & pf - > pdev - > dev , " Use PF Control I/F to enable the VF \n " ) ;
2017-04-13 16:45:54 +08:00
set_bit ( I40E_VF_STATE_DISABLED , & vf - > vf_states ) ;
2013-09-11 16:40:07 +08:00
}
} else {
vf - > num_valid_msgs + + ;
2015-09-29 02:12:39 +08:00
/* reset the invalid counter, if a valid message is received. */
vf - > num_invalid_msgs = 0 ;
2013-09-11 16:40:07 +08:00
}
2014-08-02 04:27:06 +08:00
aq_ret = i40e_aq_send_msg_to_vf ( hw , abs_vf_id , v_opcode , v_retval ,
2013-11-28 14:39:41 +08:00
msg , msglen , NULL ) ;
2013-09-11 16:40:07 +08:00
if ( aq_ret ) {
2016-03-19 03:18:14 +08:00
dev_info ( & pf - > pdev - > dev ,
" Unable to send the message to VF %d aq_err %d \n " ,
vf - > vf_id , pf - > hw . aq . asq_last_status ) ;
2013-09-11 16:40:07 +08:00
return - EIO ;
}
return 0 ;
}
/**
* i40e_vc_send_resp_to_vf
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ opcode : operation code
* @ retval : return value
*
2015-02-27 17:18:34 +08:00
* send resp msg to VF
2013-09-11 16:40:07 +08:00
* */
static int i40e_vc_send_resp_to_vf ( struct i40e_vf * vf ,
2017-05-12 02:23:11 +08:00
enum virtchnl_ops opcode ,
2013-09-11 16:40:07 +08:00
i40e_status retval )
{
return i40e_vc_send_msg_to_vf ( vf , opcode , retval , NULL , 0 ) ;
}
/**
* i40e_vc_get_version_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
*
2015-02-27 17:18:34 +08:00
* called from the VF to request the API version used by the PF
2013-09-11 16:40:07 +08:00
* */
2015-06-05 04:23:57 +08:00
static int i40e_vc_get_version_msg ( struct i40e_vf * vf , u8 * msg )
2013-09-11 16:40:07 +08:00
{
2017-05-12 02:23:11 +08:00
struct virtchnl_version_info info = {
VIRTCHNL_VERSION_MAJOR , VIRTCHNL_VERSION_MINOR
2013-09-11 16:40:07 +08:00
} ;
2017-05-12 02:23:11 +08:00
vf - > vf_ver = * ( struct virtchnl_version_info * ) msg ;
2015-06-05 04:24:00 +08:00
/* VFs running the 1.0 API expect to get 1.0 back or they will cry. */
if ( VF_IS_V10 ( vf ) )
2017-05-12 02:23:11 +08:00
info . minor = VIRTCHNL_VERSION_MINOR_NO_VF_CAPS ;
return i40e_vc_send_msg_to_vf ( vf , VIRTCHNL_OP_VERSION ,
2013-09-11 16:40:07 +08:00
I40E_SUCCESS , ( u8 * ) & info ,
sizeof ( struct
2017-05-12 02:23:11 +08:00
virtchnl_version_info ) ) ;
2013-09-11 16:40:07 +08:00
}
/**
* i40e_vc_get_vf_resources_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
2015-02-27 17:18:34 +08:00
* called from the VF to request its resources
2013-09-11 16:40:07 +08:00
* */
2015-06-05 04:23:57 +08:00
static int i40e_vc_get_vf_resources_msg ( struct i40e_vf * vf , u8 * msg )
2013-09-11 16:40:07 +08:00
{
2017-05-12 02:23:11 +08:00
struct virtchnl_vf_resource * vfres = NULL ;
2013-09-11 16:40:07 +08:00
struct i40e_pf * pf = vf - > pf ;
i40e_status aq_ret = 0 ;
struct i40e_vsi * vsi ;
int num_vsis = 1 ;
2016-03-19 03:18:06 +08:00
int len = 0 ;
2013-09-11 16:40:07 +08:00
int ret ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_INIT , & vf - > vf_states ) ) {
2013-09-11 16:40:07 +08:00
aq_ret = I40E_ERR_PARAM ;
goto err ;
}
2017-05-12 02:23:11 +08:00
len = ( sizeof ( struct virtchnl_vf_resource ) +
sizeof ( struct virtchnl_vsi_resource ) * num_vsis ) ;
2013-09-11 16:40:07 +08:00
vfres = kzalloc ( len , GFP_KERNEL ) ;
if ( ! vfres ) {
aq_ret = I40E_ERR_NO_MEMORY ;
len = 0 ;
goto err ;
}
2015-06-05 04:23:57 +08:00
if ( VF_IS_V11 ( vf ) )
vf - > driver_caps = * ( u32 * ) msg ;
else
2017-05-12 02:23:11 +08:00
vf - > driver_caps = VIRTCHNL_VF_OFFLOAD_L2 |
VIRTCHNL_VF_OFFLOAD_RSS_REG |
VIRTCHNL_VF_OFFLOAD_VLAN ;
2013-09-11 16:40:07 +08:00
2017-05-12 02:23:11 +08:00
vfres - > vf_offload_flags = VIRTCHNL_VF_OFFLOAD_L2 ;
2015-03-31 15:45:05 +08:00
vsi = pf - > vsi [ vf - > lan_vsi_idx ] ;
2013-09-11 16:40:07 +08:00
if ( ! vsi - > info . pvid )
2017-05-12 02:23:11 +08:00
vfres - > vf_offload_flags | = VIRTCHNL_VF_OFFLOAD_VLAN ;
2016-01-21 03:40:01 +08:00
i40e: KISS the client interface
(KISS is Keep It Simple, Stupid. Or is it?)
The client interface vastly overengineered for what it needs to do.
It was originally designed to support multiple clients on multiple
netdevs, possibly even with multiple drivers. None of this happened,
and now we know that there will only ever be one client for i40e
(i40iw) and one for i40evf (i40iwvf). So, time for some KISS. Since
i40e and i40evf are a Dynasty, we'll simplify this one to match the
VF interface.
First, be a Destroyer and remove all of the lists and locks required
to support multiple clients. Keep one static around to keep track of
one client, and track the client instances for each netdev in the
driver's pf (or adapter) struct. Now it's Almost Human.
Since we already know the client type is iWarp, get rid of any checks
for this. Same for VSI type - it's always going to be the same type,
so it's just a Parasite.
While we're at it, fix up some comments. This makes the function
headers actually match the functions.
These changes reduce code complexity, simplify maintenance,
squash some lurking timing bugs, and allow us to Rock and Roll All
Nite.
Change-ID: I1ea79948ad73b8685272451440a34507f9a9012e
Signed-off-by: Mitch Williams <mitch.a.williams@intel.com>
Signed-off-by: Jacob Keller <jacob.e.keller@intel.com>
Tested-by: Andrew Bowers <andrewx.bowers@intel.com>
Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
2017-01-25 02:24:00 +08:00
if ( i40e_vf_client_capable ( pf , vf - > vf_id ) & &
2017-05-12 02:23:11 +08:00
( vf - > driver_caps & VIRTCHNL_VF_OFFLOAD_IWARP ) ) {
vfres - > vf_offload_flags | = VIRTCHNL_VF_OFFLOAD_IWARP ;
2017-04-13 16:45:54 +08:00
set_bit ( I40E_VF_STATE_IWARPENA , & vf - > vf_states ) ;
2016-01-21 03:40:01 +08:00
}
2017-05-12 02:23:11 +08:00
if ( vf - > driver_caps & VIRTCHNL_VF_OFFLOAD_RSS_PF ) {
vfres - > vf_offload_flags | = VIRTCHNL_VF_OFFLOAD_RSS_PF ;
2015-06-24 07:00:04 +08:00
} else {
2016-04-12 23:30:40 +08:00
if ( ( pf - > flags & I40E_FLAG_RSS_AQ_CAPABLE ) & &
2017-05-12 02:23:11 +08:00
( vf - > driver_caps & VIRTCHNL_VF_OFFLOAD_RSS_AQ ) )
2016-04-12 23:30:40 +08:00
vfres - > vf_offload_flags | =
2017-05-12 02:23:11 +08:00
VIRTCHNL_VF_OFFLOAD_RSS_AQ ;
2016-04-12 23:30:40 +08:00
else
vfres - > vf_offload_flags | =
2017-05-12 02:23:11 +08:00
VIRTCHNL_VF_OFFLOAD_RSS_REG ;
2015-06-24 07:00:04 +08:00
}
2015-09-04 05:18:53 +08:00
2015-12-23 06:25:05 +08:00
if ( pf - > flags & I40E_FLAG_MULTIPLE_TCP_UDP_RSS_PCTYPE ) {
2017-05-12 02:23:11 +08:00
if ( vf - > driver_caps & VIRTCHNL_VF_OFFLOAD_RSS_PCTYPE_V2 )
2015-12-23 06:25:05 +08:00
vfres - > vf_offload_flags | =
2017-05-12 02:23:11 +08:00
VIRTCHNL_VF_OFFLOAD_RSS_PCTYPE_V2 ;
2015-12-23 06:25:05 +08:00
}
2017-05-12 02:23:11 +08:00
if ( vf - > driver_caps & VIRTCHNL_VF_OFFLOAD_ENCAP )
vfres - > vf_offload_flags | = VIRTCHNL_VF_OFFLOAD_ENCAP ;
2017-03-28 05:43:18 +08:00
if ( ( pf - > flags & I40E_FLAG_OUTER_UDP_CSUM_CAPABLE ) & &
2017-05-12 02:23:11 +08:00
( vf - > driver_caps & VIRTCHNL_VF_OFFLOAD_ENCAP_CSUM ) )
vfres - > vf_offload_flags | = VIRTCHNL_VF_OFFLOAD_ENCAP_CSUM ;
2017-03-28 05:43:18 +08:00
2017-05-12 02:23:11 +08:00
if ( vf - > driver_caps & VIRTCHNL_VF_OFFLOAD_RX_POLLING ) {
2016-04-01 18:56:08 +08:00
if ( pf - > flags & I40E_FLAG_MFP_ENABLED ) {
dev_err ( & pf - > pdev - > dev ,
" VF %d requested polling mode: this feature is supported only when the device is running in single function per port (SFP) mode \n " ,
vf - > vf_id ) ;
ret = I40E_ERR_PARAM ;
goto err ;
}
2017-05-12 02:23:11 +08:00
vfres - > vf_offload_flags | = VIRTCHNL_VF_OFFLOAD_RX_POLLING ;
2016-04-01 18:56:08 +08:00
}
2015-09-04 05:18:53 +08:00
2015-12-23 06:25:07 +08:00
if ( pf - > flags & I40E_FLAG_WB_ON_ITR_CAPABLE ) {
2017-05-12 02:23:11 +08:00
if ( vf - > driver_caps & VIRTCHNL_VF_OFFLOAD_WB_ON_ITR )
2015-12-23 06:25:07 +08:00
vfres - > vf_offload_flags | =
2017-05-12 02:23:11 +08:00
VIRTCHNL_VF_OFFLOAD_WB_ON_ITR ;
2015-12-23 06:25:07 +08:00
}
2013-09-11 16:40:07 +08:00
vfres - > num_vsis = num_vsis ;
vfres - > num_queue_pairs = vf - > num_queue_pairs ;
vfres - > max_vectors = pf - > hw . func_caps . num_msix_vectors_vf ;
2016-04-12 23:30:40 +08:00
vfres - > rss_key_size = I40E_HKEY_ARRAY_SIZE ;
vfres - > rss_lut_size = I40E_VF_HLUT_ARRAY_SIZE ;
2015-03-31 15:45:05 +08:00
if ( vf - > lan_vsi_idx ) {
2016-03-19 03:18:06 +08:00
vfres - > vsi_res [ 0 ] . vsi_id = vf - > lan_vsi_id ;
vfres - > vsi_res [ 0 ] . vsi_type = I40E_VSI_SRIOV ;
vfres - > vsi_res [ 0 ] . num_queue_pairs = vsi - > alloc_queue_pairs ;
2015-08-29 05:55:58 +08:00
/* VFs only use TC 0 */
2016-03-19 03:18:06 +08:00
vfres - > vsi_res [ 0 ] . qset_handle
2015-08-29 05:55:58 +08:00
= le16_to_cpu ( vsi - > info . qs_handle [ 0 ] ) ;
2016-03-19 03:18:06 +08:00
ether_addr_copy ( vfres - > vsi_res [ 0 ] . default_mac_addr ,
2015-08-29 05:55:54 +08:00
vf - > default_lan_addr . addr ) ;
2013-09-11 16:40:07 +08:00
}
2017-04-13 16:45:54 +08:00
set_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) ;
2013-09-11 16:40:07 +08:00
err :
2015-02-27 17:18:34 +08:00
/* send the response back to the VF */
2017-05-12 02:23:11 +08:00
ret = i40e_vc_send_msg_to_vf ( vf , VIRTCHNL_OP_GET_VF_RESOURCES ,
2013-09-11 16:40:07 +08:00
aq_ret , ( u8 * ) vfres , len ) ;
kfree ( vfres ) ;
return ret ;
}
/**
* i40e_vc_reset_vf_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
2015-02-27 17:18:34 +08:00
* called from the VF to reset itself ,
* unlike other virtchnl messages , PF driver
* doesn ' t send the response back to the VF
2013-09-11 16:40:07 +08:00
* */
2013-11-28 14:39:27 +08:00
static void i40e_vc_reset_vf_msg ( struct i40e_vf * vf )
2013-09-11 16:40:07 +08:00
{
2017-04-13 16:45:54 +08:00
if ( test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) )
2013-11-28 14:39:27 +08:00
i40e_reset_vf ( vf , false ) ;
2013-09-11 16:40:07 +08:00
}
2016-04-12 23:30:51 +08:00
/**
* i40e_getnum_vf_vsi_vlan_filters
* @ vsi : pointer to the vsi
*
* called to get the number of VLANs offloaded on this VF
* */
static inline int i40e_getnum_vf_vsi_vlan_filters ( struct i40e_vsi * vsi )
{
struct i40e_mac_filter * f ;
2016-10-06 00:30:37 +08:00
int num_vlans = 0 , bkt ;
2016-04-12 23:30:51 +08:00
2016-10-06 00:30:37 +08:00
hash_for_each ( vsi - > mac_filter_hash , bkt , f , hlist ) {
2016-04-12 23:30:51 +08:00
if ( f - > vlan > = 0 & & f - > vlan < = I40E_MAX_VLANID )
num_vlans + + ;
}
return num_vlans ;
}
2013-09-11 16:40:07 +08:00
/**
* i40e_vc_config_promiscuous_mode_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
2015-02-27 17:18:34 +08:00
* called from the VF to configure the promiscuous mode of
* VF vsis
2013-09-11 16:40:07 +08:00
* */
static int i40e_vc_config_promiscuous_mode_msg ( struct i40e_vf * vf ,
u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_promisc_info * info =
( struct virtchnl_promisc_info * ) msg ;
2013-09-11 16:40:07 +08:00
struct i40e_pf * pf = vf - > pf ;
struct i40e_hw * hw = & pf - > hw ;
2016-04-12 23:30:51 +08:00
struct i40e_mac_filter * f ;
i40e_status aq_ret = 0 ;
2013-09-11 16:40:07 +08:00
bool allmulti = false ;
2016-04-12 23:30:51 +08:00
struct i40e_vsi * vsi ;
bool alluni = false ;
int aq_err = 0 ;
2016-10-06 00:30:37 +08:00
int bkt ;
2013-09-11 16:40:07 +08:00
2015-03-31 15:45:05 +08:00
vsi = i40e_find_vsi_from_id ( pf , info - > vsi_id ) ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) | |
2016-08-25 02:33:50 +08:00
! i40e_vc_isvalid_vsi_id ( vf , info - > vsi_id ) | |
! vsi ) {
2016-05-04 06:13:13 +08:00
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
if ( ! test_bit ( I40E_VIRTCHNL_VF_CAP_PRIVILEGE , & vf - > vf_caps ) ) {
2016-04-12 23:30:51 +08:00
dev_err ( & pf - > pdev - > dev ,
2016-05-04 06:13:13 +08:00
" Unprivileged VF %d is attempting to configure promiscuous mode \n " ,
2016-04-12 23:30:51 +08:00
vf - > vf_id ) ;
2016-05-04 06:13:13 +08:00
/* Lie to the VF on purpose. */
aq_ret = 0 ;
2013-09-11 16:40:07 +08:00
goto error_param ;
}
2016-04-12 23:30:51 +08:00
/* Multicast promiscuous handling*/
2013-09-11 16:40:07 +08:00
if ( info - > flags & I40E_FLAG_VF_MULTICAST_PROMISC )
allmulti = true ;
2016-04-12 23:30:51 +08:00
if ( vf - > port_vlan_id ) {
aq_ret = i40e_aq_set_vsi_mc_promisc_on_vlan ( hw , vsi - > seid ,
allmulti ,
vf - > port_vlan_id ,
NULL ) ;
} else if ( i40e_getnum_vf_vsi_vlan_filters ( vsi ) ) {
2016-10-06 00:30:37 +08:00
hash_for_each ( vsi - > mac_filter_hash , bkt , f , hlist ) {
2016-04-12 23:30:52 +08:00
if ( f - > vlan < 0 | | f - > vlan > I40E_MAX_VLANID )
continue ;
aq_ret = i40e_aq_set_vsi_mc_promisc_on_vlan ( hw ,
vsi - > seid ,
allmulti ,
f - > vlan ,
NULL ) ;
2016-04-12 23:30:51 +08:00
aq_err = pf - > hw . aq . asq_last_status ;
if ( aq_ret ) {
dev_err ( & pf - > pdev - > dev ,
" Could not add VLAN %d to multicast promiscuous domain err %s aq_err %s \n " ,
f - > vlan ,
i40e_stat_str ( & pf - > hw , aq_ret ) ,
i40e_aq_str ( & pf - > hw , aq_err ) ) ;
break ;
}
}
} else {
aq_ret = i40e_aq_set_vsi_multicast_promiscuous ( hw , vsi - > seid ,
allmulti , NULL ) ;
aq_err = pf - > hw . aq . asq_last_status ;
if ( aq_ret ) {
dev_err ( & pf - > pdev - > dev ,
" VF %d failed to set multicast promiscuous mode err %s aq_err %s \n " ,
vf - > vf_id ,
i40e_stat_str ( & pf - > hw , aq_ret ) ,
i40e_aq_str ( & pf - > hw , aq_err ) ) ;
2016-10-26 07:08:48 +08:00
goto error_param ;
2016-04-12 23:30:51 +08:00
}
}
if ( ! aq_ret ) {
dev_info ( & pf - > pdev - > dev ,
" VF %d successfully set multicast promiscuous mode \n " ,
vf - > vf_id ) ;
if ( allmulti )
2017-04-13 16:45:54 +08:00
set_bit ( I40E_VF_STATE_MC_PROMISC , & vf - > vf_states ) ;
2016-04-12 23:30:51 +08:00
else
2017-04-13 16:45:54 +08:00
clear_bit ( I40E_VF_STATE_MC_PROMISC , & vf - > vf_states ) ;
2016-04-12 23:30:51 +08:00
}
if ( info - > flags & I40E_FLAG_VF_UNICAST_PROMISC )
alluni = true ;
if ( vf - > port_vlan_id ) {
aq_ret = i40e_aq_set_vsi_uc_promisc_on_vlan ( hw , vsi - > seid ,
alluni ,
vf - > port_vlan_id ,
NULL ) ;
} else if ( i40e_getnum_vf_vsi_vlan_filters ( vsi ) ) {
2016-10-06 00:30:37 +08:00
hash_for_each ( vsi - > mac_filter_hash , bkt , f , hlist ) {
2016-04-12 23:30:51 +08:00
aq_ret = 0 ;
2016-04-30 01:44:05 +08:00
if ( f - > vlan > = 0 & & f - > vlan < = I40E_MAX_VLANID ) {
2016-04-12 23:30:51 +08:00
aq_ret =
i40e_aq_set_vsi_uc_promisc_on_vlan ( hw ,
vsi - > seid ,
alluni ,
f - > vlan ,
NULL ) ;
aq_err = pf - > hw . aq . asq_last_status ;
2016-04-30 01:44:05 +08:00
}
2016-04-12 23:30:51 +08:00
if ( aq_ret )
dev_err ( & pf - > pdev - > dev ,
" Could not add VLAN %d to Unicast promiscuous domain err %s aq_err %s \n " ,
f - > vlan ,
i40e_stat_str ( & pf - > hw , aq_ret ) ,
i40e_aq_str ( & pf - > hw , aq_err ) ) ;
}
} else {
aq_ret = i40e_aq_set_vsi_unicast_promiscuous ( hw , vsi - > seid ,
2016-05-04 06:13:12 +08:00
allmulti , NULL ,
true ) ;
2016-04-12 23:30:51 +08:00
aq_err = pf - > hw . aq . asq_last_status ;
2016-10-26 07:08:48 +08:00
if ( aq_ret ) {
2016-04-12 23:30:51 +08:00
dev_err ( & pf - > pdev - > dev ,
" VF %d failed to set unicast promiscuous mode %8.8x err %s aq_err %s \n " ,
vf - > vf_id , info - > flags ,
i40e_stat_str ( & pf - > hw , aq_ret ) ,
i40e_aq_str ( & pf - > hw , aq_err ) ) ;
2016-10-26 07:08:48 +08:00
goto error_param ;
}
2016-04-12 23:30:51 +08:00
}
if ( ! aq_ret ) {
dev_info ( & pf - > pdev - > dev ,
" VF %d successfully set unicast promiscuous mode \n " ,
vf - > vf_id ) ;
if ( alluni )
2017-04-13 16:45:54 +08:00
set_bit ( I40E_VF_STATE_UC_PROMISC , & vf - > vf_states ) ;
2016-04-12 23:30:51 +08:00
else
2017-04-13 16:45:54 +08:00
clear_bit ( I40E_VF_STATE_UC_PROMISC , & vf - > vf_states ) ;
2016-04-12 23:30:51 +08:00
}
2013-09-11 16:40:07 +08:00
error_param :
2015-02-27 17:18:34 +08:00
/* send the response to the VF */
2013-09-11 16:40:07 +08:00
return i40e_vc_send_resp_to_vf ( vf ,
2017-05-12 02:23:11 +08:00
VIRTCHNL_OP_CONFIG_PROMISCUOUS_MODE ,
2013-09-11 16:40:07 +08:00
aq_ret ) ;
}
/**
* i40e_vc_config_queues_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
2015-02-27 17:18:34 +08:00
* called from the VF to configure the rx / tx
2013-09-11 16:40:07 +08:00
* queues
* */
static int i40e_vc_config_queues_msg ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_vsi_queue_config_info * qci =
( struct virtchnl_vsi_queue_config_info * ) msg ;
struct virtchnl_queue_pair_info * qpi ;
2014-07-10 15:58:15 +08:00
struct i40e_pf * pf = vf - > pf ;
2013-09-11 16:40:07 +08:00
u16 vsi_id , vsi_queue_id ;
i40e_status aq_ret = 0 ;
int i ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) ) {
2013-09-11 16:40:07 +08:00
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
vsi_id = qci - > vsi_id ;
if ( ! i40e_vc_isvalid_vsi_id ( vf , vsi_id ) ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
for ( i = 0 ; i < qci - > num_queue_pairs ; i + + ) {
qpi = & qci - > qpair [ i ] ;
vsi_queue_id = qpi - > txq . queue_id ;
if ( ( qpi - > txq . vsi_id ! = vsi_id ) | |
( qpi - > rxq . vsi_id ! = vsi_id ) | |
( qpi - > rxq . queue_id ! = vsi_queue_id ) | |
! i40e_vc_isvalid_queue_id ( vf , vsi_id , vsi_queue_id ) ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
if ( i40e_config_vsi_rx_queue ( vf , vsi_id , vsi_queue_id ,
& qpi - > rxq ) | |
i40e_config_vsi_tx_queue ( vf , vsi_id , vsi_queue_id ,
& qpi - > txq ) ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
}
2015-02-27 17:18:34 +08:00
/* set vsi num_queue_pairs in use to num configured by VF */
2015-03-31 15:45:05 +08:00
pf - > vsi [ vf - > lan_vsi_idx ] - > num_queue_pairs = qci - > num_queue_pairs ;
2013-09-11 16:40:07 +08:00
error_param :
2015-02-27 17:18:34 +08:00
/* send the response to the VF */
2017-05-12 02:23:11 +08:00
return i40e_vc_send_resp_to_vf ( vf , VIRTCHNL_OP_CONFIG_VSI_QUEUES ,
2013-09-11 16:40:07 +08:00
aq_ret ) ;
}
/**
* i40e_vc_config_irq_map_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
2015-02-27 17:18:34 +08:00
* called from the VF to configure the irq to
2013-09-11 16:40:07 +08:00
* queue map
* */
static int i40e_vc_config_irq_map_msg ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_irq_map_info * irqmap_info =
( struct virtchnl_irq_map_info * ) msg ;
struct virtchnl_vector_map * map ;
2013-09-11 16:40:07 +08:00
u16 vsi_id , vsi_queue_id , vector_id ;
i40e_status aq_ret = 0 ;
unsigned long tempmap ;
int i ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) ) {
2013-09-11 16:40:07 +08:00
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
for ( i = 0 ; i < irqmap_info - > num_vectors ; i + + ) {
map = & irqmap_info - > vecmap [ i ] ;
vector_id = map - > vector_id ;
vsi_id = map - > vsi_id ;
/* validate msg params */
if ( ! i40e_vc_isvalid_vector_id ( vf , vector_id ) | |
! i40e_vc_isvalid_vsi_id ( vf , vsi_id ) ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
/* lookout for the invalid queue index */
tempmap = map - > rxq_map ;
2013-09-24 13:17:36 +08:00
for_each_set_bit ( vsi_queue_id , & tempmap , I40E_MAX_VSI_QP ) {
2013-09-11 16:40:07 +08:00
if ( ! i40e_vc_isvalid_queue_id ( vf , vsi_id ,
vsi_queue_id ) ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
}
tempmap = map - > txq_map ;
2013-09-24 13:17:36 +08:00
for_each_set_bit ( vsi_queue_id , & tempmap , I40E_MAX_VSI_QP ) {
2013-09-11 16:40:07 +08:00
if ( ! i40e_vc_isvalid_queue_id ( vf , vsi_id ,
vsi_queue_id ) ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
}
i40e_config_irq_link_list ( vf , vsi_id , map ) ;
}
error_param :
2015-02-27 17:18:34 +08:00
/* send the response to the VF */
2017-05-12 02:23:11 +08:00
return i40e_vc_send_resp_to_vf ( vf , VIRTCHNL_OP_CONFIG_IRQ_MAP ,
2013-09-11 16:40:07 +08:00
aq_ret ) ;
}
/**
* i40e_vc_enable_queues_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
2015-02-27 17:18:34 +08:00
* called from the VF to enable all or specific queue ( s )
2013-09-11 16:40:07 +08:00
* */
static int i40e_vc_enable_queues_msg ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_queue_select * vqs =
( struct virtchnl_queue_select * ) msg ;
2013-09-11 16:40:07 +08:00
struct i40e_pf * pf = vf - > pf ;
u16 vsi_id = vqs - > vsi_id ;
i40e_status aq_ret = 0 ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) ) {
2013-09-11 16:40:07 +08:00
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
if ( ! i40e_vc_isvalid_vsi_id ( vf , vsi_id ) ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
if ( ( 0 = = vqs - > rx_queues ) & & ( 0 = = vqs - > tx_queues ) ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
2015-03-31 15:45:05 +08:00
2016-10-12 06:26:58 +08:00
if ( i40e_vsi_start_rings ( pf - > vsi [ vf - > lan_vsi_idx ] ) )
2013-11-28 14:39:28 +08:00
aq_ret = I40E_ERR_TIMEOUT ;
2013-09-11 16:40:07 +08:00
error_param :
2015-02-27 17:18:34 +08:00
/* send the response to the VF */
2017-05-12 02:23:11 +08:00
return i40e_vc_send_resp_to_vf ( vf , VIRTCHNL_OP_ENABLE_QUEUES ,
2013-09-11 16:40:07 +08:00
aq_ret ) ;
}
/**
* i40e_vc_disable_queues_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
2015-02-27 17:18:34 +08:00
* called from the VF to disable all or specific
2013-09-11 16:40:07 +08:00
* queue ( s )
* */
static int i40e_vc_disable_queues_msg ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_queue_select * vqs =
( struct virtchnl_queue_select * ) msg ;
2013-09-11 16:40:07 +08:00
struct i40e_pf * pf = vf - > pf ;
i40e_status aq_ret = 0 ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) ) {
2013-09-11 16:40:07 +08:00
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
if ( ! i40e_vc_isvalid_vsi_id ( vf , vqs - > vsi_id ) ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
if ( ( 0 = = vqs - > rx_queues ) & & ( 0 = = vqs - > tx_queues ) ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
2015-03-31 15:45:05 +08:00
2016-10-12 06:26:58 +08:00
i40e_vsi_stop_rings ( pf - > vsi [ vf - > lan_vsi_idx ] ) ;
2013-09-11 16:40:07 +08:00
error_param :
2015-02-27 17:18:34 +08:00
/* send the response to the VF */
2017-05-12 02:23:11 +08:00
return i40e_vc_send_resp_to_vf ( vf , VIRTCHNL_OP_DISABLE_QUEUES ,
2013-09-11 16:40:07 +08:00
aq_ret ) ;
}
/**
* i40e_vc_get_stats_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
2015-02-27 17:18:34 +08:00
* called from the VF to get vsi stats
2013-09-11 16:40:07 +08:00
* */
static int i40e_vc_get_stats_msg ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_queue_select * vqs =
( struct virtchnl_queue_select * ) msg ;
2013-09-11 16:40:07 +08:00
struct i40e_pf * pf = vf - > pf ;
struct i40e_eth_stats stats ;
i40e_status aq_ret = 0 ;
struct i40e_vsi * vsi ;
memset ( & stats , 0 , sizeof ( struct i40e_eth_stats ) ) ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) ) {
2013-09-11 16:40:07 +08:00
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
if ( ! i40e_vc_isvalid_vsi_id ( vf , vqs - > vsi_id ) ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
2015-03-31 15:45:05 +08:00
vsi = pf - > vsi [ vf - > lan_vsi_idx ] ;
2013-09-11 16:40:07 +08:00
if ( ! vsi ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
i40e_update_eth_stats ( vsi ) ;
2013-11-28 14:39:38 +08:00
stats = vsi - > eth_stats ;
2013-09-11 16:40:07 +08:00
error_param :
2015-02-27 17:18:34 +08:00
/* send the response back to the VF */
2017-05-12 02:23:11 +08:00
return i40e_vc_send_msg_to_vf ( vf , VIRTCHNL_OP_GET_STATS , aq_ret ,
2013-09-11 16:40:07 +08:00
( u8 * ) & stats , sizeof ( stats ) ) ;
}
2016-04-13 18:08:22 +08:00
/* If the VF is not trusted restrict the number of MAC/VLAN it can program */
2017-01-31 04:29:36 +08:00
# define I40E_VC_MAX_MAC_ADDR_PER_VF 12
2016-04-13 18:08:22 +08:00
# define I40E_VC_MAX_VLAN_PER_VF 8
2013-11-28 14:39:42 +08:00
/**
* i40e_check_vf_permission
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-11-28 14:39:42 +08:00
* @ macaddr : pointer to the MAC Address being checked
*
* Check if the VF has permission to add or delete unicast MAC address
* filters and return error code - EPERM if not . Then check if the
* address filter requested is broadcast or zero and if so return
* an invalid MAC address error code .
* */
static inline int i40e_check_vf_permission ( struct i40e_vf * vf , u8 * macaddr )
{
struct i40e_pf * pf = vf - > pf ;
int ret = 0 ;
if ( is_broadcast_ether_addr ( macaddr ) | |
is_zero_ether_addr ( macaddr ) ) {
dev_err ( & pf - > pdev - > dev , " invalid VF MAC addr %pM \n " , macaddr ) ;
ret = I40E_ERR_INVALID_MAC_ADDR ;
2013-12-07 18:36:54 +08:00
} else if ( vf - > pf_set_mac & & ! is_multicast_ether_addr ( macaddr ) & &
2016-04-13 18:08:21 +08:00
! test_bit ( I40E_VIRTCHNL_VF_CAP_PRIVILEGE , & vf - > vf_caps ) & &
2013-12-07 18:36:54 +08:00
! ether_addr_equal ( macaddr , vf - > default_lan_addr . addr ) ) {
2013-11-28 14:39:42 +08:00
/* If the host VMM administrator has set the VF MAC address
* administratively via the ndo_set_vf_mac command then deny
* permission to the VF to add or delete unicast MAC addresses .
2016-04-13 18:08:21 +08:00
* Unless the VF is privileged and then it can do whatever .
2013-12-07 18:36:54 +08:00
* The VF may request to set the MAC address filter already
* assigned to it so do not return an error in that case .
2013-11-28 14:39:42 +08:00
*/
dev_err ( & pf - > pdev - > dev ,
2016-04-13 18:08:21 +08:00
" VF attempting to override administratively set MAC address, reload the VF driver to resume normal operation \n " ) ;
2013-11-28 14:39:42 +08:00
ret = - EPERM ;
2016-04-13 18:08:22 +08:00
} else if ( ( vf - > num_mac > = I40E_VC_MAX_MAC_ADDR_PER_VF ) & &
! test_bit ( I40E_VIRTCHNL_VF_CAP_PRIVILEGE , & vf - > vf_caps ) ) {
dev_err ( & pf - > pdev - > dev ,
" VF is not trusted, switch the VF to trusted to add more functionality \n " ) ;
ret = - EPERM ;
2013-11-28 14:39:42 +08:00
}
return ret ;
}
2013-09-11 16:40:07 +08:00
/**
* i40e_vc_add_mac_addr_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
* add guest mac address filter
* */
static int i40e_vc_add_mac_addr_msg ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_ether_addr_list * al =
( struct virtchnl_ether_addr_list * ) msg ;
2013-09-11 16:40:07 +08:00
struct i40e_pf * pf = vf - > pf ;
struct i40e_vsi * vsi = NULL ;
u16 vsi_id = al - > vsi_id ;
2013-11-28 14:39:42 +08:00
i40e_status ret = 0 ;
2013-09-11 16:40:07 +08:00
int i ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) | |
2013-09-11 16:40:07 +08:00
! i40e_vc_isvalid_vsi_id ( vf , vsi_id ) ) {
2013-11-28 14:39:42 +08:00
ret = I40E_ERR_PARAM ;
2013-09-11 16:40:07 +08:00
goto error_param ;
}
for ( i = 0 ; i < al - > num_elements ; i + + ) {
2013-11-28 14:39:42 +08:00
ret = i40e_check_vf_permission ( vf , al - > list [ i ] . addr ) ;
if ( ret )
2013-09-11 16:40:07 +08:00
goto error_param ;
}
2015-03-31 15:45:05 +08:00
vsi = pf - > vsi [ vf - > lan_vsi_idx ] ;
2013-09-11 16:40:07 +08:00
2015-10-01 02:09:03 +08:00
/* Lock once, because all function inside for loop accesses VSI's
* MAC filter list which needs to be protected using same lock .
*/
2016-10-06 00:30:37 +08:00
spin_lock_bh ( & vsi - > mac_filter_hash_lock ) ;
2015-10-01 02:09:03 +08:00
2013-09-11 16:40:07 +08:00
/* add new addresses to the list */
for ( i = 0 ; i < al - > num_elements ; i + + ) {
struct i40e_mac_filter * f ;
2016-10-06 00:30:31 +08:00
f = i40e_find_mac ( vsi , al - > list [ i ] . addr ) ;
2016-11-12 04:39:33 +08:00
if ( ! f )
2016-11-12 04:39:35 +08:00
f = i40e_add_mac_filter ( vsi , al - > list [ i ] . addr ) ;
2013-09-11 16:40:07 +08:00
if ( ! f ) {
dev_err ( & pf - > pdev - > dev ,
2015-10-22 07:47:11 +08:00
" Unable to add MAC filter %pM for VF %d \n " ,
al - > list [ i ] . addr , vf - > vf_id ) ;
2013-11-28 14:39:42 +08:00
ret = I40E_ERR_PARAM ;
2016-10-06 00:30:37 +08:00
spin_unlock_bh ( & vsi - > mac_filter_hash_lock ) ;
2013-09-11 16:40:07 +08:00
goto error_param ;
2016-04-13 18:08:22 +08:00
} else {
vf - > num_mac + + ;
2013-09-11 16:40:07 +08:00
}
}
2016-10-06 00:30:37 +08:00
spin_unlock_bh ( & vsi - > mac_filter_hash_lock ) ;
2013-09-11 16:40:07 +08:00
/* program the updated filter list */
2015-11-10 07:35:50 +08:00
ret = i40e_sync_vsi_filters ( vsi ) ;
if ( ret )
dev_err ( & pf - > pdev - > dev , " Unable to program VF %d MAC filters, error %d \n " ,
vf - > vf_id , ret ) ;
2013-09-11 16:40:07 +08:00
error_param :
2015-02-27 17:18:34 +08:00
/* send the response to the VF */
2017-05-12 02:23:11 +08:00
return i40e_vc_send_resp_to_vf ( vf , VIRTCHNL_OP_ADD_ETH_ADDR ,
2013-11-28 14:39:42 +08:00
ret ) ;
2013-09-11 16:40:07 +08:00
}
/**
* i40e_vc_del_mac_addr_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
* remove guest mac address filter
* */
static int i40e_vc_del_mac_addr_msg ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_ether_addr_list * al =
( struct virtchnl_ether_addr_list * ) msg ;
2013-09-11 16:40:07 +08:00
struct i40e_pf * pf = vf - > pf ;
struct i40e_vsi * vsi = NULL ;
u16 vsi_id = al - > vsi_id ;
2013-11-28 14:39:42 +08:00
i40e_status ret = 0 ;
2013-09-11 16:40:07 +08:00
int i ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) | |
2013-09-11 16:40:07 +08:00
! i40e_vc_isvalid_vsi_id ( vf , vsi_id ) ) {
2013-11-28 14:39:42 +08:00
ret = I40E_ERR_PARAM ;
2013-09-11 16:40:07 +08:00
goto error_param ;
}
2013-11-28 14:39:42 +08:00
for ( i = 0 ; i < al - > num_elements ; i + + ) {
2013-12-21 13:44:45 +08:00
if ( is_broadcast_ether_addr ( al - > list [ i ] . addr ) | |
is_zero_ether_addr ( al - > list [ i ] . addr ) ) {
2015-10-22 07:47:11 +08:00
dev_err ( & pf - > pdev - > dev , " Invalid MAC addr %pM for VF %d \n " ,
al - > list [ i ] . addr , vf - > vf_id ) ;
2013-12-21 13:44:45 +08:00
ret = I40E_ERR_INVALID_MAC_ADDR ;
2013-11-28 14:39:42 +08:00
goto error_param ;
2013-12-21 13:44:45 +08:00
}
2013-11-28 14:39:42 +08:00
}
2015-03-31 15:45:05 +08:00
vsi = pf - > vsi [ vf - > lan_vsi_idx ] ;
2013-09-11 16:40:07 +08:00
2016-10-06 00:30:37 +08:00
spin_lock_bh ( & vsi - > mac_filter_hash_lock ) ;
2013-09-11 16:40:07 +08:00
/* delete addresses from the list */
for ( i = 0 ; i < al - > num_elements ; i + + )
2016-11-12 04:39:35 +08:00
if ( i40e_del_mac_filter ( vsi , al - > list [ i ] . addr ) ) {
2015-11-20 03:34:16 +08:00
ret = I40E_ERR_INVALID_MAC_ADDR ;
2016-10-06 00:30:37 +08:00
spin_unlock_bh ( & vsi - > mac_filter_hash_lock ) ;
2015-11-20 03:34:16 +08:00
goto error_param ;
2016-04-13 18:08:22 +08:00
} else {
vf - > num_mac - - ;
2015-11-20 03:34:16 +08:00
}
2016-10-06 00:30:37 +08:00
spin_unlock_bh ( & vsi - > mac_filter_hash_lock ) ;
2013-09-11 16:40:07 +08:00
/* program the updated filter list */
2015-11-10 07:35:50 +08:00
ret = i40e_sync_vsi_filters ( vsi ) ;
if ( ret )
dev_err ( & pf - > pdev - > dev , " Unable to program VF %d MAC filters, error %d \n " ,
vf - > vf_id , ret ) ;
2013-09-11 16:40:07 +08:00
error_param :
2015-02-27 17:18:34 +08:00
/* send the response to the VF */
2017-05-12 02:23:11 +08:00
return i40e_vc_send_resp_to_vf ( vf , VIRTCHNL_OP_DEL_ETH_ADDR ,
2013-11-28 14:39:42 +08:00
ret ) ;
2013-09-11 16:40:07 +08:00
}
/**
* i40e_vc_add_vlan_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
* program guest vlan id
* */
static int i40e_vc_add_vlan_msg ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_vlan_filter_list * vfl =
( struct virtchnl_vlan_filter_list * ) msg ;
2013-09-11 16:40:07 +08:00
struct i40e_pf * pf = vf - > pf ;
struct i40e_vsi * vsi = NULL ;
u16 vsi_id = vfl - > vsi_id ;
i40e_status aq_ret = 0 ;
int i ;
2016-04-13 18:08:22 +08:00
if ( ( vf - > num_vlan > = I40E_VC_MAX_VLAN_PER_VF ) & &
! test_bit ( I40E_VIRTCHNL_VF_CAP_PRIVILEGE , & vf - > vf_caps ) ) {
dev_err ( & pf - > pdev - > dev ,
" VF is not trusted, switch the VF to trusted to add more VLAN addresses \n " ) ;
goto error_param ;
}
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) | |
2013-09-11 16:40:07 +08:00
! i40e_vc_isvalid_vsi_id ( vf , vsi_id ) ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
for ( i = 0 ; i < vfl - > num_elements ; i + + ) {
if ( vfl - > vlan_id [ i ] > I40E_MAX_VLANID ) {
aq_ret = I40E_ERR_PARAM ;
dev_err ( & pf - > pdev - > dev ,
" invalid VF VLAN id %d \n " , vfl - > vlan_id [ i ] ) ;
goto error_param ;
}
}
2015-03-31 15:45:05 +08:00
vsi = pf - > vsi [ vf - > lan_vsi_idx ] ;
2013-09-11 16:40:07 +08:00
if ( vsi - > info . pvid ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
i40e_vlan_stripping_enable ( vsi ) ;
for ( i = 0 ; i < vfl - > num_elements ; i + + ) {
/* add new VLAN filter */
int ret = i40e_vsi_add_vlan ( vsi , vfl - > vlan_id [ i ] ) ;
2016-04-13 18:08:22 +08:00
if ( ! ret )
vf - > num_vlan + + ;
2015-08-29 05:55:54 +08:00
2017-04-13 16:45:54 +08:00
if ( test_bit ( I40E_VF_STATE_UC_PROMISC , & vf - > vf_states ) )
2016-04-12 23:30:51 +08:00
i40e_aq_set_vsi_uc_promisc_on_vlan ( & pf - > hw , vsi - > seid ,
true ,
vfl - > vlan_id [ i ] ,
NULL ) ;
2017-04-13 16:45:54 +08:00
if ( test_bit ( I40E_VF_STATE_MC_PROMISC , & vf - > vf_states ) )
2016-04-12 23:30:51 +08:00
i40e_aq_set_vsi_mc_promisc_on_vlan ( & pf - > hw , vsi - > seid ,
true ,
vfl - > vlan_id [ i ] ,
NULL ) ;
2013-09-11 16:40:07 +08:00
if ( ret )
dev_err ( & pf - > pdev - > dev ,
2015-10-22 07:47:11 +08:00
" Unable to add VLAN filter %d for VF %d, error %d \n " ,
vfl - > vlan_id [ i ] , vf - > vf_id , ret ) ;
2013-09-11 16:40:07 +08:00
}
error_param :
2015-02-27 17:18:34 +08:00
/* send the response to the VF */
2017-05-12 02:23:11 +08:00
return i40e_vc_send_resp_to_vf ( vf , VIRTCHNL_OP_ADD_VLAN , aq_ret ) ;
2013-09-11 16:40:07 +08:00
}
/**
* i40e_vc_remove_vlan_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
* remove programmed guest vlan id
* */
static int i40e_vc_remove_vlan_msg ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_vlan_filter_list * vfl =
( struct virtchnl_vlan_filter_list * ) msg ;
2013-09-11 16:40:07 +08:00
struct i40e_pf * pf = vf - > pf ;
struct i40e_vsi * vsi = NULL ;
u16 vsi_id = vfl - > vsi_id ;
i40e_status aq_ret = 0 ;
int i ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) | |
2013-09-11 16:40:07 +08:00
! i40e_vc_isvalid_vsi_id ( vf , vsi_id ) ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
for ( i = 0 ; i < vfl - > num_elements ; i + + ) {
if ( vfl - > vlan_id [ i ] > I40E_MAX_VLANID ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
}
2015-03-31 15:45:05 +08:00
vsi = pf - > vsi [ vf - > lan_vsi_idx ] ;
2013-09-11 16:40:07 +08:00
if ( vsi - > info . pvid ) {
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
for ( i = 0 ; i < vfl - > num_elements ; i + + ) {
2016-10-12 06:26:58 +08:00
i40e_vsi_kill_vlan ( vsi , vfl - > vlan_id [ i ] ) ;
vf - > num_vlan - - ;
2015-08-29 05:55:54 +08:00
2017-04-13 16:45:54 +08:00
if ( test_bit ( I40E_VF_STATE_UC_PROMISC , & vf - > vf_states ) )
2016-04-12 23:30:51 +08:00
i40e_aq_set_vsi_uc_promisc_on_vlan ( & pf - > hw , vsi - > seid ,
false ,
vfl - > vlan_id [ i ] ,
NULL ) ;
2017-04-13 16:45:54 +08:00
if ( test_bit ( I40E_VF_STATE_MC_PROMISC , & vf - > vf_states ) )
2016-04-12 23:30:51 +08:00
i40e_aq_set_vsi_mc_promisc_on_vlan ( & pf - > hw , vsi - > seid ,
false ,
vfl - > vlan_id [ i ] ,
NULL ) ;
2013-09-11 16:40:07 +08:00
}
error_param :
2015-02-27 17:18:34 +08:00
/* send the response to the VF */
2017-05-12 02:23:11 +08:00
return i40e_vc_send_resp_to_vf ( vf , VIRTCHNL_OP_DEL_VLAN , aq_ret ) ;
2013-09-11 16:40:07 +08:00
}
2016-01-21 03:40:01 +08:00
/**
* i40e_vc_iwarp_msg
* @ vf : pointer to the VF info
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
* called from the VF for the iwarp msgs
* */
static int i40e_vc_iwarp_msg ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
struct i40e_pf * pf = vf - > pf ;
int abs_vf_id = vf - > vf_id + pf - > hw . func_caps . vf_base_id ;
i40e_status aq_ret = 0 ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) | |
! test_bit ( I40E_VF_STATE_IWARPENA , & vf - > vf_states ) ) {
2016-01-21 03:40:01 +08:00
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
i40e_notify_client_of_vf_msg ( pf - > vsi [ pf - > lan_vsi ] , abs_vf_id ,
msg , msglen ) ;
error_param :
/* send the response to the VF */
2017-05-12 02:23:11 +08:00
return i40e_vc_send_resp_to_vf ( vf , VIRTCHNL_OP_IWARP ,
2016-01-21 03:40:01 +08:00
aq_ret ) ;
}
/**
* i40e_vc_iwarp_qvmap_msg
* @ vf : pointer to the VF info
* @ msg : pointer to the msg buffer
* @ msglen : msg length
* @ config : config qvmap or release it
*
* called from the VF for the iwarp msgs
* */
static int i40e_vc_iwarp_qvmap_msg ( struct i40e_vf * vf , u8 * msg , u16 msglen ,
bool config )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_iwarp_qvlist_info * qvlist_info =
( struct virtchnl_iwarp_qvlist_info * ) msg ;
2016-01-21 03:40:01 +08:00
i40e_status aq_ret = 0 ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) | |
! test_bit ( I40E_VF_STATE_IWARPENA , & vf - > vf_states ) ) {
2016-01-21 03:40:01 +08:00
aq_ret = I40E_ERR_PARAM ;
goto error_param ;
}
if ( config ) {
if ( i40e_config_iwarp_qvlist ( vf , qvlist_info ) )
aq_ret = I40E_ERR_PARAM ;
} else {
i40e_release_iwarp_qvlist ( vf ) ;
}
error_param :
/* send the response to the VF */
return i40e_vc_send_resp_to_vf ( vf ,
2017-05-12 02:23:11 +08:00
config ? VIRTCHNL_OP_CONFIG_IWARP_IRQ_MAP :
VIRTCHNL_OP_RELEASE_IWARP_IRQ_MAP ,
2016-01-21 03:40:01 +08:00
aq_ret ) ;
}
2016-04-12 23:30:40 +08:00
/**
* i40e_vc_config_rss_key
* @ vf : pointer to the VF info
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
* Configure the VF ' s RSS key
* */
static int i40e_vc_config_rss_key ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_rss_key * vrk =
( struct virtchnl_rss_key * ) msg ;
2016-04-12 23:30:40 +08:00
struct i40e_pf * pf = vf - > pf ;
struct i40e_vsi * vsi = NULL ;
u16 vsi_id = vrk - > vsi_id ;
i40e_status aq_ret = 0 ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) | |
2016-04-12 23:30:40 +08:00
! i40e_vc_isvalid_vsi_id ( vf , vsi_id ) | |
( vrk - > key_len ! = I40E_HKEY_ARRAY_SIZE ) ) {
aq_ret = I40E_ERR_PARAM ;
goto err ;
}
vsi = pf - > vsi [ vf - > lan_vsi_idx ] ;
aq_ret = i40e_config_rss ( vsi , vrk - > key , NULL , 0 ) ;
err :
/* send the response to the VF */
2017-05-12 02:23:11 +08:00
return i40e_vc_send_resp_to_vf ( vf , VIRTCHNL_OP_CONFIG_RSS_KEY ,
2016-04-12 23:30:40 +08:00
aq_ret ) ;
}
/**
* i40e_vc_config_rss_lut
* @ vf : pointer to the VF info
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
* Configure the VF ' s RSS LUT
* */
static int i40e_vc_config_rss_lut ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_rss_lut * vrl =
( struct virtchnl_rss_lut * ) msg ;
2016-04-12 23:30:40 +08:00
struct i40e_pf * pf = vf - > pf ;
struct i40e_vsi * vsi = NULL ;
u16 vsi_id = vrl - > vsi_id ;
i40e_status aq_ret = 0 ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) | |
2016-04-12 23:30:40 +08:00
! i40e_vc_isvalid_vsi_id ( vf , vsi_id ) | |
( vrl - > lut_entries ! = I40E_VF_HLUT_ARRAY_SIZE ) ) {
aq_ret = I40E_ERR_PARAM ;
goto err ;
}
vsi = pf - > vsi [ vf - > lan_vsi_idx ] ;
aq_ret = i40e_config_rss ( vsi , NULL , vrl - > lut , I40E_VF_HLUT_ARRAY_SIZE ) ;
/* send the response to the VF */
err :
2017-05-12 02:23:11 +08:00
return i40e_vc_send_resp_to_vf ( vf , VIRTCHNL_OP_CONFIG_RSS_LUT ,
2016-04-12 23:30:40 +08:00
aq_ret ) ;
}
/**
* i40e_vc_get_rss_hena
* @ vf : pointer to the VF info
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
* Return the RSS HENA bits allowed by the hardware
* */
static int i40e_vc_get_rss_hena ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_rss_hena * vrh = NULL ;
2016-04-12 23:30:40 +08:00
struct i40e_pf * pf = vf - > pf ;
i40e_status aq_ret = 0 ;
int len = 0 ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) ) {
2016-04-12 23:30:40 +08:00
aq_ret = I40E_ERR_PARAM ;
goto err ;
}
2017-05-12 02:23:11 +08:00
len = sizeof ( struct virtchnl_rss_hena ) ;
2016-04-12 23:30:40 +08:00
vrh = kzalloc ( len , GFP_KERNEL ) ;
if ( ! vrh ) {
aq_ret = I40E_ERR_NO_MEMORY ;
len = 0 ;
goto err ;
}
vrh - > hena = i40e_pf_get_default_rss_hena ( pf ) ;
err :
/* send the response back to the VF */
2017-05-12 02:23:11 +08:00
aq_ret = i40e_vc_send_msg_to_vf ( vf , VIRTCHNL_OP_GET_RSS_HENA_CAPS ,
2016-04-12 23:30:40 +08:00
aq_ret , ( u8 * ) vrh , len ) ;
2016-07-28 03:02:39 +08:00
kfree ( vrh ) ;
2016-04-12 23:30:40 +08:00
return aq_ret ;
}
/**
* i40e_vc_set_rss_hena
* @ vf : pointer to the VF info
* @ msg : pointer to the msg buffer
* @ msglen : msg length
*
* Set the RSS HENA bits for the VF
* */
static int i40e_vc_set_rss_hena ( struct i40e_vf * vf , u8 * msg , u16 msglen )
{
2017-05-12 02:23:11 +08:00
struct virtchnl_rss_hena * vrh =
( struct virtchnl_rss_hena * ) msg ;
2016-04-12 23:30:40 +08:00
struct i40e_pf * pf = vf - > pf ;
struct i40e_hw * hw = & pf - > hw ;
i40e_status aq_ret = 0 ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_ACTIVE , & vf - > vf_states ) ) {
2016-04-12 23:30:40 +08:00
aq_ret = I40E_ERR_PARAM ;
goto err ;
}
i40e_write_rx_ctl ( hw , I40E_VFQF_HENA1 ( 0 , vf - > vf_id ) , ( u32 ) vrh - > hena ) ;
i40e_write_rx_ctl ( hw , I40E_VFQF_HENA1 ( 1 , vf - > vf_id ) ,
( u32 ) ( vrh - > hena > > 32 ) ) ;
/* send the response to the VF */
err :
2017-05-12 02:23:11 +08:00
return i40e_vc_send_resp_to_vf ( vf , VIRTCHNL_OP_SET_RSS_HENA ,
2016-04-12 23:30:40 +08:00
aq_ret ) ;
}
2013-09-11 16:40:07 +08:00
/**
* i40e_vc_validate_vf_msg
2015-02-27 17:18:34 +08:00
* @ vf : pointer to the VF info
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
* @ msghndl : msg handle
*
* validate msg
* */
static int i40e_vc_validate_vf_msg ( struct i40e_vf * vf , u32 v_opcode ,
u32 v_retval , u8 * msg , u16 msglen )
{
bool err_msg_format = false ;
2016-04-13 18:08:27 +08:00
int valid_len = 0 ;
2013-09-11 16:40:07 +08:00
/* Validate message length. */
switch ( v_opcode ) {
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_VERSION :
valid_len = sizeof ( struct virtchnl_version_info ) ;
2013-09-11 16:40:07 +08:00
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_RESET_VF :
2013-09-11 16:40:07 +08:00
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_GET_VF_RESOURCES :
2015-06-05 04:23:57 +08:00
if ( VF_IS_V11 ( vf ) )
valid_len = sizeof ( u32 ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_TX_QUEUE :
valid_len = sizeof ( struct virtchnl_txq_info ) ;
2013-09-11 16:40:07 +08:00
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_RX_QUEUE :
valid_len = sizeof ( struct virtchnl_rxq_info ) ;
2013-09-11 16:40:07 +08:00
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_VSI_QUEUES :
valid_len = sizeof ( struct virtchnl_vsi_queue_config_info ) ;
2013-09-11 16:40:07 +08:00
if ( msglen > = valid_len ) {
2017-05-12 02:23:11 +08:00
struct virtchnl_vsi_queue_config_info * vqc =
( struct virtchnl_vsi_queue_config_info * ) msg ;
2013-09-11 16:40:07 +08:00
valid_len + = ( vqc - > num_queue_pairs *
sizeof ( struct
2017-05-12 02:23:11 +08:00
virtchnl_queue_pair_info ) ) ;
2013-09-11 16:40:07 +08:00
if ( vqc - > num_queue_pairs = = 0 )
err_msg_format = true ;
}
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_IRQ_MAP :
valid_len = sizeof ( struct virtchnl_irq_map_info ) ;
2013-09-11 16:40:07 +08:00
if ( msglen > = valid_len ) {
2017-05-12 02:23:11 +08:00
struct virtchnl_irq_map_info * vimi =
( struct virtchnl_irq_map_info * ) msg ;
2013-09-11 16:40:07 +08:00
valid_len + = ( vimi - > num_vectors *
2017-05-12 02:23:11 +08:00
sizeof ( struct virtchnl_vector_map ) ) ;
2013-09-11 16:40:07 +08:00
if ( vimi - > num_vectors = = 0 )
err_msg_format = true ;
}
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_ENABLE_QUEUES :
case VIRTCHNL_OP_DISABLE_QUEUES :
valid_len = sizeof ( struct virtchnl_queue_select ) ;
2013-09-11 16:40:07 +08:00
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_ADD_ETH_ADDR :
case VIRTCHNL_OP_DEL_ETH_ADDR :
valid_len = sizeof ( struct virtchnl_ether_addr_list ) ;
2013-09-11 16:40:07 +08:00
if ( msglen > = valid_len ) {
2017-05-12 02:23:11 +08:00
struct virtchnl_ether_addr_list * veal =
( struct virtchnl_ether_addr_list * ) msg ;
2013-09-11 16:40:07 +08:00
valid_len + = veal - > num_elements *
2017-05-12 02:23:11 +08:00
sizeof ( struct virtchnl_ether_addr ) ;
2013-09-11 16:40:07 +08:00
if ( veal - > num_elements = = 0 )
err_msg_format = true ;
}
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_ADD_VLAN :
case VIRTCHNL_OP_DEL_VLAN :
valid_len = sizeof ( struct virtchnl_vlan_filter_list ) ;
2013-09-11 16:40:07 +08:00
if ( msglen > = valid_len ) {
2017-05-12 02:23:11 +08:00
struct virtchnl_vlan_filter_list * vfl =
( struct virtchnl_vlan_filter_list * ) msg ;
2013-09-11 16:40:07 +08:00
valid_len + = vfl - > num_elements * sizeof ( u16 ) ;
if ( vfl - > num_elements = = 0 )
err_msg_format = true ;
}
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_PROMISCUOUS_MODE :
valid_len = sizeof ( struct virtchnl_promisc_info ) ;
2013-09-11 16:40:07 +08:00
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_GET_STATS :
valid_len = sizeof ( struct virtchnl_queue_select ) ;
2013-09-11 16:40:07 +08:00
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_IWARP :
2016-01-21 03:40:01 +08:00
/* These messages are opaque to us and will be validated in
* the RDMA client code . We just need to check for nonzero
* length . The firmware will enforce max length restrictions .
*/
if ( msglen )
valid_len = msglen ;
else
err_msg_format = true ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_RELEASE_IWARP_IRQ_MAP :
2016-01-21 03:40:01 +08:00
valid_len = 0 ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_IWARP_IRQ_MAP :
valid_len = sizeof ( struct virtchnl_iwarp_qvlist_info ) ;
2016-01-21 03:40:01 +08:00
if ( msglen > = valid_len ) {
2017-05-12 02:23:11 +08:00
struct virtchnl_iwarp_qvlist_info * qv =
( struct virtchnl_iwarp_qvlist_info * ) msg ;
2016-01-21 03:40:01 +08:00
if ( qv - > num_vectors = = 0 ) {
err_msg_format = true ;
break ;
}
valid_len + = ( ( qv - > num_vectors - 1 ) *
2017-05-12 02:23:11 +08:00
sizeof ( struct virtchnl_iwarp_qv_info ) ) ;
2016-01-21 03:40:01 +08:00
}
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_RSS_KEY :
valid_len = sizeof ( struct virtchnl_rss_key ) ;
2016-04-12 23:30:40 +08:00
if ( msglen > = valid_len ) {
2017-05-12 02:23:11 +08:00
struct virtchnl_rss_key * vrk =
( struct virtchnl_rss_key * ) msg ;
2016-04-12 23:30:40 +08:00
valid_len + = vrk - > key_len - 1 ;
}
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_RSS_LUT :
valid_len = sizeof ( struct virtchnl_rss_lut ) ;
2016-04-12 23:30:40 +08:00
if ( msglen > = valid_len ) {
2017-05-12 02:23:11 +08:00
struct virtchnl_rss_lut * vrl =
( struct virtchnl_rss_lut * ) msg ;
2016-04-12 23:30:40 +08:00
valid_len + = vrl - > lut_entries - 1 ;
}
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_GET_RSS_HENA_CAPS :
2016-04-12 23:30:40 +08:00
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_SET_RSS_HENA :
valid_len = sizeof ( struct virtchnl_rss_hena ) ;
2016-04-12 23:30:40 +08:00
break ;
2013-09-11 16:40:07 +08:00
/* These are always errors coming from the VF. */
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_EVENT :
case VIRTCHNL_OP_UNKNOWN :
2013-09-11 16:40:07 +08:00
default :
return - EPERM ;
}
/* few more checks */
if ( ( valid_len ! = msglen ) | | ( err_msg_format ) ) {
i40e_vc_send_resp_to_vf ( vf , v_opcode , I40E_ERR_PARAM ) ;
return - EINVAL ;
} else {
return 0 ;
}
}
/**
* i40e_vc_process_vf_msg
2015-02-27 17:18:34 +08:00
* @ pf : pointer to the PF structure
* @ vf_id : source VF id
2013-09-11 16:40:07 +08:00
* @ msg : pointer to the msg buffer
* @ msglen : msg length
* @ msghndl : msg handle
*
* called from the common aeq / arq handler to
2015-02-27 17:18:34 +08:00
* process request from VF
2013-09-11 16:40:07 +08:00
* */
2016-04-13 18:08:29 +08:00
int i40e_vc_process_vf_msg ( struct i40e_pf * pf , s16 vf_id , u32 v_opcode ,
2013-09-11 16:40:07 +08:00
u32 v_retval , u8 * msg , u16 msglen )
{
struct i40e_hw * hw = & pf - > hw ;
2016-04-13 18:08:29 +08:00
int local_vf_id = vf_id - ( s16 ) hw - > func_caps . vf_base_id ;
2013-11-28 14:39:30 +08:00
struct i40e_vf * vf ;
2013-09-11 16:40:07 +08:00
int ret ;
pf - > vf_aq_requests + + ;
2013-11-28 14:39:41 +08:00
if ( local_vf_id > = pf - > num_alloc_vfs )
2013-11-28 14:39:30 +08:00
return - EINVAL ;
2013-11-28 14:39:41 +08:00
vf = & ( pf - > vf [ local_vf_id ] ) ;
2017-05-12 02:23:12 +08:00
/* Check if VF is disabled. */
if ( test_bit ( I40E_VF_STATE_DISABLED , & vf - > vf_states ) )
return I40E_ERR_PARAM ;
2013-09-11 16:40:07 +08:00
/* perform basic checks on the msg */
ret = i40e_vc_validate_vf_msg ( vf , v_opcode , v_retval , msg , msglen ) ;
2017-05-12 02:23:12 +08:00
/* perform additional checks specific to this driver */
if ( v_opcode = = VIRTCHNL_OP_CONFIG_RSS_KEY ) {
struct virtchnl_rss_key * vrk = ( struct virtchnl_rss_key * ) msg ;
if ( vrk - > key_len ! = I40E_HKEY_ARRAY_SIZE )
ret = - EINVAL ;
} else if ( v_opcode = = VIRTCHNL_OP_CONFIG_RSS_LUT ) {
struct virtchnl_rss_lut * vrl = ( struct virtchnl_rss_lut * ) msg ;
if ( vrl - > lut_entries ! = I40E_VF_HLUT_ARRAY_SIZE )
ret = - EINVAL ;
}
2013-09-11 16:40:07 +08:00
if ( ret ) {
2015-02-27 17:18:34 +08:00
dev_err ( & pf - > pdev - > dev , " Invalid message from VF %d, opcode %d, len %d \n " ,
2013-11-28 14:39:41 +08:00
local_vf_id , v_opcode , msglen ) ;
2013-09-11 16:40:07 +08:00
return ret ;
}
2014-01-14 16:49:49 +08:00
2013-09-11 16:40:07 +08:00
switch ( v_opcode ) {
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_VERSION :
2015-06-05 04:23:57 +08:00
ret = i40e_vc_get_version_msg ( vf , msg ) ;
2013-09-11 16:40:07 +08:00
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_GET_VF_RESOURCES :
2015-06-05 04:23:57 +08:00
ret = i40e_vc_get_vf_resources_msg ( vf , msg ) ;
2013-09-11 16:40:07 +08:00
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_RESET_VF :
2013-11-28 14:39:27 +08:00
i40e_vc_reset_vf_msg ( vf ) ;
ret = 0 ;
2013-09-11 16:40:07 +08:00
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_PROMISCUOUS_MODE :
2013-09-11 16:40:07 +08:00
ret = i40e_vc_config_promiscuous_mode_msg ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_VSI_QUEUES :
2013-09-11 16:40:07 +08:00
ret = i40e_vc_config_queues_msg ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_IRQ_MAP :
2013-09-11 16:40:07 +08:00
ret = i40e_vc_config_irq_map_msg ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_ENABLE_QUEUES :
2013-09-11 16:40:07 +08:00
ret = i40e_vc_enable_queues_msg ( vf , msg , msglen ) ;
2015-04-08 07:45:33 +08:00
i40e_vc_notify_vf_link_state ( vf ) ;
2013-09-11 16:40:07 +08:00
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_DISABLE_QUEUES :
2013-09-11 16:40:07 +08:00
ret = i40e_vc_disable_queues_msg ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_ADD_ETH_ADDR :
2013-09-11 16:40:07 +08:00
ret = i40e_vc_add_mac_addr_msg ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_DEL_ETH_ADDR :
2013-09-11 16:40:07 +08:00
ret = i40e_vc_del_mac_addr_msg ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_ADD_VLAN :
2013-09-11 16:40:07 +08:00
ret = i40e_vc_add_vlan_msg ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_DEL_VLAN :
2013-09-11 16:40:07 +08:00
ret = i40e_vc_remove_vlan_msg ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_GET_STATS :
2013-09-11 16:40:07 +08:00
ret = i40e_vc_get_stats_msg ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_IWARP :
2016-01-21 03:40:01 +08:00
ret = i40e_vc_iwarp_msg ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_IWARP_IRQ_MAP :
2016-01-21 03:40:01 +08:00
ret = i40e_vc_iwarp_qvmap_msg ( vf , msg , msglen , true ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_RELEASE_IWARP_IRQ_MAP :
2016-01-21 03:40:01 +08:00
ret = i40e_vc_iwarp_qvmap_msg ( vf , msg , msglen , false ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_RSS_KEY :
2016-04-12 23:30:40 +08:00
ret = i40e_vc_config_rss_key ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_CONFIG_RSS_LUT :
2016-04-12 23:30:40 +08:00
ret = i40e_vc_config_rss_lut ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_GET_RSS_HENA_CAPS :
2016-04-12 23:30:40 +08:00
ret = i40e_vc_get_rss_hena ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_SET_RSS_HENA :
2016-04-12 23:30:40 +08:00
ret = i40e_vc_set_rss_hena ( vf , msg , msglen ) ;
break ;
2017-05-12 02:23:11 +08:00
case VIRTCHNL_OP_UNKNOWN :
2013-09-11 16:40:07 +08:00
default :
2015-02-27 17:18:34 +08:00
dev_err ( & pf - > pdev - > dev , " Unsupported opcode %d from VF %d \n " ,
2013-11-28 14:39:41 +08:00
v_opcode , local_vf_id ) ;
2013-09-11 16:40:07 +08:00
ret = i40e_vc_send_resp_to_vf ( vf , v_opcode ,
I40E_ERR_NOT_IMPLEMENTED ) ;
break ;
}
return ret ;
}
/**
* i40e_vc_process_vflr_event
2015-02-27 17:18:34 +08:00
* @ pf : pointer to the PF structure
2013-09-11 16:40:07 +08:00
*
* called from the vlfr irq handler to
2015-02-27 17:18:34 +08:00
* free up VF resources and state variables
2013-09-11 16:40:07 +08:00
* */
int i40e_vc_process_vflr_event ( struct i40e_pf * pf )
{
struct i40e_hw * hw = & pf - > hw ;
2016-04-13 18:08:29 +08:00
u32 reg , reg_idx , bit_idx ;
2013-09-11 16:40:07 +08:00
struct i40e_vf * vf ;
2016-04-13 18:08:29 +08:00
int vf_id ;
2013-09-11 16:40:07 +08:00
2017-04-19 21:25:55 +08:00
if ( ! test_bit ( __I40E_VFLR_EVENT_PENDING , pf - > state ) )
2013-09-11 16:40:07 +08:00
return 0 ;
2016-01-16 06:33:17 +08:00
/* Re-enable the VFLR interrupt cause here, before looking for which
* VF got reset . Otherwise , if another VF gets a reset while the
* first one is being processed , that interrupt will be lost , and
* that VF will be stuck in reset forever .
*/
2014-11-11 11:15:04 +08:00
reg = rd32 ( hw , I40E_PFINT_ICR0_ENA ) ;
reg | = I40E_PFINT_ICR0_ENA_VFLR_MASK ;
wr32 ( hw , I40E_PFINT_ICR0_ENA , reg ) ;
i40e_flush ( hw ) ;
2017-04-19 21:25:55 +08:00
clear_bit ( __I40E_VFLR_EVENT_PENDING , pf - > state ) ;
2013-09-11 16:40:07 +08:00
for ( vf_id = 0 ; vf_id < pf - > num_alloc_vfs ; vf_id + + ) {
reg_idx = ( hw - > func_caps . vf_base_id + vf_id ) / 32 ;
bit_idx = ( hw - > func_caps . vf_base_id + vf_id ) % 32 ;
2015-02-27 17:18:34 +08:00
/* read GLGEN_VFLRSTAT register to find out the flr VFs */
2013-09-11 16:40:07 +08:00
vf = & pf - > vf [ vf_id ] ;
reg = rd32 ( hw , I40E_GLGEN_VFLRSTAT ( reg_idx ) ) ;
2016-03-19 03:18:09 +08:00
if ( reg & BIT ( bit_idx ) )
2016-03-11 06:59:47 +08:00
/* i40e_reset_vf will clear the bit in GLGEN_VFLRSTAT */
2016-03-19 03:18:09 +08:00
i40e_reset_vf ( vf , true ) ;
2013-09-11 16:40:07 +08:00
}
return 0 ;
}
/**
* i40e_ndo_set_vf_mac
* @ netdev : network interface device structure
2015-02-27 17:18:34 +08:00
* @ vf_id : VF identifier
2013-09-11 16:40:07 +08:00
* @ mac : mac address
*
2015-02-27 17:18:34 +08:00
* program VF mac address
2013-09-11 16:40:07 +08:00
* */
int i40e_ndo_set_vf_mac ( struct net_device * netdev , int vf_id , u8 * mac )
{
struct i40e_netdev_priv * np = netdev_priv ( netdev ) ;
struct i40e_vsi * vsi = np - > vsi ;
struct i40e_pf * pf = vsi - > back ;
struct i40e_mac_filter * f ;
struct i40e_vf * vf ;
int ret = 0 ;
2016-10-06 00:30:37 +08:00
int bkt ;
2013-09-11 16:40:07 +08:00
/* validate the request */
if ( vf_id > = pf - > num_alloc_vfs ) {
dev_err ( & pf - > pdev - > dev ,
" Invalid VF Identifier %d \n " , vf_id ) ;
ret = - EINVAL ;
goto error_param ;
}
vf = & ( pf - > vf [ vf_id ] ) ;
2015-03-31 15:45:05 +08:00
vsi = pf - > vsi [ vf - > lan_vsi_idx ] ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_INIT , & vf - > vf_states ) ) {
2015-12-23 07:34:42 +08:00
dev_err ( & pf - > pdev - > dev , " VF %d still in reset. Try again. \n " ,
vf_id ) ;
ret = - EAGAIN ;
2013-09-11 16:40:07 +08:00
goto error_param ;
}
2015-12-23 07:34:43 +08:00
if ( is_multicast_ether_addr ( mac ) ) {
2013-09-11 16:40:07 +08:00
dev_err ( & pf - > pdev - > dev ,
2015-12-23 07:34:43 +08:00
" Invalid Ethernet address %pM for VF %d \n " , mac , vf_id ) ;
2013-09-11 16:40:07 +08:00
ret = - EINVAL ;
goto error_param ;
}
2015-10-01 02:09:03 +08:00
/* Lock once because below invoked function add/del_filter requires
2016-10-06 00:30:37 +08:00
* mac_filter_hash_lock to be held
2015-10-01 02:09:03 +08:00
*/
2016-10-06 00:30:37 +08:00
spin_lock_bh ( & vsi - > mac_filter_hash_lock ) ;
2015-10-01 02:09:03 +08:00
2013-09-11 16:40:07 +08:00
/* delete the temporary mac address */
2015-12-23 07:34:43 +08:00
if ( ! is_zero_ether_addr ( vf - > default_lan_addr . addr ) )
2016-11-12 04:39:37 +08:00
i40e_del_mac_filter ( vsi , vf - > default_lan_addr . addr ) ;
2013-09-11 16:40:07 +08:00
2014-05-20 16:01:45 +08:00
/* Delete all the filters for this VSI - we're going to kill it
* anyway .
*/
2016-10-06 00:30:37 +08:00
hash_for_each ( vsi - > mac_filter_hash , bkt , f , hlist )
2016-11-12 04:39:36 +08:00
__i40e_del_filter ( vsi , f ) ;
2013-09-11 16:40:07 +08:00
2016-10-06 00:30:37 +08:00
spin_unlock_bh ( & vsi - > mac_filter_hash_lock ) ;
2015-10-01 02:09:03 +08:00
2013-09-11 16:40:07 +08:00
dev_info ( & pf - > pdev - > dev , " Setting MAC %pM on VF %d \n " , mac , vf_id ) ;
/* program mac filter */
2015-11-06 09:01:02 +08:00
if ( i40e_sync_vsi_filters ( vsi ) ) {
2013-09-11 16:40:07 +08:00
dev_err ( & pf - > pdev - > dev , " Unable to program ucast filters \n " ) ;
ret = - EIO ;
goto error_param ;
}
2014-05-22 14:32:02 +08:00
ether_addr_copy ( vf - > default_lan_addr . addr , mac ) ;
2013-11-28 14:39:42 +08:00
vf - > pf_set_mac = true ;
2014-06-04 09:23:13 +08:00
/* Force the VF driver stop so it has to reload with new MAC address */
i40e_vc_disable_vf ( pf , vf ) ;
2013-09-11 16:40:07 +08:00
dev_info ( & pf - > pdev - > dev , " Reload the VF driver to make this change effective. \n " ) ;
error_param :
return ret ;
}
/**
* i40e_ndo_set_vf_port_vlan
* @ netdev : network interface device structure
2015-02-27 17:18:34 +08:00
* @ vf_id : VF identifier
2013-09-11 16:40:07 +08:00
* @ vlan_id : mac address
* @ qos : priority setting
2016-09-22 17:11:15 +08:00
* @ vlan_proto : vlan protocol
2013-09-11 16:40:07 +08:00
*
2015-02-27 17:18:34 +08:00
* program VF vlan id and / or qos
2013-09-11 16:40:07 +08:00
* */
2016-09-22 17:11:15 +08:00
int i40e_ndo_set_vf_port_vlan ( struct net_device * netdev , int vf_id ,
u16 vlan_id , u8 qos , __be16 vlan_proto )
2013-09-11 16:40:07 +08:00
{
2015-07-24 04:54:36 +08:00
u16 vlanprio = vlan_id | ( qos < < I40E_VLAN_PRIORITY_SHIFT ) ;
2013-09-11 16:40:07 +08:00
struct i40e_netdev_priv * np = netdev_priv ( netdev ) ;
struct i40e_pf * pf = np - > vsi - > back ;
struct i40e_vsi * vsi ;
struct i40e_vf * vf ;
int ret = 0 ;
/* validate the request */
if ( vf_id > = pf - > num_alloc_vfs ) {
dev_err ( & pf - > pdev - > dev , " Invalid VF Identifier %d \n " , vf_id ) ;
ret = - EINVAL ;
goto error_pvid ;
}
if ( ( vlan_id > I40E_MAX_VLANID ) | | ( qos > 7 ) ) {
dev_err ( & pf - > pdev - > dev , " Invalid VF Parameters \n " ) ;
ret = - EINVAL ;
goto error_pvid ;
}
2016-09-22 17:11:15 +08:00
if ( vlan_proto ! = htons ( ETH_P_8021Q ) ) {
dev_err ( & pf - > pdev - > dev , " VF VLAN protocol is not supported \n " ) ;
ret = - EPROTONOSUPPORT ;
goto error_pvid ;
}
2013-09-11 16:40:07 +08:00
vf = & ( pf - > vf [ vf_id ] ) ;
2015-03-31 15:45:05 +08:00
vsi = pf - > vsi [ vf - > lan_vsi_idx ] ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_INIT , & vf - > vf_states ) ) {
2015-12-23 07:34:42 +08:00
dev_err ( & pf - > pdev - > dev , " VF %d still in reset. Try again. \n " ,
vf_id ) ;
ret = - EAGAIN ;
2013-09-11 16:40:07 +08:00
goto error_pvid ;
}
2015-07-24 04:54:36 +08:00
if ( le16_to_cpu ( vsi - > info . pvid ) = = vlanprio )
2015-04-28 02:57:10 +08:00
/* duplicate request, so just return success */
goto error_pvid ;
2016-11-12 04:39:30 +08:00
/* Locked once because multiple functions below iterate list */
2016-10-06 00:30:37 +08:00
spin_lock_bh ( & vsi - > mac_filter_hash_lock ) ;
2015-10-01 02:09:03 +08:00
2016-11-12 04:39:30 +08:00
if ( le16_to_cpu ( vsi - > info . pvid ) = = 0 & & i40e_is_vsi_in_vlan ( vsi ) ) {
2014-01-14 08:13:03 +08:00
dev_err ( & pf - > pdev - > dev ,
" VF %d has already configured VLAN filters and the administrator is requesting a port VLAN override. \n Please unload and reload the VF driver for this change to take effect. \n " ,
vf_id ) ;
2014-03-06 17:02:28 +08:00
/* Administrator Error - knock the VF offline until he does
* the right thing by reconfiguring his network correctly
* and then reloading the VF driver .
*/
i40e_vc_disable_vf ( pf , vf ) ;
2016-02-18 08:12:23 +08:00
/* During reset the VF got a new VSI, so refresh the pointer. */
vsi = pf - > vsi [ vf - > lan_vsi_idx ] ;
2014-03-06 17:02:28 +08:00
}
2014-01-14 08:13:03 +08:00
2014-01-14 08:13:04 +08:00
/* Check for condition where there was already a port VLAN ID
* filter set and now it is being deleted by setting it to zero .
2014-03-14 15:32:20 +08:00
* Additionally check for the condition where there was a port
* VLAN but now there is a new and different port VLAN being set .
2014-01-14 08:13:04 +08:00
* Before deleting all the old VLAN filters we must add new ones
* with - 1 ( I40E_VLAN_ANY ) or otherwise we ' re left with all our
* MAC addresses deleted .
*/
2014-03-14 15:32:20 +08:00
if ( ( ! ( vlan_id | | qos ) | |
2015-07-24 04:54:36 +08:00
vlanprio ! = le16_to_cpu ( vsi - > info . pvid ) ) & &
2016-11-12 04:39:30 +08:00
vsi - > info . pvid ) {
ret = i40e_add_vlan_all_mac ( vsi , I40E_VLAN_ANY ) ;
if ( ret ) {
dev_info ( & vsi - > back - > pdev - > dev ,
" add VF VLAN failed, ret=%d aq_err=%d \n " , ret ,
vsi - > back - > hw . aq . asq_last_status ) ;
spin_unlock_bh ( & vsi - > mac_filter_hash_lock ) ;
goto error_pvid ;
}
}
2014-01-14 08:13:04 +08:00
2013-09-11 16:40:07 +08:00
if ( vsi - > info . pvid ) {
2016-11-12 04:39:30 +08:00
/* remove all filters on the old VLAN */
i40e_rm_vlan_all_mac ( vsi , ( le16_to_cpu ( vsi - > info . pvid ) &
VLAN_VID_MASK ) ) ;
2013-09-11 16:40:07 +08:00
}
2016-11-12 04:39:30 +08:00
2013-09-11 16:40:07 +08:00
if ( vlan_id | | qos )
2015-07-24 04:54:36 +08:00
ret = i40e_vsi_add_pvid ( vsi , vlanprio ) ;
2013-09-11 16:40:07 +08:00
else
2013-11-28 14:39:34 +08:00
i40e_vsi_remove_pvid ( vsi ) ;
2013-09-11 16:40:07 +08:00
if ( vlan_id ) {
dev_info ( & pf - > pdev - > dev , " Setting VLAN %d, QOS 0x%x on VF %d \n " ,
vlan_id , qos , vf_id ) ;
2016-11-12 04:39:30 +08:00
/* add new VLAN filter for each MAC */
ret = i40e_add_vlan_all_mac ( vsi , vlan_id ) ;
2013-09-11 16:40:07 +08:00
if ( ret ) {
dev_info ( & vsi - > back - > pdev - > dev ,
" add VF VLAN failed, ret=%d aq_err=%d \n " , ret ,
vsi - > back - > hw . aq . asq_last_status ) ;
2016-11-12 04:39:30 +08:00
spin_unlock_bh ( & vsi - > mac_filter_hash_lock ) ;
2013-09-11 16:40:07 +08:00
goto error_pvid ;
}
2016-11-12 04:39:30 +08:00
/* remove the previously added non-VLAN MAC filters */
i40e_rm_vlan_all_mac ( vsi , I40E_VLAN_ANY ) ;
2013-09-11 16:40:07 +08:00
}
2016-11-12 04:39:30 +08:00
spin_unlock_bh ( & vsi - > mac_filter_hash_lock ) ;
/* Schedule the worker thread to take care of applying changes */
i40e_service_event_schedule ( vsi - > back ) ;
2013-09-11 16:40:07 +08:00
if ( ret ) {
dev_err ( & pf - > pdev - > dev , " Unable to update VF vsi context \n " ) ;
goto error_pvid ;
}
2016-11-12 04:39:30 +08:00
2013-11-28 14:39:34 +08:00
/* The Port VLAN needs to be saved across resets the same as the
* default LAN MAC address .
*/
vf - > port_vlan_id = le16_to_cpu ( vsi - > info . pvid ) ;
2013-09-11 16:40:07 +08:00
ret = 0 ;
error_pvid :
return ret ;
}
2014-04-09 13:58:57 +08:00
# define I40E_BW_CREDIT_DIVISOR 50 /* 50Mbps per BW credit */
# define I40E_MAX_BW_INACTIVE_ACCUM 4 /* device can accumulate 4 credits max */
2013-09-11 16:40:07 +08:00
/**
* i40e_ndo_set_vf_bw
* @ netdev : network interface device structure
2015-02-27 17:18:34 +08:00
* @ vf_id : VF identifier
* @ tx_rate : Tx rate
2013-09-11 16:40:07 +08:00
*
2015-02-27 17:18:34 +08:00
* configure VF Tx rate
2013-09-11 16:40:07 +08:00
* */
net-next:v4: Add support to configure SR-IOV VF minimum and maximum Tx rate through ip tool.
o min_tx_rate puts lower limit on the VF bandwidth. VF is guaranteed
to have a bandwidth of at least this value.
max_tx_rate puts cap on the VF bandwidth. VF can have a bandwidth
of up to this value.
o A new handler set_vf_rate for attr IFLA_VF_RATE has been introduced
which takes 4 arguments:
netdev, VF number, min_tx_rate, max_tx_rate
o ndo_set_vf_rate replaces ndo_set_vf_tx_rate handler.
o Drivers that currently implement ndo_set_vf_tx_rate should now call
ndo_set_vf_rate instead and reject attempt to set a minimum bandwidth
greater than 0 for IFLA_VF_TX_RATE when IFLA_VF_RATE is not yet
implemented by driver.
o If user enters only one of either min_tx_rate or max_tx_rate, then,
userland should read back the other value from driver and set both
for IFLA_VF_RATE.
Drivers that have not yet implemented IFLA_VF_RATE should always
return min_tx_rate as 0 when read from ip tool.
o If both IFLA_VF_TX_RATE and IFLA_VF_RATE options are specified, then
IFLA_VF_RATE should override.
o Idea is to have consistent display of rate values to user.
o Usage example: -
./ip link set p4p1 vf 0 rate 900
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 900 (Mbps), max_tx_rate 900Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 300 min_tx_rate 200
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 300 (Mbps), max_tx_rate 300Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 600 rate 300
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5, tx rate 600 (Mbps), max_tx_rate 600Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
Signed-off-by: Sucheta Chakraborty <sucheta.chakraborty@qlogic.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-05-22 21:59:05 +08:00
int i40e_ndo_set_vf_bw ( struct net_device * netdev , int vf_id , int min_tx_rate ,
int max_tx_rate )
2013-09-11 16:40:07 +08:00
{
2014-03-06 17:02:29 +08:00
struct i40e_netdev_priv * np = netdev_priv ( netdev ) ;
struct i40e_pf * pf = np - > vsi - > back ;
struct i40e_vsi * vsi ;
struct i40e_vf * vf ;
int speed = 0 ;
int ret = 0 ;
/* validate the request */
if ( vf_id > = pf - > num_alloc_vfs ) {
dev_err ( & pf - > pdev - > dev , " Invalid VF Identifier %d. \n " , vf_id ) ;
ret = - EINVAL ;
goto error ;
}
net-next:v4: Add support to configure SR-IOV VF minimum and maximum Tx rate through ip tool.
o min_tx_rate puts lower limit on the VF bandwidth. VF is guaranteed
to have a bandwidth of at least this value.
max_tx_rate puts cap on the VF bandwidth. VF can have a bandwidth
of up to this value.
o A new handler set_vf_rate for attr IFLA_VF_RATE has been introduced
which takes 4 arguments:
netdev, VF number, min_tx_rate, max_tx_rate
o ndo_set_vf_rate replaces ndo_set_vf_tx_rate handler.
o Drivers that currently implement ndo_set_vf_tx_rate should now call
ndo_set_vf_rate instead and reject attempt to set a minimum bandwidth
greater than 0 for IFLA_VF_TX_RATE when IFLA_VF_RATE is not yet
implemented by driver.
o If user enters only one of either min_tx_rate or max_tx_rate, then,
userland should read back the other value from driver and set both
for IFLA_VF_RATE.
Drivers that have not yet implemented IFLA_VF_RATE should always
return min_tx_rate as 0 when read from ip tool.
o If both IFLA_VF_TX_RATE and IFLA_VF_RATE options are specified, then
IFLA_VF_RATE should override.
o Idea is to have consistent display of rate values to user.
o Usage example: -
./ip link set p4p1 vf 0 rate 900
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 900 (Mbps), max_tx_rate 900Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 300 min_tx_rate 200
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 300 (Mbps), max_tx_rate 300Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 600 rate 300
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5, tx rate 600 (Mbps), max_tx_rate 600Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
Signed-off-by: Sucheta Chakraborty <sucheta.chakraborty@qlogic.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-05-22 21:59:05 +08:00
if ( min_tx_rate ) {
2015-02-27 17:18:34 +08:00
dev_err ( & pf - > pdev - > dev , " Invalid min tx rate (%d) (greater than 0) specified for VF %d. \n " ,
net-next:v4: Add support to configure SR-IOV VF minimum and maximum Tx rate through ip tool.
o min_tx_rate puts lower limit on the VF bandwidth. VF is guaranteed
to have a bandwidth of at least this value.
max_tx_rate puts cap on the VF bandwidth. VF can have a bandwidth
of up to this value.
o A new handler set_vf_rate for attr IFLA_VF_RATE has been introduced
which takes 4 arguments:
netdev, VF number, min_tx_rate, max_tx_rate
o ndo_set_vf_rate replaces ndo_set_vf_tx_rate handler.
o Drivers that currently implement ndo_set_vf_tx_rate should now call
ndo_set_vf_rate instead and reject attempt to set a minimum bandwidth
greater than 0 for IFLA_VF_TX_RATE when IFLA_VF_RATE is not yet
implemented by driver.
o If user enters only one of either min_tx_rate or max_tx_rate, then,
userland should read back the other value from driver and set both
for IFLA_VF_RATE.
Drivers that have not yet implemented IFLA_VF_RATE should always
return min_tx_rate as 0 when read from ip tool.
o If both IFLA_VF_TX_RATE and IFLA_VF_RATE options are specified, then
IFLA_VF_RATE should override.
o Idea is to have consistent display of rate values to user.
o Usage example: -
./ip link set p4p1 vf 0 rate 900
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 900 (Mbps), max_tx_rate 900Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 300 min_tx_rate 200
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 300 (Mbps), max_tx_rate 300Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 600 rate 300
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5, tx rate 600 (Mbps), max_tx_rate 600Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
Signed-off-by: Sucheta Chakraborty <sucheta.chakraborty@qlogic.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-05-22 21:59:05 +08:00
min_tx_rate , vf_id ) ;
return - EINVAL ;
}
2014-03-06 17:02:29 +08:00
vf = & ( pf - > vf [ vf_id ] ) ;
2015-03-31 15:45:05 +08:00
vsi = pf - > vsi [ vf - > lan_vsi_idx ] ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_INIT , & vf - > vf_states ) ) {
2015-12-23 07:34:42 +08:00
dev_err ( & pf - > pdev - > dev , " VF %d still in reset. Try again. \n " ,
vf_id ) ;
ret = - EAGAIN ;
2014-03-06 17:02:29 +08:00
goto error ;
}
switch ( pf - > hw . phy . link_info . link_speed ) {
case I40E_LINK_SPEED_40GB :
speed = 40000 ;
break ;
2016-11-22 05:03:48 +08:00
case I40E_LINK_SPEED_25GB :
speed = 25000 ;
break ;
2015-12-24 04:05:49 +08:00
case I40E_LINK_SPEED_20GB :
speed = 20000 ;
break ;
2014-03-06 17:02:29 +08:00
case I40E_LINK_SPEED_10GB :
speed = 10000 ;
break ;
case I40E_LINK_SPEED_1GB :
speed = 1000 ;
break ;
default :
break ;
}
net-next:v4: Add support to configure SR-IOV VF minimum and maximum Tx rate through ip tool.
o min_tx_rate puts lower limit on the VF bandwidth. VF is guaranteed
to have a bandwidth of at least this value.
max_tx_rate puts cap on the VF bandwidth. VF can have a bandwidth
of up to this value.
o A new handler set_vf_rate for attr IFLA_VF_RATE has been introduced
which takes 4 arguments:
netdev, VF number, min_tx_rate, max_tx_rate
o ndo_set_vf_rate replaces ndo_set_vf_tx_rate handler.
o Drivers that currently implement ndo_set_vf_tx_rate should now call
ndo_set_vf_rate instead and reject attempt to set a minimum bandwidth
greater than 0 for IFLA_VF_TX_RATE when IFLA_VF_RATE is not yet
implemented by driver.
o If user enters only one of either min_tx_rate or max_tx_rate, then,
userland should read back the other value from driver and set both
for IFLA_VF_RATE.
Drivers that have not yet implemented IFLA_VF_RATE should always
return min_tx_rate as 0 when read from ip tool.
o If both IFLA_VF_TX_RATE and IFLA_VF_RATE options are specified, then
IFLA_VF_RATE should override.
o Idea is to have consistent display of rate values to user.
o Usage example: -
./ip link set p4p1 vf 0 rate 900
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 900 (Mbps), max_tx_rate 900Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 300 min_tx_rate 200
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 300 (Mbps), max_tx_rate 300Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 600 rate 300
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5, tx rate 600 (Mbps), max_tx_rate 600Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
Signed-off-by: Sucheta Chakraborty <sucheta.chakraborty@qlogic.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-05-22 21:59:05 +08:00
if ( max_tx_rate > speed ) {
2016-10-04 19:20:16 +08:00
dev_err ( & pf - > pdev - > dev , " Invalid max tx rate %d specified for VF %d. \n " ,
net-next:v4: Add support to configure SR-IOV VF minimum and maximum Tx rate through ip tool.
o min_tx_rate puts lower limit on the VF bandwidth. VF is guaranteed
to have a bandwidth of at least this value.
max_tx_rate puts cap on the VF bandwidth. VF can have a bandwidth
of up to this value.
o A new handler set_vf_rate for attr IFLA_VF_RATE has been introduced
which takes 4 arguments:
netdev, VF number, min_tx_rate, max_tx_rate
o ndo_set_vf_rate replaces ndo_set_vf_tx_rate handler.
o Drivers that currently implement ndo_set_vf_tx_rate should now call
ndo_set_vf_rate instead and reject attempt to set a minimum bandwidth
greater than 0 for IFLA_VF_TX_RATE when IFLA_VF_RATE is not yet
implemented by driver.
o If user enters only one of either min_tx_rate or max_tx_rate, then,
userland should read back the other value from driver and set both
for IFLA_VF_RATE.
Drivers that have not yet implemented IFLA_VF_RATE should always
return min_tx_rate as 0 when read from ip tool.
o If both IFLA_VF_TX_RATE and IFLA_VF_RATE options are specified, then
IFLA_VF_RATE should override.
o Idea is to have consistent display of rate values to user.
o Usage example: -
./ip link set p4p1 vf 0 rate 900
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 900 (Mbps), max_tx_rate 900Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 300 min_tx_rate 200
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 300 (Mbps), max_tx_rate 300Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 600 rate 300
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5, tx rate 600 (Mbps), max_tx_rate 600Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
Signed-off-by: Sucheta Chakraborty <sucheta.chakraborty@qlogic.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-05-22 21:59:05 +08:00
max_tx_rate , vf - > vf_id ) ;
2014-03-06 17:02:29 +08:00
ret = - EINVAL ;
goto error ;
}
2014-04-09 13:58:56 +08:00
if ( ( max_tx_rate < 50 ) & & ( max_tx_rate > 0 ) ) {
dev_warn ( & pf - > pdev - > dev , " Setting max Tx rate to minimum usable value of 50Mbps. \n " ) ;
max_tx_rate = 50 ;
}
2014-03-06 17:02:29 +08:00
/* Tx rate credits are in values of 50Mbps, 0 is disabled*/
2014-04-09 13:58:57 +08:00
ret = i40e_aq_config_vsi_bw_limit ( & pf - > hw , vsi - > seid ,
max_tx_rate / I40E_BW_CREDIT_DIVISOR ,
I40E_MAX_BW_INACTIVE_ACCUM , NULL ) ;
2014-03-06 17:02:29 +08:00
if ( ret ) {
net-next:v4: Add support to configure SR-IOV VF minimum and maximum Tx rate through ip tool.
o min_tx_rate puts lower limit on the VF bandwidth. VF is guaranteed
to have a bandwidth of at least this value.
max_tx_rate puts cap on the VF bandwidth. VF can have a bandwidth
of up to this value.
o A new handler set_vf_rate for attr IFLA_VF_RATE has been introduced
which takes 4 arguments:
netdev, VF number, min_tx_rate, max_tx_rate
o ndo_set_vf_rate replaces ndo_set_vf_tx_rate handler.
o Drivers that currently implement ndo_set_vf_tx_rate should now call
ndo_set_vf_rate instead and reject attempt to set a minimum bandwidth
greater than 0 for IFLA_VF_TX_RATE when IFLA_VF_RATE is not yet
implemented by driver.
o If user enters only one of either min_tx_rate or max_tx_rate, then,
userland should read back the other value from driver and set both
for IFLA_VF_RATE.
Drivers that have not yet implemented IFLA_VF_RATE should always
return min_tx_rate as 0 when read from ip tool.
o If both IFLA_VF_TX_RATE and IFLA_VF_RATE options are specified, then
IFLA_VF_RATE should override.
o Idea is to have consistent display of rate values to user.
o Usage example: -
./ip link set p4p1 vf 0 rate 900
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 900 (Mbps), max_tx_rate 900Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 300 min_tx_rate 200
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 300 (Mbps), max_tx_rate 300Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 600 rate 300
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5, tx rate 600 (Mbps), max_tx_rate 600Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
Signed-off-by: Sucheta Chakraborty <sucheta.chakraborty@qlogic.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-05-22 21:59:05 +08:00
dev_err ( & pf - > pdev - > dev , " Unable to set max tx rate, error code %d. \n " ,
2014-03-06 17:02:29 +08:00
ret ) ;
ret = - EIO ;
goto error ;
}
net-next:v4: Add support to configure SR-IOV VF minimum and maximum Tx rate through ip tool.
o min_tx_rate puts lower limit on the VF bandwidth. VF is guaranteed
to have a bandwidth of at least this value.
max_tx_rate puts cap on the VF bandwidth. VF can have a bandwidth
of up to this value.
o A new handler set_vf_rate for attr IFLA_VF_RATE has been introduced
which takes 4 arguments:
netdev, VF number, min_tx_rate, max_tx_rate
o ndo_set_vf_rate replaces ndo_set_vf_tx_rate handler.
o Drivers that currently implement ndo_set_vf_tx_rate should now call
ndo_set_vf_rate instead and reject attempt to set a minimum bandwidth
greater than 0 for IFLA_VF_TX_RATE when IFLA_VF_RATE is not yet
implemented by driver.
o If user enters only one of either min_tx_rate or max_tx_rate, then,
userland should read back the other value from driver and set both
for IFLA_VF_RATE.
Drivers that have not yet implemented IFLA_VF_RATE should always
return min_tx_rate as 0 when read from ip tool.
o If both IFLA_VF_TX_RATE and IFLA_VF_RATE options are specified, then
IFLA_VF_RATE should override.
o Idea is to have consistent display of rate values to user.
o Usage example: -
./ip link set p4p1 vf 0 rate 900
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 900 (Mbps), max_tx_rate 900Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 300 min_tx_rate 200
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 300 (Mbps), max_tx_rate 300Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 600 rate 300
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5, tx rate 600 (Mbps), max_tx_rate 600Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
Signed-off-by: Sucheta Chakraborty <sucheta.chakraborty@qlogic.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-05-22 21:59:05 +08:00
vf - > tx_rate = max_tx_rate ;
2014-03-06 17:02:29 +08:00
error :
return ret ;
2013-09-11 16:40:07 +08:00
}
/**
* i40e_ndo_get_vf_config
* @ netdev : network interface device structure
2015-02-27 17:18:34 +08:00
* @ vf_id : VF identifier
* @ ivi : VF configuration structure
2013-09-11 16:40:07 +08:00
*
2015-02-27 17:18:34 +08:00
* return VF configuration
2013-09-11 16:40:07 +08:00
* */
int i40e_ndo_get_vf_config ( struct net_device * netdev ,
int vf_id , struct ifla_vf_info * ivi )
{
struct i40e_netdev_priv * np = netdev_priv ( netdev ) ;
struct i40e_vsi * vsi = np - > vsi ;
struct i40e_pf * pf = vsi - > back ;
struct i40e_vf * vf ;
int ret = 0 ;
/* validate the request */
if ( vf_id > = pf - > num_alloc_vfs ) {
dev_err ( & pf - > pdev - > dev , " Invalid VF Identifier %d \n " , vf_id ) ;
ret = - EINVAL ;
goto error_param ;
}
vf = & ( pf - > vf [ vf_id ] ) ;
/* first vsi is always the LAN vsi */
2015-03-31 15:45:05 +08:00
vsi = pf - > vsi [ vf - > lan_vsi_idx ] ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_INIT , & vf - > vf_states ) ) {
2015-12-23 07:34:42 +08:00
dev_err ( & pf - > pdev - > dev , " VF %d still in reset. Try again. \n " ,
vf_id ) ;
ret = - EAGAIN ;
2013-09-11 16:40:07 +08:00
goto error_param ;
}
ivi - > vf = vf_id ;
2015-08-29 05:55:54 +08:00
ether_addr_copy ( ivi - > mac , vf - > default_lan_addr . addr ) ;
2013-09-11 16:40:07 +08:00
net-next:v4: Add support to configure SR-IOV VF minimum and maximum Tx rate through ip tool.
o min_tx_rate puts lower limit on the VF bandwidth. VF is guaranteed
to have a bandwidth of at least this value.
max_tx_rate puts cap on the VF bandwidth. VF can have a bandwidth
of up to this value.
o A new handler set_vf_rate for attr IFLA_VF_RATE has been introduced
which takes 4 arguments:
netdev, VF number, min_tx_rate, max_tx_rate
o ndo_set_vf_rate replaces ndo_set_vf_tx_rate handler.
o Drivers that currently implement ndo_set_vf_tx_rate should now call
ndo_set_vf_rate instead and reject attempt to set a minimum bandwidth
greater than 0 for IFLA_VF_TX_RATE when IFLA_VF_RATE is not yet
implemented by driver.
o If user enters only one of either min_tx_rate or max_tx_rate, then,
userland should read back the other value from driver and set both
for IFLA_VF_RATE.
Drivers that have not yet implemented IFLA_VF_RATE should always
return min_tx_rate as 0 when read from ip tool.
o If both IFLA_VF_TX_RATE and IFLA_VF_RATE options are specified, then
IFLA_VF_RATE should override.
o Idea is to have consistent display of rate values to user.
o Usage example: -
./ip link set p4p1 vf 0 rate 900
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 900 (Mbps), max_tx_rate 900Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 300 min_tx_rate 200
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f0 brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5a, tx rate 300 (Mbps), max_tx_rate 300Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
./ip link set p4p1 vf 0 max_tx_rate 600 rate 300
./ip link show p4p1
32: p4p1: <BROADCAST,MULTICAST> mtu 1500 qdisc noop state DOWN mode
DEFAULT qlen 1000
link/ether 00:0e:1e:08:b0:f brd ff:ff:ff:ff:ff:ff
vf 0 MAC 3e:a0:ca:bd:ae:5, tx rate 600 (Mbps), max_tx_rate 600Mbps,
min_tx_rate 200Mbps
vf 1 MAC f6:c6:7c:3f:3d:6c
vf 2 MAC 56:32:43:98:d7:71
vf 3 MAC d6:be:c3:b5:85:ff
vf 4 MAC ee:a9:9a:1e:19:14
vf 5 MAC 4a:d0:4c:07:52:18
vf 6 MAC 3a:76:44:93:62:f9
vf 7 MAC 82:e9:e7:e3:15:1a
Signed-off-by: Sucheta Chakraborty <sucheta.chakraborty@qlogic.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
2014-05-22 21:59:05 +08:00
ivi - > max_tx_rate = vf - > tx_rate ;
ivi - > min_tx_rate = 0 ;
2013-09-11 16:40:07 +08:00
ivi - > vlan = le16_to_cpu ( vsi - > info . pvid ) & I40E_VLAN_MASK ;
ivi - > qos = ( le16_to_cpu ( vsi - > info . pvid ) & I40E_PRIORITY_MASK ) > >
I40E_VLAN_PRIORITY_SHIFT ;
2014-03-14 15:32:24 +08:00
if ( vf - > link_forced = = false )
ivi - > linkstate = IFLA_VF_LINK_STATE_AUTO ;
else if ( vf - > link_up = = true )
ivi - > linkstate = IFLA_VF_LINK_STATE_ENABLE ;
else
ivi - > linkstate = IFLA_VF_LINK_STATE_DISABLE ;
2014-05-20 16:01:40 +08:00
ivi - > spoofchk = vf - > spoofchk ;
2016-08-05 00:45:47 +08:00
ivi - > trusted = vf - > trusted ;
2013-09-11 16:40:07 +08:00
ret = 0 ;
error_param :
return ret ;
}
2014-02-11 16:27:49 +08:00
/**
* i40e_ndo_set_vf_link_state
* @ netdev : network interface device structure
2015-02-27 17:18:34 +08:00
* @ vf_id : VF identifier
2014-02-11 16:27:49 +08:00
* @ link : required link state
*
* Set the link state of a specified VF , regardless of physical link state
* */
int i40e_ndo_set_vf_link_state ( struct net_device * netdev , int vf_id , int link )
{
struct i40e_netdev_priv * np = netdev_priv ( netdev ) ;
struct i40e_pf * pf = np - > vsi - > back ;
2017-05-12 02:23:11 +08:00
struct virtchnl_pf_event pfe ;
2014-02-11 16:27:49 +08:00
struct i40e_hw * hw = & pf - > hw ;
struct i40e_vf * vf ;
2014-08-02 04:27:06 +08:00
int abs_vf_id ;
2014-02-11 16:27:49 +08:00
int ret = 0 ;
/* validate the request */
if ( vf_id > = pf - > num_alloc_vfs ) {
dev_err ( & pf - > pdev - > dev , " Invalid VF Identifier %d \n " , vf_id ) ;
ret = - EINVAL ;
goto error_out ;
}
vf = & pf - > vf [ vf_id ] ;
2014-08-02 04:27:06 +08:00
abs_vf_id = vf - > vf_id + hw - > func_caps . vf_base_id ;
2014-02-11 16:27:49 +08:00
2017-05-12 02:23:11 +08:00
pfe . event = VIRTCHNL_EVENT_LINK_CHANGE ;
2014-02-11 16:27:49 +08:00
pfe . severity = I40E_PF_EVENT_SEVERITY_INFO ;
switch ( link ) {
case IFLA_VF_LINK_STATE_AUTO :
vf - > link_forced = false ;
pfe . event_data . link_event . link_status =
pf - > hw . phy . link_info . link_info & I40E_AQ_LINK_UP ;
pfe . event_data . link_event . link_speed =
pf - > hw . phy . link_info . link_speed ;
break ;
case IFLA_VF_LINK_STATE_ENABLE :
vf - > link_forced = true ;
vf - > link_up = true ;
pfe . event_data . link_event . link_status = true ;
pfe . event_data . link_event . link_speed = I40E_LINK_SPEED_40GB ;
break ;
case IFLA_VF_LINK_STATE_DISABLE :
vf - > link_forced = true ;
vf - > link_up = false ;
pfe . event_data . link_event . link_status = false ;
pfe . event_data . link_event . link_speed = 0 ;
break ;
default :
ret = - EINVAL ;
goto error_out ;
}
/* Notify the VF of its new link state */
2017-05-12 02:23:11 +08:00
i40e_aq_send_msg_to_vf ( hw , abs_vf_id , VIRTCHNL_OP_EVENT ,
2014-02-11 16:27:49 +08:00
0 , ( u8 * ) & pfe , sizeof ( pfe ) , NULL ) ;
error_out :
return ret ;
}
2014-05-20 16:01:40 +08:00
/**
* i40e_ndo_set_vf_spoofchk
* @ netdev : network interface device structure
2015-02-27 17:18:34 +08:00
* @ vf_id : VF identifier
2014-05-20 16:01:40 +08:00
* @ enable : flag to enable or disable feature
*
* Enable or disable VF spoof checking
* */
2014-07-12 15:28:14 +08:00
int i40e_ndo_set_vf_spoofchk ( struct net_device * netdev , int vf_id , bool enable )
2014-05-20 16:01:40 +08:00
{
struct i40e_netdev_priv * np = netdev_priv ( netdev ) ;
struct i40e_vsi * vsi = np - > vsi ;
struct i40e_pf * pf = vsi - > back ;
struct i40e_vsi_context ctxt ;
struct i40e_hw * hw = & pf - > hw ;
struct i40e_vf * vf ;
int ret = 0 ;
/* validate the request */
if ( vf_id > = pf - > num_alloc_vfs ) {
dev_err ( & pf - > pdev - > dev , " Invalid VF Identifier %d \n " , vf_id ) ;
ret = - EINVAL ;
goto out ;
}
vf = & ( pf - > vf [ vf_id ] ) ;
2017-04-13 16:45:54 +08:00
if ( ! test_bit ( I40E_VF_STATE_INIT , & vf - > vf_states ) ) {
2015-12-23 07:34:42 +08:00
dev_err ( & pf - > pdev - > dev , " VF %d still in reset. Try again. \n " ,
vf_id ) ;
ret = - EAGAIN ;
goto out ;
}
2014-05-20 16:01:40 +08:00
if ( enable = = vf - > spoofchk )
goto out ;
vf - > spoofchk = enable ;
memset ( & ctxt , 0 , sizeof ( ctxt ) ) ;
2015-03-31 15:45:05 +08:00
ctxt . seid = pf - > vsi [ vf - > lan_vsi_idx ] - > seid ;
2014-05-20 16:01:40 +08:00
ctxt . pf_num = pf - > hw . pf_id ;
ctxt . info . valid_sections = cpu_to_le16 ( I40E_AQ_VSI_PROP_SECURITY_VALID ) ;
if ( enable )
2015-01-29 15:17:17 +08:00
ctxt . info . sec_flags | = ( I40E_AQ_VSI_SEC_FLAG_ENABLE_VLAN_CHK |
I40E_AQ_VSI_SEC_FLAG_ENABLE_MAC_CHK ) ;
2014-05-20 16:01:40 +08:00
ret = i40e_aq_update_vsi_params ( hw , & ctxt , NULL ) ;
if ( ret ) {
dev_err ( & pf - > pdev - > dev , " Error %d updating VSI parameters \n " ,
ret ) ;
ret = - EIO ;
}
out :
return ret ;
}
2016-04-01 18:56:07 +08:00
/**
* i40e_ndo_set_vf_trust
* @ netdev : network interface device structure of the pf
* @ vf_id : VF identifier
* @ setting : trust setting
*
* Enable or disable VF trust setting
* */
int i40e_ndo_set_vf_trust ( struct net_device * netdev , int vf_id , bool setting )
{
struct i40e_netdev_priv * np = netdev_priv ( netdev ) ;
struct i40e_pf * pf = np - > vsi - > back ;
struct i40e_vf * vf ;
int ret = 0 ;
/* validate the request */
if ( vf_id > = pf - > num_alloc_vfs ) {
dev_err ( & pf - > pdev - > dev , " Invalid VF Identifier %d \n " , vf_id ) ;
return - EINVAL ;
}
if ( pf - > flags & I40E_FLAG_MFP_ENABLED ) {
dev_err ( & pf - > pdev - > dev , " Trusted VF not supported in MFP mode. \n " ) ;
return - EINVAL ;
}
vf = & pf - > vf [ vf_id ] ;
if ( ! vf )
return - EINVAL ;
if ( setting = = vf - > trusted )
goto out ;
vf - > trusted = setting ;
i40e_vc_notify_vf_reset ( vf ) ;
i40e_reset_vf ( vf , false ) ;
dev_info ( & pf - > pdev - > dev , " VF %u is now %strusted \n " ,
vf_id , setting ? " " : " un " ) ;
out :
return ret ;
}