xref: /dpdk/drivers/net/atlantic/hw_atl/hw_atl_utils_fw2x.c (revision 200bc52e5aa0d72e70464c9cd22b55cf536ed13c)
1 // SPDX-License-Identifier: (BSD-3-Clause OR GPL-2.0)
2 /* Copyright (C) 2014-2017 aQuantia Corporation. */
3 
4 /* File hw_atl_utils_fw2x.c: Definition of firmware 2.x functions for
5  * Atlantic hardware abstraction layer.
6  */
7 
8 #include <rte_ether.h>
9 #include "../atl_hw_regs.h"
10 
11 #include "../atl_types.h"
12 #include "hw_atl_utils.h"
13 #include "hw_atl_llh.h"
14 
15 #define HW_ATL_FW2X_MPI_EFUSE_ADDR	0x364
16 #define HW_ATL_FW2X_MPI_MBOX_ADDR	0x360
17 #define HW_ATL_FW2X_MPI_RPC_ADDR	0x334
18 
19 #define HW_ATL_FW2X_MPI_CONTROL_ADDR	0x368
20 #define HW_ATL_FW2X_MPI_CONTROL2_ADDR	0x36C
21 #define HW_ATL_FW2X_MPI_LED_ADDR	0x31c
22 
23 #define HW_ATL_FW2X_MPI_STATE_ADDR	0x370
24 #define HW_ATL_FW2X_MPI_STATE2_ADDR	0x374
25 
26 #define HW_ATL_FW2X_CAP_SLEEP_PROXY BIT(CAPS_HI_SLEEP_PROXY)
27 #define HW_ATL_FW2X_CAP_WOL BIT(CAPS_HI_WOL)
28 
29 #define HW_ATL_FW2X_CAP_EEE_1G_MASK   BIT(CAPS_HI_1000BASET_FD_EEE)
30 #define HW_ATL_FW2X_CAP_EEE_2G5_MASK  BIT(CAPS_HI_2P5GBASET_FD_EEE)
31 #define HW_ATL_FW2X_CAP_EEE_5G_MASK   BIT(CAPS_HI_5GBASET_FD_EEE)
32 #define HW_ATL_FW2X_CAP_EEE_10G_MASK  BIT(CAPS_HI_10GBASET_FD_EEE)
33 
34 #define HAL_ATLANTIC_WOL_FILTERS_COUNT     8
35 #define HAL_ATLANTIC_UTILS_FW2X_MSG_WOL    0x0E
36 
37 #define HW_ATL_FW_FEATURE_LED 0x03010026
38 
39 struct fw2x_msg_wol_pattern {
40 	u8 mask[16];
41 	u32 crc;
42 } __attribute__((__packed__));
43 
44 struct fw2x_msg_wol {
45 	u32 msg_id;
46 	u8 hw_addr[6];
47 	u8 magic_packet_enabled;
48 	u8 filter_count;
49 	struct fw2x_msg_wol_pattern filter[HAL_ATLANTIC_WOL_FILTERS_COUNT];
50 	u8 link_up_enabled;
51 	u8 link_down_enabled;
52 	u16 reserved;
53 	u32 link_up_timeout;
54 	u32 link_down_timeout;
55 } __attribute__((__packed__));
56 
57 static int aq_fw2x_set_link_speed(struct aq_hw_s *self, u32 speed);
58 static int aq_fw2x_set_state(struct aq_hw_s *self,
59 			     enum hal_atl_utils_fw_state_e state);
60 
61 static int aq_fw2x_init(struct aq_hw_s *self)
62 {
63 	int err = 0;
64 	struct hw_aq_atl_utils_mbox mbox;
65 
66 	/* check 10 times by 1ms */
67 	AQ_HW_WAIT_FOR(0U != (self->mbox_addr =
68 		       aq_hw_read_reg(self, HW_ATL_FW2X_MPI_MBOX_ADDR)),
69 		       1000U, 10U);
70 	AQ_HW_WAIT_FOR(0U != (self->rpc_addr =
71 		       aq_hw_read_reg(self, HW_ATL_FW2X_MPI_RPC_ADDR)),
72 		       1000U, 100U);
73 
74 	/* Read caps */
75 	hw_atl_utils_mpi_read_stats(self, &mbox);
76 
77 	self->caps_lo = mbox.info.caps_lo;
78 
79 	return err;
80 }
81 
82 static int aq_fw2x_deinit(struct aq_hw_s *self)
83 {
84 	int err = aq_fw2x_set_link_speed(self, 0);
85 
86 	if (!err)
87 		err = aq_fw2x_set_state(self, MPI_DEINIT);
88 
89 	return err;
90 }
91 
92 static enum hw_atl_fw2x_rate link_speed_mask_2fw2x_ratemask(u32 speed)
93 {
94 	enum hw_atl_fw2x_rate rate = 0;
95 
96 	if (speed & AQ_NIC_RATE_10G)
97 		rate |= FW2X_RATE_10G;
98 
99 	if (speed & AQ_NIC_RATE_5G)
100 		rate |= FW2X_RATE_5G;
101 
102 	if (speed & AQ_NIC_RATE_5G5R)
103 		rate |= FW2X_RATE_5G;
104 
105 	if (speed & AQ_NIC_RATE_2G5)
106 		rate |= FW2X_RATE_2G5;
107 
108 	if (speed & AQ_NIC_RATE_1G)
109 		rate |= FW2X_RATE_1G;
110 
111 	if (speed & AQ_NIC_RATE_100M)
112 		rate |= FW2X_RATE_100M;
113 
114 	return rate;
115 }
116 
117 static u32 fw2x_to_eee_mask(u32 speed)
118 {
119 	u32 rate = 0;
120 
121 	if (speed & HW_ATL_FW2X_CAP_EEE_10G_MASK)
122 		rate |= AQ_NIC_RATE_EEE_10G;
123 
124 	if (speed & HW_ATL_FW2X_CAP_EEE_5G_MASK)
125 		rate |= AQ_NIC_RATE_EEE_5G;
126 
127 	if (speed & HW_ATL_FW2X_CAP_EEE_2G5_MASK)
128 		rate |= AQ_NIC_RATE_EEE_2G5;
129 
130 	if (speed & HW_ATL_FW2X_CAP_EEE_1G_MASK)
131 		rate |= AQ_NIC_RATE_EEE_1G;
132 
133 	return rate;
134 }
135 
136 static int aq_fw2x_set_link_speed(struct aq_hw_s *self, u32 speed)
137 {
138 	u32 rate_mask = link_speed_mask_2fw2x_ratemask(speed);
139 	u32 reg_val = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_CONTROL_ADDR);
140 	u32 val = rate_mask | ((BIT(CAPS_LO_SMBUS_READ) |
141 				BIT(CAPS_LO_SMBUS_WRITE) |
142 				BIT(CAPS_LO_MACSEC)) & reg_val);
143 
144 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_CONTROL_ADDR, val);
145 
146 	return 0;
147 }
148 
149 static void aq_fw2x_set_mpi_flow_control(struct aq_hw_s *self, u32 *mpi_state)
150 {
151 	if (self->aq_nic_cfg->flow_control & AQ_NIC_FC_RX)
152 		*mpi_state |= BIT(CAPS_HI_PAUSE);
153 	else
154 		*mpi_state &= ~BIT(CAPS_HI_PAUSE);
155 
156 	if (self->aq_nic_cfg->flow_control & AQ_NIC_FC_TX)
157 		*mpi_state |= BIT(CAPS_HI_ASYMMETRIC_PAUSE);
158 	else
159 		*mpi_state &= ~BIT(CAPS_HI_ASYMMETRIC_PAUSE);
160 }
161 
162 static int aq_fw2x_set_state(struct aq_hw_s *self,
163 			     enum hal_atl_utils_fw_state_e state)
164 {
165 	u32 mpi_state = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR);
166 
167 	switch (state) {
168 	case MPI_INIT:
169 		mpi_state &= ~BIT(CAPS_HI_LINK_DROP);
170 		aq_fw2x_set_mpi_flow_control(self, &mpi_state);
171 		break;
172 	case MPI_DEINIT:
173 		mpi_state |= BIT(CAPS_HI_LINK_DROP);
174 		break;
175 	case MPI_RESET:
176 	case MPI_POWER:
177 		/* No actions */
178 		break;
179 	}
180 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR, mpi_state);
181 	return 0;
182 }
183 
184 static int aq_fw2x_update_link_status(struct aq_hw_s *self)
185 {
186 	u32 mpi_state = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_STATE_ADDR);
187 	u32 speed = mpi_state & (FW2X_RATE_100M | FW2X_RATE_1G |
188 				FW2X_RATE_2G5 | FW2X_RATE_5G | FW2X_RATE_10G);
189 	struct aq_hw_link_status_s *link_status = &self->aq_link_status;
190 
191 	if (speed) {
192 		if (speed & FW2X_RATE_10G)
193 			link_status->mbps = 10000;
194 		else if (speed & FW2X_RATE_5G)
195 			link_status->mbps = 5000;
196 		else if (speed & FW2X_RATE_2G5)
197 			link_status->mbps = 2500;
198 		else if (speed & FW2X_RATE_1G)
199 			link_status->mbps = 1000;
200 		else if (speed & FW2X_RATE_100M)
201 			link_status->mbps = 100;
202 		else
203 			link_status->mbps = 10000;
204 	} else {
205 		link_status->mbps = 0;
206 	}
207 
208 	return 0;
209 }
210 
211 static
212 int aq_fw2x_get_mac_permanent(struct aq_hw_s *self, u8 *mac)
213 {
214 	int err = 0;
215 	u32 h = 0U;
216 	u32 l = 0U;
217 	u32 mac_addr[2] = { 0 };
218 	u32 efuse_addr = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_EFUSE_ADDR);
219 
220 	if (efuse_addr != 0) {
221 		err = hw_atl_utils_fw_downld_dwords(self,
222 						    efuse_addr + (40U * 4U),
223 						    mac_addr,
224 						    ARRAY_SIZE(mac_addr));
225 		if (err)
226 			return err;
227 		mac_addr[0] = rte_constant_bswap32(mac_addr[0]);
228 		mac_addr[1] = rte_constant_bswap32(mac_addr[1]);
229 	}
230 
231 	rte_ether_addr_copy((struct rte_ether_addr *)mac_addr,
232 			(struct rte_ether_addr *)mac);
233 
234 	if ((mac[0] & 0x01U) || ((mac[0] | mac[1] | mac[2]) == 0x00U)) {
235 		unsigned int rnd = (uint32_t)rte_rand();
236 
237 		//get_random_bytes(&rnd, sizeof(unsigned int));
238 
239 		l = 0xE3000000U
240 			| (0xFFFFU & rnd)
241 			| (0x00 << 16);
242 		h = 0x8001300EU;
243 
244 		mac[5] = (u8)(0xFFU & l);
245 		l >>= 8;
246 		mac[4] = (u8)(0xFFU & l);
247 		l >>= 8;
248 		mac[3] = (u8)(0xFFU & l);
249 		l >>= 8;
250 		mac[2] = (u8)(0xFFU & l);
251 		mac[1] = (u8)(0xFFU & h);
252 		h >>= 8;
253 		mac[0] = (u8)(0xFFU & h);
254 	}
255 	return err;
256 }
257 
258 static int aq_fw2x_update_stats(struct aq_hw_s *self)
259 {
260 	int err = 0;
261 	u32 mpi_opts = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR);
262 	u32 orig_stats_val = mpi_opts & BIT(CAPS_HI_STATISTICS);
263 
264 	/* Toggle statistics bit for FW to update */
265 	mpi_opts = mpi_opts ^ BIT(CAPS_HI_STATISTICS);
266 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR, mpi_opts);
267 
268 	/* Wait FW to report back */
269 	AQ_HW_WAIT_FOR(orig_stats_val !=
270 		       (aq_hw_read_reg(self, HW_ATL_FW2X_MPI_STATE2_ADDR) &
271 				       BIT(CAPS_HI_STATISTICS)),
272 		       1U, 10000U);
273 	if (err)
274 		return err;
275 
276 	return hw_atl_utils_update_stats(self);
277 }
278 
279 static int aq_fw2x_get_temp(struct aq_hw_s *self, int *temp)
280 {
281 	int err = 0;
282 	u32 mpi_opts = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR);
283 	u32 temp_val = mpi_opts & BIT(CAPS_HI_TEMPERATURE);
284 	u32 temp_res;
285 
286 	/* Toggle statistics bit for FW to 0x36C.18 (CAPS_HI_TEMPERATURE) */
287 	mpi_opts = mpi_opts ^ BIT(CAPS_HI_TEMPERATURE);
288 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR, mpi_opts);
289 
290 	/* Wait FW to report back */
291 	AQ_HW_WAIT_FOR(temp_val !=
292 			(aq_hw_read_reg(self, HW_ATL_FW2X_MPI_STATE2_ADDR) &
293 					BIT(CAPS_HI_TEMPERATURE)), 1U, 10000U);
294 	err = hw_atl_utils_fw_downld_dwords(self,
295 				self->mbox_addr +
296 				offsetof(struct hw_aq_atl_utils_mbox, info) +
297 				offsetof(struct hw_aq_info, phy_temperature),
298 				&temp_res,
299 				sizeof(temp_res) / sizeof(u32));
300 
301 	if (err)
302 		return err;
303 
304 	*temp = temp_res  * 100 / 256;
305 	return 0;
306 }
307 
308 static int aq_fw2x_get_cable_len(struct aq_hw_s *self, int *cable_len)
309 {
310 	int err = 0;
311 	u32 cable_len_res;
312 
313 	err = hw_atl_utils_fw_downld_dwords(self,
314 				self->mbox_addr +
315 				offsetof(struct hw_aq_atl_utils_mbox, info) +
316 				offsetof(struct hw_aq_info, phy_temperature),
317 				&cable_len_res,
318 				sizeof(cable_len_res) / sizeof(u32));
319 
320 	if (err)
321 		return err;
322 
323 	*cable_len = (cable_len_res >> 16) & 0xFF;
324 	return 0;
325 }
326 
327 #ifndef ETH_ALEN
328 #define ETH_ALEN 6
329 #endif
330 
331 static int aq_fw2x_set_sleep_proxy(struct aq_hw_s *self, u8 *mac)
332 {
333 	int err = 0;
334 	struct hw_aq_atl_utils_fw_rpc *rpc = NULL;
335 	struct offload_info *cfg = NULL;
336 	unsigned int rpc_size = 0U;
337 	u32 mpi_opts;
338 
339 	rpc_size = sizeof(rpc->msg_id) + sizeof(*cfg);
340 
341 	err = hw_atl_utils_fw_rpc_wait(self, &rpc);
342 	if (err < 0)
343 		goto err_exit;
344 
345 	memset(rpc, 0, rpc_size);
346 	cfg = (struct offload_info *)(&rpc->msg_id + 1);
347 
348 	memcpy(cfg->mac_addr, mac, ETH_ALEN);
349 	cfg->len = sizeof(*cfg);
350 
351 	/* Clear bit 0x36C.23 */
352 	mpi_opts = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR);
353 	mpi_opts &= ~HW_ATL_FW2X_CAP_SLEEP_PROXY;
354 
355 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR, mpi_opts);
356 
357 	err = hw_atl_utils_fw_rpc_call(self, rpc_size);
358 	if (err < 0)
359 		goto err_exit;
360 
361 	/* Set bit 0x36C.23 */
362 	mpi_opts |= HW_ATL_FW2X_CAP_SLEEP_PROXY;
363 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR, mpi_opts);
364 
365 	AQ_HW_WAIT_FOR((aq_hw_read_reg(self, HW_ATL_FW2X_MPI_STATE2_ADDR) &
366 			HW_ATL_FW2X_CAP_SLEEP_PROXY), 1U, 10000U);
367 err_exit:
368 	return err;
369 }
370 
371 static int aq_fw2x_set_wol_params(struct aq_hw_s *self, u8 *mac)
372 {
373 	int err = 0;
374 	struct fw2x_msg_wol *msg = NULL;
375 	u32 mpi_opts;
376 
377 	struct hw_aq_atl_utils_fw_rpc *rpc = NULL;
378 
379 	err = hw_atl_utils_fw_rpc_wait(self, &rpc);
380 	if (err < 0)
381 		goto err_exit;
382 
383 	msg = (struct fw2x_msg_wol *)rpc;
384 
385 	msg->msg_id = HAL_ATLANTIC_UTILS_FW2X_MSG_WOL;
386 	msg->magic_packet_enabled = true;
387 	memcpy(msg->hw_addr, mac, ETH_ALEN);
388 
389 	mpi_opts = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR);
390 	mpi_opts &= ~(HW_ATL_FW2X_CAP_SLEEP_PROXY | HW_ATL_FW2X_CAP_WOL);
391 
392 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR, mpi_opts);
393 
394 	err = hw_atl_utils_fw_rpc_call(self, sizeof(*msg));
395 	if (err < 0)
396 		goto err_exit;
397 
398 	/* Set bit 0x36C.24 */
399 	mpi_opts |= HW_ATL_FW2X_CAP_WOL;
400 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR, mpi_opts);
401 
402 	AQ_HW_WAIT_FOR((aq_hw_read_reg(self, HW_ATL_FW2X_MPI_STATE2_ADDR) &
403 			HW_ATL_FW2X_CAP_WOL), 1U, 10000U);
404 err_exit:
405 	return err;
406 }
407 
408 static int aq_fw2x_set_power(struct aq_hw_s *self,
409 			     unsigned int power_state __rte_unused,
410 			     u8 *mac)
411 {
412 	int err = 0;
413 
414 	if (self->aq_nic_cfg->wol & AQ_NIC_WOL_ENABLED) {
415 		err = aq_fw2x_set_sleep_proxy(self, mac);
416 		if (err < 0)
417 			goto err_exit;
418 		err = aq_fw2x_set_wol_params(self, mac);
419 		if (err < 0)
420 			goto err_exit;
421 	}
422 err_exit:
423 	return err;
424 }
425 
426 static int aq_fw2x_set_eee_rate(struct aq_hw_s *self, u32 speed)
427 {
428 	u32 mpi_opts = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR);
429 	mpi_opts &= ~(HW_ATL_FW2X_CAP_EEE_1G_MASK |
430 		HW_ATL_FW2X_CAP_EEE_2G5_MASK | HW_ATL_FW2X_CAP_EEE_5G_MASK |
431 		HW_ATL_FW2X_CAP_EEE_10G_MASK);
432 
433 	if (speed & AQ_NIC_RATE_EEE_10G)
434 		mpi_opts |= HW_ATL_FW2X_CAP_EEE_10G_MASK;
435 
436 	if (speed & AQ_NIC_RATE_EEE_5G)
437 		mpi_opts |= HW_ATL_FW2X_CAP_EEE_5G_MASK;
438 
439 	if (speed & AQ_NIC_RATE_EEE_2G5)
440 		mpi_opts |= HW_ATL_FW2X_CAP_EEE_2G5_MASK;
441 
442 	if (speed & AQ_NIC_RATE_EEE_1G)
443 		mpi_opts |= HW_ATL_FW2X_CAP_EEE_1G_MASK;
444 
445 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR, mpi_opts);
446 
447 	return 0;
448 }
449 
450 static int aq_fw2x_get_eee_rate(struct aq_hw_s *self, u32 *rate,
451 					u32 *supported_rates)
452 {
453 	int err = 0;
454 	u32 caps_hi;
455 	u32 mpi_state;
456 
457 	err = hw_atl_utils_fw_downld_dwords(self,
458 				self->mbox_addr +
459 				offsetof(struct hw_aq_atl_utils_mbox, info) +
460 				offsetof(struct hw_aq_info, caps_hi),
461 				&caps_hi,
462 				sizeof(caps_hi) / sizeof(u32));
463 
464 	if (err)
465 		return err;
466 
467 	*supported_rates = fw2x_to_eee_mask(caps_hi);
468 
469 	mpi_state = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_STATE2_ADDR);
470 	*rate = fw2x_to_eee_mask(mpi_state);
471 
472 	return err;
473 }
474 
475 static int aq_fw2x_get_flow_control(struct aq_hw_s *self, u32 *fc)
476 {
477 	u32 mpi_state = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR);
478 
479 	*fc = ((mpi_state & BIT(CAPS_HI_PAUSE)) ? AQ_NIC_FC_RX : 0) |
480 	      ((mpi_state & BIT(CAPS_HI_ASYMMETRIC_PAUSE)) ? AQ_NIC_FC_TX : 0);
481 
482 	return 0;
483 }
484 
485 static int aq_fw2x_set_flow_control(struct aq_hw_s *self)
486 {
487 	u32 mpi_state = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR);
488 
489 	aq_fw2x_set_mpi_flow_control(self, &mpi_state);
490 
491 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_CONTROL2_ADDR, mpi_state);
492 
493 	return 0;
494 }
495 
496 static int aq_fw2x_led_control(struct aq_hw_s *self, u32 mode)
497 {
498 	if (self->fw_ver_actual < HW_ATL_FW_FEATURE_LED)
499 		return -EOPNOTSUPP;
500 
501 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_LED_ADDR, mode);
502 	return 0;
503 }
504 
505 static int aq_fw2x_get_eeprom(struct aq_hw_s *self, int dev_addr,
506 			      u32 *data, u32 len, u32 offset)
507 {
508 	u32 bytes_remains = len % sizeof(u32);
509 	u32 num_dwords = len / sizeof(u32);
510 	struct smbus_request request;
511 	u32 result = 0;
512 	u32 mpi_opts;
513 	int err = 0;
514 
515 	if ((self->caps_lo & BIT(CAPS_LO_SMBUS_READ)) == 0)
516 		return -EOPNOTSUPP;
517 
518 	request.msg_id = 0;
519 	request.device_id = dev_addr;
520 	request.address = offset;
521 	request.length = len;
522 
523 	/* Write SMBUS request to cfg memory */
524 	err = hw_atl_utils_fw_upload_dwords(self, self->rpc_addr,
525 				(u32 *)(void *)&request,
526 				sizeof(request) / sizeof(u32));
527 
528 	if (err < 0)
529 		return err;
530 
531 	/* Toggle 0x368.CAPS_LO_SMBUS_READ bit */
532 	mpi_opts = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_CONTROL_ADDR);
533 	mpi_opts ^= BIT(CAPS_LO_SMBUS_READ);
534 
535 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_CONTROL_ADDR, mpi_opts);
536 
537 	/* Wait until REQUEST_BIT matched in 0x370 */
538 
539 	AQ_HW_WAIT_FOR((aq_hw_read_reg(self, HW_ATL_FW2X_MPI_STATE_ADDR) &
540 		BIT(CAPS_LO_SMBUS_READ)) == (mpi_opts & BIT(CAPS_LO_SMBUS_READ)),
541 		10U, 10000U);
542 
543 	if (err < 0)
544 		return err;
545 
546 	err = hw_atl_utils_fw_downld_dwords(self, self->rpc_addr + sizeof(u32),
547 			&result,
548 			sizeof(result) / sizeof(u32));
549 
550 	if (err < 0)
551 		return err;
552 
553 	if (result)
554 		return -EIO;
555 
556 	if (num_dwords) {
557 		err = hw_atl_utils_fw_downld_dwords(self,
558 			self->rpc_addr + sizeof(u32) * 2,
559 			data,
560 			num_dwords);
561 
562 		if (err < 0)
563 			return err;
564 	}
565 
566 	if (bytes_remains) {
567 		u32 val = 0;
568 
569 		err = hw_atl_utils_fw_downld_dwords(self,
570 			self->rpc_addr + (sizeof(u32) * 2) +
571 			(num_dwords * sizeof(u32)),
572 			&val,
573 			1);
574 
575 		if (err < 0)
576 			return err;
577 
578 		rte_memcpy((u8 *)data + len - bytes_remains,
579 				&val, bytes_remains);
580 	}
581 
582 	return 0;
583 }
584 
585 
586 static int aq_fw2x_set_eeprom(struct aq_hw_s *self, int dev_addr,
587 			      u32 *data, u32 len, u32 offset)
588 {
589 	struct smbus_request request;
590 	u32 mpi_opts, result = 0;
591 	int err = 0;
592 
593 	if ((self->caps_lo & BIT(CAPS_LO_SMBUS_WRITE)) == 0)
594 		return -EOPNOTSUPP;
595 
596 	request.msg_id = 0;
597 	request.device_id = dev_addr;
598 	request.address = offset;
599 	request.length = len;
600 
601 	/* Write SMBUS request to cfg memory */
602 	err = hw_atl_utils_fw_upload_dwords(self, self->rpc_addr,
603 				(u32 *)(void *)&request,
604 				sizeof(request) / sizeof(u32));
605 
606 	if (err < 0)
607 		return err;
608 
609 	/* Write SMBUS data to cfg memory */
610 	u32 num_dwords = len / sizeof(u32);
611 	u32 bytes_remains = len % sizeof(u32);
612 
613 	if (num_dwords) {
614 		err = hw_atl_utils_fw_upload_dwords(self,
615 			self->rpc_addr + sizeof(request),
616 			(u32 *)(void *)data,
617 			num_dwords);
618 
619 		if (err < 0)
620 			return err;
621 	}
622 
623 	if (bytes_remains) {
624 		u32 val = 0;
625 
626 		rte_memcpy(&val, (u8 *)data + (sizeof(u32) * num_dwords),
627 			   bytes_remains);
628 
629 		err = hw_atl_utils_fw_upload_dwords(self,
630 			self->rpc_addr + sizeof(request) +
631 			(num_dwords * sizeof(u32)),
632 			&val,
633 			1);
634 
635 		if (err < 0)
636 			return err;
637 	}
638 
639 	/* Toggle 0x368.CAPS_LO_SMBUS_WRITE bit */
640 	mpi_opts = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_CONTROL_ADDR);
641 	mpi_opts ^= BIT(CAPS_LO_SMBUS_WRITE);
642 
643 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_CONTROL_ADDR, mpi_opts);
644 
645 	/* Wait until REQUEST_BIT matched in 0x370 */
646 	AQ_HW_WAIT_FOR((aq_hw_read_reg(self, HW_ATL_FW2X_MPI_STATE_ADDR) &
647 		BIT(CAPS_LO_SMBUS_WRITE)) == (mpi_opts & BIT(CAPS_LO_SMBUS_WRITE)),
648 		10U, 10000U);
649 
650 	if (err < 0)
651 		return err;
652 
653 	/* Read status of write operation */
654 	err = hw_atl_utils_fw_downld_dwords(self, self->rpc_addr + sizeof(u32),
655 				&result,
656 				sizeof(result) / sizeof(u32));
657 
658 	if (err < 0)
659 		return err;
660 
661 	if (result)
662 		return -EIO;
663 
664 	return 0;
665 }
666 
667 static int aq_fw2x_send_macsec_request(struct aq_hw_s *self,
668 				struct macsec_msg_fw_request *req,
669 				struct macsec_msg_fw_response *response)
670 {
671 	int err = 0;
672 	u32 mpi_opts = 0;
673 
674 	if (!req || !response)
675 		return 0;
676 
677 	if ((self->caps_lo & BIT(CAPS_LO_MACSEC)) == 0)
678 		return -EOPNOTSUPP;
679 
680 	/* Write macsec request to cfg memory */
681 	err = hw_atl_utils_fw_upload_dwords(self, self->rpc_addr,
682 		(u32 *)(void *)req,
683 		RTE_ALIGN(sizeof(*req) / sizeof(u32), sizeof(u32)));
684 
685 	if (err < 0)
686 		return err;
687 
688 	/* Toggle 0x368.CAPS_LO_MACSEC bit */
689 	mpi_opts = aq_hw_read_reg(self, HW_ATL_FW2X_MPI_CONTROL_ADDR);
690 	mpi_opts ^= BIT(CAPS_LO_MACSEC);
691 
692 	aq_hw_write_reg(self, HW_ATL_FW2X_MPI_CONTROL_ADDR, mpi_opts);
693 
694 	/* Wait until REQUEST_BIT matched in 0x370 */
695 	AQ_HW_WAIT_FOR((aq_hw_read_reg(self, HW_ATL_FW2X_MPI_STATE_ADDR) &
696 		BIT(CAPS_LO_MACSEC)) == (mpi_opts & BIT(CAPS_LO_MACSEC)),
697 		1000U, 10000U);
698 
699 	if (err < 0)
700 		return err;
701 
702 	/* Read status of write operation */
703 	err = hw_atl_utils_fw_downld_dwords(self, self->rpc_addr + sizeof(u32),
704 		(u32 *)(void *)response,
705 		RTE_ALIGN(sizeof(*response) / sizeof(u32), sizeof(u32)));
706 
707 	return err;
708 }
709 
710 const struct aq_fw_ops aq_fw_2x_ops = {
711 	.init = aq_fw2x_init,
712 	.deinit = aq_fw2x_deinit,
713 	.reset = NULL,
714 	.get_mac_permanent = aq_fw2x_get_mac_permanent,
715 	.set_link_speed = aq_fw2x_set_link_speed,
716 	.set_state = aq_fw2x_set_state,
717 	.update_link_status = aq_fw2x_update_link_status,
718 	.update_stats = aq_fw2x_update_stats,
719 	.set_power = aq_fw2x_set_power,
720 	.get_temp = aq_fw2x_get_temp,
721 	.get_cable_len = aq_fw2x_get_cable_len,
722 	.set_eee_rate = aq_fw2x_set_eee_rate,
723 	.get_eee_rate = aq_fw2x_get_eee_rate,
724 	.get_flow_control = aq_fw2x_get_flow_control,
725 	.set_flow_control = aq_fw2x_set_flow_control,
726 	.led_control = aq_fw2x_led_control,
727 	.get_eeprom = aq_fw2x_get_eeprom,
728 	.set_eeprom = aq_fw2x_set_eeprom,
729 	.send_macsec_req = aq_fw2x_send_macsec_request,
730 };
731