xref: /freebsd/sys/dev/igc/igc_api.c (revision 7ee310c80ea7b336972f53cc48b8c3d03029941e)
1517904deSPeter Grehan /*-
2517904deSPeter Grehan  * Copyright 2021 Intel Corp
3517904deSPeter Grehan  * Copyright 2021 Rubicon Communications, LLC (Netgate)
4517904deSPeter Grehan  * SPDX-License-Identifier: BSD-3-Clause
5517904deSPeter Grehan  */
6517904deSPeter Grehan 
7517904deSPeter Grehan #include <sys/cdefs.h>
8517904deSPeter Grehan #include "igc_api.h"
9517904deSPeter Grehan 
10517904deSPeter Grehan /**
11517904deSPeter Grehan  *  igc_init_mac_params - Initialize MAC function pointers
12517904deSPeter Grehan  *  @hw: pointer to the HW structure
13517904deSPeter Grehan  *
14517904deSPeter Grehan  *  This function initializes the function pointers for the MAC
15517904deSPeter Grehan  *  set of functions.  Called by drivers or by igc_setup_init_funcs.
16517904deSPeter Grehan  **/
igc_init_mac_params(struct igc_hw * hw)17517904deSPeter Grehan s32 igc_init_mac_params(struct igc_hw *hw)
18517904deSPeter Grehan {
19517904deSPeter Grehan 	s32 ret_val = IGC_SUCCESS;
20517904deSPeter Grehan 
21517904deSPeter Grehan 	if (hw->mac.ops.init_params) {
22517904deSPeter Grehan 		ret_val = hw->mac.ops.init_params(hw);
23517904deSPeter Grehan 		if (ret_val) {
24517904deSPeter Grehan 			DEBUGOUT("MAC Initialization Error\n");
25517904deSPeter Grehan 			goto out;
26517904deSPeter Grehan 		}
27517904deSPeter Grehan 	} else {
28517904deSPeter Grehan 		DEBUGOUT("mac.init_mac_params was NULL\n");
29517904deSPeter Grehan 		ret_val = -IGC_ERR_CONFIG;
30517904deSPeter Grehan 	}
31517904deSPeter Grehan 
32517904deSPeter Grehan out:
33517904deSPeter Grehan 	return ret_val;
34517904deSPeter Grehan }
35517904deSPeter Grehan 
36517904deSPeter Grehan /**
37517904deSPeter Grehan  *  igc_init_nvm_params - Initialize NVM function pointers
38517904deSPeter Grehan  *  @hw: pointer to the HW structure
39517904deSPeter Grehan  *
40517904deSPeter Grehan  *  This function initializes the function pointers for the NVM
41517904deSPeter Grehan  *  set of functions.  Called by drivers or by igc_setup_init_funcs.
42517904deSPeter Grehan  **/
igc_init_nvm_params(struct igc_hw * hw)43517904deSPeter Grehan s32 igc_init_nvm_params(struct igc_hw *hw)
44517904deSPeter Grehan {
45517904deSPeter Grehan 	s32 ret_val = IGC_SUCCESS;
46517904deSPeter Grehan 
47517904deSPeter Grehan 	if (hw->nvm.ops.init_params) {
48517904deSPeter Grehan 		ret_val = hw->nvm.ops.init_params(hw);
49517904deSPeter Grehan 		if (ret_val) {
50517904deSPeter Grehan 			DEBUGOUT("NVM Initialization Error\n");
51517904deSPeter Grehan 			goto out;
52517904deSPeter Grehan 		}
53517904deSPeter Grehan 	} else {
54517904deSPeter Grehan 		DEBUGOUT("nvm.init_nvm_params was NULL\n");
55517904deSPeter Grehan 		ret_val = -IGC_ERR_CONFIG;
56517904deSPeter Grehan 	}
57517904deSPeter Grehan 
58517904deSPeter Grehan out:
59517904deSPeter Grehan 	return ret_val;
60517904deSPeter Grehan }
61517904deSPeter Grehan 
62517904deSPeter Grehan /**
63517904deSPeter Grehan  *  igc_init_phy_params - Initialize PHY function pointers
64517904deSPeter Grehan  *  @hw: pointer to the HW structure
65517904deSPeter Grehan  *
66517904deSPeter Grehan  *  This function initializes the function pointers for the PHY
67517904deSPeter Grehan  *  set of functions.  Called by drivers or by igc_setup_init_funcs.
68517904deSPeter Grehan  **/
igc_init_phy_params(struct igc_hw * hw)69517904deSPeter Grehan s32 igc_init_phy_params(struct igc_hw *hw)
70517904deSPeter Grehan {
71517904deSPeter Grehan 	s32 ret_val = IGC_SUCCESS;
72517904deSPeter Grehan 
73517904deSPeter Grehan 	if (hw->phy.ops.init_params) {
74517904deSPeter Grehan 		ret_val = hw->phy.ops.init_params(hw);
75517904deSPeter Grehan 		if (ret_val) {
76517904deSPeter Grehan 			DEBUGOUT("PHY Initialization Error\n");
77517904deSPeter Grehan 			goto out;
78517904deSPeter Grehan 		}
79517904deSPeter Grehan 	} else {
80517904deSPeter Grehan 		DEBUGOUT("phy.init_phy_params was NULL\n");
81517904deSPeter Grehan 		ret_val =  -IGC_ERR_CONFIG;
82517904deSPeter Grehan 	}
83517904deSPeter Grehan 
84517904deSPeter Grehan out:
85517904deSPeter Grehan 	return ret_val;
86517904deSPeter Grehan }
87517904deSPeter Grehan 
88517904deSPeter Grehan /**
89517904deSPeter Grehan  *  igc_set_mac_type - Sets MAC type
90517904deSPeter Grehan  *  @hw: pointer to the HW structure
91517904deSPeter Grehan  *
92517904deSPeter Grehan  *  This function sets the mac type of the adapter based on the
93517904deSPeter Grehan  *  device ID stored in the hw structure.
94517904deSPeter Grehan  *  MUST BE FIRST FUNCTION CALLED (explicitly or through
95517904deSPeter Grehan  *  igc_setup_init_funcs()).
96517904deSPeter Grehan  **/
igc_set_mac_type(struct igc_hw * hw)97517904deSPeter Grehan s32 igc_set_mac_type(struct igc_hw *hw)
98517904deSPeter Grehan {
99517904deSPeter Grehan 	struct igc_mac_info *mac = &hw->mac;
100517904deSPeter Grehan 	s32 ret_val = IGC_SUCCESS;
101517904deSPeter Grehan 
102517904deSPeter Grehan 	DEBUGFUNC("igc_set_mac_type");
103517904deSPeter Grehan 
104517904deSPeter Grehan 	switch (hw->device_id) {
105517904deSPeter Grehan 	case IGC_DEV_ID_I225_LM:
106517904deSPeter Grehan 	case IGC_DEV_ID_I225_V:
107517904deSPeter Grehan 	case IGC_DEV_ID_I225_K:
108517904deSPeter Grehan 	case IGC_DEV_ID_I225_I:
109517904deSPeter Grehan 	case IGC_DEV_ID_I220_V:
110517904deSPeter Grehan 	case IGC_DEV_ID_I225_K2:
111517904deSPeter Grehan 	case IGC_DEV_ID_I225_LMVP:
112*7ee310c8SEric Joyner 	case IGC_DEV_ID_I226_K:
113*7ee310c8SEric Joyner 	case IGC_DEV_ID_I226_LMVP:
114517904deSPeter Grehan 	case IGC_DEV_ID_I225_IT:
115517904deSPeter Grehan 	case IGC_DEV_ID_I226_LM:
116517904deSPeter Grehan 	case IGC_DEV_ID_I226_V:
117517904deSPeter Grehan 	case IGC_DEV_ID_I226_IT:
118517904deSPeter Grehan 	case IGC_DEV_ID_I221_V:
119517904deSPeter Grehan 	case IGC_DEV_ID_I226_BLANK_NVM:
120517904deSPeter Grehan 	case IGC_DEV_ID_I225_BLANK_NVM:
121517904deSPeter Grehan 		mac->type = igc_i225;
122517904deSPeter Grehan 		break;
123517904deSPeter Grehan 	default:
124517904deSPeter Grehan 		/* Should never have loaded on this device */
125517904deSPeter Grehan 		ret_val = -IGC_ERR_MAC_INIT;
126517904deSPeter Grehan 		break;
127517904deSPeter Grehan 	}
128517904deSPeter Grehan 
129517904deSPeter Grehan 	return ret_val;
130517904deSPeter Grehan }
131517904deSPeter Grehan 
132517904deSPeter Grehan /**
133517904deSPeter Grehan  *  igc_setup_init_funcs - Initializes function pointers
134517904deSPeter Grehan  *  @hw: pointer to the HW structure
135517904deSPeter Grehan  *  @init_device: true will initialize the rest of the function pointers
136517904deSPeter Grehan  *		  getting the device ready for use.  FALSE will only set
137517904deSPeter Grehan  *		  MAC type and the function pointers for the other init
138517904deSPeter Grehan  *		  functions.  Passing FALSE will not generate any hardware
139517904deSPeter Grehan  *		  reads or writes.
140517904deSPeter Grehan  *
141517904deSPeter Grehan  *  This function must be called by a driver in order to use the rest
142517904deSPeter Grehan  *  of the 'shared' code files. Called by drivers only.
143517904deSPeter Grehan  **/
igc_setup_init_funcs(struct igc_hw * hw,bool init_device)144517904deSPeter Grehan s32 igc_setup_init_funcs(struct igc_hw *hw, bool init_device)
145517904deSPeter Grehan {
146517904deSPeter Grehan 	s32 ret_val;
147517904deSPeter Grehan 
148517904deSPeter Grehan 	/* Can't do much good without knowing the MAC type. */
149517904deSPeter Grehan 	ret_val = igc_set_mac_type(hw);
150517904deSPeter Grehan 	if (ret_val) {
151517904deSPeter Grehan 		DEBUGOUT("ERROR: MAC type could not be set properly.\n");
152517904deSPeter Grehan 		goto out;
153517904deSPeter Grehan 	}
154517904deSPeter Grehan 
155517904deSPeter Grehan 	if (!hw->hw_addr) {
156517904deSPeter Grehan 		DEBUGOUT("ERROR: Registers not mapped\n");
157517904deSPeter Grehan 		ret_val = -IGC_ERR_CONFIG;
158517904deSPeter Grehan 		goto out;
159517904deSPeter Grehan 	}
160517904deSPeter Grehan 
161517904deSPeter Grehan 	/*
162517904deSPeter Grehan 	 * Init function pointers to generic implementations. We do this first
163517904deSPeter Grehan 	 * allowing a driver module to override it afterward.
164517904deSPeter Grehan 	 */
165517904deSPeter Grehan 	igc_init_mac_ops_generic(hw);
166517904deSPeter Grehan 	igc_init_phy_ops_generic(hw);
167517904deSPeter Grehan 	igc_init_nvm_ops_generic(hw);
168517904deSPeter Grehan 
169517904deSPeter Grehan 	/*
170517904deSPeter Grehan 	 * Set up the init function pointers. These are functions within the
171517904deSPeter Grehan 	 * adapter family file that sets up function pointers for the rest of
172517904deSPeter Grehan 	 * the functions in that family.
173517904deSPeter Grehan 	 */
174517904deSPeter Grehan 	switch (hw->mac.type) {
175517904deSPeter Grehan 	case igc_i225:
176517904deSPeter Grehan 		igc_init_function_pointers_i225(hw);
177517904deSPeter Grehan 		break;
178517904deSPeter Grehan 	default:
179517904deSPeter Grehan 		DEBUGOUT("Hardware not supported\n");
180517904deSPeter Grehan 		ret_val = -IGC_ERR_CONFIG;
181517904deSPeter Grehan 		break;
182517904deSPeter Grehan 	}
183517904deSPeter Grehan 
184517904deSPeter Grehan 	/*
185517904deSPeter Grehan 	 * Initialize the rest of the function pointers. These require some
186517904deSPeter Grehan 	 * register reads/writes in some cases.
187517904deSPeter Grehan 	 */
188517904deSPeter Grehan 	if (!(ret_val) && init_device) {
189517904deSPeter Grehan 		ret_val = igc_init_mac_params(hw);
190517904deSPeter Grehan 		if (ret_val)
191517904deSPeter Grehan 			goto out;
192517904deSPeter Grehan 
193517904deSPeter Grehan 		ret_val = igc_init_nvm_params(hw);
194517904deSPeter Grehan 		if (ret_val)
195517904deSPeter Grehan 			goto out;
196517904deSPeter Grehan 
197517904deSPeter Grehan 		ret_val = igc_init_phy_params(hw);
198517904deSPeter Grehan 		if (ret_val)
199517904deSPeter Grehan 			goto out;
200517904deSPeter Grehan 	}
201517904deSPeter Grehan 
202517904deSPeter Grehan out:
203517904deSPeter Grehan 	return ret_val;
204517904deSPeter Grehan }
205517904deSPeter Grehan 
206517904deSPeter Grehan /**
207517904deSPeter Grehan  *  igc_get_bus_info - Obtain bus information for adapter
208517904deSPeter Grehan  *  @hw: pointer to the HW structure
209517904deSPeter Grehan  *
210517904deSPeter Grehan  *  This will obtain information about the HW bus for which the
211517904deSPeter Grehan  *  adapter is attached and stores it in the hw structure. This is a
212517904deSPeter Grehan  *  function pointer entry point called by drivers.
213517904deSPeter Grehan  **/
igc_get_bus_info(struct igc_hw * hw)214517904deSPeter Grehan s32 igc_get_bus_info(struct igc_hw *hw)
215517904deSPeter Grehan {
216517904deSPeter Grehan 	if (hw->mac.ops.get_bus_info)
217517904deSPeter Grehan 		return hw->mac.ops.get_bus_info(hw);
218517904deSPeter Grehan 
219517904deSPeter Grehan 	return IGC_SUCCESS;
220517904deSPeter Grehan }
221517904deSPeter Grehan 
222517904deSPeter Grehan /**
223517904deSPeter Grehan  *  igc_clear_vfta - Clear VLAN filter table
224517904deSPeter Grehan  *  @hw: pointer to the HW structure
225517904deSPeter Grehan  *
226517904deSPeter Grehan  *  This clears the VLAN filter table on the adapter. This is a function
227517904deSPeter Grehan  *  pointer entry point called by drivers.
228517904deSPeter Grehan  **/
igc_clear_vfta(struct igc_hw * hw)229517904deSPeter Grehan void igc_clear_vfta(struct igc_hw *hw)
230517904deSPeter Grehan {
231517904deSPeter Grehan 	if (hw->mac.ops.clear_vfta)
232517904deSPeter Grehan 		hw->mac.ops.clear_vfta(hw);
233517904deSPeter Grehan }
234517904deSPeter Grehan 
235517904deSPeter Grehan /**
236517904deSPeter Grehan  *  igc_write_vfta - Write value to VLAN filter table
237517904deSPeter Grehan  *  @hw: pointer to the HW structure
238517904deSPeter Grehan  *  @offset: the 32-bit offset in which to write the value to.
239517904deSPeter Grehan  *  @value: the 32-bit value to write at location offset.
240517904deSPeter Grehan  *
241517904deSPeter Grehan  *  This writes a 32-bit value to a 32-bit offset in the VLAN filter
242517904deSPeter Grehan  *  table. This is a function pointer entry point called by drivers.
243517904deSPeter Grehan  **/
igc_write_vfta(struct igc_hw * hw,u32 offset,u32 value)244517904deSPeter Grehan void igc_write_vfta(struct igc_hw *hw, u32 offset, u32 value)
245517904deSPeter Grehan {
246517904deSPeter Grehan 	if (hw->mac.ops.write_vfta)
247517904deSPeter Grehan 		hw->mac.ops.write_vfta(hw, offset, value);
248517904deSPeter Grehan }
249517904deSPeter Grehan 
250517904deSPeter Grehan /**
251517904deSPeter Grehan  *  igc_update_mc_addr_list - Update Multicast addresses
252517904deSPeter Grehan  *  @hw: pointer to the HW structure
253517904deSPeter Grehan  *  @mc_addr_list: array of multicast addresses to program
254517904deSPeter Grehan  *  @mc_addr_count: number of multicast addresses to program
255517904deSPeter Grehan  *
256517904deSPeter Grehan  *  Updates the Multicast Table Array.
257517904deSPeter Grehan  *  The caller must have a packed mc_addr_list of multicast addresses.
258517904deSPeter Grehan  **/
igc_update_mc_addr_list(struct igc_hw * hw,u8 * mc_addr_list,u32 mc_addr_count)259517904deSPeter Grehan void igc_update_mc_addr_list(struct igc_hw *hw, u8 *mc_addr_list,
260517904deSPeter Grehan 			       u32 mc_addr_count)
261517904deSPeter Grehan {
262517904deSPeter Grehan 	if (hw->mac.ops.update_mc_addr_list)
263517904deSPeter Grehan 		hw->mac.ops.update_mc_addr_list(hw, mc_addr_list,
264517904deSPeter Grehan 						mc_addr_count);
265517904deSPeter Grehan }
266517904deSPeter Grehan 
267517904deSPeter Grehan /**
268517904deSPeter Grehan  *  igc_force_mac_fc - Force MAC flow control
269517904deSPeter Grehan  *  @hw: pointer to the HW structure
270517904deSPeter Grehan  *
271517904deSPeter Grehan  *  Force the MAC's flow control settings. Currently no func pointer exists
272517904deSPeter Grehan  *  and all implementations are handled in the generic version of this
273517904deSPeter Grehan  *  function.
274517904deSPeter Grehan  **/
igc_force_mac_fc(struct igc_hw * hw)275517904deSPeter Grehan s32 igc_force_mac_fc(struct igc_hw *hw)
276517904deSPeter Grehan {
277517904deSPeter Grehan 	return igc_force_mac_fc_generic(hw);
278517904deSPeter Grehan }
279517904deSPeter Grehan 
280517904deSPeter Grehan /**
281517904deSPeter Grehan  *  igc_check_for_link - Check/Store link connection
282517904deSPeter Grehan  *  @hw: pointer to the HW structure
283517904deSPeter Grehan  *
284517904deSPeter Grehan  *  This checks the link condition of the adapter and stores the
285517904deSPeter Grehan  *  results in the hw->mac structure. This is a function pointer entry
286517904deSPeter Grehan  *  point called by drivers.
287517904deSPeter Grehan  **/
igc_check_for_link(struct igc_hw * hw)288517904deSPeter Grehan s32 igc_check_for_link(struct igc_hw *hw)
289517904deSPeter Grehan {
290517904deSPeter Grehan 	if (hw->mac.ops.check_for_link)
291517904deSPeter Grehan 		return hw->mac.ops.check_for_link(hw);
292517904deSPeter Grehan 
293517904deSPeter Grehan 	return -IGC_ERR_CONFIG;
294517904deSPeter Grehan }
295517904deSPeter Grehan 
296517904deSPeter Grehan /**
297517904deSPeter Grehan  *  igc_reset_hw - Reset hardware
298517904deSPeter Grehan  *  @hw: pointer to the HW structure
299517904deSPeter Grehan  *
300517904deSPeter Grehan  *  This resets the hardware into a known state. This is a function pointer
301517904deSPeter Grehan  *  entry point called by drivers.
302517904deSPeter Grehan  **/
igc_reset_hw(struct igc_hw * hw)303517904deSPeter Grehan s32 igc_reset_hw(struct igc_hw *hw)
304517904deSPeter Grehan {
305517904deSPeter Grehan 	if (hw->mac.ops.reset_hw)
306517904deSPeter Grehan 		return hw->mac.ops.reset_hw(hw);
307517904deSPeter Grehan 
308517904deSPeter Grehan 	return -IGC_ERR_CONFIG;
309517904deSPeter Grehan }
310517904deSPeter Grehan 
311517904deSPeter Grehan /**
312517904deSPeter Grehan  *  igc_init_hw - Initialize hardware
313517904deSPeter Grehan  *  @hw: pointer to the HW structure
314517904deSPeter Grehan  *
315517904deSPeter Grehan  *  This inits the hardware readying it for operation. This is a function
316517904deSPeter Grehan  *  pointer entry point called by drivers.
317517904deSPeter Grehan  **/
igc_init_hw(struct igc_hw * hw)318517904deSPeter Grehan s32 igc_init_hw(struct igc_hw *hw)
319517904deSPeter Grehan {
320517904deSPeter Grehan 	if (hw->mac.ops.init_hw)
321517904deSPeter Grehan 		return hw->mac.ops.init_hw(hw);
322517904deSPeter Grehan 
323517904deSPeter Grehan 	return -IGC_ERR_CONFIG;
324517904deSPeter Grehan }
325517904deSPeter Grehan 
326517904deSPeter Grehan /**
327517904deSPeter Grehan  *  igc_setup_link - Configures link and flow control
328517904deSPeter Grehan  *  @hw: pointer to the HW structure
329517904deSPeter Grehan  *
330517904deSPeter Grehan  *  This configures link and flow control settings for the adapter. This
331517904deSPeter Grehan  *  is a function pointer entry point called by drivers. While modules can
332517904deSPeter Grehan  *  also call this, they probably call their own version of this function.
333517904deSPeter Grehan  **/
igc_setup_link(struct igc_hw * hw)334517904deSPeter Grehan s32 igc_setup_link(struct igc_hw *hw)
335517904deSPeter Grehan {
336517904deSPeter Grehan 	if (hw->mac.ops.setup_link)
337517904deSPeter Grehan 		return hw->mac.ops.setup_link(hw);
338517904deSPeter Grehan 
339517904deSPeter Grehan 	return -IGC_ERR_CONFIG;
340517904deSPeter Grehan }
341517904deSPeter Grehan 
342517904deSPeter Grehan /**
343517904deSPeter Grehan  *  igc_get_speed_and_duplex - Returns current speed and duplex
344517904deSPeter Grehan  *  @hw: pointer to the HW structure
345517904deSPeter Grehan  *  @speed: pointer to a 16-bit value to store the speed
346517904deSPeter Grehan  *  @duplex: pointer to a 16-bit value to store the duplex.
347517904deSPeter Grehan  *
348517904deSPeter Grehan  *  This returns the speed and duplex of the adapter in the two 'out'
349517904deSPeter Grehan  *  variables passed in. This is a function pointer entry point called
350517904deSPeter Grehan  *  by drivers.
351517904deSPeter Grehan  **/
igc_get_speed_and_duplex(struct igc_hw * hw,u16 * speed,u16 * duplex)352517904deSPeter Grehan s32 igc_get_speed_and_duplex(struct igc_hw *hw, u16 *speed, u16 *duplex)
353517904deSPeter Grehan {
354517904deSPeter Grehan 	if (hw->mac.ops.get_link_up_info)
355517904deSPeter Grehan 		return hw->mac.ops.get_link_up_info(hw, speed, duplex);
356517904deSPeter Grehan 
357517904deSPeter Grehan 	return -IGC_ERR_CONFIG;
358517904deSPeter Grehan }
359517904deSPeter Grehan 
360517904deSPeter Grehan /**
361517904deSPeter Grehan  *  igc_disable_pcie_master - Disable PCI-Express master access
362517904deSPeter Grehan  *  @hw: pointer to the HW structure
363517904deSPeter Grehan  *
364517904deSPeter Grehan  *  Disables PCI-Express master access and verifies there are no pending
365517904deSPeter Grehan  *  requests. Currently no func pointer exists and all implementations are
366517904deSPeter Grehan  *  handled in the generic version of this function.
367517904deSPeter Grehan  **/
igc_disable_pcie_master(struct igc_hw * hw)368517904deSPeter Grehan s32 igc_disable_pcie_master(struct igc_hw *hw)
369517904deSPeter Grehan {
370517904deSPeter Grehan 	return igc_disable_pcie_master_generic(hw);
371517904deSPeter Grehan }
372517904deSPeter Grehan 
373517904deSPeter Grehan /**
374517904deSPeter Grehan  *  igc_config_collision_dist - Configure collision distance
375517904deSPeter Grehan  *  @hw: pointer to the HW structure
376517904deSPeter Grehan  *
377517904deSPeter Grehan  *  Configures the collision distance to the default value and is used
378517904deSPeter Grehan  *  during link setup.
379517904deSPeter Grehan  **/
igc_config_collision_dist(struct igc_hw * hw)380517904deSPeter Grehan void igc_config_collision_dist(struct igc_hw *hw)
381517904deSPeter Grehan {
382517904deSPeter Grehan 	if (hw->mac.ops.config_collision_dist)
383517904deSPeter Grehan 		hw->mac.ops.config_collision_dist(hw);
384517904deSPeter Grehan }
385517904deSPeter Grehan 
386517904deSPeter Grehan /**
387517904deSPeter Grehan  *  igc_rar_set - Sets a receive address register
388517904deSPeter Grehan  *  @hw: pointer to the HW structure
389517904deSPeter Grehan  *  @addr: address to set the RAR to
390517904deSPeter Grehan  *  @index: the RAR to set
391517904deSPeter Grehan  *
392517904deSPeter Grehan  *  Sets a Receive Address Register (RAR) to the specified address.
393517904deSPeter Grehan  **/
igc_rar_set(struct igc_hw * hw,u8 * addr,u32 index)394517904deSPeter Grehan int igc_rar_set(struct igc_hw *hw, u8 *addr, u32 index)
395517904deSPeter Grehan {
396517904deSPeter Grehan 	if (hw->mac.ops.rar_set)
397517904deSPeter Grehan 		return hw->mac.ops.rar_set(hw, addr, index);
398517904deSPeter Grehan 
399517904deSPeter Grehan 	return IGC_SUCCESS;
400517904deSPeter Grehan }
401517904deSPeter Grehan 
402517904deSPeter Grehan /**
403517904deSPeter Grehan  *  igc_validate_mdi_setting - Ensures valid MDI/MDIX SW state
404517904deSPeter Grehan  *  @hw: pointer to the HW structure
405517904deSPeter Grehan  *
406517904deSPeter Grehan  *  Ensures that the MDI/MDIX SW state is valid.
407517904deSPeter Grehan  **/
igc_validate_mdi_setting(struct igc_hw * hw)408517904deSPeter Grehan s32 igc_validate_mdi_setting(struct igc_hw *hw)
409517904deSPeter Grehan {
410517904deSPeter Grehan 	if (hw->mac.ops.validate_mdi_setting)
411517904deSPeter Grehan 		return hw->mac.ops.validate_mdi_setting(hw);
412517904deSPeter Grehan 
413517904deSPeter Grehan 	return IGC_SUCCESS;
414517904deSPeter Grehan }
415517904deSPeter Grehan 
416517904deSPeter Grehan /**
417517904deSPeter Grehan  *  igc_hash_mc_addr - Determines address location in multicast table
418517904deSPeter Grehan  *  @hw: pointer to the HW structure
419517904deSPeter Grehan  *  @mc_addr: Multicast address to hash.
420517904deSPeter Grehan  *
421517904deSPeter Grehan  *  This hashes an address to determine its location in the multicast
422517904deSPeter Grehan  *  table. Currently no func pointer exists and all implementations
423517904deSPeter Grehan  *  are handled in the generic version of this function.
424517904deSPeter Grehan  **/
igc_hash_mc_addr(struct igc_hw * hw,u8 * mc_addr)425517904deSPeter Grehan u32 igc_hash_mc_addr(struct igc_hw *hw, u8 *mc_addr)
426517904deSPeter Grehan {
427517904deSPeter Grehan 	return igc_hash_mc_addr_generic(hw, mc_addr);
428517904deSPeter Grehan }
429517904deSPeter Grehan 
430517904deSPeter Grehan /**
431517904deSPeter Grehan  *  igc_check_reset_block - Verifies PHY can be reset
432517904deSPeter Grehan  *  @hw: pointer to the HW structure
433517904deSPeter Grehan  *
434517904deSPeter Grehan  *  Checks if the PHY is in a state that can be reset or if manageability
435517904deSPeter Grehan  *  has it tied up. This is a function pointer entry point called by drivers.
436517904deSPeter Grehan  **/
igc_check_reset_block(struct igc_hw * hw)437517904deSPeter Grehan s32 igc_check_reset_block(struct igc_hw *hw)
438517904deSPeter Grehan {
439517904deSPeter Grehan 	if (hw->phy.ops.check_reset_block)
440517904deSPeter Grehan 		return hw->phy.ops.check_reset_block(hw);
441517904deSPeter Grehan 
442517904deSPeter Grehan 	return IGC_SUCCESS;
443517904deSPeter Grehan }
444517904deSPeter Grehan 
445517904deSPeter Grehan /**
446517904deSPeter Grehan  *  igc_read_phy_reg - Reads PHY register
447517904deSPeter Grehan  *  @hw: pointer to the HW structure
448517904deSPeter Grehan  *  @offset: the register to read
449517904deSPeter Grehan  *  @data: the buffer to store the 16-bit read.
450517904deSPeter Grehan  *
451517904deSPeter Grehan  *  Reads the PHY register and returns the value in data.
452517904deSPeter Grehan  *  This is a function pointer entry point called by drivers.
453517904deSPeter Grehan  **/
igc_read_phy_reg(struct igc_hw * hw,u32 offset,u16 * data)454517904deSPeter Grehan s32 igc_read_phy_reg(struct igc_hw *hw, u32 offset, u16 *data)
455517904deSPeter Grehan {
456517904deSPeter Grehan 	if (hw->phy.ops.read_reg)
457517904deSPeter Grehan 		return hw->phy.ops.read_reg(hw, offset, data);
458517904deSPeter Grehan 
459517904deSPeter Grehan 	return IGC_SUCCESS;
460517904deSPeter Grehan }
461517904deSPeter Grehan 
462517904deSPeter Grehan /**
463517904deSPeter Grehan  *  igc_write_phy_reg - Writes PHY register
464517904deSPeter Grehan  *  @hw: pointer to the HW structure
465517904deSPeter Grehan  *  @offset: the register to write
466517904deSPeter Grehan  *  @data: the value to write.
467517904deSPeter Grehan  *
468517904deSPeter Grehan  *  Writes the PHY register at offset with the value in data.
469517904deSPeter Grehan  *  This is a function pointer entry point called by drivers.
470517904deSPeter Grehan  **/
igc_write_phy_reg(struct igc_hw * hw,u32 offset,u16 data)471517904deSPeter Grehan s32 igc_write_phy_reg(struct igc_hw *hw, u32 offset, u16 data)
472517904deSPeter Grehan {
473517904deSPeter Grehan 	if (hw->phy.ops.write_reg)
474517904deSPeter Grehan 		return hw->phy.ops.write_reg(hw, offset, data);
475517904deSPeter Grehan 
476517904deSPeter Grehan 	return IGC_SUCCESS;
477517904deSPeter Grehan }
478517904deSPeter Grehan 
479517904deSPeter Grehan /**
480517904deSPeter Grehan  *  igc_release_phy - Generic release PHY
481517904deSPeter Grehan  *  @hw: pointer to the HW structure
482517904deSPeter Grehan  *
483517904deSPeter Grehan  *  Return if silicon family does not require a semaphore when accessing the
484517904deSPeter Grehan  *  PHY.
485517904deSPeter Grehan  **/
igc_release_phy(struct igc_hw * hw)486517904deSPeter Grehan void igc_release_phy(struct igc_hw *hw)
487517904deSPeter Grehan {
488517904deSPeter Grehan 	if (hw->phy.ops.release)
489517904deSPeter Grehan 		hw->phy.ops.release(hw);
490517904deSPeter Grehan }
491517904deSPeter Grehan 
492517904deSPeter Grehan /**
493517904deSPeter Grehan  *  igc_acquire_phy - Generic acquire PHY
494517904deSPeter Grehan  *  @hw: pointer to the HW structure
495517904deSPeter Grehan  *
496517904deSPeter Grehan  *  Return success if silicon family does not require a semaphore when
497517904deSPeter Grehan  *  accessing the PHY.
498517904deSPeter Grehan  **/
igc_acquire_phy(struct igc_hw * hw)499517904deSPeter Grehan s32 igc_acquire_phy(struct igc_hw *hw)
500517904deSPeter Grehan {
501517904deSPeter Grehan 	if (hw->phy.ops.acquire)
502517904deSPeter Grehan 		return hw->phy.ops.acquire(hw);
503517904deSPeter Grehan 
504517904deSPeter Grehan 	return IGC_SUCCESS;
505517904deSPeter Grehan }
506517904deSPeter Grehan 
507517904deSPeter Grehan /**
508517904deSPeter Grehan  *  igc_get_phy_info - Retrieves PHY information from registers
509517904deSPeter Grehan  *  @hw: pointer to the HW structure
510517904deSPeter Grehan  *
511517904deSPeter Grehan  *  This function gets some information from various PHY registers and
512517904deSPeter Grehan  *  populates hw->phy values with it. This is a function pointer entry
513517904deSPeter Grehan  *  point called by drivers.
514517904deSPeter Grehan  **/
igc_get_phy_info(struct igc_hw * hw)515517904deSPeter Grehan s32 igc_get_phy_info(struct igc_hw *hw)
516517904deSPeter Grehan {
517517904deSPeter Grehan 	if (hw->phy.ops.get_info)
518517904deSPeter Grehan 		return hw->phy.ops.get_info(hw);
519517904deSPeter Grehan 
520517904deSPeter Grehan 	return IGC_SUCCESS;
521517904deSPeter Grehan }
522517904deSPeter Grehan 
523517904deSPeter Grehan /**
524517904deSPeter Grehan  *  igc_phy_hw_reset - Hard PHY reset
525517904deSPeter Grehan  *  @hw: pointer to the HW structure
526517904deSPeter Grehan  *
527517904deSPeter Grehan  *  Performs a hard PHY reset. This is a function pointer entry point called
528517904deSPeter Grehan  *  by drivers.
529517904deSPeter Grehan  **/
igc_phy_hw_reset(struct igc_hw * hw)530517904deSPeter Grehan s32 igc_phy_hw_reset(struct igc_hw *hw)
531517904deSPeter Grehan {
532517904deSPeter Grehan 	if (hw->phy.ops.reset)
533517904deSPeter Grehan 		return hw->phy.ops.reset(hw);
534517904deSPeter Grehan 
535517904deSPeter Grehan 	return IGC_SUCCESS;
536517904deSPeter Grehan }
537517904deSPeter Grehan 
538517904deSPeter Grehan /**
539517904deSPeter Grehan  *  igc_set_d0_lplu_state - Sets low power link up state for D0
540517904deSPeter Grehan  *  @hw: pointer to the HW structure
541517904deSPeter Grehan  *  @active: boolean used to enable/disable lplu
542517904deSPeter Grehan  *
543517904deSPeter Grehan  *  Success returns 0, Failure returns 1
544517904deSPeter Grehan  *
545517904deSPeter Grehan  *  The low power link up (lplu) state is set to the power management level D0
546517904deSPeter Grehan  *  and SmartSpeed is disabled when active is true, else clear lplu for D0
547517904deSPeter Grehan  *  and enable Smartspeed.  LPLU and Smartspeed are mutually exclusive.  LPLU
548517904deSPeter Grehan  *  is used during Dx states where the power conservation is most important.
549517904deSPeter Grehan  *  During driver activity, SmartSpeed should be enabled so performance is
550517904deSPeter Grehan  *  maintained.  This is a function pointer entry point called by drivers.
551517904deSPeter Grehan  **/
igc_set_d0_lplu_state(struct igc_hw * hw,bool active)552517904deSPeter Grehan s32 igc_set_d0_lplu_state(struct igc_hw *hw, bool active)
553517904deSPeter Grehan {
554517904deSPeter Grehan 	if (hw->phy.ops.set_d0_lplu_state)
555517904deSPeter Grehan 		return hw->phy.ops.set_d0_lplu_state(hw, active);
556517904deSPeter Grehan 
557517904deSPeter Grehan 	return IGC_SUCCESS;
558517904deSPeter Grehan }
559517904deSPeter Grehan 
560517904deSPeter Grehan /**
561517904deSPeter Grehan  *  igc_set_d3_lplu_state - Sets low power link up state for D3
562517904deSPeter Grehan  *  @hw: pointer to the HW structure
563517904deSPeter Grehan  *  @active: boolean used to enable/disable lplu
564517904deSPeter Grehan  *
565517904deSPeter Grehan  *  Success returns 0, Failure returns 1
566517904deSPeter Grehan  *
567517904deSPeter Grehan  *  The low power link up (lplu) state is set to the power management level D3
568517904deSPeter Grehan  *  and SmartSpeed is disabled when active is true, else clear lplu for D3
569517904deSPeter Grehan  *  and enable Smartspeed.  LPLU and Smartspeed are mutually exclusive.  LPLU
570517904deSPeter Grehan  *  is used during Dx states where the power conservation is most important.
571517904deSPeter Grehan  *  During driver activity, SmartSpeed should be enabled so performance is
572517904deSPeter Grehan  *  maintained.  This is a function pointer entry point called by drivers.
573517904deSPeter Grehan  **/
igc_set_d3_lplu_state(struct igc_hw * hw,bool active)574517904deSPeter Grehan s32 igc_set_d3_lplu_state(struct igc_hw *hw, bool active)
575517904deSPeter Grehan {
576517904deSPeter Grehan 	if (hw->phy.ops.set_d3_lplu_state)
577517904deSPeter Grehan 		return hw->phy.ops.set_d3_lplu_state(hw, active);
578517904deSPeter Grehan 
579517904deSPeter Grehan 	return IGC_SUCCESS;
580517904deSPeter Grehan }
581517904deSPeter Grehan 
582517904deSPeter Grehan /**
583517904deSPeter Grehan  *  igc_read_mac_addr - Reads MAC address
584517904deSPeter Grehan  *  @hw: pointer to the HW structure
585517904deSPeter Grehan  *
586517904deSPeter Grehan  *  Reads the MAC address out of the adapter and stores it in the HW structure.
587517904deSPeter Grehan  *  Currently no func pointer exists and all implementations are handled in the
588517904deSPeter Grehan  *  generic version of this function.
589517904deSPeter Grehan  **/
igc_read_mac_addr(struct igc_hw * hw)590517904deSPeter Grehan s32 igc_read_mac_addr(struct igc_hw *hw)
591517904deSPeter Grehan {
592517904deSPeter Grehan 	if (hw->mac.ops.read_mac_addr)
593517904deSPeter Grehan 		return hw->mac.ops.read_mac_addr(hw);
594517904deSPeter Grehan 
595517904deSPeter Grehan 	return igc_read_mac_addr_generic(hw);
596517904deSPeter Grehan }
597517904deSPeter Grehan 
598517904deSPeter Grehan /**
599517904deSPeter Grehan  *  igc_read_pba_string - Read device part number string
600517904deSPeter Grehan  *  @hw: pointer to the HW structure
601517904deSPeter Grehan  *  @pba_num: pointer to device part number
602517904deSPeter Grehan  *  @pba_num_size: size of part number buffer
603517904deSPeter Grehan  *
604517904deSPeter Grehan  *  Reads the product board assembly (PBA) number from the EEPROM and stores
605517904deSPeter Grehan  *  the value in pba_num.
606517904deSPeter Grehan  *  Currently no func pointer exists and all implementations are handled in the
607517904deSPeter Grehan  *  generic version of this function.
608517904deSPeter Grehan  **/
igc_read_pba_string(struct igc_hw * hw,u8 * pba_num,u32 pba_num_size)609517904deSPeter Grehan s32 igc_read_pba_string(struct igc_hw *hw, u8 *pba_num, u32 pba_num_size)
610517904deSPeter Grehan {
611517904deSPeter Grehan 	return igc_read_pba_string_generic(hw, pba_num, pba_num_size);
612517904deSPeter Grehan }
613517904deSPeter Grehan 
614517904deSPeter Grehan /**
615517904deSPeter Grehan  *  igc_validate_nvm_checksum - Verifies NVM (EEPROM) checksum
616517904deSPeter Grehan  *  @hw: pointer to the HW structure
617517904deSPeter Grehan  *
618517904deSPeter Grehan  *  Validates the NVM checksum is correct. This is a function pointer entry
619517904deSPeter Grehan  *  point called by drivers.
620517904deSPeter Grehan  **/
igc_validate_nvm_checksum(struct igc_hw * hw)621517904deSPeter Grehan s32 igc_validate_nvm_checksum(struct igc_hw *hw)
622517904deSPeter Grehan {
623517904deSPeter Grehan 	if (hw->nvm.ops.validate)
624517904deSPeter Grehan 		return hw->nvm.ops.validate(hw);
625517904deSPeter Grehan 
626517904deSPeter Grehan 	return -IGC_ERR_CONFIG;
627517904deSPeter Grehan }
628517904deSPeter Grehan 
629517904deSPeter Grehan /**
630517904deSPeter Grehan  *  igc_update_nvm_checksum - Updates NVM (EEPROM) checksum
631517904deSPeter Grehan  *  @hw: pointer to the HW structure
632517904deSPeter Grehan  *
633517904deSPeter Grehan  *  Updates the NVM checksum. Currently no func pointer exists and all
634517904deSPeter Grehan  *  implementations are handled in the generic version of this function.
635517904deSPeter Grehan  **/
igc_update_nvm_checksum(struct igc_hw * hw)636517904deSPeter Grehan s32 igc_update_nvm_checksum(struct igc_hw *hw)
637517904deSPeter Grehan {
638517904deSPeter Grehan 	if (hw->nvm.ops.update)
639517904deSPeter Grehan 		return hw->nvm.ops.update(hw);
640517904deSPeter Grehan 
641517904deSPeter Grehan 	return -IGC_ERR_CONFIG;
642517904deSPeter Grehan }
643517904deSPeter Grehan 
644517904deSPeter Grehan /**
645517904deSPeter Grehan  *  igc_reload_nvm - Reloads EEPROM
646517904deSPeter Grehan  *  @hw: pointer to the HW structure
647517904deSPeter Grehan  *
648517904deSPeter Grehan  *  Reloads the EEPROM by setting the "Reinitialize from EEPROM" bit in the
649517904deSPeter Grehan  *  extended control register.
650517904deSPeter Grehan  **/
igc_reload_nvm(struct igc_hw * hw)651517904deSPeter Grehan void igc_reload_nvm(struct igc_hw *hw)
652517904deSPeter Grehan {
653517904deSPeter Grehan 	if (hw->nvm.ops.reload)
654517904deSPeter Grehan 		hw->nvm.ops.reload(hw);
655517904deSPeter Grehan }
656517904deSPeter Grehan 
657517904deSPeter Grehan /**
658517904deSPeter Grehan  *  igc_read_nvm - Reads NVM (EEPROM)
659517904deSPeter Grehan  *  @hw: pointer to the HW structure
660517904deSPeter Grehan  *  @offset: the word offset to read
661517904deSPeter Grehan  *  @words: number of 16-bit words to read
662517904deSPeter Grehan  *  @data: pointer to the properly sized buffer for the data.
663517904deSPeter Grehan  *
664517904deSPeter Grehan  *  Reads 16-bit chunks of data from the NVM (EEPROM). This is a function
665517904deSPeter Grehan  *  pointer entry point called by drivers.
666517904deSPeter Grehan  **/
igc_read_nvm(struct igc_hw * hw,u16 offset,u16 words,u16 * data)667517904deSPeter Grehan s32 igc_read_nvm(struct igc_hw *hw, u16 offset, u16 words, u16 *data)
668517904deSPeter Grehan {
669517904deSPeter Grehan 	if (hw->nvm.ops.read)
670517904deSPeter Grehan 		return hw->nvm.ops.read(hw, offset, words, data);
671517904deSPeter Grehan 
672517904deSPeter Grehan 	return -IGC_ERR_CONFIG;
673517904deSPeter Grehan }
674517904deSPeter Grehan 
675517904deSPeter Grehan /**
676517904deSPeter Grehan  *  igc_write_nvm - Writes to NVM (EEPROM)
677517904deSPeter Grehan  *  @hw: pointer to the HW structure
678517904deSPeter Grehan  *  @offset: the word offset to read
679517904deSPeter Grehan  *  @words: number of 16-bit words to write
680517904deSPeter Grehan  *  @data: pointer to the properly sized buffer for the data.
681517904deSPeter Grehan  *
682517904deSPeter Grehan  *  Writes 16-bit chunks of data to the NVM (EEPROM). This is a function
683517904deSPeter Grehan  *  pointer entry point called by drivers.
684517904deSPeter Grehan  **/
igc_write_nvm(struct igc_hw * hw,u16 offset,u16 words,u16 * data)685517904deSPeter Grehan s32 igc_write_nvm(struct igc_hw *hw, u16 offset, u16 words, u16 *data)
686517904deSPeter Grehan {
687517904deSPeter Grehan 	if (hw->nvm.ops.write)
688517904deSPeter Grehan 		return hw->nvm.ops.write(hw, offset, words, data);
689517904deSPeter Grehan 
690517904deSPeter Grehan 	return IGC_SUCCESS;
691517904deSPeter Grehan }
692517904deSPeter Grehan 
693517904deSPeter Grehan /**
694517904deSPeter Grehan  * igc_power_up_phy - Restores link in case of PHY power down
695517904deSPeter Grehan  * @hw: pointer to the HW structure
696517904deSPeter Grehan  *
697517904deSPeter Grehan  * The phy may be powered down to save power, to turn off link when the
698517904deSPeter Grehan  * driver is unloaded, or wake on lan is not enabled (among others).
699517904deSPeter Grehan  **/
igc_power_up_phy(struct igc_hw * hw)700517904deSPeter Grehan void igc_power_up_phy(struct igc_hw *hw)
701517904deSPeter Grehan {
702517904deSPeter Grehan 	if (hw->phy.ops.power_up)
703517904deSPeter Grehan 		hw->phy.ops.power_up(hw);
704517904deSPeter Grehan 
705517904deSPeter Grehan 	igc_setup_link(hw);
706517904deSPeter Grehan }
707517904deSPeter Grehan 
708517904deSPeter Grehan /**
709517904deSPeter Grehan  * igc_power_down_phy - Power down PHY
710517904deSPeter Grehan  * @hw: pointer to the HW structure
711517904deSPeter Grehan  *
712517904deSPeter Grehan  * The phy may be powered down to save power, to turn off link when the
713517904deSPeter Grehan  * driver is unloaded, or wake on lan is not enabled (among others).
714517904deSPeter Grehan  **/
igc_power_down_phy(struct igc_hw * hw)715517904deSPeter Grehan void igc_power_down_phy(struct igc_hw *hw)
716517904deSPeter Grehan {
717517904deSPeter Grehan 	if (hw->phy.ops.power_down)
718517904deSPeter Grehan 		hw->phy.ops.power_down(hw);
719517904deSPeter Grehan }
720517904deSPeter Grehan 
721