mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	i40evf: init code and hardware support
This patch implements the hardware specific init and management. Signed-off-by: Mitch Williams <mitch.a.williams@intel.com> Signed-off-by: Greg Rose <gregory.v.rose@intel.com> Tested-by: Sibai Li <sibai.li@intel.com> Signed-off-by: Jeff Kirsher <jeffrey.t.kirsher@intel.com>
This commit is contained in:
		
							parent
							
								
									5321a21c1c
								
							
						
					
					
						commit
						d358aa9a7a
					
				
							
								
								
									
										927
									
								
								drivers/net/ethernet/intel/i40evf/i40e_adminq.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										927
									
								
								drivers/net/ethernet/intel/i40evf/i40e_adminq.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,927 @@ | ||||
| /*******************************************************************************
 | ||||
|  * | ||||
|  * Intel Ethernet Controller XL710 Family Linux Virtual Function Driver | ||||
|  * Copyright(c) 2013 Intel Corporation. | ||||
|  * | ||||
|  * 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. | ||||
|  * | ||||
|  * 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_status.h" | ||||
| #include "i40e_type.h" | ||||
| #include "i40e_register.h" | ||||
| #include "i40e_adminq.h" | ||||
| #include "i40e_prototype.h" | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_adminq_init_regs - Initialize AdminQ registers | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  * | ||||
|  *  This assumes the alloc_asq and alloc_arq functions have already been called | ||||
|  **/ | ||||
| static void i40e_adminq_init_regs(struct i40e_hw *hw) | ||||
| { | ||||
| 	/* set head and tail registers in our local struct */ | ||||
| 	if (hw->mac.type == I40E_MAC_VF) { | ||||
| 		hw->aq.asq.tail = I40E_VF_ATQT1; | ||||
| 		hw->aq.asq.head = I40E_VF_ATQH1; | ||||
| 		hw->aq.asq.len  = I40E_VF_ATQLEN1; | ||||
| 		hw->aq.arq.tail = I40E_VF_ARQT1; | ||||
| 		hw->aq.arq.head = I40E_VF_ARQH1; | ||||
| 		hw->aq.arq.len  = I40E_VF_ARQLEN1; | ||||
| 	} else { | ||||
| 		hw->aq.asq.tail = I40E_PF_ATQT; | ||||
| 		hw->aq.asq.head = I40E_PF_ATQH; | ||||
| 		hw->aq.asq.len  = I40E_PF_ATQLEN; | ||||
| 		hw->aq.arq.tail = I40E_PF_ARQT; | ||||
| 		hw->aq.arq.head = I40E_PF_ARQH; | ||||
| 		hw->aq.arq.len  = I40E_PF_ARQLEN; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_alloc_adminq_asq_ring - Allocate Admin Queue send rings | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  **/ | ||||
| static i40e_status i40e_alloc_adminq_asq_ring(struct i40e_hw *hw) | ||||
| { | ||||
| 	i40e_status ret_code; | ||||
| 
 | ||||
| 	ret_code = i40e_allocate_dma_mem(hw, &hw->aq.asq.desc_buf, | ||||
| 					 i40e_mem_atq_ring, | ||||
| 					 (hw->aq.num_asq_entries * | ||||
| 					 sizeof(struct i40e_aq_desc)), | ||||
| 					 I40E_ADMINQ_DESC_ALIGNMENT); | ||||
| 	if (ret_code) | ||||
| 		return ret_code; | ||||
| 
 | ||||
| 	ret_code = i40e_allocate_virt_mem(hw, &hw->aq.asq.cmd_buf, | ||||
| 					  (hw->aq.num_asq_entries * | ||||
| 					  sizeof(struct i40e_asq_cmd_details))); | ||||
| 	if (ret_code) { | ||||
| 		i40e_free_dma_mem(hw, &hw->aq.asq.desc_buf); | ||||
| 		return ret_code; | ||||
| 	} | ||||
| 
 | ||||
| 	return ret_code; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_alloc_adminq_arq_ring - Allocate Admin Queue receive rings | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  **/ | ||||
| static i40e_status i40e_alloc_adminq_arq_ring(struct i40e_hw *hw) | ||||
| { | ||||
| 	i40e_status ret_code; | ||||
| 
 | ||||
| 	ret_code = i40e_allocate_dma_mem(hw, &hw->aq.arq.desc_buf, | ||||
| 					 i40e_mem_arq_ring, | ||||
| 					 (hw->aq.num_arq_entries * | ||||
| 					 sizeof(struct i40e_aq_desc)), | ||||
| 					 I40E_ADMINQ_DESC_ALIGNMENT); | ||||
| 
 | ||||
| 	return ret_code; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_free_adminq_asq - Free Admin Queue send rings | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  * | ||||
|  *  This assumes the posted send buffers have already been cleaned | ||||
|  *  and de-allocated | ||||
|  **/ | ||||
| static void i40e_free_adminq_asq(struct i40e_hw *hw) | ||||
| { | ||||
| 	i40e_free_dma_mem(hw, &hw->aq.asq.desc_buf); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_free_adminq_arq - Free Admin Queue receive rings | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  * | ||||
|  *  This assumes the posted receive buffers have already been cleaned | ||||
|  *  and de-allocated | ||||
|  **/ | ||||
| static void i40e_free_adminq_arq(struct i40e_hw *hw) | ||||
| { | ||||
| 	i40e_free_dma_mem(hw, &hw->aq.arq.desc_buf); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_alloc_arq_bufs - Allocate pre-posted buffers for the receive queue | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  **/ | ||||
| static i40e_status i40e_alloc_arq_bufs(struct i40e_hw *hw) | ||||
| { | ||||
| 	i40e_status ret_code; | ||||
| 	struct i40e_aq_desc *desc; | ||||
| 	struct i40e_dma_mem *bi; | ||||
| 	int i; | ||||
| 
 | ||||
| 	/* We'll be allocating the buffer info memory first, then we can
 | ||||
| 	 * allocate the mapped buffers for the event processing | ||||
| 	 */ | ||||
| 
 | ||||
| 	/* buffer_info structures do not need alignment */ | ||||
| 	ret_code = i40e_allocate_virt_mem(hw, &hw->aq.arq.dma_head, | ||||
| 		(hw->aq.num_arq_entries * sizeof(struct i40e_dma_mem))); | ||||
| 	if (ret_code) | ||||
| 		goto alloc_arq_bufs; | ||||
| 	hw->aq.arq.r.arq_bi = (struct i40e_dma_mem *)hw->aq.arq.dma_head.va; | ||||
| 
 | ||||
| 	/* allocate the mapped buffers */ | ||||
| 	for (i = 0; i < hw->aq.num_arq_entries; i++) { | ||||
| 		bi = &hw->aq.arq.r.arq_bi[i]; | ||||
| 		ret_code = i40e_allocate_dma_mem(hw, bi, | ||||
| 						 i40e_mem_arq_buf, | ||||
| 						 hw->aq.arq_buf_size, | ||||
| 						 I40E_ADMINQ_DESC_ALIGNMENT); | ||||
| 		if (ret_code) | ||||
| 			goto unwind_alloc_arq_bufs; | ||||
| 
 | ||||
| 		/* now configure the descriptors for use */ | ||||
| 		desc = I40E_ADMINQ_DESC(hw->aq.arq, i); | ||||
| 
 | ||||
| 		desc->flags = cpu_to_le16(I40E_AQ_FLAG_BUF); | ||||
| 		if (hw->aq.arq_buf_size > I40E_AQ_LARGE_BUF) | ||||
| 			desc->flags |= cpu_to_le16(I40E_AQ_FLAG_LB); | ||||
| 		desc->opcode = 0; | ||||
| 		/* This is in accordance with Admin queue design, there is no
 | ||||
| 		 * register for buffer size configuration | ||||
| 		 */ | ||||
| 		desc->datalen = cpu_to_le16((u16)bi->size); | ||||
| 		desc->retval = 0; | ||||
| 		desc->cookie_high = 0; | ||||
| 		desc->cookie_low = 0; | ||||
| 		desc->params.external.addr_high = | ||||
| 			cpu_to_le32(upper_32_bits(bi->pa)); | ||||
| 		desc->params.external.addr_low = | ||||
| 			cpu_to_le32(lower_32_bits(bi->pa)); | ||||
| 		desc->params.external.param0 = 0; | ||||
| 		desc->params.external.param1 = 0; | ||||
| 	} | ||||
| 
 | ||||
| alloc_arq_bufs: | ||||
| 	return ret_code; | ||||
| 
 | ||||
| unwind_alloc_arq_bufs: | ||||
| 	/* don't try to free the one that failed... */ | ||||
| 	i--; | ||||
| 	for (; i >= 0; i--) | ||||
| 		i40e_free_dma_mem(hw, &hw->aq.arq.r.arq_bi[i]); | ||||
| 	i40e_free_virt_mem(hw, &hw->aq.arq.dma_head); | ||||
| 
 | ||||
| 	return ret_code; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_alloc_asq_bufs - Allocate empty buffer structs for the send queue | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  **/ | ||||
| static i40e_status i40e_alloc_asq_bufs(struct i40e_hw *hw) | ||||
| { | ||||
| 	i40e_status ret_code; | ||||
| 	struct i40e_dma_mem *bi; | ||||
| 	int i; | ||||
| 
 | ||||
| 	/* No mapped memory needed yet, just the buffer info structures */ | ||||
| 	ret_code = i40e_allocate_virt_mem(hw, &hw->aq.asq.dma_head, | ||||
| 		(hw->aq.num_asq_entries * sizeof(struct i40e_dma_mem))); | ||||
| 	if (ret_code) | ||||
| 		goto alloc_asq_bufs; | ||||
| 	hw->aq.asq.r.asq_bi = (struct i40e_dma_mem *)hw->aq.asq.dma_head.va; | ||||
| 
 | ||||
| 	/* allocate the mapped buffers */ | ||||
| 	for (i = 0; i < hw->aq.num_asq_entries; i++) { | ||||
| 		bi = &hw->aq.asq.r.asq_bi[i]; | ||||
| 		ret_code = i40e_allocate_dma_mem(hw, bi, | ||||
| 						 i40e_mem_asq_buf, | ||||
| 						 hw->aq.asq_buf_size, | ||||
| 						 I40E_ADMINQ_DESC_ALIGNMENT); | ||||
| 		if (ret_code) | ||||
| 			goto unwind_alloc_asq_bufs; | ||||
| 	} | ||||
| alloc_asq_bufs: | ||||
| 	return ret_code; | ||||
| 
 | ||||
| unwind_alloc_asq_bufs: | ||||
| 	/* don't try to free the one that failed... */ | ||||
| 	i--; | ||||
| 	for (; i >= 0; i--) | ||||
| 		i40e_free_dma_mem(hw, &hw->aq.asq.r.asq_bi[i]); | ||||
| 	i40e_free_virt_mem(hw, &hw->aq.asq.dma_head); | ||||
| 
 | ||||
| 	return ret_code; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_free_arq_bufs - Free receive queue buffer info elements | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  **/ | ||||
| static void i40e_free_arq_bufs(struct i40e_hw *hw) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	/* free descriptors */ | ||||
| 	for (i = 0; i < hw->aq.num_arq_entries; i++) | ||||
| 		i40e_free_dma_mem(hw, &hw->aq.arq.r.arq_bi[i]); | ||||
| 
 | ||||
| 	/* free the descriptor memory */ | ||||
| 	i40e_free_dma_mem(hw, &hw->aq.arq.desc_buf); | ||||
| 
 | ||||
| 	/* free the dma header */ | ||||
| 	i40e_free_virt_mem(hw, &hw->aq.arq.dma_head); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_free_asq_bufs - Free send queue buffer info elements | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  **/ | ||||
| static void i40e_free_asq_bufs(struct i40e_hw *hw) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	/* only unmap if the address is non-NULL */ | ||||
| 	for (i = 0; i < hw->aq.num_asq_entries; i++) | ||||
| 		if (hw->aq.asq.r.asq_bi[i].pa) | ||||
| 			i40e_free_dma_mem(hw, &hw->aq.asq.r.asq_bi[i]); | ||||
| 
 | ||||
| 	/* free the buffer info list */ | ||||
| 	i40e_free_virt_mem(hw, &hw->aq.asq.cmd_buf); | ||||
| 
 | ||||
| 	/* free the descriptor memory */ | ||||
| 	i40e_free_dma_mem(hw, &hw->aq.asq.desc_buf); | ||||
| 
 | ||||
| 	/* free the dma header */ | ||||
| 	i40e_free_virt_mem(hw, &hw->aq.asq.dma_head); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_config_asq_regs - configure ASQ registers | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  * | ||||
|  *  Configure base address and length registers for the transmit queue | ||||
|  **/ | ||||
| static void i40e_config_asq_regs(struct i40e_hw *hw) | ||||
| { | ||||
| 	if (hw->mac.type == I40E_MAC_VF) { | ||||
| 		/* configure the transmit queue */ | ||||
| 		wr32(hw, I40E_VF_ATQBAH1, | ||||
| 		    upper_32_bits(hw->aq.asq.desc_buf.pa)); | ||||
| 		wr32(hw, I40E_VF_ATQBAL1, | ||||
| 		    lower_32_bits(hw->aq.asq.desc_buf.pa)); | ||||
| 		wr32(hw, I40E_VF_ATQLEN1, (hw->aq.num_asq_entries | | ||||
| 					  I40E_VF_ATQLEN1_ATQENABLE_MASK)); | ||||
| 	} else { | ||||
| 		/* configure the transmit queue */ | ||||
| 		wr32(hw, I40E_PF_ATQBAH, | ||||
| 		    upper_32_bits(hw->aq.asq.desc_buf.pa)); | ||||
| 		wr32(hw, I40E_PF_ATQBAL, | ||||
| 		    lower_32_bits(hw->aq.asq.desc_buf.pa)); | ||||
| 		wr32(hw, I40E_PF_ATQLEN, (hw->aq.num_asq_entries | | ||||
| 					  I40E_PF_ATQLEN_ATQENABLE_MASK)); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_config_arq_regs - ARQ register configuration | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  * | ||||
|  * Configure base address and length registers for the receive (event queue) | ||||
|  **/ | ||||
| static void i40e_config_arq_regs(struct i40e_hw *hw) | ||||
| { | ||||
| 	if (hw->mac.type == I40E_MAC_VF) { | ||||
| 		/* configure the receive queue */ | ||||
| 		wr32(hw, I40E_VF_ARQBAH1, | ||||
| 		    upper_32_bits(hw->aq.arq.desc_buf.pa)); | ||||
| 		wr32(hw, I40E_VF_ARQBAL1, | ||||
| 		    lower_32_bits(hw->aq.arq.desc_buf.pa)); | ||||
| 		wr32(hw, I40E_VF_ARQLEN1, (hw->aq.num_arq_entries | | ||||
| 					  I40E_VF_ARQLEN1_ARQENABLE_MASK)); | ||||
| 	} else { | ||||
| 		/* configure the receive queue */ | ||||
| 		wr32(hw, I40E_PF_ARQBAH, | ||||
| 		    upper_32_bits(hw->aq.arq.desc_buf.pa)); | ||||
| 		wr32(hw, I40E_PF_ARQBAL, | ||||
| 		    lower_32_bits(hw->aq.arq.desc_buf.pa)); | ||||
| 		wr32(hw, I40E_PF_ARQLEN, (hw->aq.num_arq_entries | | ||||
| 					  I40E_PF_ARQLEN_ARQENABLE_MASK)); | ||||
| 	} | ||||
| 
 | ||||
| 	/* Update tail in the HW to post pre-allocated buffers */ | ||||
| 	wr32(hw, hw->aq.arq.tail, hw->aq.num_arq_entries - 1); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_init_asq - main initialization routine for ASQ | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  * | ||||
|  *  This is the main initialization routine for the Admin Send Queue | ||||
|  *  Prior to calling this function, drivers *MUST* set the following fields | ||||
|  *  in the hw->aq structure: | ||||
|  *     - hw->aq.num_asq_entries | ||||
|  *     - hw->aq.arq_buf_size | ||||
|  * | ||||
|  *  Do *NOT* hold the lock when calling this as the memory allocation routines | ||||
|  *  called are not going to be atomic context safe | ||||
|  **/ | ||||
| static i40e_status i40e_init_asq(struct i40e_hw *hw) | ||||
| { | ||||
| 	i40e_status ret_code = 0; | ||||
| 
 | ||||
| 	if (hw->aq.asq.count > 0) { | ||||
| 		/* queue already initialized */ | ||||
| 		ret_code = I40E_ERR_NOT_READY; | ||||
| 		goto init_adminq_exit; | ||||
| 	} | ||||
| 
 | ||||
| 	/* verify input for valid configuration */ | ||||
| 	if ((hw->aq.num_asq_entries == 0) || | ||||
| 	    (hw->aq.asq_buf_size == 0)) { | ||||
| 		ret_code = I40E_ERR_CONFIG; | ||||
| 		goto init_adminq_exit; | ||||
| 	} | ||||
| 
 | ||||
| 	hw->aq.asq.next_to_use = 0; | ||||
| 	hw->aq.asq.next_to_clean = 0; | ||||
| 	hw->aq.asq.count = hw->aq.num_asq_entries; | ||||
| 
 | ||||
| 	/* allocate the ring memory */ | ||||
| 	ret_code = i40e_alloc_adminq_asq_ring(hw); | ||||
| 	if (ret_code) | ||||
| 		goto init_adminq_exit; | ||||
| 
 | ||||
| 	/* allocate buffers in the rings */ | ||||
| 	ret_code = i40e_alloc_asq_bufs(hw); | ||||
| 	if (ret_code) | ||||
| 		goto init_adminq_free_rings; | ||||
| 
 | ||||
| 	/* initialize base registers */ | ||||
| 	i40e_config_asq_regs(hw); | ||||
| 
 | ||||
| 	/* success! */ | ||||
| 	goto init_adminq_exit; | ||||
| 
 | ||||
| init_adminq_free_rings: | ||||
| 	i40e_free_adminq_asq(hw); | ||||
| 
 | ||||
| init_adminq_exit: | ||||
| 	return ret_code; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_init_arq - initialize ARQ | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  * | ||||
|  *  The main initialization routine for the Admin Receive (Event) Queue. | ||||
|  *  Prior to calling this function, drivers *MUST* set the following fields | ||||
|  *  in the hw->aq structure: | ||||
|  *     - hw->aq.num_asq_entries | ||||
|  *     - hw->aq.arq_buf_size | ||||
|  * | ||||
|  *  Do *NOT* hold the lock when calling this as the memory allocation routines | ||||
|  *  called are not going to be atomic context safe | ||||
|  **/ | ||||
| static i40e_status i40e_init_arq(struct i40e_hw *hw) | ||||
| { | ||||
| 	i40e_status ret_code = 0; | ||||
| 
 | ||||
| 	if (hw->aq.arq.count > 0) { | ||||
| 		/* queue already initialized */ | ||||
| 		ret_code = I40E_ERR_NOT_READY; | ||||
| 		goto init_adminq_exit; | ||||
| 	} | ||||
| 
 | ||||
| 	/* verify input for valid configuration */ | ||||
| 	if ((hw->aq.num_arq_entries == 0) || | ||||
| 	    (hw->aq.arq_buf_size == 0)) { | ||||
| 		ret_code = I40E_ERR_CONFIG; | ||||
| 		goto init_adminq_exit; | ||||
| 	} | ||||
| 
 | ||||
| 	hw->aq.arq.next_to_use = 0; | ||||
| 	hw->aq.arq.next_to_clean = 0; | ||||
| 	hw->aq.arq.count = hw->aq.num_arq_entries; | ||||
| 
 | ||||
| 	/* allocate the ring memory */ | ||||
| 	ret_code = i40e_alloc_adminq_arq_ring(hw); | ||||
| 	if (ret_code) | ||||
| 		goto init_adminq_exit; | ||||
| 
 | ||||
| 	/* allocate buffers in the rings */ | ||||
| 	ret_code = i40e_alloc_arq_bufs(hw); | ||||
| 	if (ret_code) | ||||
| 		goto init_adminq_free_rings; | ||||
| 
 | ||||
| 	/* initialize base registers */ | ||||
| 	i40e_config_arq_regs(hw); | ||||
| 
 | ||||
| 	/* success! */ | ||||
| 	goto init_adminq_exit; | ||||
| 
 | ||||
| init_adminq_free_rings: | ||||
| 	i40e_free_adminq_arq(hw); | ||||
| 
 | ||||
| init_adminq_exit: | ||||
| 	return ret_code; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_shutdown_asq - shutdown the ASQ | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  * | ||||
|  *  The main shutdown routine for the Admin Send Queue | ||||
|  **/ | ||||
| static i40e_status i40e_shutdown_asq(struct i40e_hw *hw) | ||||
| { | ||||
| 	i40e_status ret_code = 0; | ||||
| 
 | ||||
| 	if (hw->aq.asq.count == 0) | ||||
| 		return I40E_ERR_NOT_READY; | ||||
| 
 | ||||
| 	/* Stop firmware AdminQ processing */ | ||||
| 	wr32(hw, hw->aq.asq.head, 0); | ||||
| 	wr32(hw, hw->aq.asq.tail, 0); | ||||
| 	wr32(hw, hw->aq.asq.len, 0); | ||||
| 
 | ||||
| 	/* make sure lock is available */ | ||||
| 	mutex_lock(&hw->aq.asq_mutex); | ||||
| 
 | ||||
| 	hw->aq.asq.count = 0; /* to indicate uninitialized queue */ | ||||
| 
 | ||||
| 	/* free ring buffers */ | ||||
| 	i40e_free_asq_bufs(hw); | ||||
| 
 | ||||
| 	mutex_unlock(&hw->aq.asq_mutex); | ||||
| 
 | ||||
| 	return ret_code; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_shutdown_arq - shutdown ARQ | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  * | ||||
|  *  The main shutdown routine for the Admin Receive Queue | ||||
|  **/ | ||||
| static i40e_status i40e_shutdown_arq(struct i40e_hw *hw) | ||||
| { | ||||
| 	i40e_status ret_code = 0; | ||||
| 
 | ||||
| 	if (hw->aq.arq.count == 0) | ||||
| 		return I40E_ERR_NOT_READY; | ||||
| 
 | ||||
| 	/* Stop firmware AdminQ processing */ | ||||
| 	wr32(hw, hw->aq.arq.head, 0); | ||||
| 	wr32(hw, hw->aq.arq.tail, 0); | ||||
| 	wr32(hw, hw->aq.arq.len, 0); | ||||
| 
 | ||||
| 	/* make sure lock is available */ | ||||
| 	mutex_lock(&hw->aq.arq_mutex); | ||||
| 
 | ||||
| 	hw->aq.arq.count = 0; /* to indicate uninitialized queue */ | ||||
| 
 | ||||
| 	/* free ring buffers */ | ||||
| 	i40e_free_arq_bufs(hw); | ||||
| 
 | ||||
| 	mutex_unlock(&hw->aq.arq_mutex); | ||||
| 
 | ||||
| 	return ret_code; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40evf_init_adminq - main initialization routine for Admin Queue | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  * | ||||
|  *  Prior to calling this function, drivers *MUST* set the following fields | ||||
|  *  in the hw->aq structure: | ||||
|  *     - hw->aq.num_asq_entries | ||||
|  *     - hw->aq.num_arq_entries | ||||
|  *     - hw->aq.arq_buf_size | ||||
|  *     - hw->aq.asq_buf_size | ||||
|  **/ | ||||
| i40e_status i40evf_init_adminq(struct i40e_hw *hw) | ||||
| { | ||||
| 	i40e_status ret_code; | ||||
| 
 | ||||
| 	/* verify input for valid configuration */ | ||||
| 	if ((hw->aq.num_arq_entries == 0) || | ||||
| 	    (hw->aq.num_asq_entries == 0) || | ||||
| 	    (hw->aq.arq_buf_size == 0) || | ||||
| 	    (hw->aq.asq_buf_size == 0)) { | ||||
| 		ret_code = I40E_ERR_CONFIG; | ||||
| 		goto init_adminq_exit; | ||||
| 	} | ||||
| 
 | ||||
| 	/* initialize locks */ | ||||
| 	mutex_init(&hw->aq.asq_mutex); | ||||
| 	mutex_init(&hw->aq.arq_mutex); | ||||
| 
 | ||||
| 	/* Set up register offsets */ | ||||
| 	i40e_adminq_init_regs(hw); | ||||
| 
 | ||||
| 	/* allocate the ASQ */ | ||||
| 	ret_code = i40e_init_asq(hw); | ||||
| 	if (ret_code) | ||||
| 		goto init_adminq_destroy_locks; | ||||
| 
 | ||||
| 	/* allocate the ARQ */ | ||||
| 	ret_code = i40e_init_arq(hw); | ||||
| 	if (ret_code) | ||||
| 		goto init_adminq_free_asq; | ||||
| 
 | ||||
| 	/* success! */ | ||||
| 	goto init_adminq_exit; | ||||
| 
 | ||||
| init_adminq_free_asq: | ||||
| 	i40e_shutdown_asq(hw); | ||||
| init_adminq_destroy_locks: | ||||
| 
 | ||||
| init_adminq_exit: | ||||
| 	return ret_code; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40evf_shutdown_adminq - shutdown routine for the Admin Queue | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  **/ | ||||
| i40e_status i40evf_shutdown_adminq(struct i40e_hw *hw) | ||||
| { | ||||
| 	i40e_status ret_code = 0; | ||||
| 
 | ||||
| 	if (i40evf_check_asq_alive(hw)) | ||||
| 		i40evf_aq_queue_shutdown(hw, true); | ||||
| 
 | ||||
| 	i40e_shutdown_asq(hw); | ||||
| 	i40e_shutdown_arq(hw); | ||||
| 
 | ||||
| 	/* destroy the locks */ | ||||
| 
 | ||||
| 	return ret_code; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40e_clean_asq - cleans Admin send queue | ||||
|  *  @hw: pointer to the hardware structure | ||||
|  * | ||||
|  *  returns the number of free desc | ||||
|  **/ | ||||
| static u16 i40e_clean_asq(struct i40e_hw *hw) | ||||
| { | ||||
| 	struct i40e_adminq_ring *asq = &(hw->aq.asq); | ||||
| 	struct i40e_asq_cmd_details *details; | ||||
| 	u16 ntc = asq->next_to_clean; | ||||
| 	struct i40e_aq_desc desc_cb; | ||||
| 	struct i40e_aq_desc *desc; | ||||
| 
 | ||||
| 	desc = I40E_ADMINQ_DESC(*asq, ntc); | ||||
| 	details = I40E_ADMINQ_DETAILS(*asq, ntc); | ||||
| 	while (rd32(hw, hw->aq.asq.head) != ntc) { | ||||
| 		if (details->callback) { | ||||
| 			I40E_ADMINQ_CALLBACK cb_func = | ||||
| 					(I40E_ADMINQ_CALLBACK)details->callback; | ||||
| 			desc_cb = *desc; | ||||
| 			cb_func(hw, &desc_cb); | ||||
| 		} | ||||
| 		memset((void *)desc, 0, sizeof(struct i40e_aq_desc)); | ||||
| 		memset((void *)details, 0, | ||||
| 		       sizeof(struct i40e_asq_cmd_details)); | ||||
| 		ntc++; | ||||
| 		if (ntc == asq->count) | ||||
| 			ntc = 0; | ||||
| 		desc = I40E_ADMINQ_DESC(*asq, ntc); | ||||
| 		details = I40E_ADMINQ_DETAILS(*asq, ntc); | ||||
| 	} | ||||
| 
 | ||||
| 	asq->next_to_clean = ntc; | ||||
| 
 | ||||
| 	return I40E_DESC_UNUSED(asq); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40evf_asq_done - check if FW has processed the Admin Send Queue | ||||
|  *  @hw: pointer to the hw struct | ||||
|  * | ||||
|  *  Returns true if the firmware has processed all descriptors on the | ||||
|  *  admin send queue. Returns false if there are still requests pending. | ||||
|  **/ | ||||
| bool i40evf_asq_done(struct i40e_hw *hw) | ||||
| { | ||||
| 	/* AQ designers suggest use of head for better
 | ||||
| 	 * timing reliability than DD bit | ||||
| 	 */ | ||||
| 	return rd32(hw, hw->aq.asq.head) == hw->aq.asq.next_to_use; | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40evf_asq_send_command - send command to Admin Queue | ||||
|  *  @hw: pointer to the hw struct | ||||
|  *  @desc: prefilled descriptor describing the command (non DMA mem) | ||||
|  *  @buff: buffer to use for indirect commands | ||||
|  *  @buff_size: size of buffer for indirect commands | ||||
|  *  @cmd_details: pointer to command details structure | ||||
|  * | ||||
|  *  This is the main send command driver routine for the Admin Queue send | ||||
|  *  queue.  It runs the queue, cleans the queue, etc | ||||
|  **/ | ||||
| i40e_status i40evf_asq_send_command(struct i40e_hw *hw, | ||||
| 				struct i40e_aq_desc *desc, | ||||
| 				void *buff, /* can be NULL */ | ||||
| 				u16  buff_size, | ||||
| 				struct i40e_asq_cmd_details *cmd_details) | ||||
| { | ||||
| 	i40e_status status = 0; | ||||
| 	struct i40e_dma_mem *dma_buff = NULL; | ||||
| 	struct i40e_asq_cmd_details *details; | ||||
| 	struct i40e_aq_desc *desc_on_ring; | ||||
| 	bool cmd_completed = false; | ||||
| 	u16  retval = 0; | ||||
| 
 | ||||
| 	if (hw->aq.asq.count == 0) { | ||||
| 		i40e_debug(hw, I40E_DEBUG_AQ_MESSAGE, | ||||
| 			   "AQTX: Admin queue not initialized.\n"); | ||||
| 		status = I40E_ERR_QUEUE_EMPTY; | ||||
| 		goto asq_send_command_exit; | ||||
| 	} | ||||
| 
 | ||||
| 	details = I40E_ADMINQ_DETAILS(hw->aq.asq, hw->aq.asq.next_to_use); | ||||
| 	if (cmd_details) { | ||||
| 		*details = *cmd_details; | ||||
| 
 | ||||
| 		/* If the cmd_details are defined copy the cookie.  The
 | ||||
| 		 * cpu_to_le32 is not needed here because the data is ignored | ||||
| 		 * by the FW, only used by the driver | ||||
| 		 */ | ||||
| 		if (details->cookie) { | ||||
| 			desc->cookie_high = | ||||
| 				cpu_to_le32(upper_32_bits(details->cookie)); | ||||
| 			desc->cookie_low = | ||||
| 				cpu_to_le32(lower_32_bits(details->cookie)); | ||||
| 		} | ||||
| 	} else { | ||||
| 		memset(details, 0, sizeof(struct i40e_asq_cmd_details)); | ||||
| 	} | ||||
| 
 | ||||
| 	/* clear requested flags and then set additional flags if defined */ | ||||
| 	desc->flags &= ~cpu_to_le16(details->flags_dis); | ||||
| 	desc->flags |= cpu_to_le16(details->flags_ena); | ||||
| 
 | ||||
| 	mutex_lock(&hw->aq.asq_mutex); | ||||
| 
 | ||||
| 	if (buff_size > hw->aq.asq_buf_size) { | ||||
| 		i40e_debug(hw, | ||||
| 			   I40E_DEBUG_AQ_MESSAGE, | ||||
| 			   "AQTX: Invalid buffer size: %d.\n", | ||||
| 			   buff_size); | ||||
| 		status = I40E_ERR_INVALID_SIZE; | ||||
| 		goto asq_send_command_error; | ||||
| 	} | ||||
| 
 | ||||
| 	if (details->postpone && !details->async) { | ||||
| 		i40e_debug(hw, | ||||
| 			   I40E_DEBUG_AQ_MESSAGE, | ||||
| 			   "AQTX: Async flag not set along with postpone flag"); | ||||
| 		status = I40E_ERR_PARAM; | ||||
| 		goto asq_send_command_error; | ||||
| 	} | ||||
| 
 | ||||
| 	/* call clean and check queue available function to reclaim the
 | ||||
| 	 * descriptors that were processed by FW, the function returns the | ||||
| 	 * number of desc available | ||||
| 	 */ | ||||
| 	/* the clean function called here could be called in a separate thread
 | ||||
| 	 * in case of asynchronous completions | ||||
| 	 */ | ||||
| 	if (i40e_clean_asq(hw) == 0) { | ||||
| 		i40e_debug(hw, | ||||
| 			   I40E_DEBUG_AQ_MESSAGE, | ||||
| 			   "AQTX: Error queue is full.\n"); | ||||
| 		status = I40E_ERR_ADMIN_QUEUE_FULL; | ||||
| 		goto asq_send_command_error; | ||||
| 	} | ||||
| 
 | ||||
| 	/* initialize the temp desc pointer with the right desc */ | ||||
| 	desc_on_ring = I40E_ADMINQ_DESC(hw->aq.asq, hw->aq.asq.next_to_use); | ||||
| 
 | ||||
| 	/* if the desc is available copy the temp desc to the right place */ | ||||
| 	*desc_on_ring = *desc; | ||||
| 
 | ||||
| 	/* if buff is not NULL assume indirect command */ | ||||
| 	if (buff != NULL) { | ||||
| 		dma_buff = &(hw->aq.asq.r.asq_bi[hw->aq.asq.next_to_use]); | ||||
| 		/* copy the user buff into the respective DMA buff */ | ||||
| 		memcpy(dma_buff->va, buff, buff_size); | ||||
| 		desc_on_ring->datalen = cpu_to_le16(buff_size); | ||||
| 
 | ||||
| 		/* Update the address values in the desc with the pa value
 | ||||
| 		 * for respective buffer | ||||
| 		 */ | ||||
| 		desc_on_ring->params.external.addr_high = | ||||
| 				cpu_to_le32(upper_32_bits(dma_buff->pa)); | ||||
| 		desc_on_ring->params.external.addr_low = | ||||
| 				cpu_to_le32(lower_32_bits(dma_buff->pa)); | ||||
| 	} | ||||
| 
 | ||||
| 	/* bump the tail */ | ||||
| 	i40evf_debug_aq(hw, I40E_DEBUG_AQ_COMMAND, (void *)desc_on_ring, buff); | ||||
| 	(hw->aq.asq.next_to_use)++; | ||||
| 	if (hw->aq.asq.next_to_use == hw->aq.asq.count) | ||||
| 		hw->aq.asq.next_to_use = 0; | ||||
| 	if (!details->postpone) | ||||
| 		wr32(hw, hw->aq.asq.tail, hw->aq.asq.next_to_use); | ||||
| 
 | ||||
| 	/* if cmd_details are not defined or async flag is not set,
 | ||||
| 	 * we need to wait for desc write back | ||||
| 	 */ | ||||
| 	if (!details->async && !details->postpone) { | ||||
| 		u32 total_delay = 0; | ||||
| 		u32 delay_len = 10; | ||||
| 
 | ||||
| 		do { | ||||
| 			/* AQ designers suggest use of head for better
 | ||||
| 			 * timing reliability than DD bit | ||||
| 			 */ | ||||
| 			if (i40evf_asq_done(hw)) | ||||
| 				break; | ||||
| 			/* ugh! delay while spin_lock */ | ||||
| 			udelay(delay_len); | ||||
| 			total_delay += delay_len; | ||||
| 		} while (total_delay <  I40E_ASQ_CMD_TIMEOUT); | ||||
| 	} | ||||
| 
 | ||||
| 	/* if ready, copy the desc back to temp */ | ||||
| 	if (i40evf_asq_done(hw)) { | ||||
| 		*desc = *desc_on_ring; | ||||
| 		if (buff != NULL) | ||||
| 			memcpy(buff, dma_buff->va, buff_size); | ||||
| 		retval = le16_to_cpu(desc->retval); | ||||
| 		if (retval != 0) { | ||||
| 			i40e_debug(hw, | ||||
| 				   I40E_DEBUG_AQ_MESSAGE, | ||||
| 				   "AQTX: Command completed with error 0x%X.\n", | ||||
| 				   retval); | ||||
| 			/* strip off FW internal code */ | ||||
| 			retval &= 0xff; | ||||
| 		} | ||||
| 		cmd_completed = true; | ||||
| 		if ((enum i40e_admin_queue_err)retval == I40E_AQ_RC_OK) | ||||
| 			status = 0; | ||||
| 		else | ||||
| 			status = I40E_ERR_ADMIN_QUEUE_ERROR; | ||||
| 		hw->aq.asq_last_status = (enum i40e_admin_queue_err)retval; | ||||
| 	} | ||||
| 
 | ||||
| 	/* update the error if time out occurred */ | ||||
| 	if ((!cmd_completed) && | ||||
| 	    (!details->async && !details->postpone)) { | ||||
| 		i40e_debug(hw, | ||||
| 			   I40E_DEBUG_AQ_MESSAGE, | ||||
| 			   "AQTX: Writeback timeout.\n"); | ||||
| 		status = I40E_ERR_ADMIN_QUEUE_TIMEOUT; | ||||
| 	} | ||||
| 
 | ||||
| asq_send_command_error: | ||||
| 	mutex_unlock(&hw->aq.asq_mutex); | ||||
| asq_send_command_exit: | ||||
| 	return status; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40evf_fill_default_direct_cmd_desc - AQ descriptor helper function | ||||
|  *  @desc:     pointer to the temp descriptor (non DMA mem) | ||||
|  *  @opcode:   the opcode can be used to decide which flags to turn off or on | ||||
|  * | ||||
|  *  Fill the desc with default values | ||||
|  **/ | ||||
| void i40evf_fill_default_direct_cmd_desc(struct i40e_aq_desc *desc, | ||||
| 				       u16 opcode) | ||||
| { | ||||
| 	/* zero out the desc */ | ||||
| 	memset((void *)desc, 0, sizeof(struct i40e_aq_desc)); | ||||
| 	desc->opcode = cpu_to_le16(opcode); | ||||
| 	desc->flags = cpu_to_le16(I40E_AQ_FLAG_SI); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  *  i40evf_clean_arq_element | ||||
|  *  @hw: pointer to the hw struct | ||||
|  *  @e: event info from the receive descriptor, includes any buffers | ||||
|  *  @pending: number of events that could be left to process | ||||
|  * | ||||
|  *  This function cleans one Admin Receive Queue element and returns | ||||
|  *  the contents through e.  It can also return how many events are | ||||
|  *  left to process through 'pending' | ||||
|  **/ | ||||
| i40e_status i40evf_clean_arq_element(struct i40e_hw *hw, | ||||
| 					     struct i40e_arq_event_info *e, | ||||
| 					     u16 *pending) | ||||
| { | ||||
| 	i40e_status ret_code = 0; | ||||
| 	u16 ntc = hw->aq.arq.next_to_clean; | ||||
| 	struct i40e_aq_desc *desc; | ||||
| 	struct i40e_dma_mem *bi; | ||||
| 	u16 desc_idx; | ||||
| 	u16 datalen; | ||||
| 	u16 flags; | ||||
| 	u16 ntu; | ||||
| 
 | ||||
| 	/* take the lock before we start messing with the ring */ | ||||
| 	mutex_lock(&hw->aq.arq_mutex); | ||||
| 
 | ||||
| 	/* set next_to_use to head */ | ||||
| 	ntu = (rd32(hw, hw->aq.arq.head) & I40E_PF_ARQH_ARQH_MASK); | ||||
| 	if (ntu == ntc) { | ||||
| 		/* nothing to do - shouldn't need to update ring's values */ | ||||
| 		i40e_debug(hw, | ||||
| 			   I40E_DEBUG_AQ_MESSAGE, | ||||
| 			   "AQRX: Queue is empty.\n"); | ||||
| 		ret_code = I40E_ERR_ADMIN_QUEUE_NO_WORK; | ||||
| 		goto clean_arq_element_out; | ||||
| 	} | ||||
| 
 | ||||
| 	/* now clean the next descriptor */ | ||||
| 	desc = I40E_ADMINQ_DESC(hw->aq.arq, ntc); | ||||
| 	desc_idx = ntc; | ||||
| 	i40evf_debug_aq(hw, | ||||
| 		      I40E_DEBUG_AQ_COMMAND, | ||||
| 		      (void *)desc, | ||||
| 		      hw->aq.arq.r.arq_bi[desc_idx].va); | ||||
| 
 | ||||
| 	flags = le16_to_cpu(desc->flags); | ||||
| 	if (flags & I40E_AQ_FLAG_ERR) { | ||||
| 		ret_code = I40E_ERR_ADMIN_QUEUE_ERROR; | ||||
| 		hw->aq.arq_last_status = | ||||
| 			(enum i40e_admin_queue_err)le16_to_cpu(desc->retval); | ||||
| 		i40e_debug(hw, | ||||
| 			   I40E_DEBUG_AQ_MESSAGE, | ||||
| 			   "AQRX: Event received with error 0x%X.\n", | ||||
| 			   hw->aq.arq_last_status); | ||||
| 	} else { | ||||
| 		e->desc = *desc; | ||||
| 		datalen = le16_to_cpu(desc->datalen); | ||||
| 		e->msg_size = min(datalen, e->msg_size); | ||||
| 		if (e->msg_buf != NULL && (e->msg_size != 0)) | ||||
| 			memcpy(e->msg_buf, hw->aq.arq.r.arq_bi[desc_idx].va, | ||||
| 			       e->msg_size); | ||||
| 	} | ||||
| 
 | ||||
| 	/* Restore the original datalen and buffer address in the desc,
 | ||||
| 	 * FW updates datalen to indicate the event message | ||||
| 	 * size | ||||
| 	 */ | ||||
| 	bi = &hw->aq.arq.r.arq_bi[ntc]; | ||||
| 	memset((void *)desc, 0, sizeof(struct i40e_aq_desc)); | ||||
| 
 | ||||
| 	desc->flags = cpu_to_le16(I40E_AQ_FLAG_BUF); | ||||
| 	if (hw->aq.arq_buf_size > I40E_AQ_LARGE_BUF) | ||||
| 		desc->flags |= cpu_to_le16(I40E_AQ_FLAG_LB); | ||||
| 	desc->datalen = cpu_to_le16((u16)bi->size); | ||||
| 	desc->params.external.addr_high = cpu_to_le32(upper_32_bits(bi->pa)); | ||||
| 	desc->params.external.addr_low = cpu_to_le32(lower_32_bits(bi->pa)); | ||||
| 
 | ||||
| 	/* set tail = the last cleaned desc index. */ | ||||
| 	wr32(hw, hw->aq.arq.tail, ntc); | ||||
| 	/* ntc is updated to tail + 1 */ | ||||
| 	ntc++; | ||||
| 	if (ntc == hw->aq.num_arq_entries) | ||||
| 		ntc = 0; | ||||
| 	hw->aq.arq.next_to_clean = ntc; | ||||
| 	hw->aq.arq.next_to_use = ntu; | ||||
| 
 | ||||
| clean_arq_element_out: | ||||
| 	/* Set pending if needed, unlock and return */ | ||||
| 	if (pending != NULL) | ||||
| 		*pending = (ntc > ntu ? hw->aq.arq.count : 0) + (ntu - ntc); | ||||
| 	mutex_unlock(&hw->aq.arq_mutex); | ||||
| 
 | ||||
| 	return ret_code; | ||||
| } | ||||
| 
 | ||||
| void i40evf_resume_aq(struct i40e_hw *hw) | ||||
| { | ||||
| 	/* Registers are reset after PF reset */ | ||||
| 	hw->aq.asq.next_to_use = 0; | ||||
| 	hw->aq.asq.next_to_clean = 0; | ||||
| 
 | ||||
| 	i40e_config_asq_regs(hw); | ||||
| 
 | ||||
| 	hw->aq.arq.next_to_use = 0; | ||||
| 	hw->aq.arq.next_to_clean = 0; | ||||
| 
 | ||||
| 	i40e_config_arq_regs(hw); | ||||
| } | ||||
							
								
								
									
										106
									
								
								drivers/net/ethernet/intel/i40evf/i40e_adminq.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										106
									
								
								drivers/net/ethernet/intel/i40evf/i40e_adminq.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,106 @@ | ||||
| /*******************************************************************************
 | ||||
|  * | ||||
|  * Intel Ethernet Controller XL710 Family Linux Virtual Function Driver | ||||
|  * Copyright(c) 2013 Intel Corporation. | ||||
|  * | ||||
|  * 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. | ||||
|  * | ||||
|  * 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 | ||||
|  * | ||||
|  ******************************************************************************/ | ||||
| 
 | ||||
| #ifndef _I40E_ADMINQ_H_ | ||||
| #define _I40E_ADMINQ_H_ | ||||
| 
 | ||||
| #include "i40e_osdep.h" | ||||
| #include "i40e_adminq_cmd.h" | ||||
| 
 | ||||
| #define I40E_ADMINQ_DESC(R, i)   \ | ||||
| 	(&(((struct i40e_aq_desc *)((R).desc_buf.va))[i])) | ||||
| 
 | ||||
| #define I40E_ADMINQ_DESC_ALIGNMENT 4096 | ||||
| 
 | ||||
| struct i40e_adminq_ring { | ||||
| 	struct i40e_virt_mem dma_head;	/* space for dma structures */ | ||||
| 	struct i40e_dma_mem desc_buf;	/* descriptor ring memory */ | ||||
| 	struct i40e_virt_mem cmd_buf;	/* command buffer memory */ | ||||
| 
 | ||||
| 	union { | ||||
| 		struct i40e_dma_mem *asq_bi; | ||||
| 		struct i40e_dma_mem *arq_bi; | ||||
| 	} r; | ||||
| 
 | ||||
| 	u16 count;		/* Number of descriptors */ | ||||
| 	u16 rx_buf_len;		/* Admin Receive Queue buffer length */ | ||||
| 
 | ||||
| 	/* used for interrupt processing */ | ||||
| 	u16 next_to_use; | ||||
| 	u16 next_to_clean; | ||||
| 
 | ||||
| 	/* used for queue tracking */ | ||||
| 	u32 head; | ||||
| 	u32 tail; | ||||
| 	u32 len; | ||||
| }; | ||||
| 
 | ||||
| /* ASQ transaction details */ | ||||
| struct i40e_asq_cmd_details { | ||||
| 	void *callback; /* cast from type I40E_ADMINQ_CALLBACK */ | ||||
| 	u64 cookie; | ||||
| 	u16 flags_ena; | ||||
| 	u16 flags_dis; | ||||
| 	bool async; | ||||
| 	bool postpone; | ||||
| }; | ||||
| 
 | ||||
| #define I40E_ADMINQ_DETAILS(R, i)   \ | ||||
| 	(&(((struct i40e_asq_cmd_details *)((R).cmd_buf.va))[i])) | ||||
| 
 | ||||
| /* ARQ event information */ | ||||
| struct i40e_arq_event_info { | ||||
| 	struct i40e_aq_desc desc; | ||||
| 	u16 msg_size; | ||||
| 	u8 *msg_buf; | ||||
| }; | ||||
| 
 | ||||
| /* Admin Queue information */ | ||||
| struct i40e_adminq_info { | ||||
| 	struct i40e_adminq_ring arq;    /* receive queue */ | ||||
| 	struct i40e_adminq_ring asq;    /* send queue */ | ||||
| 	u16 num_arq_entries;            /* receive queue depth */ | ||||
| 	u16 num_asq_entries;            /* send queue depth */ | ||||
| 	u16 arq_buf_size;               /* receive queue buffer size */ | ||||
| 	u16 asq_buf_size;               /* send queue buffer size */ | ||||
| 	u16 fw_maj_ver;                 /* firmware major version */ | ||||
| 	u16 fw_min_ver;                 /* firmware minor version */ | ||||
| 	u16 api_maj_ver;                /* api major version */ | ||||
| 	u16 api_min_ver;                /* api minor version */ | ||||
| 
 | ||||
| 	struct mutex asq_mutex; /* Send queue lock */ | ||||
| 	struct mutex arq_mutex; /* Receive queue lock */ | ||||
| 
 | ||||
| 	/* last status values on send and receive queues */ | ||||
| 	enum i40e_admin_queue_err asq_last_status; | ||||
| 	enum i40e_admin_queue_err arq_last_status; | ||||
| }; | ||||
| 
 | ||||
| /* general information */ | ||||
| #define I40E_AQ_LARGE_BUF	512 | ||||
| #define I40E_ASQ_CMD_TIMEOUT	100000  /* usecs */ | ||||
| 
 | ||||
| void i40evf_fill_default_direct_cmd_desc(struct i40e_aq_desc *desc, | ||||
| 				       u16 opcode); | ||||
| 
 | ||||
| #endif /* _I40E_ADMINQ_H_ */ | ||||
							
								
								
									
										2129
									
								
								drivers/net/ethernet/intel/i40evf/i40e_adminq_cmd.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2129
									
								
								drivers/net/ethernet/intel/i40evf/i40e_adminq_cmd.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										55
									
								
								drivers/net/ethernet/intel/i40evf/i40e_alloc.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								drivers/net/ethernet/intel/i40evf/i40e_alloc.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,55 @@ | ||||
| /*******************************************************************************
 | ||||
|  * | ||||
|  * Intel Ethernet Controller XL710 Family Linux Virtual Function Driver | ||||
|  * Copyright(c) 2013 Intel Corporation. | ||||
|  * | ||||
|  * 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. | ||||
|  * | ||||
|  * 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 | ||||
|  * | ||||
|  ******************************************************************************/ | ||||
| 
 | ||||
| #ifndef _I40E_ALLOC_H_ | ||||
| #define _I40E_ALLOC_H_ | ||||
| 
 | ||||
| struct i40e_hw; | ||||
| 
 | ||||
| /* Memory allocation types */ | ||||
| enum i40e_memory_type { | ||||
| 	i40e_mem_arq_buf = 0,		/* ARQ indirect command buffer */ | ||||
| 	i40e_mem_asq_buf = 1, | ||||
| 	i40e_mem_atq_buf = 2,		/* ATQ indirect command buffer */ | ||||
| 	i40e_mem_arq_ring = 3,		/* ARQ descriptor ring */ | ||||
| 	i40e_mem_atq_ring = 4,		/* ATQ descriptor ring */ | ||||
| 	i40e_mem_pd = 5,		/* Page Descriptor */ | ||||
| 	i40e_mem_bp = 6,		/* Backing Page - 4KB */ | ||||
| 	i40e_mem_bp_jumbo = 7,		/* Backing Page - > 4KB */ | ||||
| 	i40e_mem_reserved | ||||
| }; | ||||
| 
 | ||||
| /* prototype for functions used for dynamic memory allocation */ | ||||
| i40e_status i40e_allocate_dma_mem(struct i40e_hw *hw, | ||||
| 					    struct i40e_dma_mem *mem, | ||||
| 					    enum i40e_memory_type type, | ||||
| 					    u64 size, u32 alignment); | ||||
| i40e_status i40e_free_dma_mem(struct i40e_hw *hw, | ||||
| 					struct i40e_dma_mem *mem); | ||||
| i40e_status i40e_allocate_virt_mem(struct i40e_hw *hw, | ||||
| 					     struct i40e_virt_mem *mem, | ||||
| 					     u32 size); | ||||
| i40e_status i40e_free_virt_mem(struct i40e_hw *hw, | ||||
| 					 struct i40e_virt_mem *mem); | ||||
| 
 | ||||
| #endif /* _I40E_ALLOC_H_ */ | ||||
							
								
								
									
										254
									
								
								drivers/net/ethernet/intel/i40evf/i40e_common.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										254
									
								
								drivers/net/ethernet/intel/i40evf/i40e_common.c
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,254 @@ | ||||
| /*******************************************************************************
 | ||||
|  * | ||||
|  * Intel Ethernet Controller XL710 Family Linux Virtual Function Driver | ||||
|  * Copyright(c) 2013 Intel Corporation. | ||||
|  * | ||||
|  * 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. | ||||
|  * | ||||
|  * 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_type.h" | ||||
| #include "i40e_adminq.h" | ||||
| #include "i40e_prototype.h" | ||||
| #include "i40e_virtchnl.h" | ||||
| 
 | ||||
| /**
 | ||||
|  * i40e_set_mac_type - Sets MAC type | ||||
|  * @hw: pointer to the HW structure | ||||
|  * | ||||
|  * This function sets the mac type of the adapter based on the | ||||
|  * vendor ID and device ID stored in the hw structure. | ||||
|  **/ | ||||
| i40e_status i40e_set_mac_type(struct i40e_hw *hw) | ||||
| { | ||||
| 	i40e_status status = 0; | ||||
| 
 | ||||
| 	if (hw->vendor_id == PCI_VENDOR_ID_INTEL) { | ||||
| 		switch (hw->device_id) { | ||||
| 		case I40E_SFP_XL710_DEVICE_ID: | ||||
| 		case I40E_SFP_X710_DEVICE_ID: | ||||
| 		case I40E_QEMU_DEVICE_ID: | ||||
| 		case I40E_KX_A_DEVICE_ID: | ||||
| 		case I40E_KX_B_DEVICE_ID: | ||||
| 		case I40E_KX_C_DEVICE_ID: | ||||
| 		case I40E_KX_D_DEVICE_ID: | ||||
| 		case I40E_QSFP_A_DEVICE_ID: | ||||
| 		case I40E_QSFP_B_DEVICE_ID: | ||||
| 		case I40E_QSFP_C_DEVICE_ID: | ||||
| 			hw->mac.type = I40E_MAC_XL710; | ||||
| 			break; | ||||
| 		case I40E_VF_DEVICE_ID: | ||||
| 		case I40E_VF_HV_DEVICE_ID: | ||||
| 			hw->mac.type = I40E_MAC_VF; | ||||
| 			break; | ||||
| 		default: | ||||
| 			hw->mac.type = I40E_MAC_GENERIC; | ||||
| 			break; | ||||
| 		} | ||||
| 	} else { | ||||
| 		status = I40E_ERR_DEVICE_NOT_SUPPORTED; | ||||
| 	} | ||||
| 
 | ||||
| 	hw_dbg(hw, "i40e_set_mac_type found mac: %d, returns: %d\n", | ||||
| 		  hw->mac.type, status); | ||||
| 	return status; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * i40evf_debug_aq | ||||
|  * @hw: debug mask related to admin queue | ||||
|  * @mask: debug mask | ||||
|  * @desc: pointer to admin queue descriptor | ||||
|  * @buffer: pointer to command buffer | ||||
|  * | ||||
|  * Dumps debug log about adminq command with descriptor contents. | ||||
|  **/ | ||||
| void i40evf_debug_aq(struct i40e_hw *hw, enum i40e_debug_mask mask, void *desc, | ||||
| 		   void *buffer) | ||||
| { | ||||
| 	struct i40e_aq_desc *aq_desc = (struct i40e_aq_desc *)desc; | ||||
| 	u8 *aq_buffer = (u8 *)buffer; | ||||
| 	u32 data[4]; | ||||
| 	u32 i = 0; | ||||
| 
 | ||||
| 	if ((!(mask & hw->debug_mask)) || (desc == NULL)) | ||||
| 		return; | ||||
| 
 | ||||
| 	i40e_debug(hw, mask, | ||||
| 		   "AQ CMD: opcode 0x%04X, flags 0x%04X, datalen 0x%04X, retval 0x%04X\n", | ||||
| 		   aq_desc->opcode, aq_desc->flags, aq_desc->datalen, | ||||
| 		   aq_desc->retval); | ||||
| 	i40e_debug(hw, mask, "\tcookie (h,l) 0x%08X 0x%08X\n", | ||||
| 		   aq_desc->cookie_high, aq_desc->cookie_low); | ||||
| 	i40e_debug(hw, mask, "\tparam (0,1)  0x%08X 0x%08X\n", | ||||
| 		   aq_desc->params.internal.param0, | ||||
| 		   aq_desc->params.internal.param1); | ||||
| 	i40e_debug(hw, mask, "\taddr (h,l)   0x%08X 0x%08X\n", | ||||
| 		   aq_desc->params.external.addr_high, | ||||
| 		   aq_desc->params.external.addr_low); | ||||
| 
 | ||||
| 	if ((buffer != NULL) && (aq_desc->datalen != 0)) { | ||||
| 		memset(data, 0, sizeof(data)); | ||||
| 		i40e_debug(hw, mask, "AQ CMD Buffer:\n"); | ||||
| 		for (i = 0; i < le16_to_cpu(aq_desc->datalen); i++) { | ||||
| 			data[((i % 16) / 4)] |= | ||||
| 				((u32)aq_buffer[i]) << (8 * (i % 4)); | ||||
| 			if ((i % 16) == 15) { | ||||
| 				i40e_debug(hw, mask, | ||||
| 					   "\t0x%04X  %08X %08X %08X %08X\n", | ||||
| 					   i - 15, data[0], data[1], data[2], | ||||
| 					   data[3]); | ||||
| 				memset(data, 0, sizeof(data)); | ||||
| 			} | ||||
| 		} | ||||
| 		if ((i % 16) != 0) | ||||
| 			i40e_debug(hw, mask, "\t0x%04X  %08X %08X %08X %08X\n", | ||||
| 				   i - (i % 16), data[0], data[1], data[2], | ||||
| 				   data[3]); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * i40evf_check_asq_alive | ||||
|  * @hw: pointer to the hw struct | ||||
|  * | ||||
|  * Returns true if Queue is enabled else false. | ||||
|  **/ | ||||
| bool i40evf_check_asq_alive(struct i40e_hw *hw) | ||||
| { | ||||
| 	return !!(rd32(hw, hw->aq.asq.len) & I40E_PF_ATQLEN_ATQENABLE_MASK); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * i40evf_aq_queue_shutdown | ||||
|  * @hw: pointer to the hw struct | ||||
|  * @unloading: is the driver unloading itself | ||||
|  * | ||||
|  * Tell the Firmware that we're shutting down the AdminQ and whether | ||||
|  * or not the driver is unloading as well. | ||||
|  **/ | ||||
| i40e_status i40evf_aq_queue_shutdown(struct i40e_hw *hw, | ||||
| 					     bool unloading) | ||||
| { | ||||
| 	struct i40e_aq_desc desc; | ||||
| 	struct i40e_aqc_queue_shutdown *cmd = | ||||
| 		(struct i40e_aqc_queue_shutdown *)&desc.params.raw; | ||||
| 	i40e_status status; | ||||
| 
 | ||||
| 	i40evf_fill_default_direct_cmd_desc(&desc, | ||||
| 					  i40e_aqc_opc_queue_shutdown); | ||||
| 
 | ||||
| 	if (unloading) | ||||
| 		cmd->driver_unloading = cpu_to_le32(I40E_AQ_DRIVER_UNLOADING); | ||||
| 	status = i40evf_asq_send_command(hw, &desc, NULL, 0, NULL); | ||||
| 
 | ||||
| 	return status; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| /**
 | ||||
|  * i40e_aq_send_msg_to_pf | ||||
|  * @hw: pointer to the hardware structure | ||||
|  * @v_opcode: opcodes for VF-PF communication | ||||
|  * @v_retval: return error code | ||||
|  * @msg: pointer to the msg buffer | ||||
|  * @msglen: msg length | ||||
|  * @cmd_details: pointer to command details | ||||
|  * | ||||
|  * Send message to PF driver using admin queue. By default, this message | ||||
|  * is sent asynchronously, i.e. i40evf_asq_send_command() does not wait for | ||||
|  * completion before returning. | ||||
|  **/ | ||||
| i40e_status i40e_aq_send_msg_to_pf(struct i40e_hw *hw, | ||||
| 				enum i40e_virtchnl_ops v_opcode, | ||||
| 				i40e_status v_retval, | ||||
| 				u8 *msg, u16 msglen, | ||||
| 				struct i40e_asq_cmd_details *cmd_details) | ||||
| { | ||||
| 	struct i40e_aq_desc desc; | ||||
| 	i40e_status status; | ||||
| 
 | ||||
| 	i40evf_fill_default_direct_cmd_desc(&desc, i40e_aqc_opc_send_msg_to_pf); | ||||
| 	desc.flags |= cpu_to_le16((u16)I40E_AQ_FLAG_SI); | ||||
| 	desc.cookie_high = cpu_to_le32(v_opcode); | ||||
| 	desc.cookie_low = cpu_to_le32(v_retval); | ||||
| 	if (msglen) { | ||||
| 		desc.flags |= cpu_to_le16((u16)(I40E_AQ_FLAG_BUF | ||||
| 						| I40E_AQ_FLAG_RD)); | ||||
| 		if (msglen > I40E_AQ_LARGE_BUF) | ||||
| 			desc.flags |= cpu_to_le16((u16)I40E_AQ_FLAG_LB); | ||||
| 		desc.datalen = cpu_to_le16(msglen); | ||||
| 	} | ||||
| 	if (!cmd_details) { | ||||
| 		struct i40e_asq_cmd_details details; | ||||
| 		memset(&details, 0, sizeof(details)); | ||||
| 		details.async = true; | ||||
| 		cmd_details = &details; | ||||
| 	} | ||||
| 	status = i40evf_asq_send_command(hw, (struct i40e_aq_desc *)&desc, msg, | ||||
| 				       msglen, cmd_details); | ||||
| 	return status; | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * i40e_vf_parse_hw_config | ||||
|  * @hw: pointer to the hardware structure | ||||
|  * @msg: pointer to the virtual channel VF resource structure | ||||
|  * | ||||
|  * Given a VF resource message from the PF, populate the hw struct | ||||
|  * with appropriate information. | ||||
|  **/ | ||||
| void i40e_vf_parse_hw_config(struct i40e_hw *hw, | ||||
| 			     struct i40e_virtchnl_vf_resource *msg) | ||||
| { | ||||
| 	struct i40e_virtchnl_vsi_resource *vsi_res; | ||||
| 	int i; | ||||
| 
 | ||||
| 	vsi_res = &msg->vsi_res[0]; | ||||
| 
 | ||||
| 	hw->dev_caps.num_vsis = msg->num_vsis; | ||||
| 	hw->dev_caps.num_rx_qp = msg->num_queue_pairs; | ||||
| 	hw->dev_caps.num_tx_qp = msg->num_queue_pairs; | ||||
| 	hw->dev_caps.num_msix_vectors_vf = msg->max_vectors; | ||||
| 	hw->dev_caps.dcb = msg->vf_offload_flags & | ||||
| 			   I40E_VIRTCHNL_VF_OFFLOAD_L2; | ||||
| 	hw->dev_caps.fcoe = (msg->vf_offload_flags & | ||||
| 			     I40E_VIRTCHNL_VF_OFFLOAD_FCOE) ? 1 : 0; | ||||
| 	for (i = 0; i < msg->num_vsis; i++) { | ||||
| 		if (vsi_res->vsi_type == I40E_VSI_SRIOV) { | ||||
| 			memcpy(hw->mac.perm_addr, vsi_res->default_mac_addr, | ||||
| 			       ETH_ALEN); | ||||
| 			memcpy(hw->mac.addr, vsi_res->default_mac_addr, | ||||
| 			       ETH_ALEN); | ||||
| 		} | ||||
| 		vsi_res++; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * i40e_vf_reset | ||||
|  * @hw: pointer to the hardware structure | ||||
|  * | ||||
|  * Send a VF_RESET message to the PF. Does not wait for response from PF | ||||
|  * as none will be forthcoming. Immediately after calling this function, | ||||
|  * the admin queue should be shut down and (optionally) reinitialized. | ||||
|  **/ | ||||
| i40e_status i40e_vf_reset(struct i40e_hw *hw) | ||||
| { | ||||
| 	return i40e_aq_send_msg_to_pf(hw, I40E_VIRTCHNL_OP_RESET_VF, | ||||
| 				      0, NULL, 0, NULL); | ||||
| } | ||||
							
								
								
									
										238
									
								
								drivers/net/ethernet/intel/i40evf/i40e_hmc.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										238
									
								
								drivers/net/ethernet/intel/i40evf/i40e_hmc.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,238 @@ | ||||
| /*******************************************************************************
 | ||||
|  * | ||||
|  * Intel Ethernet Controller XL710 Family Linux Virtual Function Driver | ||||
|  * Copyright(c) 2013 Intel Corporation. | ||||
|  * | ||||
|  * 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. | ||||
|  * | ||||
|  * 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 | ||||
|  * | ||||
|  ******************************************************************************/ | ||||
| 
 | ||||
| #ifndef _I40E_HMC_H_ | ||||
| #define _I40E_HMC_H_ | ||||
| 
 | ||||
| #define I40E_HMC_MAX_BP_COUNT 512 | ||||
| 
 | ||||
| /* forward-declare the HW struct for the compiler */ | ||||
| struct i40e_hw; | ||||
| 
 | ||||
| #define I40E_HMC_INFO_SIGNATURE		0x484D5347 /* HMSG */ | ||||
| #define I40E_HMC_PD_CNT_IN_SD		512 | ||||
| #define I40E_HMC_DIRECT_BP_SIZE		0x200000 /* 2M */ | ||||
| #define I40E_HMC_PAGED_BP_SIZE		4096 | ||||
| #define I40E_HMC_PD_BP_BUF_ALIGNMENT	4096 | ||||
| #define I40E_FIRST_VF_FPM_ID		16 | ||||
| 
 | ||||
| struct i40e_hmc_obj_info { | ||||
| 	u64 base;	/* base addr in FPM */ | ||||
| 	u32 max_cnt;	/* max count available for this hmc func */ | ||||
| 	u32 cnt;	/* count of objects driver actually wants to create */ | ||||
| 	u64 size;	/* size in bytes of one object */ | ||||
| }; | ||||
| 
 | ||||
| enum i40e_sd_entry_type { | ||||
| 	I40E_SD_TYPE_INVALID = 0, | ||||
| 	I40E_SD_TYPE_PAGED   = 1, | ||||
| 	I40E_SD_TYPE_DIRECT  = 2 | ||||
| }; | ||||
| 
 | ||||
| struct i40e_hmc_bp { | ||||
| 	enum i40e_sd_entry_type entry_type; | ||||
| 	struct i40e_dma_mem addr; /* populate to be used by hw */ | ||||
| 	u32 sd_pd_index; | ||||
| 	u32 ref_cnt; | ||||
| }; | ||||
| 
 | ||||
| struct i40e_hmc_pd_entry { | ||||
| 	struct i40e_hmc_bp bp; | ||||
| 	u32 sd_index; | ||||
| 	bool valid; | ||||
| }; | ||||
| 
 | ||||
| struct i40e_hmc_pd_table { | ||||
| 	struct i40e_dma_mem pd_page_addr; /* populate to be used by hw */ | ||||
| 	struct i40e_hmc_pd_entry  *pd_entry; /* [512] for sw book keeping */ | ||||
| 	struct i40e_virt_mem pd_entry_virt_mem; /* virt mem for pd_entry */ | ||||
| 
 | ||||
| 	u32 ref_cnt; | ||||
| 	u32 sd_index; | ||||
| }; | ||||
| 
 | ||||
| struct i40e_hmc_sd_entry { | ||||
| 	enum i40e_sd_entry_type entry_type; | ||||
| 	bool valid; | ||||
| 
 | ||||
| 	union { | ||||
| 		struct i40e_hmc_pd_table pd_table; | ||||
| 		struct i40e_hmc_bp bp; | ||||
| 	} u; | ||||
| }; | ||||
| 
 | ||||
| struct i40e_hmc_sd_table { | ||||
| 	struct i40e_virt_mem addr; /* used to track sd_entry allocations */ | ||||
| 	u32 sd_cnt; | ||||
| 	u32 ref_cnt; | ||||
| 	struct i40e_hmc_sd_entry *sd_entry; /* (sd_cnt*512) entries max */ | ||||
| }; | ||||
| 
 | ||||
| struct i40e_hmc_info { | ||||
| 	u32 signature; | ||||
| 	/* equals to pci func num for PF and dynamically allocated for VFs */ | ||||
| 	u8 hmc_fn_id; | ||||
| 	u16 first_sd_index; /* index of the first available SD */ | ||||
| 
 | ||||
| 	/* hmc objects */ | ||||
| 	struct i40e_hmc_obj_info *hmc_obj; | ||||
| 	struct i40e_virt_mem hmc_obj_virt_mem; | ||||
| 	struct i40e_hmc_sd_table sd_table; | ||||
| }; | ||||
| 
 | ||||
| #define I40E_INC_SD_REFCNT(sd_table)	((sd_table)->ref_cnt++) | ||||
| #define I40E_INC_PD_REFCNT(pd_table)	((pd_table)->ref_cnt++) | ||||
| #define I40E_INC_BP_REFCNT(bp)		((bp)->ref_cnt++) | ||||
| 
 | ||||
| #define I40E_DEC_SD_REFCNT(sd_table)	((sd_table)->ref_cnt--) | ||||
| #define I40E_DEC_PD_REFCNT(pd_table)	((pd_table)->ref_cnt--) | ||||
| #define I40E_DEC_BP_REFCNT(bp)		((bp)->ref_cnt--) | ||||
| 
 | ||||
| /**
 | ||||
|  * I40E_SET_PF_SD_ENTRY - marks the sd entry as valid in the hardware | ||||
|  * @hw: pointer to our hw struct | ||||
|  * @pa: pointer to physical address | ||||
|  * @sd_index: segment descriptor index | ||||
|  * @type: if sd entry is direct or paged | ||||
|  **/ | ||||
| #define I40E_SET_PF_SD_ENTRY(hw, pa, sd_index, type)			\ | ||||
| {									\ | ||||
| 	u32 val1, val2, val3;						\ | ||||
| 	val1 = (u32)(upper_32_bits(pa));				\ | ||||
| 	val2 = (u32)(pa) | (I40E_HMC_MAX_BP_COUNT <<			\ | ||||
| 		 I40E_PFHMC_SDDATALOW_PMSDBPCOUNT_SHIFT) |		\ | ||||
| 		((((type) == I40E_SD_TYPE_PAGED) ? 0 : 1) <<		\ | ||||
| 		I40E_PFHMC_SDDATALOW_PMSDTYPE_SHIFT) |			\ | ||||
| 		(1 << I40E_PFHMC_SDDATALOW_PMSDVALID_SHIFT);		\ | ||||
| 	val3 = (sd_index) | (1 << I40E_PFHMC_SDCMD_PMSDWR_SHIFT);	\ | ||||
| 	wr32((hw), I40E_PFHMC_SDDATAHIGH, val1);			\ | ||||
| 	wr32((hw), I40E_PFHMC_SDDATALOW, val2);				\ | ||||
| 	wr32((hw), I40E_PFHMC_SDCMD, val3);				\ | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * I40E_CLEAR_PF_SD_ENTRY - marks the sd entry as invalid in the hardware | ||||
|  * @hw: pointer to our hw struct | ||||
|  * @sd_index: segment descriptor index | ||||
|  * @type: if sd entry is direct or paged | ||||
|  **/ | ||||
| #define I40E_CLEAR_PF_SD_ENTRY(hw, sd_index, type)			\ | ||||
| {									\ | ||||
| 	u32 val2, val3;							\ | ||||
| 	val2 = (I40E_HMC_MAX_BP_COUNT <<				\ | ||||
| 		I40E_PFHMC_SDDATALOW_PMSDBPCOUNT_SHIFT) |		\ | ||||
| 		((((type) == I40E_SD_TYPE_PAGED) ? 0 : 1) <<		\ | ||||
| 		I40E_PFHMC_SDDATALOW_PMSDTYPE_SHIFT);			\ | ||||
| 	val3 = (sd_index) | (1 << I40E_PFHMC_SDCMD_PMSDWR_SHIFT);	\ | ||||
| 	wr32((hw), I40E_PFHMC_SDDATAHIGH, 0);				\ | ||||
| 	wr32((hw), I40E_PFHMC_SDDATALOW, val2);				\ | ||||
| 	wr32((hw), I40E_PFHMC_SDCMD, val3);				\ | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * I40E_INVALIDATE_PF_HMC_PD - Invalidates the pd cache in the hardware | ||||
|  * @hw: pointer to our hw struct | ||||
|  * @sd_idx: segment descriptor index | ||||
|  * @pd_idx: page descriptor index | ||||
|  **/ | ||||
| #define I40E_INVALIDATE_PF_HMC_PD(hw, sd_idx, pd_idx)			\ | ||||
| 	wr32((hw), I40E_PFHMC_PDINV,					\ | ||||
| 	    (((sd_idx) << I40E_PFHMC_PDINV_PMSDIDX_SHIFT) |		\ | ||||
| 	     ((pd_idx) << I40E_PFHMC_PDINV_PMPDIDX_SHIFT))) | ||||
| 
 | ||||
| #define I40E_INVALIDATE_VF_HMC_PD(hw, sd_idx, pd_idx, hmc_fn_id)	   \ | ||||
| 	wr32((hw), I40E_GLHMC_VFPDINV((hmc_fn_id) - I40E_FIRST_VF_FPM_ID), \ | ||||
| 	     (((sd_idx) << I40E_PFHMC_PDINV_PMSDIDX_SHIFT) |		   \ | ||||
| 	      ((pd_idx) << I40E_PFHMC_PDINV_PMPDIDX_SHIFT))) | ||||
| 
 | ||||
| /**
 | ||||
|  * I40E_FIND_SD_INDEX_LIMIT - finds segment descriptor index limit | ||||
|  * @hmc_info: pointer to the HMC configuration information structure | ||||
|  * @type: type of HMC resources we're searching | ||||
|  * @index: starting index for the object | ||||
|  * @cnt: number of objects we're trying to create | ||||
|  * @sd_idx: pointer to return index of the segment descriptor in question | ||||
|  * @sd_limit: pointer to return the maximum number of segment descriptors | ||||
|  * | ||||
|  * This function calculates the segment descriptor index and index limit | ||||
|  * for the resource defined by i40e_hmc_rsrc_type. | ||||
|  **/ | ||||
| #define I40E_FIND_SD_INDEX_LIMIT(hmc_info, type, index, cnt, sd_idx, sd_limit)\ | ||||
| {									\ | ||||
| 	u64 fpm_addr, fpm_limit;					\ | ||||
| 	fpm_addr = (hmc_info)->hmc_obj[(type)].base +			\ | ||||
| 		   (hmc_info)->hmc_obj[(type)].size * (index);		\ | ||||
| 	fpm_limit = fpm_addr + (hmc_info)->hmc_obj[(type)].size * (cnt);\ | ||||
| 	*(sd_idx) = (u32)(fpm_addr / I40E_HMC_DIRECT_BP_SIZE);		\ | ||||
| 	*(sd_limit) = (u32)((fpm_limit - 1) / I40E_HMC_DIRECT_BP_SIZE);	\ | ||||
| 	/* add one more to the limit to correct our range */		\ | ||||
| 	*(sd_limit) += 1;						\ | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * I40E_FIND_PD_INDEX_LIMIT - finds page descriptor index limit | ||||
|  * @hmc_info: pointer to the HMC configuration information struct | ||||
|  * @type: HMC resource type we're examining | ||||
|  * @idx: starting index for the object | ||||
|  * @cnt: number of objects we're trying to create | ||||
|  * @pd_index: pointer to return page descriptor index | ||||
|  * @pd_limit: pointer to return page descriptor index limit | ||||
|  * | ||||
|  * Calculates the page descriptor index and index limit for the resource | ||||
|  * defined by i40e_hmc_rsrc_type. | ||||
|  **/ | ||||
| #define I40E_FIND_PD_INDEX_LIMIT(hmc_info, type, idx, cnt, pd_index, pd_limit)\ | ||||
| {									\ | ||||
| 	u64 fpm_adr, fpm_limit;						\ | ||||
| 	fpm_adr = (hmc_info)->hmc_obj[(type)].base +			\ | ||||
| 		  (hmc_info)->hmc_obj[(type)].size * (idx);		\ | ||||
| 	fpm_limit = fpm_adr + (hmc_info)->hmc_obj[(type)].size * (cnt);	\ | ||||
| 	*(pd_index) = (u32)(fpm_adr / I40E_HMC_PAGED_BP_SIZE);		\ | ||||
| 	*(pd_limit) = (u32)((fpm_limit - 1) / I40E_HMC_PAGED_BP_SIZE);	\ | ||||
| 	/* add one more to the limit to correct our range */		\ | ||||
| 	*(pd_limit) += 1;						\ | ||||
| } | ||||
| i40e_status i40e_add_sd_table_entry(struct i40e_hw *hw, | ||||
| 					      struct i40e_hmc_info *hmc_info, | ||||
| 					      u32 sd_index, | ||||
| 					      enum i40e_sd_entry_type type, | ||||
| 					      u64 direct_mode_sz); | ||||
| 
 | ||||
| i40e_status i40e_add_pd_table_entry(struct i40e_hw *hw, | ||||
| 					      struct i40e_hmc_info *hmc_info, | ||||
| 					      u32 pd_index); | ||||
| i40e_status i40e_remove_pd_bp(struct i40e_hw *hw, | ||||
| 					struct i40e_hmc_info *hmc_info, | ||||
| 					u32 idx, bool is_pf); | ||||
| i40e_status i40e_prep_remove_sd_bp(struct i40e_hmc_info *hmc_info, | ||||
| 					     u32 idx); | ||||
| i40e_status i40e_remove_sd_bp_new(struct i40e_hw *hw, | ||||
| 					    struct i40e_hmc_info *hmc_info, | ||||
| 					    u32 idx, bool is_pf); | ||||
| i40e_status i40e_prep_remove_pd_page(struct i40e_hmc_info *hmc_info, | ||||
| 					       u32 idx); | ||||
| i40e_status i40e_remove_pd_page_new(struct i40e_hw *hw, | ||||
| 					      struct i40e_hmc_info *hmc_info, | ||||
| 					      u32 idx, bool is_pf); | ||||
| 
 | ||||
| #endif /* _I40E_HMC_H_ */ | ||||
							
								
								
									
										165
									
								
								drivers/net/ethernet/intel/i40evf/i40e_lan_hmc.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										165
									
								
								drivers/net/ethernet/intel/i40evf/i40e_lan_hmc.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,165 @@ | ||||
| /*******************************************************************************
 | ||||
|  * | ||||
|  * Intel Ethernet Controller XL710 Family Linux Virtual Function Driver | ||||
|  * Copyright(c) 2013 Intel Corporation. | ||||
|  * | ||||
|  * 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. | ||||
|  * | ||||
|  * 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 | ||||
|  * | ||||
|  ******************************************************************************/ | ||||
| 
 | ||||
| #ifndef _I40E_LAN_HMC_H_ | ||||
| #define _I40E_LAN_HMC_H_ | ||||
| 
 | ||||
| /* forward-declare the HW struct for the compiler */ | ||||
| struct i40e_hw; | ||||
| 
 | ||||
| /* HMC element context information */ | ||||
| 
 | ||||
| /* Rx queue context data */ | ||||
| struct i40e_hmc_obj_rxq { | ||||
| 	u16 head; | ||||
| 	u8  cpuid; | ||||
| 	u64 base; | ||||
| 	u16 qlen; | ||||
| #define I40E_RXQ_CTX_DBUFF_SHIFT 7 | ||||
| 	u8  dbuff; | ||||
| #define I40E_RXQ_CTX_HBUFF_SHIFT 6 | ||||
| 	u8  hbuff; | ||||
| 	u8  dtype; | ||||
| 	u8  dsize; | ||||
| 	u8  crcstrip; | ||||
| 	u8  fc_ena; | ||||
| 	u8  l2tsel; | ||||
| 	u8  hsplit_0; | ||||
| 	u8  hsplit_1; | ||||
| 	u8  showiv; | ||||
| 	u16 rxmax; | ||||
| 	u8  tphrdesc_ena; | ||||
| 	u8  tphwdesc_ena; | ||||
| 	u8  tphdata_ena; | ||||
| 	u8  tphhead_ena; | ||||
| 	u8  lrxqthresh; | ||||
| }; | ||||
| 
 | ||||
| /* Tx queue context data */ | ||||
| struct i40e_hmc_obj_txq { | ||||
| 	u16 head; | ||||
| 	u8  new_context; | ||||
| 	u64 base; | ||||
| 	u8  fc_ena; | ||||
| 	u8  timesync_ena; | ||||
| 	u8  fd_ena; | ||||
| 	u8  alt_vlan_ena; | ||||
| 	u16 thead_wb; | ||||
| 	u16 cpuid; | ||||
| 	u8  head_wb_ena; | ||||
| 	u16 qlen; | ||||
| 	u8  tphrdesc_ena; | ||||
| 	u8  tphrpacket_ena; | ||||
| 	u8  tphwdesc_ena; | ||||
| 	u64 head_wb_addr; | ||||
| 	u32 crc; | ||||
| 	u16 rdylist; | ||||
| 	u8  rdylist_act; | ||||
| }; | ||||
| 
 | ||||
| /* for hsplit_0 field of Rx HMC context */ | ||||
| enum i40e_hmc_obj_rx_hsplit_0 { | ||||
| 	I40E_HMC_OBJ_RX_HSPLIT_0_NO_SPLIT      = 0, | ||||
| 	I40E_HMC_OBJ_RX_HSPLIT_0_SPLIT_L2      = 1, | ||||
| 	I40E_HMC_OBJ_RX_HSPLIT_0_SPLIT_IP      = 2, | ||||
| 	I40E_HMC_OBJ_RX_HSPLIT_0_SPLIT_TCP_UDP = 4, | ||||
| 	I40E_HMC_OBJ_RX_HSPLIT_0_SPLIT_SCTP    = 8, | ||||
| }; | ||||
| 
 | ||||
| /* fcoe_cntx and fcoe_filt are for debugging purpose only */ | ||||
| struct i40e_hmc_obj_fcoe_cntx { | ||||
| 	u32 rsv[32]; | ||||
| }; | ||||
| 
 | ||||
| struct i40e_hmc_obj_fcoe_filt { | ||||
| 	u32 rsv[8]; | ||||
| }; | ||||
| 
 | ||||
| /* Context sizes for LAN objects */ | ||||
| enum i40e_hmc_lan_object_size { | ||||
| 	I40E_HMC_LAN_OBJ_SZ_8   = 0x3, | ||||
| 	I40E_HMC_LAN_OBJ_SZ_16  = 0x4, | ||||
| 	I40E_HMC_LAN_OBJ_SZ_32  = 0x5, | ||||
| 	I40E_HMC_LAN_OBJ_SZ_64  = 0x6, | ||||
| 	I40E_HMC_LAN_OBJ_SZ_128 = 0x7, | ||||
| 	I40E_HMC_LAN_OBJ_SZ_256 = 0x8, | ||||
| 	I40E_HMC_LAN_OBJ_SZ_512 = 0x9, | ||||
| }; | ||||
| 
 | ||||
| #define I40E_HMC_L2OBJ_BASE_ALIGNMENT 512 | ||||
| #define I40E_HMC_OBJ_SIZE_TXQ         128 | ||||
| #define I40E_HMC_OBJ_SIZE_RXQ         32 | ||||
| #define I40E_HMC_OBJ_SIZE_FCOE_CNTX   64 | ||||
| #define I40E_HMC_OBJ_SIZE_FCOE_FILT   64 | ||||
| 
 | ||||
| enum i40e_hmc_lan_rsrc_type { | ||||
| 	I40E_HMC_LAN_FULL  = 0, | ||||
| 	I40E_HMC_LAN_TX    = 1, | ||||
| 	I40E_HMC_LAN_RX    = 2, | ||||
| 	I40E_HMC_FCOE_CTX  = 3, | ||||
| 	I40E_HMC_FCOE_FILT = 4, | ||||
| 	I40E_HMC_LAN_MAX   = 5 | ||||
| }; | ||||
| 
 | ||||
| enum i40e_hmc_model { | ||||
| 	I40E_HMC_MODEL_DIRECT_PREFERRED = 0, | ||||
| 	I40E_HMC_MODEL_DIRECT_ONLY      = 1, | ||||
| 	I40E_HMC_MODEL_PAGED_ONLY       = 2, | ||||
| 	I40E_HMC_MODEL_UNKNOWN, | ||||
| }; | ||||
| 
 | ||||
| struct i40e_hmc_lan_create_obj_info { | ||||
| 	struct i40e_hmc_info *hmc_info; | ||||
| 	u32 rsrc_type; | ||||
| 	u32 start_idx; | ||||
| 	u32 count; | ||||
| 	enum i40e_sd_entry_type entry_type; | ||||
| 	u64 direct_mode_sz; | ||||
| }; | ||||
| 
 | ||||
| struct i40e_hmc_lan_delete_obj_info { | ||||
| 	struct i40e_hmc_info *hmc_info; | ||||
| 	u32 rsrc_type; | ||||
| 	u32 start_idx; | ||||
| 	u32 count; | ||||
| }; | ||||
| 
 | ||||
| i40e_status i40e_init_lan_hmc(struct i40e_hw *hw, u32 txq_num, | ||||
| 					u32 rxq_num, u32 fcoe_cntx_num, | ||||
| 					u32 fcoe_filt_num); | ||||
| i40e_status i40e_configure_lan_hmc(struct i40e_hw *hw, | ||||
| 					     enum i40e_hmc_model model); | ||||
| i40e_status i40e_shutdown_lan_hmc(struct i40e_hw *hw); | ||||
| 
 | ||||
| i40e_status i40e_clear_lan_tx_queue_context(struct i40e_hw *hw, | ||||
| 						      u16 queue); | ||||
| i40e_status i40e_set_lan_tx_queue_context(struct i40e_hw *hw, | ||||
| 						    u16 queue, | ||||
| 						    struct i40e_hmc_obj_txq *s); | ||||
| i40e_status i40e_clear_lan_rx_queue_context(struct i40e_hw *hw, | ||||
| 						      u16 queue); | ||||
| i40e_status i40e_set_lan_rx_queue_context(struct i40e_hw *hw, | ||||
| 						    u16 queue, | ||||
| 						    struct i40e_hmc_obj_rxq *s); | ||||
| 
 | ||||
| #endif /* _I40E_LAN_HMC_H_ */ | ||||
							
								
								
									
										84
									
								
								drivers/net/ethernet/intel/i40evf/i40e_prototype.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										84
									
								
								drivers/net/ethernet/intel/i40evf/i40e_prototype.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,84 @@ | ||||
| /*******************************************************************************
 | ||||
|  * | ||||
|  * Intel Ethernet Controller XL710 Family Linux Virtual Function Driver | ||||
|  * Copyright(c) 2013 Intel Corporation. | ||||
|  * | ||||
|  * 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. | ||||
|  * | ||||
|  * 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 | ||||
|  * | ||||
|  ******************************************************************************/ | ||||
| 
 | ||||
| #ifndef _I40E_PROTOTYPE_H_ | ||||
| #define _I40E_PROTOTYPE_H_ | ||||
| 
 | ||||
| #include "i40e_type.h" | ||||
| #include "i40e_alloc.h" | ||||
| #include "i40e_virtchnl.h" | ||||
| 
 | ||||
| /* Prototypes for shared code functions that are not in
 | ||||
|  * the standard function pointer structures.  These are | ||||
|  * mostly because they are needed even before the init | ||||
|  * has happened and will assist in the early SW and FW | ||||
|  * setup. | ||||
|  */ | ||||
| 
 | ||||
| /* adminq functions */ | ||||
| i40e_status i40evf_init_adminq(struct i40e_hw *hw); | ||||
| i40e_status i40evf_shutdown_adminq(struct i40e_hw *hw); | ||||
| void i40e_adminq_init_ring_data(struct i40e_hw *hw); | ||||
| i40e_status i40evf_clean_arq_element(struct i40e_hw *hw, | ||||
| 					     struct i40e_arq_event_info *e, | ||||
| 					     u16 *events_pending); | ||||
| i40e_status i40evf_asq_send_command(struct i40e_hw *hw, | ||||
| 				struct i40e_aq_desc *desc, | ||||
| 				void *buff, /* can be NULL */ | ||||
| 				u16  buff_size, | ||||
| 				struct i40e_asq_cmd_details *cmd_details); | ||||
| bool i40evf_asq_done(struct i40e_hw *hw); | ||||
| 
 | ||||
| /* debug function for adminq */ | ||||
| void i40evf_debug_aq(struct i40e_hw *hw, | ||||
| 		   enum i40e_debug_mask mask, | ||||
| 		   void *desc, | ||||
| 		   void *buffer); | ||||
| 
 | ||||
| void i40e_idle_aq(struct i40e_hw *hw); | ||||
| void i40evf_resume_aq(struct i40e_hw *hw); | ||||
| bool i40evf_check_asq_alive(struct i40e_hw *hw); | ||||
| i40e_status i40evf_aq_queue_shutdown(struct i40e_hw *hw, | ||||
| 					     bool unloading); | ||||
| 
 | ||||
| i40e_status i40e_set_mac_type(struct i40e_hw *hw); | ||||
| 
 | ||||
| /* prototype for functions used for SW locks */ | ||||
| 
 | ||||
| /* i40e_common for VF drivers*/ | ||||
| void i40e_vf_parse_hw_config(struct i40e_hw *hw, | ||||
| 			     struct i40e_virtchnl_vf_resource *msg); | ||||
| i40e_status i40e_vf_reset(struct i40e_hw *hw); | ||||
| i40e_status i40e_aq_send_msg_to_pf(struct i40e_hw *hw, | ||||
| 				enum i40e_virtchnl_ops v_opcode, | ||||
| 				i40e_status v_retval, | ||||
| 				u8 *msg, u16 msglen, | ||||
| 				struct i40e_asq_cmd_details *cmd_details); | ||||
| i40e_status i40e_set_filter_control(struct i40e_hw *hw, | ||||
| 				struct i40e_filter_control_settings *settings); | ||||
| i40e_status i40e_aq_add_rem_control_packet_filter(struct i40e_hw *hw, | ||||
| 				u8 *mac_addr, u16 ethtype, u16 flags, | ||||
| 				u16 vsi_seid, u16 queue, bool is_add, | ||||
| 				struct i40e_control_filter_stats *stats, | ||||
| 				struct i40e_asq_cmd_details *cmd_details); | ||||
| #endif /* _I40E_PROTOTYPE_H_ */ | ||||
							
								
								
									
										4661
									
								
								drivers/net/ethernet/intel/i40evf/i40e_register.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4661
									
								
								drivers/net/ethernet/intel/i40evf/i40e_register.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										97
									
								
								drivers/net/ethernet/intel/i40evf/i40e_status.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								drivers/net/ethernet/intel/i40evf/i40e_status.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,97 @@ | ||||
| /*******************************************************************************
 | ||||
|  * | ||||
|  * Intel Ethernet Controller XL710 Family Linux Virtual Function Driver | ||||
|  * Copyright(c) 2013 Intel Corporation. | ||||
|  * | ||||
|  * 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. | ||||
|  * | ||||
|  * 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 | ||||
|  * | ||||
|  ******************************************************************************/ | ||||
| 
 | ||||
| #ifndef _I40E_STATUS_H_ | ||||
| #define _I40E_STATUS_H_ | ||||
| 
 | ||||
| /* Error Codes */ | ||||
| enum i40e_status_code { | ||||
| 	I40E_SUCCESS				= 0, | ||||
| 	I40E_ERR_NVM				= -1, | ||||
| 	I40E_ERR_NVM_CHECKSUM			= -2, | ||||
| 	I40E_ERR_PHY				= -3, | ||||
| 	I40E_ERR_CONFIG				= -4, | ||||
| 	I40E_ERR_PARAM				= -5, | ||||
| 	I40E_ERR_MAC_TYPE			= -6, | ||||
| 	I40E_ERR_UNKNOWN_PHY			= -7, | ||||
| 	I40E_ERR_LINK_SETUP			= -8, | ||||
| 	I40E_ERR_ADAPTER_STOPPED		= -9, | ||||
| 	I40E_ERR_INVALID_MAC_ADDR		= -10, | ||||
| 	I40E_ERR_DEVICE_NOT_SUPPORTED		= -11, | ||||
| 	I40E_ERR_MASTER_REQUESTS_PENDING	= -12, | ||||
| 	I40E_ERR_INVALID_LINK_SETTINGS		= -13, | ||||
| 	I40E_ERR_AUTONEG_NOT_COMPLETE		= -14, | ||||
| 	I40E_ERR_RESET_FAILED			= -15, | ||||
| 	I40E_ERR_SWFW_SYNC			= -16, | ||||
| 	I40E_ERR_NO_AVAILABLE_VSI		= -17, | ||||
| 	I40E_ERR_NO_MEMORY			= -18, | ||||
| 	I40E_ERR_BAD_PTR			= -19, | ||||
| 	I40E_ERR_RING_FULL			= -20, | ||||
| 	I40E_ERR_INVALID_PD_ID			= -21, | ||||
| 	I40E_ERR_INVALID_QP_ID			= -22, | ||||
| 	I40E_ERR_INVALID_CQ_ID			= -23, | ||||
| 	I40E_ERR_INVALID_CEQ_ID			= -24, | ||||
| 	I40E_ERR_INVALID_AEQ_ID			= -25, | ||||
| 	I40E_ERR_INVALID_SIZE			= -26, | ||||
| 	I40E_ERR_INVALID_ARP_INDEX		= -27, | ||||
| 	I40E_ERR_INVALID_FPM_FUNC_ID		= -28, | ||||
| 	I40E_ERR_QP_INVALID_MSG_SIZE		= -29, | ||||
| 	I40E_ERR_QP_TOOMANY_WRS_POSTED		= -30, | ||||
| 	I40E_ERR_INVALID_FRAG_COUNT		= -31, | ||||
| 	I40E_ERR_QUEUE_EMPTY			= -32, | ||||
| 	I40E_ERR_INVALID_ALIGNMENT		= -33, | ||||
| 	I40E_ERR_FLUSHED_QUEUE			= -34, | ||||
| 	I40E_ERR_INVALID_PUSH_PAGE_INDEX	= -35, | ||||
| 	I40E_ERR_INVALID_IMM_DATA_SIZE		= -36, | ||||
| 	I40E_ERR_TIMEOUT			= -37, | ||||
| 	I40E_ERR_OPCODE_MISMATCH		= -38, | ||||
| 	I40E_ERR_CQP_COMPL_ERROR		= -39, | ||||
| 	I40E_ERR_INVALID_VF_ID			= -40, | ||||
| 	I40E_ERR_INVALID_HMCFN_ID		= -41, | ||||
| 	I40E_ERR_BACKING_PAGE_ERROR		= -42, | ||||
| 	I40E_ERR_NO_PBLCHUNKS_AVAILABLE		= -43, | ||||
| 	I40E_ERR_INVALID_PBLE_INDEX		= -44, | ||||
| 	I40E_ERR_INVALID_SD_INDEX		= -45, | ||||
| 	I40E_ERR_INVALID_PAGE_DESC_INDEX	= -46, | ||||
| 	I40E_ERR_INVALID_SD_TYPE		= -47, | ||||
| 	I40E_ERR_MEMCPY_FAILED			= -48, | ||||
| 	I40E_ERR_INVALID_HMC_OBJ_INDEX		= -49, | ||||
| 	I40E_ERR_INVALID_HMC_OBJ_COUNT		= -50, | ||||
| 	I40E_ERR_INVALID_SRQ_ARM_LIMIT		= -51, | ||||
| 	I40E_ERR_SRQ_ENABLED			= -52, | ||||
| 	I40E_ERR_ADMIN_QUEUE_ERROR		= -53, | ||||
| 	I40E_ERR_ADMIN_QUEUE_TIMEOUT		= -54, | ||||
| 	I40E_ERR_BUF_TOO_SHORT			= -55, | ||||
| 	I40E_ERR_ADMIN_QUEUE_FULL		= -56, | ||||
| 	I40E_ERR_ADMIN_QUEUE_NO_WORK		= -57, | ||||
| 	I40E_ERR_BAD_IWARP_CQE			= -58, | ||||
| 	I40E_ERR_NVM_BLANK_MODE			= -59, | ||||
| 	I40E_ERR_NOT_IMPLEMENTED		= -60, | ||||
| 	I40E_ERR_PE_DOORBELL_NOT_ENABLED	= -61, | ||||
| 	I40E_ERR_DIAG_TEST_FAILED		= -62, | ||||
| 	I40E_ERR_NOT_READY			= -63, | ||||
| 	I40E_NOT_SUPPORTED			= -64, | ||||
| 	I40E_ERR_FIRMWARE_API_VERSION		= -65, | ||||
| }; | ||||
| 
 | ||||
| #endif /* _I40E_STATUS_H_ */ | ||||
							
								
								
									
										1152
									
								
								drivers/net/ethernet/intel/i40evf/i40e_type.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1152
									
								
								drivers/net/ethernet/intel/i40evf/i40e_type.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Greg Rose
						Greg Rose