2002-11-03 08:24:07 +08:00
|
|
|
/*
|
2010-03-31 23:56:08 +08:00
|
|
|
* (C) Copyright 2001-2010
|
2002-11-03 08:24:07 +08:00
|
|
|
* Wolfgang Denk, DENX Software Engineering, wd@denx.de.
|
|
|
|
*
|
2013-07-08 15:37:19 +08:00
|
|
|
* SPDX-License-Identifier: GPL-2.0+
|
2002-11-03 08:24:07 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <common.h>
|
|
|
|
#include <command.h>
|
|
|
|
#include <net.h>
|
2005-12-01 01:06:04 +08:00
|
|
|
#include <miiphy.h>
|
Create PHY Lib for U-Boot
Extends the mii_dev structure to participate in a full-blown MDIO and
PHY driver scheme. The mii_dev structure and miiphy calls are modified
in such a way to allow the original mii command and miiphy
infrastructure to work as before, but also to support a new set of APIs
which allow (among other things) sharing of PHY driver code and 10G support
The mii command will continue to support normal PHY management functions
(Clause 22 of 802.3), but will not be changed to support 10G
(Clause 45).
The basic design is similar to PHY Lib from Linux, but simplified for
U-Boot's network and driver infrastructure.
We now have MDIO drivers and PHY drivers
An MDIO driver provides:
read
write
reset
A PHY driver provides:
(optionally): probe
config - initial setup, starting of auto-negotiation
startup - waiting for AN, and reading link state
shutdown - any cleanup needed
The ethernet drivers interact with the PHY Lib using these functions:
phy_connect()
phy_config()
phy_startup()
phy_shutdown()
Each PHY driver can be configured separately, or all at once using
config_phylib_all_drivers.h (added in the patch which adds the drivers)
We also provide generic drivers for Clause 22 (10/100/1000), and
Clause 45 (10G) PHYs.
We also implement phy_reset(), and call it in phy_connect(). Because
phy_reset() is essentially the same as miiphy_reset, but:
a) must support 10G PHYs, and
b) should use the phylib primitives,
we implement miiphy_reset, using phy_reset(), but only when
CONFIG_PHYLIB is set. Otherwise, we just use the old version. In this
way, we save on compile size, even if we don't manage to save code size.
Pulled ethtool.h and mdio.h from:
git://git.kernel.org/pub/scm/linux/kernel/git/davem/net-next-2.6
782d640afd15af7a1faf01cfe566ca4ac511319d
With many, many deletions so as to enable compilation under u-boot
Signed-off-by: Andy Fleming <afleming@freescale.com>
Signed-off-by: Kumar Gala <galak@kernel.crashing.org>
Acked-by: Detlev Zundel <dzu@denx.de>
2011-04-08 15:10:27 +08:00
|
|
|
#include <phy.h>
|
2002-11-03 08:24:07 +08:00
|
|
|
|
2009-01-30 08:43:44 +08:00
|
|
|
void eth_parse_enetaddr(const char *addr, uchar *enetaddr)
|
|
|
|
{
|
|
|
|
char *end;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < 6; ++i) {
|
|
|
|
enetaddr[i] = addr ? simple_strtoul(addr, &end, 16) : 0;
|
|
|
|
if (addr)
|
|
|
|
addr = (*end) ? end + 1 : end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int eth_getenv_enetaddr(char *name, uchar *enetaddr)
|
|
|
|
{
|
|
|
|
eth_parse_enetaddr(getenv(name), enetaddr);
|
|
|
|
return is_valid_ether_addr(enetaddr);
|
|
|
|
}
|
|
|
|
|
|
|
|
int eth_setenv_enetaddr(char *name, const uchar *enetaddr)
|
|
|
|
{
|
|
|
|
char buf[20];
|
|
|
|
|
|
|
|
sprintf(buf, "%pM", enetaddr);
|
|
|
|
|
|
|
|
return setenv(name, buf);
|
|
|
|
}
|
2009-07-16 09:31:28 +08:00
|
|
|
|
2011-06-14 07:13:10 +08:00
|
|
|
int eth_getenv_enetaddr_by_index(const char *base_name, int index,
|
|
|
|
uchar *enetaddr)
|
2009-07-16 09:31:28 +08:00
|
|
|
{
|
|
|
|
char enetvar[32];
|
2011-06-14 07:13:10 +08:00
|
|
|
sprintf(enetvar, index ? "%s%daddr" : "%saddr", base_name, index);
|
2009-07-16 09:31:28 +08:00
|
|
|
return eth_getenv_enetaddr(enetvar, enetaddr);
|
|
|
|
}
|
2009-01-30 08:43:44 +08:00
|
|
|
|
2012-07-11 05:23:22 +08:00
|
|
|
static inline int eth_setenv_enetaddr_by_index(const char *base_name, int index,
|
2012-04-15 02:06:49 +08:00
|
|
|
uchar *enetaddr)
|
|
|
|
{
|
|
|
|
char enetvar[32];
|
|
|
|
sprintf(enetvar, index ? "%s%daddr" : "%saddr", base_name, index);
|
|
|
|
return eth_setenv_enetaddr(enetvar, enetaddr);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-04-27 02:11:46 +08:00
|
|
|
static int eth_mac_skip(int index)
|
|
|
|
{
|
|
|
|
char enetvar[15];
|
|
|
|
char *skip_state;
|
|
|
|
sprintf(enetvar, index ? "eth%dmacskip" : "ethmacskip", index);
|
|
|
|
return ((skip_state = getenv(enetvar)) != NULL);
|
|
|
|
}
|
|
|
|
|
2012-06-05 19:33:16 +08:00
|
|
|
#ifdef CONFIG_RANDOM_MACADDR
|
|
|
|
void eth_random_enetaddr(uchar *enetaddr)
|
|
|
|
{
|
|
|
|
uint32_t rval;
|
|
|
|
|
|
|
|
srand(get_timer(0));
|
|
|
|
|
|
|
|
rval = rand();
|
|
|
|
enetaddr[0] = rval & 0xff;
|
|
|
|
enetaddr[1] = (rval >> 8) & 0xff;
|
|
|
|
enetaddr[2] = (rval >> 16) & 0xff;
|
|
|
|
|
|
|
|
rval = rand();
|
|
|
|
enetaddr[3] = rval & 0xff;
|
|
|
|
enetaddr[4] = (rval >> 8) & 0xff;
|
|
|
|
enetaddr[5] = (rval >> 16) & 0xff;
|
|
|
|
|
|
|
|
/* make sure it's local and unicast */
|
|
|
|
enetaddr[0] = (enetaddr[0] | 0x02) & ~0x01;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2008-06-24 13:57:27 +08:00
|
|
|
/*
|
|
|
|
* CPU and board-specific Ethernet initializations. Aliased function
|
|
|
|
* signals caller to move on
|
|
|
|
*/
|
|
|
|
static int __def_eth_init(bd_t *bis)
|
|
|
|
{
|
|
|
|
return -1;
|
|
|
|
}
|
2009-04-21 00:08:46 +08:00
|
|
|
int cpu_eth_init(bd_t *bis) __attribute__((weak, alias("__def_eth_init")));
|
|
|
|
int board_eth_init(bd_t *bis) __attribute__((weak, alias("__def_eth_init")));
|
2008-06-24 13:57:27 +08:00
|
|
|
|
2007-12-28 01:19:02 +08:00
|
|
|
#ifdef CONFIG_API
|
|
|
|
static struct {
|
|
|
|
uchar data[PKTSIZE];
|
|
|
|
int length;
|
|
|
|
} eth_rcv_bufs[PKTBUFSRX];
|
|
|
|
|
2012-05-15 16:59:07 +08:00
|
|
|
static unsigned int eth_rcv_current, eth_rcv_last;
|
2007-12-28 01:19:02 +08:00
|
|
|
#endif
|
|
|
|
|
2012-08-03 18:59:08 +08:00
|
|
|
static struct eth_device *eth_devices;
|
|
|
|
struct eth_device *eth_current;
|
2002-11-03 08:24:07 +08:00
|
|
|
|
2010-07-30 03:56:11 +08:00
|
|
|
struct eth_device *eth_get_dev_by_name(const char *devname)
|
2005-10-29 04:30:33 +08:00
|
|
|
{
|
|
|
|
struct eth_device *dev, *target_dev;
|
|
|
|
|
2011-08-22 08:17:17 +08:00
|
|
|
BUG_ON(devname == NULL);
|
|
|
|
|
2005-10-29 04:30:33 +08:00
|
|
|
if (!eth_devices)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
dev = eth_devices;
|
|
|
|
target_dev = NULL;
|
|
|
|
do {
|
|
|
|
if (strcmp(devname, dev->name) == 0) {
|
|
|
|
target_dev = dev;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
dev = dev->next;
|
|
|
|
} while (dev != eth_devices);
|
|
|
|
|
|
|
|
return target_dev;
|
|
|
|
}
|
|
|
|
|
2009-02-12 05:07:24 +08:00
|
|
|
struct eth_device *eth_get_dev_by_index(int index)
|
|
|
|
{
|
|
|
|
struct eth_device *dev, *target_dev;
|
|
|
|
|
|
|
|
if (!eth_devices)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
dev = eth_devices;
|
|
|
|
target_dev = NULL;
|
|
|
|
do {
|
2011-10-27 19:31:35 +08:00
|
|
|
if (dev->index == index) {
|
2009-02-12 05:07:24 +08:00
|
|
|
target_dev = dev;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
dev = dev->next;
|
|
|
|
} while (dev != eth_devices);
|
|
|
|
|
|
|
|
return target_dev;
|
|
|
|
}
|
|
|
|
|
2012-05-15 16:59:07 +08:00
|
|
|
int eth_get_dev_index(void)
|
2002-11-03 08:24:07 +08:00
|
|
|
{
|
2012-05-15 16:59:07 +08:00
|
|
|
if (!eth_current)
|
2011-10-27 19:31:35 +08:00
|
|
|
return -1;
|
2002-11-03 08:24:07 +08:00
|
|
|
|
2011-10-27 19:31:35 +08:00
|
|
|
return eth_current->index;
|
2002-11-03 08:24:07 +08:00
|
|
|
}
|
|
|
|
|
2011-02-17 03:14:33 +08:00
|
|
|
static void eth_current_changed(void)
|
2002-11-03 08:24:07 +08:00
|
|
|
{
|
2011-10-02 18:01:27 +08:00
|
|
|
char *act = getenv("ethact");
|
|
|
|
/* update current ethernet name */
|
|
|
|
if (eth_current) {
|
|
|
|
if (act == NULL || strcmp(act, eth_current->name) != 0)
|
|
|
|
setenv("ethact", eth_current->name);
|
2011-02-17 03:14:33 +08:00
|
|
|
}
|
2011-10-02 18:01:27 +08:00
|
|
|
/*
|
|
|
|
* remove the variable completely if there is no active
|
|
|
|
* interface
|
|
|
|
*/
|
|
|
|
else if (act != NULL)
|
|
|
|
setenv("ethact", NULL);
|
2011-02-17 03:14:33 +08:00
|
|
|
}
|
|
|
|
|
2011-06-14 07:13:10 +08:00
|
|
|
int eth_write_hwaddr(struct eth_device *dev, const char *base_name,
|
|
|
|
int eth_number)
|
|
|
|
{
|
|
|
|
unsigned char env_enetaddr[6];
|
|
|
|
int ret = 0;
|
|
|
|
|
net/eth.c: fix eth_write_hwaddr() to use dev->enetaddr as fall back
Ignore the return value of eth_getenv_enetaddr_by_index(), and if it
fails, fall back to use dev->enetaddr, which could be filled up by
the ethernet device driver:
With the current code, introduced with below commit, eth_write_hwaddr()
will fail immediately if there is no eth<n>addr in the environment variables.
However, e.g. for an overo based product that uses the SMSC911x ethernet
chip (with the MAC address set via EEPROM connected to the SMSC911x chip),
the MAC address is still OK.
On mx28 boards that are depending on the OCOTP bits to set the MAC address
(like the Denx m28 board), the OCOTP bits should be used instead of
failing on the environment variables.
Actually, this was the original behavior, and was later changed by
commit 7616e7850804c7c69e0a22c179dfcba9e8f3f587.
Signed-off-by: Eric Miao <eric.miao@linaro.org>
Acked-by: Simon Glass <sjg@chromium.org>
Acked-by: Dirk Behme <dirk.behme@de.bosch.com>
CC: Stefan Roese <sr@denx.de>
CC: Eric Miao <eric.miao@linaro.org>
CC: Wolfgang Denk <wd@denx.de>
CC: Philip Balister <philip@balister.org>
CC: Zach Sadecki <zach@itwatchdogs.com>
2012-01-19 06:56:33 +08:00
|
|
|
eth_getenv_enetaddr_by_index(base_name, eth_number, env_enetaddr);
|
2011-06-14 07:13:10 +08:00
|
|
|
|
|
|
|
if (memcmp(env_enetaddr, "\0\0\0\0\0\0", 6)) {
|
|
|
|
if (memcmp(dev->enetaddr, "\0\0\0\0\0\0", 6) &&
|
2012-05-15 16:59:07 +08:00
|
|
|
memcmp(dev->enetaddr, env_enetaddr, 6)) {
|
2011-06-14 07:13:10 +08:00
|
|
|
printf("\nWarning: %s MAC addresses don't match:\n",
|
|
|
|
dev->name);
|
|
|
|
printf("Address in SROM is %pM\n",
|
|
|
|
dev->enetaddr);
|
|
|
|
printf("Address in environment is %pM\n",
|
|
|
|
env_enetaddr);
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(dev->enetaddr, env_enetaddr, 6);
|
2012-04-15 02:06:49 +08:00
|
|
|
} else if (is_valid_ether_addr(dev->enetaddr)) {
|
|
|
|
eth_setenv_enetaddr_by_index(base_name, eth_number,
|
|
|
|
dev->enetaddr);
|
|
|
|
printf("\nWarning: %s using MAC address from net device\n",
|
|
|
|
dev->name);
|
2011-06-14 07:13:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (dev->write_hwaddr &&
|
2012-08-10 15:56:21 +08:00
|
|
|
!eth_mac_skip(eth_number)) {
|
|
|
|
if (!is_valid_ether_addr(dev->enetaddr))
|
|
|
|
return -1;
|
|
|
|
|
2011-06-14 07:13:10 +08:00
|
|
|
ret = dev->write_hwaddr(dev);
|
2012-08-10 15:56:21 +08:00
|
|
|
}
|
2011-06-14 07:13:10 +08:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2011-02-17 03:14:33 +08:00
|
|
|
int eth_register(struct eth_device *dev)
|
|
|
|
{
|
|
|
|
struct eth_device *d;
|
2012-05-15 16:59:07 +08:00
|
|
|
static int index;
|
2011-08-30 07:30:13 +08:00
|
|
|
|
2011-11-10 22:11:04 +08:00
|
|
|
assert(strlen(dev->name) < sizeof(dev->name));
|
2011-08-30 07:30:13 +08:00
|
|
|
|
2011-02-17 03:14:33 +08:00
|
|
|
if (!eth_devices) {
|
|
|
|
eth_current = eth_devices = dev;
|
|
|
|
eth_current_changed();
|
2002-11-03 08:24:07 +08:00
|
|
|
} else {
|
2012-05-15 16:59:07 +08:00
|
|
|
for (d = eth_devices; d->next != eth_devices; d = d->next)
|
2010-03-31 23:56:08 +08:00
|
|
|
;
|
2002-11-03 08:24:07 +08:00
|
|
|
d->next = dev;
|
|
|
|
}
|
|
|
|
|
|
|
|
dev->state = ETH_STATE_INIT;
|
|
|
|
dev->next = eth_devices;
|
2011-10-27 19:31:35 +08:00
|
|
|
dev->index = index++;
|
2002-11-03 08:24:07 +08:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-01-09 16:32:36 +08:00
|
|
|
int eth_unregister(struct eth_device *dev)
|
|
|
|
{
|
|
|
|
struct eth_device *cur;
|
|
|
|
|
|
|
|
/* No device */
|
|
|
|
if (!eth_devices)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
for (cur = eth_devices; cur->next != eth_devices && cur->next != dev;
|
|
|
|
cur = cur->next)
|
|
|
|
;
|
|
|
|
|
|
|
|
/* Device not found */
|
|
|
|
if (cur->next != dev)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
cur->next = dev->next;
|
|
|
|
|
|
|
|
if (eth_devices == dev)
|
|
|
|
eth_devices = dev->next == eth_devices ? NULL : dev->next;
|
|
|
|
|
|
|
|
if (eth_current == dev) {
|
|
|
|
eth_current = eth_devices;
|
|
|
|
eth_current_changed();
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2012-04-05 02:53:41 +08:00
|
|
|
static void eth_env_init(bd_t *bis)
|
|
|
|
{
|
|
|
|
const char *s;
|
|
|
|
|
|
|
|
if ((s = getenv("bootfile")) != NULL)
|
|
|
|
copy_filename(BootFile, s, sizeof(BootFile));
|
|
|
|
}
|
|
|
|
|
2002-11-03 08:24:07 +08:00
|
|
|
int eth_initialize(bd_t *bis)
|
|
|
|
{
|
2011-10-27 19:31:35 +08:00
|
|
|
int num_devices = 0;
|
2002-11-03 08:24:07 +08:00
|
|
|
eth_devices = NULL;
|
|
|
|
eth_current = NULL;
|
|
|
|
|
2012-02-13 21:51:18 +08:00
|
|
|
bootstage_mark(BOOTSTAGE_ID_NET_ETH_START);
|
2007-07-10 06:45:14 +08:00
|
|
|
#if defined(CONFIG_MII) || defined(CONFIG_CMD_MII)
|
2005-12-01 01:06:04 +08:00
|
|
|
miiphy_init();
|
|
|
|
#endif
|
Create PHY Lib for U-Boot
Extends the mii_dev structure to participate in a full-blown MDIO and
PHY driver scheme. The mii_dev structure and miiphy calls are modified
in such a way to allow the original mii command and miiphy
infrastructure to work as before, but also to support a new set of APIs
which allow (among other things) sharing of PHY driver code and 10G support
The mii command will continue to support normal PHY management functions
(Clause 22 of 802.3), but will not be changed to support 10G
(Clause 45).
The basic design is similar to PHY Lib from Linux, but simplified for
U-Boot's network and driver infrastructure.
We now have MDIO drivers and PHY drivers
An MDIO driver provides:
read
write
reset
A PHY driver provides:
(optionally): probe
config - initial setup, starting of auto-negotiation
startup - waiting for AN, and reading link state
shutdown - any cleanup needed
The ethernet drivers interact with the PHY Lib using these functions:
phy_connect()
phy_config()
phy_startup()
phy_shutdown()
Each PHY driver can be configured separately, or all at once using
config_phylib_all_drivers.h (added in the patch which adds the drivers)
We also provide generic drivers for Clause 22 (10/100/1000), and
Clause 45 (10G) PHYs.
We also implement phy_reset(), and call it in phy_connect(). Because
phy_reset() is essentially the same as miiphy_reset, but:
a) must support 10G PHYs, and
b) should use the phylib primitives,
we implement miiphy_reset, using phy_reset(), but only when
CONFIG_PHYLIB is set. Otherwise, we just use the old version. In this
way, we save on compile size, even if we don't manage to save code size.
Pulled ethtool.h and mdio.h from:
git://git.kernel.org/pub/scm/linux/kernel/git/davem/net-next-2.6
782d640afd15af7a1faf01cfe566ca4ac511319d
With many, many deletions so as to enable compilation under u-boot
Signed-off-by: Andy Fleming <afleming@freescale.com>
Signed-off-by: Kumar Gala <galak@kernel.crashing.org>
Acked-by: Detlev Zundel <dzu@denx.de>
2011-04-08 15:10:27 +08:00
|
|
|
|
|
|
|
#ifdef CONFIG_PHYLIB
|
|
|
|
phy_init();
|
|
|
|
#endif
|
|
|
|
|
2012-04-05 02:53:41 +08:00
|
|
|
eth_env_init(bis);
|
|
|
|
|
2010-09-01 14:05:04 +08:00
|
|
|
/*
|
|
|
|
* If board-specific initialization exists, call it.
|
|
|
|
* If not, call a CPU-specific one
|
|
|
|
*/
|
|
|
|
if (board_eth_init != __def_eth_init) {
|
|
|
|
if (board_eth_init(bis) < 0)
|
|
|
|
printf("Board Net Initialization Failed\n");
|
|
|
|
} else if (cpu_eth_init != __def_eth_init) {
|
|
|
|
if (cpu_eth_init(bis) < 0)
|
|
|
|
printf("CPU Net Initialization Failed\n");
|
|
|
|
} else
|
|
|
|
printf("Net Initialization Skipped\n");
|
2005-12-01 01:06:04 +08:00
|
|
|
|
2002-11-03 08:24:07 +08:00
|
|
|
if (!eth_devices) {
|
2012-05-15 16:59:07 +08:00
|
|
|
puts("No ethernet found.\n");
|
2012-02-13 21:51:18 +08:00
|
|
|
bootstage_error(BOOTSTAGE_ID_NET_ETH_START);
|
2002-11-03 08:24:07 +08:00
|
|
|
} else {
|
|
|
|
struct eth_device *dev = eth_devices;
|
2012-05-15 16:59:07 +08:00
|
|
|
char *ethprime = getenv("ethprime");
|
2002-11-03 08:24:07 +08:00
|
|
|
|
2012-02-13 21:51:18 +08:00
|
|
|
bootstage_mark(BOOTSTAGE_ID_NET_ETH_INIT);
|
2002-11-03 08:24:07 +08:00
|
|
|
do {
|
2011-10-27 19:31:35 +08:00
|
|
|
if (dev->index)
|
2012-05-15 16:59:07 +08:00
|
|
|
puts(", ");
|
2002-11-03 08:24:07 +08:00
|
|
|
|
|
|
|
printf("%s", dev->name);
|
|
|
|
|
2012-05-15 16:59:07 +08:00
|
|
|
if (ethprime && strcmp(dev->name, ethprime) == 0) {
|
2002-11-03 08:24:07 +08:00
|
|
|
eth_current = dev;
|
2012-05-15 16:59:07 +08:00
|
|
|
puts(" [PRIME]");
|
2002-11-03 08:24:07 +08:00
|
|
|
}
|
|
|
|
|
2010-06-10 10:10:48 +08:00
|
|
|
if (strchr(dev->name, ' '))
|
2012-05-15 16:59:07 +08:00
|
|
|
puts("\nWarning: eth device name has a space!"
|
|
|
|
"\n");
|
2010-06-10 10:10:48 +08:00
|
|
|
|
2011-10-27 19:31:35 +08:00
|
|
|
if (eth_write_hwaddr(dev, "eth", dev->index))
|
2011-09-07 09:57:48 +08:00
|
|
|
puts("\nWarning: failed to set MAC address\n");
|
2002-11-03 08:24:07 +08:00
|
|
|
|
|
|
|
dev = dev->next;
|
2011-10-27 19:31:35 +08:00
|
|
|
num_devices++;
|
2012-05-15 16:59:07 +08:00
|
|
|
} while (dev != eth_devices);
|
2002-11-03 08:24:07 +08:00
|
|
|
|
2011-02-17 03:14:33 +08:00
|
|
|
eth_current_changed();
|
2012-05-15 16:59:07 +08:00
|
|
|
putc('\n');
|
2002-11-03 08:24:07 +08:00
|
|
|
}
|
|
|
|
|
2011-10-27 19:31:35 +08:00
|
|
|
return num_devices;
|
2002-11-03 08:24:07 +08:00
|
|
|
}
|
|
|
|
|
2007-06-11 23:41:07 +08:00
|
|
|
#ifdef CONFIG_MCAST_TFTP
|
|
|
|
/* Multicast.
|
|
|
|
* mcast_addr: multicast ipaddr from which multicast Mac is made
|
2007-08-14 15:47:27 +08:00
|
|
|
* join: 1=join, 0=leave.
|
2007-06-11 23:41:07 +08:00
|
|
|
*/
|
2012-05-15 16:59:07 +08:00
|
|
|
int eth_mcast_join(IPaddr_t mcast_ip, u8 join)
|
2007-06-11 23:41:07 +08:00
|
|
|
{
|
2012-05-15 16:59:07 +08:00
|
|
|
u8 mcast_mac[6];
|
2007-08-14 15:47:27 +08:00
|
|
|
if (!eth_current || !eth_current->mcast)
|
2007-06-11 23:41:07 +08:00
|
|
|
return -1;
|
|
|
|
mcast_mac[5] = htonl(mcast_ip) & 0xff;
|
|
|
|
mcast_mac[4] = (htonl(mcast_ip)>>8) & 0xff;
|
|
|
|
mcast_mac[3] = (htonl(mcast_ip)>>16) & 0x7f;
|
|
|
|
mcast_mac[2] = 0x5e;
|
|
|
|
mcast_mac[1] = 0x0;
|
|
|
|
mcast_mac[0] = 0x1;
|
|
|
|
return eth_current->mcast(eth_current, mcast_mac, join);
|
|
|
|
}
|
|
|
|
|
2007-08-14 15:47:27 +08:00
|
|
|
/* the 'way' for ethernet-CRC-32. Spliced in from Linux lib/crc32.c
|
|
|
|
* and this is the ethernet-crc method needed for TSEC -- and perhaps
|
2007-06-11 23:41:07 +08:00
|
|
|
* some other adapter -- hash tables
|
|
|
|
*/
|
|
|
|
#define CRCPOLY_LE 0xedb88320
|
2012-05-15 16:59:07 +08:00
|
|
|
u32 ether_crc(size_t len, unsigned char const *p)
|
2007-06-11 23:41:07 +08:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
u32 crc;
|
|
|
|
crc = ~0;
|
|
|
|
while (len--) {
|
|
|
|
crc ^= *p++;
|
|
|
|
for (i = 0; i < 8; i++)
|
|
|
|
crc = (crc >> 1) ^ ((crc & 1) ? CRCPOLY_LE : 0);
|
|
|
|
}
|
|
|
|
/* an reverse the bits, cuz of way they arrive -- last-first */
|
|
|
|
crc = (crc >> 16) | (crc << 16);
|
|
|
|
crc = (crc >> 8 & 0x00ff00ff) | (crc << 8 & 0xff00ff00);
|
|
|
|
crc = (crc >> 4 & 0x0f0f0f0f) | (crc << 4 & 0xf0f0f0f0);
|
|
|
|
crc = (crc >> 2 & 0x33333333) | (crc << 2 & 0xcccccccc);
|
|
|
|
crc = (crc >> 1 & 0x55555555) | (crc << 1 & 0xaaaaaaaa);
|
|
|
|
return crc;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2002-11-03 08:24:07 +08:00
|
|
|
|
|
|
|
int eth_init(bd_t *bis)
|
|
|
|
{
|
2009-07-16 09:31:28 +08:00
|
|
|
struct eth_device *old_current, *dev;
|
2002-11-03 08:24:07 +08:00
|
|
|
|
2008-03-05 00:40:41 +08:00
|
|
|
if (!eth_current) {
|
2012-05-15 16:59:07 +08:00
|
|
|
puts("No ethernet found.\n");
|
2007-11-29 14:46:13 +08:00
|
|
|
return -1;
|
2008-03-05 00:40:41 +08:00
|
|
|
}
|
2002-11-03 08:24:07 +08:00
|
|
|
|
2009-07-16 09:31:28 +08:00
|
|
|
/* Sync environment with network devices */
|
|
|
|
dev = eth_devices;
|
|
|
|
do {
|
|
|
|
uchar env_enetaddr[6];
|
|
|
|
|
2011-10-27 19:31:35 +08:00
|
|
|
if (eth_getenv_enetaddr_by_index("eth", dev->index,
|
2011-06-14 07:13:10 +08:00
|
|
|
env_enetaddr))
|
2009-07-16 09:31:28 +08:00
|
|
|
memcpy(dev->enetaddr, env_enetaddr, 6);
|
|
|
|
|
|
|
|
dev = dev->next;
|
|
|
|
} while (dev != eth_devices);
|
|
|
|
|
2002-11-03 08:24:07 +08:00
|
|
|
old_current = eth_current;
|
|
|
|
do {
|
2009-07-23 15:01:03 +08:00
|
|
|
debug("Trying %s\n", eth_current->name);
|
2002-11-03 08:24:07 +08:00
|
|
|
|
2012-05-15 16:59:07 +08:00
|
|
|
if (eth_current->init(eth_current, bis) >= 0) {
|
2002-11-03 08:24:07 +08:00
|
|
|
eth_current->state = ETH_STATE_ACTIVE;
|
|
|
|
|
2007-11-29 14:46:13 +08:00
|
|
|
return 0;
|
2002-11-03 08:24:07 +08:00
|
|
|
}
|
2009-07-23 15:01:03 +08:00
|
|
|
debug("FAIL\n");
|
2002-11-03 08:24:07 +08:00
|
|
|
|
|
|
|
eth_try_another(0);
|
|
|
|
} while (old_current != eth_current);
|
|
|
|
|
2007-11-29 14:46:13 +08:00
|
|
|
return -1;
|
2002-11-03 08:24:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void eth_halt(void)
|
|
|
|
{
|
|
|
|
if (!eth_current)
|
|
|
|
return;
|
|
|
|
|
|
|
|
eth_current->halt(eth_current);
|
|
|
|
|
|
|
|
eth_current->state = ETH_STATE_PASSIVE;
|
|
|
|
}
|
|
|
|
|
2012-05-15 16:59:04 +08:00
|
|
|
int eth_send(void *packet, int length)
|
2002-11-03 08:24:07 +08:00
|
|
|
{
|
|
|
|
if (!eth_current)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return eth_current->send(eth_current, packet, length);
|
|
|
|
}
|
|
|
|
|
|
|
|
int eth_rx(void)
|
|
|
|
{
|
|
|
|
if (!eth_current)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return eth_current->recv(eth_current);
|
|
|
|
}
|
|
|
|
|
2007-12-28 01:19:02 +08:00
|
|
|
#ifdef CONFIG_API
|
2012-05-15 16:59:04 +08:00
|
|
|
static void eth_save_packet(void *packet, int length)
|
2007-12-28 01:19:02 +08:00
|
|
|
{
|
2012-05-15 16:59:04 +08:00
|
|
|
char *p = packet;
|
2007-12-28 01:19:02 +08:00
|
|
|
int i;
|
|
|
|
|
|
|
|
if ((eth_rcv_last+1) % PKTBUFSRX == eth_rcv_current)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (PKTSIZE < length)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < length; i++)
|
|
|
|
eth_rcv_bufs[eth_rcv_last].data[i] = p[i];
|
|
|
|
|
|
|
|
eth_rcv_bufs[eth_rcv_last].length = length;
|
|
|
|
eth_rcv_last = (eth_rcv_last + 1) % PKTBUFSRX;
|
|
|
|
}
|
|
|
|
|
2012-05-15 16:59:04 +08:00
|
|
|
int eth_receive(void *packet, int length)
|
2007-12-28 01:19:02 +08:00
|
|
|
{
|
2012-05-15 16:59:04 +08:00
|
|
|
char *p = packet;
|
2007-12-28 01:19:02 +08:00
|
|
|
void *pp = push_packet;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (eth_rcv_current == eth_rcv_last) {
|
|
|
|
push_packet = eth_save_packet;
|
|
|
|
eth_rx();
|
|
|
|
push_packet = pp;
|
|
|
|
|
|
|
|
if (eth_rcv_current == eth_rcv_last)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2012-06-22 19:24:28 +08:00
|
|
|
length = min(eth_rcv_bufs[eth_rcv_current].length, length);
|
2007-12-28 01:19:02 +08:00
|
|
|
|
|
|
|
for (i = 0; i < length; i++)
|
|
|
|
p[i] = eth_rcv_bufs[eth_rcv_current].data[i];
|
|
|
|
|
|
|
|
eth_rcv_current = (eth_rcv_current + 1) % PKTBUFSRX;
|
|
|
|
return length;
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_API */
|
|
|
|
|
2002-11-03 08:24:07 +08:00
|
|
|
void eth_try_another(int first_restart)
|
|
|
|
{
|
2012-05-15 16:59:07 +08:00
|
|
|
static struct eth_device *first_failed;
|
2011-02-20 03:15:14 +08:00
|
|
|
char *ethrotate;
|
2008-01-17 14:45:05 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Do not rotate between network interfaces when
|
|
|
|
* 'ethrotate' variable is set to 'no'.
|
|
|
|
*/
|
2012-05-15 16:59:07 +08:00
|
|
|
ethrotate = getenv("ethrotate");
|
|
|
|
if ((ethrotate != NULL) && (strcmp(ethrotate, "no") == 0))
|
2008-01-17 14:45:05 +08:00
|
|
|
return;
|
2002-11-03 08:24:07 +08:00
|
|
|
|
|
|
|
if (!eth_current)
|
|
|
|
return;
|
|
|
|
|
2012-05-15 16:59:07 +08:00
|
|
|
if (first_restart)
|
2002-11-03 08:24:07 +08:00
|
|
|
first_failed = eth_current;
|
|
|
|
|
|
|
|
eth_current = eth_current->next;
|
|
|
|
|
2011-02-17 03:14:33 +08:00
|
|
|
eth_current_changed();
|
2004-04-16 05:48:45 +08:00
|
|
|
|
2012-05-15 16:59:07 +08:00
|
|
|
if (first_failed == eth_current)
|
2002-11-03 08:24:07 +08:00
|
|
|
NetRestartWrap = 1;
|
|
|
|
}
|
|
|
|
|
2004-04-16 05:48:45 +08:00
|
|
|
void eth_set_current(void)
|
|
|
|
{
|
2012-05-15 16:59:07 +08:00
|
|
|
static char *act;
|
|
|
|
static int env_changed_id;
|
|
|
|
struct eth_device *old_current;
|
2009-02-10 16:38:52 +08:00
|
|
|
int env_id;
|
2004-04-16 05:48:45 +08:00
|
|
|
|
|
|
|
if (!eth_current) /* XXX no current */
|
|
|
|
return;
|
|
|
|
|
2009-02-10 16:38:52 +08:00
|
|
|
env_id = get_env_id();
|
|
|
|
if ((act == NULL) || (env_changed_id != env_id)) {
|
|
|
|
act = getenv("ethact");
|
|
|
|
env_changed_id = env_id;
|
|
|
|
}
|
2004-04-16 05:48:45 +08:00
|
|
|
if (act != NULL) {
|
|
|
|
old_current = eth_current;
|
|
|
|
do {
|
|
|
|
if (strcmp(eth_current->name, act) == 0)
|
|
|
|
return;
|
|
|
|
eth_current = eth_current->next;
|
|
|
|
} while (old_current != eth_current);
|
|
|
|
}
|
|
|
|
|
2011-02-17 03:14:33 +08:00
|
|
|
eth_current_changed();
|
2004-04-16 05:48:45 +08:00
|
|
|
}
|
|
|
|
|
2012-05-15 16:59:07 +08:00
|
|
|
char *eth_get_name(void)
|
2003-11-18 05:14:37 +08:00
|
|
|
{
|
2012-05-15 16:59:07 +08:00
|
|
|
return eth_current ? eth_current->name : "unknown";
|
2003-11-18 05:14:37 +08:00
|
|
|
}
|