/* * Copyright 2008-2013 Freescale Semiconductor Inc. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * Neither the name of Freescale Semiconductor nor the * names of its contributors may be used to endorse or promote products * derived from this software without specific prior written permission. * * * ALTERNATIVELY, this software may be distributed under the terms of the * GNU General Public License ("GPL") as published by the Free Software * Foundation, either version 2 of that License or (at your option) any * later version. * * THIS SOFTWARE IS PROVIDED BY Freescale Semiconductor ``AS IS'' AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL Freescale Semiconductor BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "common/general.h" #include "fsl_fman_dtsec_mii_acc.h" /** * dtsec_mii_get_div() - calculates the value of the dtsec mii divider * @dtsec_freq: dtsec clock frequency (in Mhz) * * This function calculates the dtsec mii clock divider that determines * the MII MDC clock. MII MDC clock will be set to work in the range * of 1.5 to 2.5Mhz * The output of this function is the value of MIIMCFG[MgmtClk] which * implicitly determines the divider value. * Note: the dTSEC system clock is equal to 1/2 of the FMan clock. * * The table below which reflects dtsec_mii_get_div() functionality * shows the relations among dtsec_freq, MgmtClk, actual divider * and the MII frequency: * * dtsec freq MgmtClk div MII freq Mhz * [0.....80] 1 (1/4)(1/8) [0 to 2.5] * [81...120] 2 (1/6)(1/8) [1.6 to 2.5] * [121..160] 3 (1/8)(1/8) [1.8 to 2.5] * [161..200] 4 (1/10)(1/8) [2.0 to 2.5] * [201..280] 5 (1/14)(1/8) [1.8 to 2.5] * [281..400] 6 (1/20)(1/8) [1.1 to 2.5] * [401..560] 7 (1/28)(1/8) [1.8 to 2.5] * [560..frq] 7 (1/28)(1/8) [frq/224] * * Returns: the MIIMCFG[MgmtClk] appropriate value */ static uint8_t dtsec_mii_get_div(uint16_t dtsec_freq) { uint16_t mgmt_clk; if (dtsec_freq < 80) mgmt_clk = 1; else if (dtsec_freq < 120) mgmt_clk = 2; else if (dtsec_freq < 160) mgmt_clk = 3; else if (dtsec_freq < 200) mgmt_clk = 4; else if (dtsec_freq < 280) mgmt_clk = 5; else if (dtsec_freq < 400) mgmt_clk = 6; else mgmt_clk = 7; return (uint8_t)mgmt_clk; } void fman_dtsec_mii_reset(struct dtsec_mii_reg *regs) { /* Reset the management interface */ iowrite32be(ioread32be(®s->miimcfg) | MIIMCFG_RESET_MGMT, ®s->miimcfg); iowrite32be(ioread32be(®s->miimcfg) & ~MIIMCFG_RESET_MGMT, ®s->miimcfg); } int fman_dtsec_mii_write_reg(struct dtsec_mii_reg *regs, uint8_t addr, uint8_t reg, uint16_t data, uint16_t dtsec_freq) { uint32_t tmp; /* Setup the MII Mgmt clock speed */ iowrite32be((uint32_t)dtsec_mii_get_div(dtsec_freq), ®s->miimcfg); wmb(); /* Stop the MII management read cycle */ iowrite32be(0, ®s->miimcom); /* Dummy read to make sure MIIMCOM is written */ tmp = ioread32be(®s->miimcom); wmb(); /* Setting up MII Management Address Register */ tmp = (uint32_t)((addr << MIIMADD_PHY_ADDR_SHIFT) | reg); iowrite32be(tmp, ®s->miimadd); wmb(); /* Setting up MII Management Control Register with data */ iowrite32be((uint32_t)data, ®s->miimcon); /* Dummy read to make sure MIIMCON is written */ tmp = ioread32be(®s->miimcon); wmb(); /* Wait until MII management write is complete */ /* todo: a timeout could be useful here */ while ((ioread32be(®s->miimind)) & MIIMIND_BUSY) /* busy wait */; return 0; } int fman_dtsec_mii_read_reg(struct dtsec_mii_reg *regs, uint8_t addr, uint8_t reg, uint16_t *data, uint16_t dtsec_freq) { uint32_t tmp; /* Setup the MII Mgmt clock speed */ iowrite32be((uint32_t)dtsec_mii_get_div(dtsec_freq), ®s->miimcfg); wmb(); /* Setting up the MII Management Address Register */ tmp = (uint32_t)((addr << MIIMADD_PHY_ADDR_SHIFT) | reg); iowrite32be(tmp, ®s->miimadd); wmb(); /* Perform an MII management read cycle */ iowrite32be(MIIMCOM_READ_CYCLE, ®s->miimcom); /* Dummy read to make sure MIIMCOM is written */ tmp = ioread32be(®s->miimcom); wmb(); /* Wait until MII management read is complete */ /* todo: a timeout could be useful here */ while ((ioread32be(®s->miimind)) & MIIMIND_BUSY) /* busy wait */; /* Read MII management status */ *data = (uint16_t)ioread32be(®s->miimstat); wmb(); iowrite32be(0, ®s->miimcom); /* Dummy read to make sure MIIMCOM is written */ tmp = ioread32be(®s->miimcom); if (*data == 0xffff) return -ENXIO; return 0; }