Hi, On Thu, Sep 05, 2013 at 11:42:02PM +0200, Peter Korsgaard wrote: > When booting with CONFIG_ARM_APPENDED_DTB (either because of using an old > U-Boot, not wanting the hassle of 2 files or when using Falcon fast boot > mode in U-Boot), nothing updates the ethernet hwaddr specified for the > CPSW slaves, causing the driver to use a random hwaddr, which is some times > troublesome. > > The am33xx has unique ethernet hwaddrs programmed in the efuse, so it makes > more sense to default to these rather than random ones. Add a fixup step > which adds mac-address dt properties using the efuse addresses if the DTB > didn't contain valid ones. > > Signed-off-by: Peter Korsgaard <jacmet@xxxxxxxxxx> > Acked-by: Mugunthan V N <mugunthanvnm@xxxxxx> > Tested-by: Mark Jackson <mpfj-list@xxxxxxxxxxxxx> > Tested-by: Koen Kooi <koen@xxxxxxxxxxxxxxxxxxxxx> > Tested-by: Matt Porter <matt.porter@xxxxxxxxxx> > --- > diff --git a/arch/arm/mach-omap2/am33xx-cpsw.c b/arch/arm/mach-omap2/am33xx-cpsw.c > new file mode 100644 > index 0000000..eec29a4 > --- /dev/null > +++ b/arch/arm/mach-omap2/am33xx-cpsw.c > @@ -0,0 +1,94 @@ > +/* > + * am335x specific cpsw dt fixups > + * > + * 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 <linux/kernel.h> > +#include <linux/init.h> > +#include <linux/etherdevice.h> > +#include <linux/of.h> > +#include <linux/of_net.h> > + > +#include "soc.h" > +#include "control.h" > + > +/** > + * am33xx_dt_cpsw_set_mac_from_efuse - Add mac-address property using > + * ethernet hwaddr from efuse > + * @np: Pointer to the cpsw slave to set mac address of > + * @idx: Mac address index to use from efuse > + */ > +static void am33xx_dt_cpsw_set_mac_from_efuse(struct device_node *np, int idx) > +{ > + struct property *prop; > + u32 lo, hi; > + u8 *mac; > + > + switch (idx) { > + case 0: > + lo = omap_ctrl_readl(AM33XX_CONTROL_MAC_ID0_LOW); > + hi = omap_ctrl_readl(AM33XX_CONTROL_MAC_ID0_HIGH); > + break; > + > + case 1: > + lo = omap_ctrl_readl(AM33XX_CONTROL_MAC_ID1_LOW); > + hi = omap_ctrl_readl(AM33XX_CONTROL_MAC_ID1_HIGH); > + break; > + > + default: > + pr_err("cpsw.%d: too many slaves found\n", idx); > + return; > + } > + > + prop = kzalloc(sizeof(*prop) + ETH_ALEN, GFP_KERNEL); > + if (!prop) > + return; > + > + prop->value = prop + 1; > + prop->length = ETH_ALEN; > + prop->name = kstrdup("mac-address", GFP_KERNEL); Hmm. Do we want this to be mac-address or local-mac-address? If it's mac-address, then this will override anything set by u-boot (by means of priority in of_net). I don't think that's desireable. So it should probably set local-mac-address instead. > + if (!prop->name) { > + kfree(prop); > + return; > + } > + > + mac = prop->value; > + > + mac[0] = hi; > + mac[1] = hi >> 8; > + mac[2] = hi >> 16; > + mac[3] = hi >> 24; > + mac[4] = lo; > + mac[5] = lo >> 8; > + > + of_update_property(np, prop); mach-mxs does this too, I wonder if it's worth creating a small helper for it. Doesn't have to be done as part of this patch but it's still worth doing. > + > + pr_info("cpsw.%d: No hwaddr in dt. Using %pM from efuse\n", idx, mac); > +} > + > +static int __init am33xx_dt_cpsw_mac_fixup(void) > +{ > + struct device_node *np, *slave; > + int idx = 0; > + > + if (!soc_is_am33xx()) > + return -ENODEV; > + > + for_each_compatible_node(np, NULL, "ti,cpsw") > + for_each_node_by_name(slave, "slave") { The binding doesn't enforce "slave" node names for the subnodes, so you should probably just iterate over children. Right now they are named slave@0 and slave@1, but lack reg properties (and a notion of what said reg property would symbolize). > + if (!of_get_mac_address(slave)) > + am33xx_dt_cpsw_set_mac_from_efuse(slave, idx); > + idx++; You're assuming that you get the children in order here, which is not guaranteed. Maybe best is to adjust the binding, to make "reg" mean something (i.e. the MAC index for this hardware) and use the <reg> property of the childnode to tell which efuse to read. -Olof -- To unsubscribe from this list: send the line "unsubscribe linux-omap" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html