2010/11/26 Wojciech Dubowik <dubowoj@xxxxxxxxxxx>: > From: Felix Fietkau <nbd@xxxxxxxxxxx> > > On WiSoc we cannot access mac register before it is resetted. > It will crash hardware otherwise. > > Signed-off-by: Felix Fietkau <nbd@xxxxxxxxxxx> > Signed-off-by: Wojciech Dubowik <Wojciech.Dubowik@xxxxxxxxxxx> > --- > Âdrivers/net/wireless/ath/ath5k/base.c Â|  Â7 ++- > Âdrivers/net/wireless/ath/ath5k/reset.c | Â114 ++++++++++++++++++++++++------- > Â2 files changed, 94 insertions(+), 27 deletions(-) > > diff --git a/drivers/net/wireless/ath/ath5k/base.c b/drivers/net/wireless/ath/ath5k/base.c > index 4d5ac71..87a4bb6 100644 > --- a/drivers/net/wireless/ath/ath5k/base.c > +++ b/drivers/net/wireless/ath/ath5k/base.c > @@ -2169,7 +2169,8 @@ ath5k_intr(int irq, void *dev_id) >    Âunsigned int counter = 1000; > >    Âif (unlikely(test_bit(ATH_STAT_INVALID, sc->status) || > -                !ath5k_hw_is_intr_pending(ah))) > +        ((ath5k_get_bus_type(ah) != ATH_AHB) && > +                !ath5k_hw_is_intr_pending(ah)))) >        Âreturn IRQ_NONE; > >    Âdo { > @@ -2235,6 +2236,10 @@ ath5k_intr(int irq, void *dev_id) >                Âtasklet_schedule(&sc->rf_kill.toggleq); > >        Â} > + > +        if (ath5k_get_bus_type(ah) == ATH_AHB) > +            break; > + >    Â} while (ath5k_hw_is_intr_pending(ah) && --counter > 0); > >    Âif (unlikely(!counter)) > diff --git a/drivers/net/wireless/ath/ath5k/reset.c b/drivers/net/wireless/ath/ath5k/reset.c > index 198a146..4f54655 100644 > --- a/drivers/net/wireless/ath/ath5k/reset.c > +++ b/drivers/net/wireless/ath/ath5k/reset.c > @@ -27,6 +27,7 @@ > > Â#include <linux/pci.h>         /* To determine if a card is pci-e */ > Â#include <linux/log2.h> > +#include <linux/platform_device.h> > Â#include "ath5k.h" > Â#include "reg.h" > Â#include "base.h" > @@ -198,31 +199,74 @@ static inline void ath5k_hw_write_rate_duration(struct ath5k_hw *ah, > Â*/ > Âstatic int ath5k_hw_nic_reset(struct ath5k_hw *ah, u32 val) > Â{ > -    int ret; > +    int ret = 0; >    Âu32 mask = val ? val : ~0U; > >    Â/* Read-and-clear RX Descriptor Pointer*/ > -    ath5k_hw_reg_read(ah, AR5K_RXDP); > +    if (!(mask & AR5K_RESET_CTL_MAC)) > +        ath5k_hw_reg_read(ah, AR5K_RXDP); > >    Â/* >     * Reset the device and wait until success >     */ > -    ath5k_hw_reg_write(ah, val, AR5K_RESET_CTL); > +    if (ath5k_get_bus_type(ah) == ATH_AHB) { > +        volatile u32 *reg; > +        u32 regval; > +        val = 0; > + > +        /* ah->ah_mac_srev is not available at this point yet */ > +        if (ah->ah_sc->devid >= AR5K_SREV_AR2315_R6) { > +            reg = (u32 *) AR5K_AR2315_RESET; > +            if (mask & AR5K_RESET_CTL_MAC) > +                val |= AR5K_AR2315_RESET_WMAC; > +            if (mask & AR5K_RESET_CTL_BASEBAND) > +                val |= AR5K_AR2315_RESET_BB_WARM; > +        } else { > +            reg = (u32 *) AR5K_AR5312_RESET; > +            if (to_platform_device(ah->ah_sc->dev)->id == 0) { > +                if (mask & AR5K_RESET_CTL_MAC) > +                    val |= AR5K_AR5312_RESET_WMAC0; > +                if (mask & AR5K_RESET_CTL_BASEBAND) > +                    val |= AR5K_AR5312_RESET_BB0_COLD | > +                       ÂAR5K_AR5312_RESET_BB0_WARM; > +            } else { > +                if (mask & AR5K_RESET_CTL_MAC) > +                    val |= AR5K_AR5312_RESET_WMAC1; > +                if (mask & AR5K_RESET_CTL_BASEBAND) > +                    val |= AR5K_AR5312_RESET_BB1_COLD | > +                       ÂAR5K_AR5312_RESET_BB1_WARM; > +            } > +        } > > -    /* Wait at least 128 PCI clocks */ > -    udelay(15); > +        /* Put BB/MAC into reset */ > +        regval = __raw_readl(reg); > +        __raw_writel(regval | val, reg); > +        regval = __raw_readl(reg); > +        udelay(100); > > -    if (ah->ah_version == AR5K_AR5210) { > -        val &= AR5K_RESET_CTL_PCU | AR5K_RESET_CTL_DMA > -            | AR5K_RESET_CTL_MAC | AR5K_RESET_CTL_PHY; > -        mask &= AR5K_RESET_CTL_PCU | AR5K_RESET_CTL_DMA > -            | AR5K_RESET_CTL_MAC | AR5K_RESET_CTL_PHY; > +        /* Bring BB/MAC out of reset */ > +        __raw_writel(regval & ~val, reg); > +        regval = __raw_readl(reg); >    Â} else { > -        val &= AR5K_RESET_CTL_PCU | AR5K_RESET_CTL_BASEBAND; > -        mask &= AR5K_RESET_CTL_PCU | AR5K_RESET_CTL_BASEBAND; > -    } > > -    ret = ath5k_hw_register_timeout(ah, AR5K_RESET_CTL, mask, val, false); > +        ath5k_hw_reg_write(ah, val, AR5K_RESET_CTL); > + > +        /* Wait at least 128 PCI clocks */ > +        udelay(15); > + > +        if (ah->ah_version == AR5K_AR5210) { > +            val &= AR5K_RESET_CTL_PCU | AR5K_RESET_CTL_DMA > +                | AR5K_RESET_CTL_MAC | AR5K_RESET_CTL_PHY; > +            mask &= AR5K_RESET_CTL_PCU | AR5K_RESET_CTL_DMA > +                | AR5K_RESET_CTL_MAC | AR5K_RESET_CTL_PHY; > +        } else { > +            val &= AR5K_RESET_CTL_PCU | AR5K_RESET_CTL_BASEBAND; > +            mask &= AR5K_RESET_CTL_PCU | AR5K_RESET_CTL_BASEBAND; > +        } > + > +        ret = ath5k_hw_register_timeout(ah, AR5K_RESET_CTL, > +                mask, val, false); > +    } > I think it would be much cleaner if we had a different function to handle wisoc reset instead of putting both on nic_reset. How about having a ath5k_hw_wisoc_reset and call that instead ? >    Â/* >     * Reset configuration register (for hw byte-swap). Note that this > @@ -334,6 +378,9 @@ int ath5k_hw_on_hold(struct ath5k_hw *ah) >    Âu32 bus_flags; >    Âint ret; > > +    if (ath5k_get_bus_type(ah) == ATH_AHB) > +        return 0; > + >    Â/* Make sure device is awake */ >    Âret = ath5k_hw_set_power(ah, AR5K_PM_AWAKE, true, 0); >    Âif (ret) { > @@ -390,22 +437,30 @@ int ath5k_hw_nic_wakeup(struct ath5k_hw *ah, int flags, bool initial) >    Âmode = 0; >    Âclock = 0; > > -    /* Wakeup the device */ > -    ret = ath5k_hw_set_power(ah, AR5K_PM_AWAKE, true, 0); > -    if (ret) { > -        ATH5K_ERR(ah->ah_sc, "failed to wakeup the MAC Chip\n"); > -        return ret; > +    if (ath5k_get_bus_type(ah) == ATH_AHB && !initial) { > +        /* Wakeup the device */ > +        ret = ath5k_hw_set_power(ah, AR5K_PM_AWAKE, true, 0); > +        if (ret) { > +            ATH5K_ERR(ah->ah_sc, "failed to wakeup the MAC Chip\n"); > +            return ret; > +        } >    Â} > You only call ath5k_hw_set_power for AHB devices this way ! >    Â/* >     * Put chipset on warm reset... >     * > -    Â* Note: putting PCI core on warm reset on PCI-E cards > -    Â* results card to hang and always return 0xffff... so > -    Â* we ingore that flag for PCI-E cards. On PCI cards > -    Â* this flag gets cleared after 64 PCI clocks. >     */ > -    bus_flags = (pdev && pdev->is_pcie) ? 0 : AR5K_RESET_CTL_PCI; > +    if (ath5k_get_bus_type(ah) == ATH_AHB) { > +        /* Reset MAC on WiSoc devices */ > +        bus_flags = (initial) ? AR5K_RESET_CTL_MAC : 0; > +    } else { > +        /* Note: putting PCI core on warm reset on PCI-E cards > +        Â* results card to hang and always return 0xffff... so > +        Â* we ingore that flag for PCI-E cards. On PCI cards > +        Â* this flag gets cleared after 64 PCI clocks. > +        Â*/ > +        bus_flags = (pdev && pdev->is_pcie) ? 0 : AR5K_RESET_CTL_PCI; > +    } > This is wrong... #define AR5K_RESET_CTL_MAC 0x00000004 /* MAC reset (PCU+Baseband ?) [5210] */ this bit was only available on earlier chips. To reset mac on later chips (WiSoC too) you need to use AR5K_RESET_CTL_PCU and we already do that below. This is something I have to clean up actually, bit 0 is reset mac (both PCU + DMA) and bit 1 resets Baseband. Note there is poor documentation on that, documentation on AR2317 eg. says mac reset for bit 0 and "warm reset to baseband logic" for bit 1 (which is correct) but on description for bit 1 says "PCU and DMA but not baseband" (that's actually the description of bit 0) and above says "in order to reset both baseband and mac and pci write 0x13" (0x10 is pci) that doesn't make sense because according to descriptions none of the 2 first bits resets baseband :P If you go on AR5213, documentation is correct, it says that bit 0 is for PCU and DMA and bit 1 is for baseband and that also works on all post-5211 chips. That's why I've put the comments on reg.h, all [5210] are only available on AR5210, only bits/registers marked with [5211+] are available on later macs. So what you are really doing here is activate bit 3 that is reserved according to docs and should be zeroed ! We already do what's needed to reset both mac and baseband later. val &= AR5K_RESET_CTL_PCU | AR5K_RESET_CTL_BASEBAND; Notice that AR5K_RESET_CTL_PCU | AR5K_RESET_CTL_BASEBAND = 0x13 as documentation suggests. >    Âif (ah->ah_version == AR5K_AR5210) { >        Âret = ath5k_hw_nic_reset(ah, AR5K_RESET_CTL_PCU | > @@ -536,6 +591,9 @@ static void ath5k_hw_set_sleep_clock(struct ath5k_hw *ah, bool enable) >    Âstruct ath5k_eeprom_info *ee = &ah->ah_capabilities.cap_eeprom; >    Âu32 scal, spending, usec32; > > +    if (ath5k_get_bus_type(ah) == ATH_AHB) > +        enable = false; > + >    Â/* Only set 32KHz settings if we have an external >     * 32KHz crystal present */ >    Âif ((AR5K_EEPROM_HAS32KHZCRYSTAL(ee->ee_misc1) || > @@ -607,6 +665,7 @@ static void ath5k_hw_set_sleep_clock(struct ath5k_hw *ah, bool enable) > >        Âif ((ah->ah_radio == AR5K_RF5112) || >        Â(ah->ah_radio == AR5K_RF5413) || > +        (ah->ah_radio == AR5K_RF2316) || >        Â(ah->ah_mac_version == (AR5K_SREV_AR2417 >> 4))) >            Âspending = 0x14; >        Âelse > @@ -614,7 +673,9 @@ static void ath5k_hw_set_sleep_clock(struct ath5k_hw *ah, bool enable) >        Âath5k_hw_reg_write(ah, spending, AR5K_PHY_SPENDING); > >        Âif ((ah->ah_radio == AR5K_RF5112) || > -        (ah->ah_radio == AR5K_RF5413)) > +        (ah->ah_radio == AR5K_RF5413) || > +        (ah->ah_radio == AR5K_RF2316) || > +        (ah->ah_radio == AR5K_RF2317)) >            Âusec32 = 39; >        Âelse >            Âusec32 = 31; > @@ -678,7 +739,8 @@ static void ath5k_hw_tweak_initval_settings(struct ath5k_hw *ah, > >    Â/* Set fast ADC */ >    Âif ((ah->ah_radio == AR5K_RF5413) || > -    (ah->ah_mac_version == (AR5K_SREV_AR2417 >> 4))) { > +        (ah->ah_radio == AR5K_RF2317) || > +        (ah->ah_mac_version == (AR5K_SREV_AR2417 >> 4))) { >        Âu32 fast_adc = true; > >        Âif (channel->center_freq == 2462 || > -- > 1.7.1 > > -- > To unsubscribe from this list: send the line "unsubscribe linux-wireless" in > the body of a message to majordomo@xxxxxxxxxxxxxxx > More majordomo info at Âhttp://vger.kernel.org/majordomo-info.html > -- GPG ID: 0xD21DB2DB As you read this post global entropy rises. Have Fun ;-) Nick ÿô.nÇ·®+%˱é¥wÿº{.nÇ·¥{±ÿ«zW¬³ø¡Ü}©²ÆzÚj:+v¨þø®w¥þàÞ¨è&¢)ß«a¶Úÿûz¹ÞúÝjÿwèf