162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Copyright(c) 2015 EZchip Technologies. 462306a36Sopenharmony_ci */ 562306a36Sopenharmony_ci 662306a36Sopenharmony_ci#include <linux/module.h> 762306a36Sopenharmony_ci#include <linux/etherdevice.h> 862306a36Sopenharmony_ci#include <linux/interrupt.h> 962306a36Sopenharmony_ci#include <linux/mod_devicetable.h> 1062306a36Sopenharmony_ci#include <linux/of_net.h> 1162306a36Sopenharmony_ci#include <linux/platform_device.h> 1262306a36Sopenharmony_ci#include "nps_enet.h" 1362306a36Sopenharmony_ci 1462306a36Sopenharmony_ci#define DRV_NAME "nps_mgt_enet" 1562306a36Sopenharmony_ci 1662306a36Sopenharmony_cistatic inline bool nps_enet_is_tx_pending(struct nps_enet_priv *priv) 1762306a36Sopenharmony_ci{ 1862306a36Sopenharmony_ci u32 tx_ctrl_value = nps_enet_reg_get(priv, NPS_ENET_REG_TX_CTL); 1962306a36Sopenharmony_ci u32 tx_ctrl_ct = (tx_ctrl_value & TX_CTL_CT_MASK) >> TX_CTL_CT_SHIFT; 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_ci return (!tx_ctrl_ct && priv->tx_skb); 2262306a36Sopenharmony_ci} 2362306a36Sopenharmony_ci 2462306a36Sopenharmony_cistatic void nps_enet_clean_rx_fifo(struct net_device *ndev, u32 frame_len) 2562306a36Sopenharmony_ci{ 2662306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 2762306a36Sopenharmony_ci u32 i, len = DIV_ROUND_UP(frame_len, sizeof(u32)); 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_ci /* Empty Rx FIFO buffer by reading all words */ 3062306a36Sopenharmony_ci for (i = 0; i < len; i++) 3162306a36Sopenharmony_ci nps_enet_reg_get(priv, NPS_ENET_REG_RX_BUF); 3262306a36Sopenharmony_ci} 3362306a36Sopenharmony_ci 3462306a36Sopenharmony_cistatic void nps_enet_read_rx_fifo(struct net_device *ndev, 3562306a36Sopenharmony_ci unsigned char *dst, u32 length) 3662306a36Sopenharmony_ci{ 3762306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 3862306a36Sopenharmony_ci s32 i, last = length & (sizeof(u32) - 1); 3962306a36Sopenharmony_ci u32 *reg = (u32 *)dst, len = length / sizeof(u32); 4062306a36Sopenharmony_ci bool dst_is_aligned = IS_ALIGNED((unsigned long)dst, sizeof(u32)); 4162306a36Sopenharmony_ci 4262306a36Sopenharmony_ci /* In case dst is not aligned we need an intermediate buffer */ 4362306a36Sopenharmony_ci if (dst_is_aligned) { 4462306a36Sopenharmony_ci ioread32_rep(priv->regs_base + NPS_ENET_REG_RX_BUF, reg, len); 4562306a36Sopenharmony_ci reg += len; 4662306a36Sopenharmony_ci } else { /* !dst_is_aligned */ 4762306a36Sopenharmony_ci for (i = 0; i < len; i++, reg++) { 4862306a36Sopenharmony_ci u32 buf = nps_enet_reg_get(priv, NPS_ENET_REG_RX_BUF); 4962306a36Sopenharmony_ci 5062306a36Sopenharmony_ci put_unaligned_be32(buf, reg); 5162306a36Sopenharmony_ci } 5262306a36Sopenharmony_ci } 5362306a36Sopenharmony_ci /* copy last bytes (if any) */ 5462306a36Sopenharmony_ci if (last) { 5562306a36Sopenharmony_ci u32 buf; 5662306a36Sopenharmony_ci 5762306a36Sopenharmony_ci ioread32_rep(priv->regs_base + NPS_ENET_REG_RX_BUF, &buf, 1); 5862306a36Sopenharmony_ci memcpy((u8 *)reg, &buf, last); 5962306a36Sopenharmony_ci } 6062306a36Sopenharmony_ci} 6162306a36Sopenharmony_ci 6262306a36Sopenharmony_cistatic u32 nps_enet_rx_handler(struct net_device *ndev) 6362306a36Sopenharmony_ci{ 6462306a36Sopenharmony_ci u32 frame_len, err = 0; 6562306a36Sopenharmony_ci u32 work_done = 0; 6662306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 6762306a36Sopenharmony_ci struct sk_buff *skb; 6862306a36Sopenharmony_ci u32 rx_ctrl_value = nps_enet_reg_get(priv, NPS_ENET_REG_RX_CTL); 6962306a36Sopenharmony_ci u32 rx_ctrl_cr = (rx_ctrl_value & RX_CTL_CR_MASK) >> RX_CTL_CR_SHIFT; 7062306a36Sopenharmony_ci u32 rx_ctrl_er = (rx_ctrl_value & RX_CTL_ER_MASK) >> RX_CTL_ER_SHIFT; 7162306a36Sopenharmony_ci u32 rx_ctrl_crc = (rx_ctrl_value & RX_CTL_CRC_MASK) >> RX_CTL_CRC_SHIFT; 7262306a36Sopenharmony_ci 7362306a36Sopenharmony_ci frame_len = (rx_ctrl_value & RX_CTL_NR_MASK) >> RX_CTL_NR_SHIFT; 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_ci /* Check if we got RX */ 7662306a36Sopenharmony_ci if (!rx_ctrl_cr) 7762306a36Sopenharmony_ci return work_done; 7862306a36Sopenharmony_ci 7962306a36Sopenharmony_ci /* If we got here there is a work for us */ 8062306a36Sopenharmony_ci work_done++; 8162306a36Sopenharmony_ci 8262306a36Sopenharmony_ci /* Check Rx error */ 8362306a36Sopenharmony_ci if (rx_ctrl_er) { 8462306a36Sopenharmony_ci ndev->stats.rx_errors++; 8562306a36Sopenharmony_ci err = 1; 8662306a36Sopenharmony_ci } 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_ci /* Check Rx CRC error */ 8962306a36Sopenharmony_ci if (rx_ctrl_crc) { 9062306a36Sopenharmony_ci ndev->stats.rx_crc_errors++; 9162306a36Sopenharmony_ci ndev->stats.rx_dropped++; 9262306a36Sopenharmony_ci err = 1; 9362306a36Sopenharmony_ci } 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_ci /* Check Frame length Min 64b */ 9662306a36Sopenharmony_ci if (unlikely(frame_len < ETH_ZLEN)) { 9762306a36Sopenharmony_ci ndev->stats.rx_length_errors++; 9862306a36Sopenharmony_ci ndev->stats.rx_dropped++; 9962306a36Sopenharmony_ci err = 1; 10062306a36Sopenharmony_ci } 10162306a36Sopenharmony_ci 10262306a36Sopenharmony_ci if (err) 10362306a36Sopenharmony_ci goto rx_irq_clean; 10462306a36Sopenharmony_ci 10562306a36Sopenharmony_ci /* Skb allocation */ 10662306a36Sopenharmony_ci skb = netdev_alloc_skb_ip_align(ndev, frame_len); 10762306a36Sopenharmony_ci if (unlikely(!skb)) { 10862306a36Sopenharmony_ci ndev->stats.rx_errors++; 10962306a36Sopenharmony_ci ndev->stats.rx_dropped++; 11062306a36Sopenharmony_ci goto rx_irq_clean; 11162306a36Sopenharmony_ci } 11262306a36Sopenharmony_ci 11362306a36Sopenharmony_ci /* Copy frame from Rx fifo into the skb */ 11462306a36Sopenharmony_ci nps_enet_read_rx_fifo(ndev, skb->data, frame_len); 11562306a36Sopenharmony_ci 11662306a36Sopenharmony_ci skb_put(skb, frame_len); 11762306a36Sopenharmony_ci skb->protocol = eth_type_trans(skb, ndev); 11862306a36Sopenharmony_ci skb->ip_summed = CHECKSUM_UNNECESSARY; 11962306a36Sopenharmony_ci 12062306a36Sopenharmony_ci ndev->stats.rx_packets++; 12162306a36Sopenharmony_ci ndev->stats.rx_bytes += frame_len; 12262306a36Sopenharmony_ci netif_receive_skb(skb); 12362306a36Sopenharmony_ci 12462306a36Sopenharmony_ci goto rx_irq_frame_done; 12562306a36Sopenharmony_ci 12662306a36Sopenharmony_cirx_irq_clean: 12762306a36Sopenharmony_ci /* Clean Rx fifo */ 12862306a36Sopenharmony_ci nps_enet_clean_rx_fifo(ndev, frame_len); 12962306a36Sopenharmony_ci 13062306a36Sopenharmony_cirx_irq_frame_done: 13162306a36Sopenharmony_ci /* Ack Rx ctrl register */ 13262306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_RX_CTL, 0); 13362306a36Sopenharmony_ci 13462306a36Sopenharmony_ci return work_done; 13562306a36Sopenharmony_ci} 13662306a36Sopenharmony_ci 13762306a36Sopenharmony_cistatic void nps_enet_tx_handler(struct net_device *ndev) 13862306a36Sopenharmony_ci{ 13962306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 14062306a36Sopenharmony_ci u32 tx_ctrl_value = nps_enet_reg_get(priv, NPS_ENET_REG_TX_CTL); 14162306a36Sopenharmony_ci u32 tx_ctrl_et = (tx_ctrl_value & TX_CTL_ET_MASK) >> TX_CTL_ET_SHIFT; 14262306a36Sopenharmony_ci u32 tx_ctrl_nt = (tx_ctrl_value & TX_CTL_NT_MASK) >> TX_CTL_NT_SHIFT; 14362306a36Sopenharmony_ci 14462306a36Sopenharmony_ci /* Check if we got TX */ 14562306a36Sopenharmony_ci if (!nps_enet_is_tx_pending(priv)) 14662306a36Sopenharmony_ci return; 14762306a36Sopenharmony_ci 14862306a36Sopenharmony_ci /* Ack Tx ctrl register */ 14962306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_TX_CTL, 0); 15062306a36Sopenharmony_ci 15162306a36Sopenharmony_ci /* Check Tx transmit error */ 15262306a36Sopenharmony_ci if (unlikely(tx_ctrl_et)) { 15362306a36Sopenharmony_ci ndev->stats.tx_errors++; 15462306a36Sopenharmony_ci } else { 15562306a36Sopenharmony_ci ndev->stats.tx_packets++; 15662306a36Sopenharmony_ci ndev->stats.tx_bytes += tx_ctrl_nt; 15762306a36Sopenharmony_ci } 15862306a36Sopenharmony_ci 15962306a36Sopenharmony_ci dev_kfree_skb(priv->tx_skb); 16062306a36Sopenharmony_ci priv->tx_skb = NULL; 16162306a36Sopenharmony_ci 16262306a36Sopenharmony_ci if (netif_queue_stopped(ndev)) 16362306a36Sopenharmony_ci netif_wake_queue(ndev); 16462306a36Sopenharmony_ci} 16562306a36Sopenharmony_ci 16662306a36Sopenharmony_ci/** 16762306a36Sopenharmony_ci * nps_enet_poll - NAPI poll handler. 16862306a36Sopenharmony_ci * @napi: Pointer to napi_struct structure. 16962306a36Sopenharmony_ci * @budget: How many frames to process on one call. 17062306a36Sopenharmony_ci * 17162306a36Sopenharmony_ci * returns: Number of processed frames 17262306a36Sopenharmony_ci */ 17362306a36Sopenharmony_cistatic int nps_enet_poll(struct napi_struct *napi, int budget) 17462306a36Sopenharmony_ci{ 17562306a36Sopenharmony_ci struct net_device *ndev = napi->dev; 17662306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 17762306a36Sopenharmony_ci u32 work_done; 17862306a36Sopenharmony_ci 17962306a36Sopenharmony_ci nps_enet_tx_handler(ndev); 18062306a36Sopenharmony_ci work_done = nps_enet_rx_handler(ndev); 18162306a36Sopenharmony_ci if ((work_done < budget) && napi_complete_done(napi, work_done)) { 18262306a36Sopenharmony_ci u32 buf_int_enable_value = 0; 18362306a36Sopenharmony_ci 18462306a36Sopenharmony_ci /* set tx_done and rx_rdy bits */ 18562306a36Sopenharmony_ci buf_int_enable_value |= NPS_ENET_ENABLE << RX_RDY_SHIFT; 18662306a36Sopenharmony_ci buf_int_enable_value |= NPS_ENET_ENABLE << TX_DONE_SHIFT; 18762306a36Sopenharmony_ci 18862306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_BUF_INT_ENABLE, 18962306a36Sopenharmony_ci buf_int_enable_value); 19062306a36Sopenharmony_ci 19162306a36Sopenharmony_ci /* in case we will get a tx interrupt while interrupts 19262306a36Sopenharmony_ci * are masked, we will lose it since the tx is edge interrupt. 19362306a36Sopenharmony_ci * specifically, while executing the code section above, 19462306a36Sopenharmony_ci * between nps_enet_tx_handler and the interrupts enable, all 19562306a36Sopenharmony_ci * tx requests will be stuck until we will get an rx interrupt. 19662306a36Sopenharmony_ci * the two code lines below will solve this situation by 19762306a36Sopenharmony_ci * re-adding ourselves to the poll list. 19862306a36Sopenharmony_ci */ 19962306a36Sopenharmony_ci if (nps_enet_is_tx_pending(priv)) { 20062306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_BUF_INT_ENABLE, 0); 20162306a36Sopenharmony_ci napi_reschedule(napi); 20262306a36Sopenharmony_ci } 20362306a36Sopenharmony_ci } 20462306a36Sopenharmony_ci 20562306a36Sopenharmony_ci return work_done; 20662306a36Sopenharmony_ci} 20762306a36Sopenharmony_ci 20862306a36Sopenharmony_ci/** 20962306a36Sopenharmony_ci * nps_enet_irq_handler - Global interrupt handler for ENET. 21062306a36Sopenharmony_ci * @irq: irq number. 21162306a36Sopenharmony_ci * @dev_instance: device instance. 21262306a36Sopenharmony_ci * 21362306a36Sopenharmony_ci * returns: IRQ_HANDLED for all cases. 21462306a36Sopenharmony_ci * 21562306a36Sopenharmony_ci * EZchip ENET has 2 interrupt causes, and depending on bits raised in 21662306a36Sopenharmony_ci * CTRL registers we may tell what is a reason for interrupt to fire up. 21762306a36Sopenharmony_ci * We got one for RX and the other for TX (completion). 21862306a36Sopenharmony_ci */ 21962306a36Sopenharmony_cistatic irqreturn_t nps_enet_irq_handler(s32 irq, void *dev_instance) 22062306a36Sopenharmony_ci{ 22162306a36Sopenharmony_ci struct net_device *ndev = dev_instance; 22262306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 22362306a36Sopenharmony_ci u32 rx_ctrl_value = nps_enet_reg_get(priv, NPS_ENET_REG_RX_CTL); 22462306a36Sopenharmony_ci u32 rx_ctrl_cr = (rx_ctrl_value & RX_CTL_CR_MASK) >> RX_CTL_CR_SHIFT; 22562306a36Sopenharmony_ci 22662306a36Sopenharmony_ci if (nps_enet_is_tx_pending(priv) || rx_ctrl_cr) 22762306a36Sopenharmony_ci if (likely(napi_schedule_prep(&priv->napi))) { 22862306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_BUF_INT_ENABLE, 0); 22962306a36Sopenharmony_ci __napi_schedule(&priv->napi); 23062306a36Sopenharmony_ci } 23162306a36Sopenharmony_ci 23262306a36Sopenharmony_ci return IRQ_HANDLED; 23362306a36Sopenharmony_ci} 23462306a36Sopenharmony_ci 23562306a36Sopenharmony_cistatic void nps_enet_set_hw_mac_address(struct net_device *ndev) 23662306a36Sopenharmony_ci{ 23762306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 23862306a36Sopenharmony_ci u32 ge_mac_cfg_1_value = 0; 23962306a36Sopenharmony_ci u32 *ge_mac_cfg_2_value = &priv->ge_mac_cfg_2_value; 24062306a36Sopenharmony_ci 24162306a36Sopenharmony_ci /* set MAC address in HW */ 24262306a36Sopenharmony_ci ge_mac_cfg_1_value |= ndev->dev_addr[0] << CFG_1_OCTET_0_SHIFT; 24362306a36Sopenharmony_ci ge_mac_cfg_1_value |= ndev->dev_addr[1] << CFG_1_OCTET_1_SHIFT; 24462306a36Sopenharmony_ci ge_mac_cfg_1_value |= ndev->dev_addr[2] << CFG_1_OCTET_2_SHIFT; 24562306a36Sopenharmony_ci ge_mac_cfg_1_value |= ndev->dev_addr[3] << CFG_1_OCTET_3_SHIFT; 24662306a36Sopenharmony_ci *ge_mac_cfg_2_value = (*ge_mac_cfg_2_value & ~CFG_2_OCTET_4_MASK) 24762306a36Sopenharmony_ci | ndev->dev_addr[4] << CFG_2_OCTET_4_SHIFT; 24862306a36Sopenharmony_ci *ge_mac_cfg_2_value = (*ge_mac_cfg_2_value & ~CFG_2_OCTET_5_MASK) 24962306a36Sopenharmony_ci | ndev->dev_addr[5] << CFG_2_OCTET_5_SHIFT; 25062306a36Sopenharmony_ci 25162306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_GE_MAC_CFG_1, 25262306a36Sopenharmony_ci ge_mac_cfg_1_value); 25362306a36Sopenharmony_ci 25462306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_GE_MAC_CFG_2, 25562306a36Sopenharmony_ci *ge_mac_cfg_2_value); 25662306a36Sopenharmony_ci} 25762306a36Sopenharmony_ci 25862306a36Sopenharmony_ci/** 25962306a36Sopenharmony_ci * nps_enet_hw_reset - Reset the network device. 26062306a36Sopenharmony_ci * @ndev: Pointer to the network device. 26162306a36Sopenharmony_ci * 26262306a36Sopenharmony_ci * This function reset the PCS and TX fifo. 26362306a36Sopenharmony_ci * The programming model is to set the relevant reset bits 26462306a36Sopenharmony_ci * wait for some time for this to propagate and then unset 26562306a36Sopenharmony_ci * the reset bits. This way we ensure that reset procedure 26662306a36Sopenharmony_ci * is done successfully by device. 26762306a36Sopenharmony_ci */ 26862306a36Sopenharmony_cistatic void nps_enet_hw_reset(struct net_device *ndev) 26962306a36Sopenharmony_ci{ 27062306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 27162306a36Sopenharmony_ci u32 ge_rst_value = 0, phase_fifo_ctl_value = 0; 27262306a36Sopenharmony_ci 27362306a36Sopenharmony_ci /* Pcs reset sequence*/ 27462306a36Sopenharmony_ci ge_rst_value |= NPS_ENET_ENABLE << RST_GMAC_0_SHIFT; 27562306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_GE_RST, ge_rst_value); 27662306a36Sopenharmony_ci usleep_range(10, 20); 27762306a36Sopenharmony_ci ge_rst_value = 0; 27862306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_GE_RST, ge_rst_value); 27962306a36Sopenharmony_ci 28062306a36Sopenharmony_ci /* Tx fifo reset sequence */ 28162306a36Sopenharmony_ci phase_fifo_ctl_value |= NPS_ENET_ENABLE << PHASE_FIFO_CTL_RST_SHIFT; 28262306a36Sopenharmony_ci phase_fifo_ctl_value |= NPS_ENET_ENABLE << PHASE_FIFO_CTL_INIT_SHIFT; 28362306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_PHASE_FIFO_CTL, 28462306a36Sopenharmony_ci phase_fifo_ctl_value); 28562306a36Sopenharmony_ci usleep_range(10, 20); 28662306a36Sopenharmony_ci phase_fifo_ctl_value = 0; 28762306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_PHASE_FIFO_CTL, 28862306a36Sopenharmony_ci phase_fifo_ctl_value); 28962306a36Sopenharmony_ci} 29062306a36Sopenharmony_ci 29162306a36Sopenharmony_cistatic void nps_enet_hw_enable_control(struct net_device *ndev) 29262306a36Sopenharmony_ci{ 29362306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 29462306a36Sopenharmony_ci u32 ge_mac_cfg_0_value = 0, buf_int_enable_value = 0; 29562306a36Sopenharmony_ci u32 *ge_mac_cfg_2_value = &priv->ge_mac_cfg_2_value; 29662306a36Sopenharmony_ci u32 *ge_mac_cfg_3_value = &priv->ge_mac_cfg_3_value; 29762306a36Sopenharmony_ci s32 max_frame_length; 29862306a36Sopenharmony_ci 29962306a36Sopenharmony_ci /* Enable Rx and Tx statistics */ 30062306a36Sopenharmony_ci *ge_mac_cfg_2_value = (*ge_mac_cfg_2_value & ~CFG_2_STAT_EN_MASK) 30162306a36Sopenharmony_ci | NPS_ENET_GE_MAC_CFG_2_STAT_EN << CFG_2_STAT_EN_SHIFT; 30262306a36Sopenharmony_ci 30362306a36Sopenharmony_ci /* Discard packets with different MAC address */ 30462306a36Sopenharmony_ci *ge_mac_cfg_2_value = (*ge_mac_cfg_2_value & ~CFG_2_DISK_DA_MASK) 30562306a36Sopenharmony_ci | NPS_ENET_ENABLE << CFG_2_DISK_DA_SHIFT; 30662306a36Sopenharmony_ci 30762306a36Sopenharmony_ci /* Discard multicast packets */ 30862306a36Sopenharmony_ci *ge_mac_cfg_2_value = (*ge_mac_cfg_2_value & ~CFG_2_DISK_MC_MASK) 30962306a36Sopenharmony_ci | NPS_ENET_ENABLE << CFG_2_DISK_MC_SHIFT; 31062306a36Sopenharmony_ci 31162306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_GE_MAC_CFG_2, 31262306a36Sopenharmony_ci *ge_mac_cfg_2_value); 31362306a36Sopenharmony_ci 31462306a36Sopenharmony_ci /* Discard Packets bigger than max frame length */ 31562306a36Sopenharmony_ci max_frame_length = ETH_HLEN + ndev->mtu + ETH_FCS_LEN; 31662306a36Sopenharmony_ci if (max_frame_length <= NPS_ENET_MAX_FRAME_LENGTH) { 31762306a36Sopenharmony_ci *ge_mac_cfg_3_value = 31862306a36Sopenharmony_ci (*ge_mac_cfg_3_value & ~CFG_3_MAX_LEN_MASK) 31962306a36Sopenharmony_ci | max_frame_length << CFG_3_MAX_LEN_SHIFT; 32062306a36Sopenharmony_ci } 32162306a36Sopenharmony_ci 32262306a36Sopenharmony_ci /* Enable interrupts */ 32362306a36Sopenharmony_ci buf_int_enable_value |= NPS_ENET_ENABLE << RX_RDY_SHIFT; 32462306a36Sopenharmony_ci buf_int_enable_value |= NPS_ENET_ENABLE << TX_DONE_SHIFT; 32562306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_BUF_INT_ENABLE, 32662306a36Sopenharmony_ci buf_int_enable_value); 32762306a36Sopenharmony_ci 32862306a36Sopenharmony_ci /* Write device MAC address to HW */ 32962306a36Sopenharmony_ci nps_enet_set_hw_mac_address(ndev); 33062306a36Sopenharmony_ci 33162306a36Sopenharmony_ci /* Rx and Tx HW features */ 33262306a36Sopenharmony_ci ge_mac_cfg_0_value |= NPS_ENET_ENABLE << CFG_0_TX_PAD_EN_SHIFT; 33362306a36Sopenharmony_ci ge_mac_cfg_0_value |= NPS_ENET_ENABLE << CFG_0_TX_CRC_EN_SHIFT; 33462306a36Sopenharmony_ci ge_mac_cfg_0_value |= NPS_ENET_ENABLE << CFG_0_RX_CRC_STRIP_SHIFT; 33562306a36Sopenharmony_ci 33662306a36Sopenharmony_ci /* IFG configuration */ 33762306a36Sopenharmony_ci ge_mac_cfg_0_value |= 33862306a36Sopenharmony_ci NPS_ENET_GE_MAC_CFG_0_RX_IFG << CFG_0_RX_IFG_SHIFT; 33962306a36Sopenharmony_ci ge_mac_cfg_0_value |= 34062306a36Sopenharmony_ci NPS_ENET_GE_MAC_CFG_0_TX_IFG << CFG_0_TX_IFG_SHIFT; 34162306a36Sopenharmony_ci 34262306a36Sopenharmony_ci /* preamble configuration */ 34362306a36Sopenharmony_ci ge_mac_cfg_0_value |= NPS_ENET_ENABLE << CFG_0_RX_PR_CHECK_EN_SHIFT; 34462306a36Sopenharmony_ci ge_mac_cfg_0_value |= 34562306a36Sopenharmony_ci NPS_ENET_GE_MAC_CFG_0_TX_PR_LEN << CFG_0_TX_PR_LEN_SHIFT; 34662306a36Sopenharmony_ci 34762306a36Sopenharmony_ci /* enable flow control frames */ 34862306a36Sopenharmony_ci ge_mac_cfg_0_value |= NPS_ENET_ENABLE << CFG_0_TX_FC_EN_SHIFT; 34962306a36Sopenharmony_ci ge_mac_cfg_0_value |= NPS_ENET_ENABLE << CFG_0_RX_FC_EN_SHIFT; 35062306a36Sopenharmony_ci ge_mac_cfg_0_value |= 35162306a36Sopenharmony_ci NPS_ENET_GE_MAC_CFG_0_TX_FC_RETR << CFG_0_TX_FC_RETR_SHIFT; 35262306a36Sopenharmony_ci *ge_mac_cfg_3_value = (*ge_mac_cfg_3_value & ~CFG_3_CF_DROP_MASK) 35362306a36Sopenharmony_ci | NPS_ENET_ENABLE << CFG_3_CF_DROP_SHIFT; 35462306a36Sopenharmony_ci 35562306a36Sopenharmony_ci /* Enable Rx and Tx */ 35662306a36Sopenharmony_ci ge_mac_cfg_0_value |= NPS_ENET_ENABLE << CFG_0_RX_EN_SHIFT; 35762306a36Sopenharmony_ci ge_mac_cfg_0_value |= NPS_ENET_ENABLE << CFG_0_TX_EN_SHIFT; 35862306a36Sopenharmony_ci 35962306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_GE_MAC_CFG_3, 36062306a36Sopenharmony_ci *ge_mac_cfg_3_value); 36162306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_GE_MAC_CFG_0, 36262306a36Sopenharmony_ci ge_mac_cfg_0_value); 36362306a36Sopenharmony_ci} 36462306a36Sopenharmony_ci 36562306a36Sopenharmony_cistatic void nps_enet_hw_disable_control(struct net_device *ndev) 36662306a36Sopenharmony_ci{ 36762306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 36862306a36Sopenharmony_ci 36962306a36Sopenharmony_ci /* Disable interrupts */ 37062306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_BUF_INT_ENABLE, 0); 37162306a36Sopenharmony_ci 37262306a36Sopenharmony_ci /* Disable Rx and Tx */ 37362306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_GE_MAC_CFG_0, 0); 37462306a36Sopenharmony_ci} 37562306a36Sopenharmony_ci 37662306a36Sopenharmony_cistatic void nps_enet_send_frame(struct net_device *ndev, 37762306a36Sopenharmony_ci struct sk_buff *skb) 37862306a36Sopenharmony_ci{ 37962306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 38062306a36Sopenharmony_ci u32 tx_ctrl_value = 0; 38162306a36Sopenharmony_ci short length = skb->len; 38262306a36Sopenharmony_ci u32 i, len = DIV_ROUND_UP(length, sizeof(u32)); 38362306a36Sopenharmony_ci u32 *src = (void *)skb->data; 38462306a36Sopenharmony_ci bool src_is_aligned = IS_ALIGNED((unsigned long)src, sizeof(u32)); 38562306a36Sopenharmony_ci 38662306a36Sopenharmony_ci /* In case src is not aligned we need an intermediate buffer */ 38762306a36Sopenharmony_ci if (src_is_aligned) 38862306a36Sopenharmony_ci iowrite32_rep(priv->regs_base + NPS_ENET_REG_TX_BUF, src, len); 38962306a36Sopenharmony_ci else /* !src_is_aligned */ 39062306a36Sopenharmony_ci for (i = 0; i < len; i++, src++) 39162306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_TX_BUF, 39262306a36Sopenharmony_ci get_unaligned_be32(src)); 39362306a36Sopenharmony_ci 39462306a36Sopenharmony_ci /* Write the length of the Frame */ 39562306a36Sopenharmony_ci tx_ctrl_value |= length << TX_CTL_NT_SHIFT; 39662306a36Sopenharmony_ci 39762306a36Sopenharmony_ci tx_ctrl_value |= NPS_ENET_ENABLE << TX_CTL_CT_SHIFT; 39862306a36Sopenharmony_ci /* Send Frame */ 39962306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_TX_CTL, tx_ctrl_value); 40062306a36Sopenharmony_ci} 40162306a36Sopenharmony_ci 40262306a36Sopenharmony_ci/** 40362306a36Sopenharmony_ci * nps_enet_set_mac_address - Set the MAC address for this device. 40462306a36Sopenharmony_ci * @ndev: Pointer to net_device structure. 40562306a36Sopenharmony_ci * @p: 6 byte Address to be written as MAC address. 40662306a36Sopenharmony_ci * 40762306a36Sopenharmony_ci * This function copies the HW address from the sockaddr structure to the 40862306a36Sopenharmony_ci * net_device structure and updates the address in HW. 40962306a36Sopenharmony_ci * 41062306a36Sopenharmony_ci * returns: -EBUSY if the net device is busy or 0 if the address is set 41162306a36Sopenharmony_ci * successfully. 41262306a36Sopenharmony_ci */ 41362306a36Sopenharmony_cistatic s32 nps_enet_set_mac_address(struct net_device *ndev, void *p) 41462306a36Sopenharmony_ci{ 41562306a36Sopenharmony_ci struct sockaddr *addr = p; 41662306a36Sopenharmony_ci s32 res; 41762306a36Sopenharmony_ci 41862306a36Sopenharmony_ci if (netif_running(ndev)) 41962306a36Sopenharmony_ci return -EBUSY; 42062306a36Sopenharmony_ci 42162306a36Sopenharmony_ci res = eth_mac_addr(ndev, p); 42262306a36Sopenharmony_ci if (!res) { 42362306a36Sopenharmony_ci eth_hw_addr_set(ndev, addr->sa_data); 42462306a36Sopenharmony_ci nps_enet_set_hw_mac_address(ndev); 42562306a36Sopenharmony_ci } 42662306a36Sopenharmony_ci 42762306a36Sopenharmony_ci return res; 42862306a36Sopenharmony_ci} 42962306a36Sopenharmony_ci 43062306a36Sopenharmony_ci/** 43162306a36Sopenharmony_ci * nps_enet_set_rx_mode - Change the receive filtering mode. 43262306a36Sopenharmony_ci * @ndev: Pointer to the network device. 43362306a36Sopenharmony_ci * 43462306a36Sopenharmony_ci * This function enables/disables promiscuous mode 43562306a36Sopenharmony_ci */ 43662306a36Sopenharmony_cistatic void nps_enet_set_rx_mode(struct net_device *ndev) 43762306a36Sopenharmony_ci{ 43862306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 43962306a36Sopenharmony_ci u32 ge_mac_cfg_2_value = priv->ge_mac_cfg_2_value; 44062306a36Sopenharmony_ci 44162306a36Sopenharmony_ci if (ndev->flags & IFF_PROMISC) { 44262306a36Sopenharmony_ci ge_mac_cfg_2_value = (ge_mac_cfg_2_value & ~CFG_2_DISK_DA_MASK) 44362306a36Sopenharmony_ci | NPS_ENET_DISABLE << CFG_2_DISK_DA_SHIFT; 44462306a36Sopenharmony_ci ge_mac_cfg_2_value = (ge_mac_cfg_2_value & ~CFG_2_DISK_MC_MASK) 44562306a36Sopenharmony_ci | NPS_ENET_DISABLE << CFG_2_DISK_MC_SHIFT; 44662306a36Sopenharmony_ci 44762306a36Sopenharmony_ci } else { 44862306a36Sopenharmony_ci ge_mac_cfg_2_value = (ge_mac_cfg_2_value & ~CFG_2_DISK_DA_MASK) 44962306a36Sopenharmony_ci | NPS_ENET_ENABLE << CFG_2_DISK_DA_SHIFT; 45062306a36Sopenharmony_ci ge_mac_cfg_2_value = (ge_mac_cfg_2_value & ~CFG_2_DISK_MC_MASK) 45162306a36Sopenharmony_ci | NPS_ENET_ENABLE << CFG_2_DISK_MC_SHIFT; 45262306a36Sopenharmony_ci } 45362306a36Sopenharmony_ci 45462306a36Sopenharmony_ci nps_enet_reg_set(priv, NPS_ENET_REG_GE_MAC_CFG_2, ge_mac_cfg_2_value); 45562306a36Sopenharmony_ci} 45662306a36Sopenharmony_ci 45762306a36Sopenharmony_ci/** 45862306a36Sopenharmony_ci * nps_enet_open - Open the network device. 45962306a36Sopenharmony_ci * @ndev: Pointer to the network device. 46062306a36Sopenharmony_ci * 46162306a36Sopenharmony_ci * returns: 0, on success or non-zero error value on failure. 46262306a36Sopenharmony_ci * 46362306a36Sopenharmony_ci * This function sets the MAC address, requests and enables an IRQ 46462306a36Sopenharmony_ci * for the ENET device and starts the Tx queue. 46562306a36Sopenharmony_ci */ 46662306a36Sopenharmony_cistatic s32 nps_enet_open(struct net_device *ndev) 46762306a36Sopenharmony_ci{ 46862306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 46962306a36Sopenharmony_ci s32 err; 47062306a36Sopenharmony_ci 47162306a36Sopenharmony_ci /* Reset private variables */ 47262306a36Sopenharmony_ci priv->tx_skb = NULL; 47362306a36Sopenharmony_ci priv->ge_mac_cfg_2_value = 0; 47462306a36Sopenharmony_ci priv->ge_mac_cfg_3_value = 0; 47562306a36Sopenharmony_ci 47662306a36Sopenharmony_ci /* ge_mac_cfg_3 default values */ 47762306a36Sopenharmony_ci priv->ge_mac_cfg_3_value |= 47862306a36Sopenharmony_ci NPS_ENET_GE_MAC_CFG_3_RX_IFG_TH << CFG_3_RX_IFG_TH_SHIFT; 47962306a36Sopenharmony_ci 48062306a36Sopenharmony_ci priv->ge_mac_cfg_3_value |= 48162306a36Sopenharmony_ci NPS_ENET_GE_MAC_CFG_3_MAX_LEN << CFG_3_MAX_LEN_SHIFT; 48262306a36Sopenharmony_ci 48362306a36Sopenharmony_ci /* Disable HW device */ 48462306a36Sopenharmony_ci nps_enet_hw_disable_control(ndev); 48562306a36Sopenharmony_ci 48662306a36Sopenharmony_ci /* irq Rx allocation */ 48762306a36Sopenharmony_ci err = request_irq(priv->irq, nps_enet_irq_handler, 48862306a36Sopenharmony_ci 0, "enet-rx-tx", ndev); 48962306a36Sopenharmony_ci if (err) 49062306a36Sopenharmony_ci return err; 49162306a36Sopenharmony_ci 49262306a36Sopenharmony_ci napi_enable(&priv->napi); 49362306a36Sopenharmony_ci 49462306a36Sopenharmony_ci /* Enable HW device */ 49562306a36Sopenharmony_ci nps_enet_hw_reset(ndev); 49662306a36Sopenharmony_ci nps_enet_hw_enable_control(ndev); 49762306a36Sopenharmony_ci 49862306a36Sopenharmony_ci netif_start_queue(ndev); 49962306a36Sopenharmony_ci 50062306a36Sopenharmony_ci return 0; 50162306a36Sopenharmony_ci} 50262306a36Sopenharmony_ci 50362306a36Sopenharmony_ci/** 50462306a36Sopenharmony_ci * nps_enet_stop - Close the network device. 50562306a36Sopenharmony_ci * @ndev: Pointer to the network device. 50662306a36Sopenharmony_ci * 50762306a36Sopenharmony_ci * This function stops the Tx queue, disables interrupts for the ENET device. 50862306a36Sopenharmony_ci */ 50962306a36Sopenharmony_cistatic s32 nps_enet_stop(struct net_device *ndev) 51062306a36Sopenharmony_ci{ 51162306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 51262306a36Sopenharmony_ci 51362306a36Sopenharmony_ci napi_disable(&priv->napi); 51462306a36Sopenharmony_ci netif_stop_queue(ndev); 51562306a36Sopenharmony_ci nps_enet_hw_disable_control(ndev); 51662306a36Sopenharmony_ci free_irq(priv->irq, ndev); 51762306a36Sopenharmony_ci 51862306a36Sopenharmony_ci return 0; 51962306a36Sopenharmony_ci} 52062306a36Sopenharmony_ci 52162306a36Sopenharmony_ci/** 52262306a36Sopenharmony_ci * nps_enet_start_xmit - Starts the data transmission. 52362306a36Sopenharmony_ci * @skb: sk_buff pointer that contains data to be Transmitted. 52462306a36Sopenharmony_ci * @ndev: Pointer to net_device structure. 52562306a36Sopenharmony_ci * 52662306a36Sopenharmony_ci * returns: NETDEV_TX_OK, on success 52762306a36Sopenharmony_ci * NETDEV_TX_BUSY, if any of the descriptors are not free. 52862306a36Sopenharmony_ci * 52962306a36Sopenharmony_ci * This function is invoked from upper layers to initiate transmission. 53062306a36Sopenharmony_ci */ 53162306a36Sopenharmony_cistatic netdev_tx_t nps_enet_start_xmit(struct sk_buff *skb, 53262306a36Sopenharmony_ci struct net_device *ndev) 53362306a36Sopenharmony_ci{ 53462306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 53562306a36Sopenharmony_ci 53662306a36Sopenharmony_ci /* This driver handles one frame at a time */ 53762306a36Sopenharmony_ci netif_stop_queue(ndev); 53862306a36Sopenharmony_ci 53962306a36Sopenharmony_ci priv->tx_skb = skb; 54062306a36Sopenharmony_ci 54162306a36Sopenharmony_ci /* make sure tx_skb is actually written to the memory 54262306a36Sopenharmony_ci * before the HW is informed and the IRQ is fired. 54362306a36Sopenharmony_ci */ 54462306a36Sopenharmony_ci wmb(); 54562306a36Sopenharmony_ci 54662306a36Sopenharmony_ci nps_enet_send_frame(ndev, skb); 54762306a36Sopenharmony_ci 54862306a36Sopenharmony_ci return NETDEV_TX_OK; 54962306a36Sopenharmony_ci} 55062306a36Sopenharmony_ci 55162306a36Sopenharmony_ci#ifdef CONFIG_NET_POLL_CONTROLLER 55262306a36Sopenharmony_cistatic void nps_enet_poll_controller(struct net_device *ndev) 55362306a36Sopenharmony_ci{ 55462306a36Sopenharmony_ci disable_irq(ndev->irq); 55562306a36Sopenharmony_ci nps_enet_irq_handler(ndev->irq, ndev); 55662306a36Sopenharmony_ci enable_irq(ndev->irq); 55762306a36Sopenharmony_ci} 55862306a36Sopenharmony_ci#endif 55962306a36Sopenharmony_ci 56062306a36Sopenharmony_cistatic const struct net_device_ops nps_netdev_ops = { 56162306a36Sopenharmony_ci .ndo_open = nps_enet_open, 56262306a36Sopenharmony_ci .ndo_stop = nps_enet_stop, 56362306a36Sopenharmony_ci .ndo_start_xmit = nps_enet_start_xmit, 56462306a36Sopenharmony_ci .ndo_set_mac_address = nps_enet_set_mac_address, 56562306a36Sopenharmony_ci .ndo_set_rx_mode = nps_enet_set_rx_mode, 56662306a36Sopenharmony_ci#ifdef CONFIG_NET_POLL_CONTROLLER 56762306a36Sopenharmony_ci .ndo_poll_controller = nps_enet_poll_controller, 56862306a36Sopenharmony_ci#endif 56962306a36Sopenharmony_ci}; 57062306a36Sopenharmony_ci 57162306a36Sopenharmony_cistatic s32 nps_enet_probe(struct platform_device *pdev) 57262306a36Sopenharmony_ci{ 57362306a36Sopenharmony_ci struct device *dev = &pdev->dev; 57462306a36Sopenharmony_ci struct net_device *ndev; 57562306a36Sopenharmony_ci struct nps_enet_priv *priv; 57662306a36Sopenharmony_ci s32 err = 0; 57762306a36Sopenharmony_ci 57862306a36Sopenharmony_ci if (!dev->of_node) 57962306a36Sopenharmony_ci return -ENODEV; 58062306a36Sopenharmony_ci 58162306a36Sopenharmony_ci ndev = alloc_etherdev(sizeof(struct nps_enet_priv)); 58262306a36Sopenharmony_ci if (!ndev) 58362306a36Sopenharmony_ci return -ENOMEM; 58462306a36Sopenharmony_ci 58562306a36Sopenharmony_ci platform_set_drvdata(pdev, ndev); 58662306a36Sopenharmony_ci SET_NETDEV_DEV(ndev, dev); 58762306a36Sopenharmony_ci priv = netdev_priv(ndev); 58862306a36Sopenharmony_ci 58962306a36Sopenharmony_ci /* The EZ NET specific entries in the device structure. */ 59062306a36Sopenharmony_ci ndev->netdev_ops = &nps_netdev_ops; 59162306a36Sopenharmony_ci ndev->watchdog_timeo = (400 * HZ / 1000); 59262306a36Sopenharmony_ci /* FIXME :: no multicast support yet */ 59362306a36Sopenharmony_ci ndev->flags &= ~IFF_MULTICAST; 59462306a36Sopenharmony_ci 59562306a36Sopenharmony_ci priv->regs_base = devm_platform_ioremap_resource(pdev, 0); 59662306a36Sopenharmony_ci if (IS_ERR(priv->regs_base)) { 59762306a36Sopenharmony_ci err = PTR_ERR(priv->regs_base); 59862306a36Sopenharmony_ci goto out_netdev; 59962306a36Sopenharmony_ci } 60062306a36Sopenharmony_ci dev_dbg(dev, "Registers base address is 0x%p\n", priv->regs_base); 60162306a36Sopenharmony_ci 60262306a36Sopenharmony_ci /* set kernel MAC address to dev */ 60362306a36Sopenharmony_ci err = of_get_ethdev_address(dev->of_node, ndev); 60462306a36Sopenharmony_ci if (err) 60562306a36Sopenharmony_ci eth_hw_addr_random(ndev); 60662306a36Sopenharmony_ci 60762306a36Sopenharmony_ci /* Get IRQ number */ 60862306a36Sopenharmony_ci priv->irq = platform_get_irq(pdev, 0); 60962306a36Sopenharmony_ci if (priv->irq < 0) { 61062306a36Sopenharmony_ci err = -ENODEV; 61162306a36Sopenharmony_ci goto out_netdev; 61262306a36Sopenharmony_ci } 61362306a36Sopenharmony_ci 61462306a36Sopenharmony_ci netif_napi_add_weight(ndev, &priv->napi, nps_enet_poll, 61562306a36Sopenharmony_ci NPS_ENET_NAPI_POLL_WEIGHT); 61662306a36Sopenharmony_ci 61762306a36Sopenharmony_ci /* Register the driver. Should be the last thing in probe */ 61862306a36Sopenharmony_ci err = register_netdev(ndev); 61962306a36Sopenharmony_ci if (err) { 62062306a36Sopenharmony_ci dev_err(dev, "Failed to register ndev for %s, err = 0x%08x\n", 62162306a36Sopenharmony_ci ndev->name, (s32)err); 62262306a36Sopenharmony_ci goto out_netif_api; 62362306a36Sopenharmony_ci } 62462306a36Sopenharmony_ci 62562306a36Sopenharmony_ci dev_info(dev, "(rx/tx=%d)\n", priv->irq); 62662306a36Sopenharmony_ci return 0; 62762306a36Sopenharmony_ci 62862306a36Sopenharmony_ciout_netif_api: 62962306a36Sopenharmony_ci netif_napi_del(&priv->napi); 63062306a36Sopenharmony_ciout_netdev: 63162306a36Sopenharmony_ci free_netdev(ndev); 63262306a36Sopenharmony_ci 63362306a36Sopenharmony_ci return err; 63462306a36Sopenharmony_ci} 63562306a36Sopenharmony_ci 63662306a36Sopenharmony_cistatic s32 nps_enet_remove(struct platform_device *pdev) 63762306a36Sopenharmony_ci{ 63862306a36Sopenharmony_ci struct net_device *ndev = platform_get_drvdata(pdev); 63962306a36Sopenharmony_ci struct nps_enet_priv *priv = netdev_priv(ndev); 64062306a36Sopenharmony_ci 64162306a36Sopenharmony_ci unregister_netdev(ndev); 64262306a36Sopenharmony_ci netif_napi_del(&priv->napi); 64362306a36Sopenharmony_ci free_netdev(ndev); 64462306a36Sopenharmony_ci 64562306a36Sopenharmony_ci return 0; 64662306a36Sopenharmony_ci} 64762306a36Sopenharmony_ci 64862306a36Sopenharmony_cistatic const struct of_device_id nps_enet_dt_ids[] = { 64962306a36Sopenharmony_ci { .compatible = "ezchip,nps-mgt-enet" }, 65062306a36Sopenharmony_ci { /* Sentinel */ } 65162306a36Sopenharmony_ci}; 65262306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, nps_enet_dt_ids); 65362306a36Sopenharmony_ci 65462306a36Sopenharmony_cistatic struct platform_driver nps_enet_driver = { 65562306a36Sopenharmony_ci .probe = nps_enet_probe, 65662306a36Sopenharmony_ci .remove = nps_enet_remove, 65762306a36Sopenharmony_ci .driver = { 65862306a36Sopenharmony_ci .name = DRV_NAME, 65962306a36Sopenharmony_ci .of_match_table = nps_enet_dt_ids, 66062306a36Sopenharmony_ci }, 66162306a36Sopenharmony_ci}; 66262306a36Sopenharmony_ci 66362306a36Sopenharmony_cimodule_platform_driver(nps_enet_driver); 66462306a36Sopenharmony_ci 66562306a36Sopenharmony_ciMODULE_AUTHOR("EZchip Semiconductor"); 66662306a36Sopenharmony_ciMODULE_LICENSE("GPL v2"); 667