Commit 1f975aba authored by David S. Miller's avatar David S. Miller

Merge branch 'pch_gbe-cleanups'

Andy Shevchenko says:

====================
net: pch_gbe: fix and a few cleanups

The series provides one fix (patch 1) for GPIO to be able to wait for
the GPIO driver to appear. This is separated from the conversion to
the GPIO descriptors (patch 2) in order to have a possibility for
backporting. Patches 3 and 4 fix minor warnings from Sparse while
moving to a new APIs. Patch 5 is MODULE_VERSION() clean up.

Tested on Intel Minnowboard (v1).

Since v3:
- rebased on top of v5.13-rc1
- added Tested-by (Flavio)
- added Reported-by to certain changes (LKP)

Since v2:
- added a few cleanups on top of the fix
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents 27d64141 40b161bb
...@@ -596,8 +596,6 @@ struct pch_gbe_adapter { ...@@ -596,8 +596,6 @@ struct pch_gbe_adapter {
#define pch_gbe_hw_to_adapter(hw) container_of(hw, struct pch_gbe_adapter, hw) #define pch_gbe_hw_to_adapter(hw) container_of(hw, struct pch_gbe_adapter, hw)
extern const char pch_driver_version[];
/* pch_gbe_main.c */ /* pch_gbe_main.c */
int pch_gbe_up(struct pch_gbe_adapter *adapter); int pch_gbe_up(struct pch_gbe_adapter *adapter);
void pch_gbe_down(struct pch_gbe_adapter *adapter); void pch_gbe_down(struct pch_gbe_adapter *adapter);
......
...@@ -8,6 +8,8 @@ ...@@ -8,6 +8,8 @@
#include "pch_gbe.h" #include "pch_gbe.h"
#include "pch_gbe_phy.h" #include "pch_gbe_phy.h"
static const char pch_driver_version[] = "1.01";
/* /*
* pch_gbe_stats - Stats item information * pch_gbe_stats - Stats item information
*/ */
......
...@@ -8,15 +8,16 @@ ...@@ -8,15 +8,16 @@
#include "pch_gbe.h" #include "pch_gbe.h"
#include "pch_gbe_phy.h" #include "pch_gbe_phy.h"
#include <linux/gpio/consumer.h>
#include <linux/gpio/machine.h>
#include <linux/iopoll.h>
#include <linux/module.h> #include <linux/module.h>
#include <linux/net_tstamp.h> #include <linux/net_tstamp.h>
#include <linux/ptp_classify.h> #include <linux/ptp_classify.h>
#include <linux/ptp_pch.h> #include <linux/ptp_pch.h>
#include <linux/gpio.h> #include <linux/gpio.h>
#define DRV_VERSION "1.01"
const char pch_driver_version[] = DRV_VERSION;
#define PCH_GBE_MAR_ENTRIES 16 #define PCH_GBE_MAR_ENTRIES 16
#define PCH_GBE_SHORT_PKT 64 #define PCH_GBE_SHORT_PKT 64
#define DSC_INIT16 0xC000 #define DSC_INIT16 0xC000
...@@ -97,8 +98,6 @@ const char pch_driver_version[] = DRV_VERSION; ...@@ -97,8 +98,6 @@ const char pch_driver_version[] = DRV_VERSION;
#define PTP_L4_MULTICAST_SA "01:00:5e:00:01:81" #define PTP_L4_MULTICAST_SA "01:00:5e:00:01:81"
#define PTP_L2_MULTICAST_SA "01:1b:19:00:00:00" #define PTP_L2_MULTICAST_SA "01:1b:19:00:00:00"
#define MINNOW_PHY_RESET_GPIO 13
static int pch_gbe_mdio_read(struct net_device *netdev, int addr, int reg); static int pch_gbe_mdio_read(struct net_device *netdev, int addr, int reg);
static void pch_gbe_mdio_write(struct net_device *netdev, int addr, int reg, static void pch_gbe_mdio_write(struct net_device *netdev, int addr, int reg,
int data); int data);
...@@ -108,7 +107,7 @@ static int pch_ptp_match(struct sk_buff *skb, u16 uid_hi, u32 uid_lo, u16 seqid) ...@@ -108,7 +107,7 @@ static int pch_ptp_match(struct sk_buff *skb, u16 uid_hi, u32 uid_lo, u16 seqid)
{ {
u8 *data = skb->data; u8 *data = skb->data;
unsigned int offset; unsigned int offset;
u16 *hi, *id; u16 hi, id;
u32 lo; u32 lo;
if (ptp_classify_raw(skb) == PTP_CLASS_NONE) if (ptp_classify_raw(skb) == PTP_CLASS_NONE)
...@@ -119,14 +118,11 @@ static int pch_ptp_match(struct sk_buff *skb, u16 uid_hi, u32 uid_lo, u16 seqid) ...@@ -119,14 +118,11 @@ static int pch_ptp_match(struct sk_buff *skb, u16 uid_hi, u32 uid_lo, u16 seqid)
if (skb->len < offset + OFF_PTP_SEQUENCE_ID + sizeof(seqid)) if (skb->len < offset + OFF_PTP_SEQUENCE_ID + sizeof(seqid))
return 0; return 0;
hi = (u16 *)(data + offset + OFF_PTP_SOURCE_UUID); hi = get_unaligned_be16(data + offset + OFF_PTP_SOURCE_UUID + 0);
id = (u16 *)(data + offset + OFF_PTP_SEQUENCE_ID); lo = get_unaligned_be32(data + offset + OFF_PTP_SOURCE_UUID + 2);
id = get_unaligned_be16(data + offset + OFF_PTP_SEQUENCE_ID);
memcpy(&lo, &hi[1], sizeof(lo));
return (uid_hi == *hi && return (uid_hi == hi && uid_lo == lo && seqid == id);
uid_lo == lo &&
seqid == *id);
} }
static void static void
...@@ -136,7 +132,6 @@ pch_rx_timestamp(struct pch_gbe_adapter *adapter, struct sk_buff *skb) ...@@ -136,7 +132,6 @@ pch_rx_timestamp(struct pch_gbe_adapter *adapter, struct sk_buff *skb)
struct pci_dev *pdev; struct pci_dev *pdev;
u64 ns; u64 ns;
u32 hi, lo, val; u32 hi, lo, val;
u16 uid, seq;
if (!adapter->hwts_rx_en) if (!adapter->hwts_rx_en)
return; return;
...@@ -152,10 +147,7 @@ pch_rx_timestamp(struct pch_gbe_adapter *adapter, struct sk_buff *skb) ...@@ -152,10 +147,7 @@ pch_rx_timestamp(struct pch_gbe_adapter *adapter, struct sk_buff *skb)
lo = pch_src_uuid_lo_read(pdev); lo = pch_src_uuid_lo_read(pdev);
hi = pch_src_uuid_hi_read(pdev); hi = pch_src_uuid_hi_read(pdev);
uid = hi & 0xffff; if (!pch_ptp_match(skb, hi, lo, hi >> 16))
seq = (hi >> 16) & 0xffff;
if (!pch_ptp_match(skb, htons(uid), htonl(lo), htons(seq)))
goto out; goto out;
ns = pch_rx_snap_read(pdev); ns = pch_rx_snap_read(pdev);
...@@ -298,15 +290,12 @@ static s32 pch_gbe_mac_read_mac_addr(struct pch_gbe_hw *hw) ...@@ -298,15 +290,12 @@ static s32 pch_gbe_mac_read_mac_addr(struct pch_gbe_hw *hw)
* @reg: Pointer of register * @reg: Pointer of register
* @bit: Busy bit * @bit: Busy bit
*/ */
static void pch_gbe_wait_clr_bit(void *reg, u32 bit) static void pch_gbe_wait_clr_bit(void __iomem *reg, u32 bit)
{ {
u32 tmp; u32 tmp;
/* wait busy */ /* wait busy */
tmp = 1000; if (readx_poll_timeout_atomic(ioread32, reg, tmp, !(tmp & bit), 0, 10))
while ((ioread32(reg) & bit) && --tmp)
cpu_relax();
if (!tmp)
pr_err("Error: busy bit is not cleared\n"); pr_err("Error: busy bit is not cleared\n");
} }
...@@ -490,18 +479,13 @@ u16 pch_gbe_mac_ctrl_miim(struct pch_gbe_hw *hw, u32 addr, u32 dir, u32 reg, ...@@ -490,18 +479,13 @@ u16 pch_gbe_mac_ctrl_miim(struct pch_gbe_hw *hw, u32 addr, u32 dir, u32 reg,
u16 data) u16 data)
{ {
struct pch_gbe_adapter *adapter = pch_gbe_hw_to_adapter(hw); struct pch_gbe_adapter *adapter = pch_gbe_hw_to_adapter(hw);
u32 data_out = 0;
unsigned int i;
unsigned long flags; unsigned long flags;
u32 data_out;
spin_lock_irqsave(&hw->miim_lock, flags); spin_lock_irqsave(&hw->miim_lock, flags);
for (i = 100; i; --i) { if (readx_poll_timeout_atomic(ioread32, &hw->reg->MIIM, data_out,
if ((ioread32(&hw->reg->MIIM) & PCH_GBE_MIIM_OPER_READY)) data_out & PCH_GBE_MIIM_OPER_READY, 20, 2000)) {
break;
udelay(20);
}
if (i == 0) {
netdev_err(adapter->netdev, "pch-gbe.miim won't go Ready\n"); netdev_err(adapter->netdev, "pch-gbe.miim won't go Ready\n");
spin_unlock_irqrestore(&hw->miim_lock, flags); spin_unlock_irqrestore(&hw->miim_lock, flags);
return 0; /* No way to indicate timeout error */ return 0; /* No way to indicate timeout error */
...@@ -509,12 +493,8 @@ u16 pch_gbe_mac_ctrl_miim(struct pch_gbe_hw *hw, u32 addr, u32 dir, u32 reg, ...@@ -509,12 +493,8 @@ u16 pch_gbe_mac_ctrl_miim(struct pch_gbe_hw *hw, u32 addr, u32 dir, u32 reg,
iowrite32(((reg << PCH_GBE_MIIM_REG_ADDR_SHIFT) | iowrite32(((reg << PCH_GBE_MIIM_REG_ADDR_SHIFT) |
(addr << PCH_GBE_MIIM_PHY_ADDR_SHIFT) | (addr << PCH_GBE_MIIM_PHY_ADDR_SHIFT) |
dir | data), &hw->reg->MIIM); dir | data), &hw->reg->MIIM);
for (i = 0; i < 100; i++) { readx_poll_timeout_atomic(ioread32, &hw->reg->MIIM, data_out,
udelay(20); data_out & PCH_GBE_MIIM_OPER_READY, 20, 2000);
data_out = ioread32(&hw->reg->MIIM);
if ((data_out & PCH_GBE_MIIM_OPER_READY))
break;
}
spin_unlock_irqrestore(&hw->miim_lock, flags); spin_unlock_irqrestore(&hw->miim_lock, flags);
netdev_dbg(adapter->netdev, "PHY %s: reg=%d, data=0x%04X\n", netdev_dbg(adapter->netdev, "PHY %s: reg=%d, data=0x%04X\n",
...@@ -2532,9 +2512,13 @@ static int pch_gbe_probe(struct pci_dev *pdev, ...@@ -2532,9 +2512,13 @@ static int pch_gbe_probe(struct pci_dev *pdev,
adapter->pdev = pdev; adapter->pdev = pdev;
adapter->hw.back = adapter; adapter->hw.back = adapter;
adapter->hw.reg = pcim_iomap_table(pdev)[PCH_GBE_PCI_BAR]; adapter->hw.reg = pcim_iomap_table(pdev)[PCH_GBE_PCI_BAR];
adapter->pdata = (struct pch_gbe_privdata *)pci_id->driver_data; adapter->pdata = (struct pch_gbe_privdata *)pci_id->driver_data;
if (adapter->pdata && adapter->pdata->platform_init) if (adapter->pdata && adapter->pdata->platform_init) {
adapter->pdata->platform_init(pdev); ret = adapter->pdata->platform_init(pdev);
if (ret)
goto err_free_netdev;
}
adapter->ptp_pdev = adapter->ptp_pdev =
pci_get_domain_bus_and_slot(pci_domain_nr(adapter->pdev->bus), pci_get_domain_bus_and_slot(pci_domain_nr(adapter->pdev->bus),
...@@ -2624,26 +2608,45 @@ static int pch_gbe_probe(struct pci_dev *pdev, ...@@ -2624,26 +2608,45 @@ static int pch_gbe_probe(struct pci_dev *pdev,
return ret; return ret;
} }
static void pch_gbe_gpio_remove_table(void *table)
{
gpiod_remove_lookup_table(table);
}
static int pch_gbe_gpio_add_table(struct device *dev, void *table)
{
gpiod_add_lookup_table(table);
return devm_add_action_or_reset(dev, pch_gbe_gpio_remove_table, table);
}
static struct gpiod_lookup_table pch_gbe_minnow_gpio_table = {
.dev_id = "0000:02:00.1",
.table = {
GPIO_LOOKUP("sch_gpio.33158", 13, NULL, GPIO_ACTIVE_LOW),
{}
},
};
/* The AR803X PHY on the MinnowBoard requires a physical pin to be toggled to /* The AR803X PHY on the MinnowBoard requires a physical pin to be toggled to
* ensure it is awake for probe and init. Request the line and reset the PHY. * ensure it is awake for probe and init. Request the line and reset the PHY.
*/ */
static int pch_gbe_minnow_platform_init(struct pci_dev *pdev) static int pch_gbe_minnow_platform_init(struct pci_dev *pdev)
{ {
unsigned long flags = GPIOF_DIR_OUT | GPIOF_INIT_HIGH | GPIOF_EXPORT; struct gpio_desc *gpiod;
unsigned gpio = MINNOW_PHY_RESET_GPIO;
int ret; int ret;
ret = devm_gpio_request_one(&pdev->dev, gpio, flags, ret = pch_gbe_gpio_add_table(&pdev->dev, &pch_gbe_minnow_gpio_table);
"minnow_phy_reset"); if (ret)
if (ret) {
dev_err(&pdev->dev,
"ERR: Can't request PHY reset GPIO line '%d'\n", gpio);
return ret; return ret;
}
gpio_set_value(gpio, 0); gpiod = devm_gpiod_get(&pdev->dev, NULL, GPIOD_OUT_HIGH);
if (IS_ERR(gpiod))
return dev_err_probe(&pdev->dev, PTR_ERR(gpiod),
"Can't request PHY reset GPIO line\n");
gpiod_set_value(gpiod, 1);
usleep_range(1250, 1500); usleep_range(1250, 1500);
gpio_set_value(gpio, 1); gpiod_set_value(gpiod, 0);
usleep_range(1250, 1500); usleep_range(1250, 1500);
return ret; return ret;
...@@ -2722,7 +2725,6 @@ module_pci_driver(pch_gbe_driver); ...@@ -2722,7 +2725,6 @@ module_pci_driver(pch_gbe_driver);
MODULE_DESCRIPTION("EG20T PCH Gigabit ethernet Driver"); MODULE_DESCRIPTION("EG20T PCH Gigabit ethernet Driver");
MODULE_AUTHOR("LAPIS SEMICONDUCTOR, <tshimizu818@gmail.com>"); MODULE_AUTHOR("LAPIS SEMICONDUCTOR, <tshimizu818@gmail.com>");
MODULE_LICENSE("GPL"); MODULE_LICENSE("GPL");
MODULE_VERSION(DRV_VERSION);
MODULE_DEVICE_TABLE(pci, pch_gbe_pcidev_id); MODULE_DEVICE_TABLE(pci, pch_gbe_pcidev_id);
/* pch_gbe_main.c */ /* pch_gbe_main.c */
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment