6f8d3f3338
This patch add support for 1G Ethernet driver based on Keystone NetCP hardware. The gigabit Ethernet (GbE) switch subsystem is one of the main components of the network coprocessor (NETCP) peripheral. The purpose of the gigabit Ethernet switch subsystem in the NETCP is to provide an interface to transfer data between the host device and another connected device in compliance with the Ethernet protocol. GbE consists of 5 port Ethernet Switch module, 4 Serial Gigabit Media Independent Interface (SGMII) modules, MDIO module and SerDes. Driver for 5 port GbE switch and SGMII module is added in this patch. These hardware modules along with netcp core driver provides Network driver functions for 1G Ethernet. Detailed hardware spec is available at http://www.ti.com/lit/ug/sprugv9d/sprugv9d.pdf Cc: David Miller <davem@davemloft.net> Cc: Rob Herring <robh+dt@kernel.org> Cc: Grant Likely <grant.likely@linaro.org> Cc: Santosh Shilimkar <santosh.shilimkar@kernel.org> Cc: Pawel Moll <pawel.moll@arm.com> Cc: Mark Rutland <mark.rutland@arm.com> Cc: Ian Campbell <ijc+devicetree@hellion.org.uk> Cc: Kumar Gala <galak@codeaurora.org> Signed-off-by: Wingman Kwok <w-kwok2@ti.com> Signed-off-by: Murali Karicheri <m-karicheri2@ti.com> Signed-off-by: David S. Miller <davem@davemloft.net>
132 lines
3.3 KiB
C
132 lines
3.3 KiB
C
/*
|
|
* SGMI module initialisation
|
|
*
|
|
* Copyright (C) 2014 Texas Instruments Incorporated
|
|
* Authors: Sandeep Nair <sandeep_n@ti.com>
|
|
* Sandeep Paulraj <s-paulraj@ti.com>
|
|
* Wingman Kwok <w-kwok2@ti.com>
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation version 2.
|
|
*
|
|
* This program is distributed "as is" WITHOUT ANY WARRANTY of any
|
|
* kind, whether express or implied; without even the implied warranty
|
|
* of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*/
|
|
|
|
#include "netcp.h"
|
|
|
|
#define SGMII_REG_STATUS_LOCK BIT(4)
|
|
#define SGMII_REG_STATUS_LINK BIT(0)
|
|
#define SGMII_REG_STATUS_AUTONEG BIT(2)
|
|
#define SGMII_REG_CONTROL_AUTONEG BIT(0)
|
|
|
|
#define SGMII23_OFFSET(x) ((x - 2) * 0x100)
|
|
#define SGMII_OFFSET(x) ((x <= 1) ? (x * 0x100) : (SGMII23_OFFSET(x)))
|
|
|
|
/* SGMII registers */
|
|
#define SGMII_SRESET_REG(x) (SGMII_OFFSET(x) + 0x004)
|
|
#define SGMII_CTL_REG(x) (SGMII_OFFSET(x) + 0x010)
|
|
#define SGMII_STATUS_REG(x) (SGMII_OFFSET(x) + 0x014)
|
|
#define SGMII_MRADV_REG(x) (SGMII_OFFSET(x) + 0x018)
|
|
|
|
static void sgmii_write_reg(void __iomem *base, int reg, u32 val)
|
|
{
|
|
writel(val, base + reg);
|
|
}
|
|
|
|
static u32 sgmii_read_reg(void __iomem *base, int reg)
|
|
{
|
|
return readl(base + reg);
|
|
}
|
|
|
|
static void sgmii_write_reg_bit(void __iomem *base, int reg, u32 val)
|
|
{
|
|
writel((readl(base + reg) | val), base + reg);
|
|
}
|
|
|
|
/* port is 0 based */
|
|
int netcp_sgmii_reset(void __iomem *sgmii_ofs, int port)
|
|
{
|
|
/* Soft reset */
|
|
sgmii_write_reg_bit(sgmii_ofs, SGMII_SRESET_REG(port), 0x1);
|
|
while (sgmii_read_reg(sgmii_ofs, SGMII_SRESET_REG(port)) != 0x0)
|
|
;
|
|
return 0;
|
|
}
|
|
|
|
int netcp_sgmii_get_port_link(void __iomem *sgmii_ofs, int port)
|
|
{
|
|
u32 status = 0, link = 0;
|
|
|
|
status = sgmii_read_reg(sgmii_ofs, SGMII_STATUS_REG(port));
|
|
if ((status & SGMII_REG_STATUS_LINK) != 0)
|
|
link = 1;
|
|
return link;
|
|
}
|
|
|
|
int netcp_sgmii_config(void __iomem *sgmii_ofs, int port, u32 interface)
|
|
{
|
|
unsigned int i, status, mask;
|
|
u32 mr_adv_ability;
|
|
u32 control;
|
|
|
|
switch (interface) {
|
|
case SGMII_LINK_MAC_MAC_AUTONEG:
|
|
mr_adv_ability = 0x9801;
|
|
control = 0x21;
|
|
break;
|
|
|
|
case SGMII_LINK_MAC_PHY:
|
|
case SGMII_LINK_MAC_PHY_NO_MDIO:
|
|
mr_adv_ability = 1;
|
|
control = 1;
|
|
break;
|
|
|
|
case SGMII_LINK_MAC_MAC_FORCED:
|
|
mr_adv_ability = 0x9801;
|
|
control = 0x20;
|
|
break;
|
|
|
|
case SGMII_LINK_MAC_FIBER:
|
|
mr_adv_ability = 0x20;
|
|
control = 0x1;
|
|
break;
|
|
|
|
default:
|
|
WARN_ONCE(1, "Invalid sgmii interface: %d\n", interface);
|
|
return -EINVAL;
|
|
}
|
|
|
|
sgmii_write_reg(sgmii_ofs, SGMII_CTL_REG(port), 0);
|
|
|
|
/* Wait for the SerDes pll to lock */
|
|
for (i = 0; i < 1000; i++) {
|
|
usleep_range(1000, 2000);
|
|
status = sgmii_read_reg(sgmii_ofs, SGMII_STATUS_REG(port));
|
|
if ((status & SGMII_REG_STATUS_LOCK) != 0)
|
|
break;
|
|
}
|
|
|
|
if ((status & SGMII_REG_STATUS_LOCK) == 0)
|
|
pr_err("serdes PLL not locked\n");
|
|
|
|
sgmii_write_reg(sgmii_ofs, SGMII_MRADV_REG(port), mr_adv_ability);
|
|
sgmii_write_reg(sgmii_ofs, SGMII_CTL_REG(port), control);
|
|
|
|
mask = SGMII_REG_STATUS_LINK;
|
|
if (control & SGMII_REG_CONTROL_AUTONEG)
|
|
mask |= SGMII_REG_STATUS_AUTONEG;
|
|
|
|
for (i = 0; i < 1000; i++) {
|
|
usleep_range(200, 500);
|
|
status = sgmii_read_reg(sgmii_ofs, SGMII_STATUS_REG(port));
|
|
if ((status & mask) == mask)
|
|
break;
|
|
}
|
|
|
|
return 0;
|
|
}
|