[U-Boot-Users] [PATCH 5/5] mpc83xx: UEC: add support for Broadcom BCM5481 Gigabit PHY
Ben Warren
biggerbadderben at gmail.com
Fri Jan 11 06:26:45 CET 2008
Peter Barada wrote:
> On Thu, 2008-01-10 at 15:41 -0500, Ben Warren wrote:
>
>> Anton Vorontsov wrote:
>>
>>> This patch adds basic support for Broadcom BCM5481 PHY,
>>> with the quirk needed for at least MPC8360E-RDK.
>>>
>>> Quirk comes from MPC8360E-RDK BSP source, I think author is
>>> Peter Barada <peterb at logicpd.com>, but I'm not sure.
>>>
>>> There are no openly available specifications for that PHY.
>>>
>>> Signed-off-by: Anton Vorontsov <avorontsov at ru.mvista.com>
>>> ---
>>> drivers/qe/uec_phy.c | 77 ++++++++++++++++++++++++++++++++++++++++++++++++++
>>> drivers/qe/uec_phy.h | 5 +++
>>> 2 files changed, 82 insertions(+), 0 deletions(-)
>>>
>>> diff --git a/drivers/qe/uec_phy.c b/drivers/qe/uec_phy.c
>>> index ca6faa6..6882d03 100644
>>> --- a/drivers/qe/uec_phy.c
>>> +++ b/drivers/qe/uec_phy.c
>>> @@ -237,6 +237,44 @@ static int gbit_config_aneg (struct uec_mii_info *mii_info)
>>> return 0;
>>> }
>>>
>>> +static int gbit_read_status(struct uec_mii_info *mii_info)
>>> +{
>>> + u16 status;
>>> + int err;
>>> +
>>> + err = genmii_update_link(mii_info);
>>> + if (err)
>>> + return err;
>>> +
>>> + if (mii_info->autoneg) {
>>> + mii_info->pause = 0;
>>> + status = phy_read(mii_info, MII_1000BASETSTATUS);
>>> + if (status & (LPA_1000FULL | LPA_1000HALF)) {
>>> + mii_info->speed = SPEED_1000;
>>> + if (status & LPA_1000FULL)
>>> + mii_info->duplex = DUPLEX_FULL;
>>> + else
>>> + mii_info->duplex = DUPLEX_HALF;
>>> + } else {
>>> + status = phy_read(mii_info, PHY_ANLPAR);
>>> +
>>> + if (status & (PHY_ANLPAR_10FD | PHY_ANLPAR_TXFD))
>>> + mii_info->duplex = DUPLEX_FULL;
>>> + else
>>> + mii_info->duplex = DUPLEX_HALF;
>>> + if (status & (PHY_ANLPAR_TXFD | PHY_ANLPAR_TX))
>>> + mii_info->speed = SPEED_100;
>>> + else
>>> + mii_info->speed = SPEED_10;
>>> + }
>>> + }
>>> + /*
>>> + * On non-aneg, we assume what we put in BMCR is the speed,
>>> + * though magic-aneg shouldn't prevent this case from occurring.
>>> + */
>>> + return 0;
>>> +}
>>> +
>>>
>>>
>> Please roll the 1Gb code into genmii_read_status. 10/100 PHYs should
>> report 0 in the
>> MII_1000BASETSTATUS register so it should work.
>>
>>> static int marvell_config_aneg (struct uec_mii_info *mii_info)
>>> {
>>> /* The Marvell PHY has an errata which requires
>>> @@ -319,6 +357,35 @@ static int genmii_read_status (struct uec_mii_info *mii_info)
>>> return 0;
>>> }
>>>
>>> +static int bcm_init(struct uec_mii_info *mii_info)
>>> +{
>>> + gbit_config_aneg(mii_info);
>>> +
>>> +#ifdef CONFIG_MPC8360ERDK
>>> + {
>>> + u16 val;
>>> + int cnt = 50;
>>> +
>>> + /* Wait for aneg to complete. */
>>> + do
>>> + val = phy_read(mii_info, PHY_BMSR);
>>> + while (--cnt && !(val & PHY_BMSR_AUTN_COMP));
>>> +
>>> + /* Set RDX clk delay. */
>>> + phy_write(mii_info, 0x18, 0x7 | (7 << 12));
>>> +
>>> + val = phy_read(mii_info, 0x18);
>>> + /* Set RDX-RXC skew. */
>>> + val |= (1<<8);
>>> + val |= (7 | (7 << 12));
>>> + /* Write bits 14:0. */
>>> + val |= (1<<15);
>>> + phy_write(mii_info, 0x18, val);
>>> + }
>>> +#endif
>>>
>>>
>> What is it that makes this specific to one board? Magic numbers are
>> baaaad, especially for industry-standard memory maps (I'm thinking of
>> offset 0x18 in case it's not obvious)
>>
>
> Its specific to the board, in that when hooked up via RGMII the setting
> in the PHY is required to supply adequate delay between the RXD and RXC
> signals instead of using trace lengths to achieve timing. I know magic
> numbers are baaaaad, but the PHY spec is only available under NDA and as
> such descriptions(including informative constants, etc) can not be
> disclosed. If it wasn't for this one register setting(after reset), the
> PHY, as used, wouldn't require any PHY-specific code(outside of ID'ng it
> as a 5481).
>
>
Is this just due to a layout problem on the reference board, or do you
always need to do this when connecting this MAC/PHY combo via RGMII? If
the former, the additional registers needs to be set in board code. If
the latter, please do something like this: Add an 'enet_interface_e
if_mode' entry to the 'struct uec_mii_info' definition, and fill it in
when you call 'uec_set_mac_if_mode()' and check the value for RGMII when
'bcm_init()' is called.
Sorry, but having a *reference* board #ifdef'd in library code is just
wrong.
Regarding the magic numbers, something like this would be better:
#define MII_BCM_BCM_PRIVATE_0 0x16
#define MII_BCM_BCM_PRIVATE_1 0x17
#define MII_BCM_BCM_PRIVATE_2 0x18
regards,
Ben
More information about the U-Boot
mailing list