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