162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0
262306a36Sopenharmony_ci/* Copyright (c) 2019 - 2022 Beijing WangXun Technology Co., Ltd. */
362306a36Sopenharmony_ci
462306a36Sopenharmony_ci#include <linux/etherdevice.h>
562306a36Sopenharmony_ci#include <linux/iopoll.h>
662306a36Sopenharmony_ci#include <linux/pci.h>
762306a36Sopenharmony_ci
862306a36Sopenharmony_ci#include "../libwx/wx_type.h"
962306a36Sopenharmony_ci#include "../libwx/wx_hw.h"
1062306a36Sopenharmony_ci#include "ngbe_type.h"
1162306a36Sopenharmony_ci#include "ngbe_hw.h"
1262306a36Sopenharmony_ci
1362306a36Sopenharmony_ciint ngbe_eeprom_chksum_hostif(struct wx *wx)
1462306a36Sopenharmony_ci{
1562306a36Sopenharmony_ci	struct wx_hic_read_shadow_ram buffer;
1662306a36Sopenharmony_ci	int status;
1762306a36Sopenharmony_ci	int tmp;
1862306a36Sopenharmony_ci
1962306a36Sopenharmony_ci	buffer.hdr.req.cmd = NGBE_FW_EEPROM_CHECKSUM_CMD;
2062306a36Sopenharmony_ci	buffer.hdr.req.buf_lenh = 0;
2162306a36Sopenharmony_ci	buffer.hdr.req.buf_lenl = 0;
2262306a36Sopenharmony_ci	buffer.hdr.req.checksum = NGBE_FW_CMD_DEFAULT_CHECKSUM;
2362306a36Sopenharmony_ci	/* convert offset from words to bytes */
2462306a36Sopenharmony_ci	buffer.address = 0;
2562306a36Sopenharmony_ci	/* one word */
2662306a36Sopenharmony_ci	buffer.length = 0;
2762306a36Sopenharmony_ci
2862306a36Sopenharmony_ci	status = wx_host_interface_command(wx, (u32 *)&buffer, sizeof(buffer),
2962306a36Sopenharmony_ci					   WX_HI_COMMAND_TIMEOUT, false);
3062306a36Sopenharmony_ci
3162306a36Sopenharmony_ci	if (status < 0)
3262306a36Sopenharmony_ci		return status;
3362306a36Sopenharmony_ci	tmp = rd32a(wx, WX_MNG_MBOX, 1);
3462306a36Sopenharmony_ci	if (tmp == NGBE_FW_CMD_ST_PASS)
3562306a36Sopenharmony_ci		return 0;
3662306a36Sopenharmony_ci	return -EIO;
3762306a36Sopenharmony_ci}
3862306a36Sopenharmony_ci
3962306a36Sopenharmony_cistatic int ngbe_reset_misc(struct wx *wx)
4062306a36Sopenharmony_ci{
4162306a36Sopenharmony_ci	wx_reset_misc(wx);
4262306a36Sopenharmony_ci	if (wx->gpio_ctrl) {
4362306a36Sopenharmony_ci		/* gpio0 is used to power on/off control*/
4462306a36Sopenharmony_ci		wr32(wx, NGBE_GPIO_DDR, 0x1);
4562306a36Sopenharmony_ci		ngbe_sfp_modules_txrx_powerctl(wx, false);
4662306a36Sopenharmony_ci	}
4762306a36Sopenharmony_ci	return 0;
4862306a36Sopenharmony_ci}
4962306a36Sopenharmony_ci
5062306a36Sopenharmony_civoid ngbe_sfp_modules_txrx_powerctl(struct wx *wx, bool swi)
5162306a36Sopenharmony_ci{
5262306a36Sopenharmony_ci	/* gpio0 is used to power on control . 0 is on */
5362306a36Sopenharmony_ci	wr32(wx, NGBE_GPIO_DR, swi ? 0 : NGBE_GPIO_DR_0);
5462306a36Sopenharmony_ci}
5562306a36Sopenharmony_ci
5662306a36Sopenharmony_ci/**
5762306a36Sopenharmony_ci *  ngbe_reset_hw - Perform hardware reset
5862306a36Sopenharmony_ci *  @wx: pointer to hardware structure
5962306a36Sopenharmony_ci *
6062306a36Sopenharmony_ci *  Resets the hardware by resetting the transmit and receive units, masks
6162306a36Sopenharmony_ci *  and clears all interrupts, perform a PHY reset, and perform a link (MAC)
6262306a36Sopenharmony_ci *  reset.
6362306a36Sopenharmony_ci **/
6462306a36Sopenharmony_ciint ngbe_reset_hw(struct wx *wx)
6562306a36Sopenharmony_ci{
6662306a36Sopenharmony_ci	u32 val = 0;
6762306a36Sopenharmony_ci	int ret = 0;
6862306a36Sopenharmony_ci
6962306a36Sopenharmony_ci	/* Call wx stop to disable tx/rx and clear interrupts */
7062306a36Sopenharmony_ci	ret = wx_stop_adapter(wx);
7162306a36Sopenharmony_ci	if (ret != 0)
7262306a36Sopenharmony_ci		return ret;
7362306a36Sopenharmony_ci
7462306a36Sopenharmony_ci	if (wx->mac_type != em_mac_type_mdi) {
7562306a36Sopenharmony_ci		val = WX_MIS_RST_LAN_RST(wx->bus.func);
7662306a36Sopenharmony_ci		wr32(wx, WX_MIS_RST, val | rd32(wx, WX_MIS_RST));
7762306a36Sopenharmony_ci
7862306a36Sopenharmony_ci		ret = read_poll_timeout(rd32, val,
7962306a36Sopenharmony_ci					!(val & (BIT(9) << wx->bus.func)), 1000,
8062306a36Sopenharmony_ci					100000, false, wx, 0x10028);
8162306a36Sopenharmony_ci		if (ret) {
8262306a36Sopenharmony_ci			wx_err(wx, "Lan reset exceed s maximum times.\n");
8362306a36Sopenharmony_ci			return ret;
8462306a36Sopenharmony_ci		}
8562306a36Sopenharmony_ci	}
8662306a36Sopenharmony_ci	ngbe_reset_misc(wx);
8762306a36Sopenharmony_ci
8862306a36Sopenharmony_ci	/* Store the permanent mac address */
8962306a36Sopenharmony_ci	wx_get_mac_addr(wx, wx->mac.perm_addr);
9062306a36Sopenharmony_ci
9162306a36Sopenharmony_ci	/* reset num_rar_entries to 128 */
9262306a36Sopenharmony_ci	wx->mac.num_rar_entries = NGBE_RAR_ENTRIES;
9362306a36Sopenharmony_ci	wx_init_rx_addrs(wx);
9462306a36Sopenharmony_ci	pci_set_master(wx->pdev);
9562306a36Sopenharmony_ci
9662306a36Sopenharmony_ci	return 0;
9762306a36Sopenharmony_ci}
98