#include "bcm43xx_main.h"
#include <linux/delay.h>
+#include <linux/sched.h>
/* Write the LocalOscillator Control (adjust) value-pair. */
}
static inline
-void assert_rfatt_and_bbatt(const struct bcm43xx_rfatt *rfatt,
- const struct bcm43xx_bbatt *bbatt)
+int assert_rfatt_and_bbatt(const struct bcm43xx_rfatt *rfatt,
+ const struct bcm43xx_bbatt *bbatt)
{
- if (BCM43xx_DEBUG) {
- int err = 0;
+ int err = 0;
- if (unlikely(rfatt->att >= 16)) {
- dprintk(KERN_ERR PFX "ERROR: invalid rf_att: %u\n",
- rfatt->att);
- err = 1;
- }
- if (unlikely(bbatt->att >= 9)) {
- dprintk(KERN_ERR PFX "ERROR: invalid bband_att: %u\n",
- bbatt->att);
- err = 1;
- }
- if (unlikely(err))
- dump_stack();
+ /* Check the attenuation values against the LO control array sizes. */
+#if BCM43xx_DEBUG
+ if (rfatt->att >= BCM43xx_NR_RF) {
+ dprintk(KERN_ERR PFX
+ "ERROR: rfatt(%u) >= size of LO array\n",
+ rfatt->att);
+ err = -EINVAL;
}
+ if (bbatt->att >= BCM43xx_NR_BB) {
+ dprintk(KERN_ERR PFX
+ "ERROR: bbatt(%u) >= size of LO array\n",
+ bbatt->att);
+ err = -EINVAL;
+ }
+ if (err)
+ dump_stack();
+#endif /* BCM43xx_DEBUG */
+
+ return err;
}
static
struct bcm43xx_phy *phy = &dev->phy;
struct bcm43xx_txpower_lo_control *lo = phy->lo_control;
- assert_rfatt_and_bbatt(rfatt, bbatt);
+ if (assert_rfatt_and_bbatt(rfatt, bbatt))
+ return &(lo->no_padmix[0][0]); /* Just prevent a crash */
return &(lo->no_padmix[bbatt->att][rfatt->att]);
}
{
struct bcm43xx_phy *phy = &dev->phy;
struct bcm43xx_txpower_lo_control *lo = phy->lo_control;
- struct bcm43xx_loctl *ret;
- assert_rfatt_and_bbatt(rfatt, bbatt);
+ if (assert_rfatt_and_bbatt(rfatt, bbatt))
+ return &(lo->no_padmix[0][0]); /* Just prevent a crash */
if (rfatt->with_padmix)
- ret = &(lo->with_padmix[bbatt->att][rfatt->att]);
- else
- ret = &(lo->no_padmix[bbatt->att][rfatt->att]);
-
- return ret;
+ return &(lo->with_padmix[bbatt->att][rfatt->att]);
+ return &(lo->no_padmix[bbatt->att][rfatt->att]);
}
/* Call a function for every possible LO control value-pair. */
-static int bcm43xx_call_for_each_loctl(struct bcm43xx_wldev *dev,
- int (*func)(struct bcm43xx_wldev *,
- struct bcm43xx_loctl *))
+static void bcm43xx_call_for_each_loctl(struct bcm43xx_wldev *dev,
+ void (*func)(struct bcm43xx_wldev *,
+ struct bcm43xx_loctl *))
{
struct bcm43xx_phy *phy = &dev->phy;
struct bcm43xx_txpower_lo_control *ctl = phy->lo_control;
int i, j;
- int err;
for (i = 0; i < BCM43xx_NR_BB; i++) {
- for (j = 0; j < BCM43xx_NR_RF; j++) {
- err = func(dev, &(ctl->with_padmix[i][j]));
- if (unlikely(err))
- return err;
- }
+ for (j = 0; j < BCM43xx_NR_RF; j++)
+ func(dev, &(ctl->with_padmix[i][j]));
}
for (i = 0; i < BCM43xx_NR_BB; i++) {
- for (j = 0; j < BCM43xx_NR_RF; j++) {
- err = func(dev, &(ctl->no_padmix[i][j]));
- if (unlikely(err))
- return err;
- }
+ for (j = 0; j < BCM43xx_NR_RF; j++)
+ func(dev, &(ctl->no_padmix[i][j]));
}
-
- return 0;
}
static u16 lo_b_r15_loop(struct bcm43xx_wldev *dev)
{
struct bcm43xx_phy *phy = &dev->phy;
u16 rfover;
+ u16 feedthrough;
if (phy->gmode) {
lna <<= BCM43xx_PHY_RFOVERVAL_LNA_SHIFT;
bcm43xx_phy_write(dev, BCM43xx_PHY_PGACTL, pga);
}
udelay(21);
+ feedthrough = bcm43xx_phy_read(dev, BCM43xx_PHY_LO_LEAKAGE);
- return bcm43xx_phy_read(dev, BCM43xx_PHY_LO_LEAKAGE);
+ /* This is a good place to check if we need to relax a bit,
+ * as this is the main function called regularly
+ * in the LO calibration. */
+ cond_resched();
+
+ return feedthrough;
}
/* TXCTL Register and Value Table.
& 0xFF00) | lo->tx_bias | lo->tx_magn);
}
} else {
- lo->tx_magn = 0; /* FIXME */
+ lo->tx_magn = 0;
lo->tx_bias = 0;
bcm43xx_radio_write16(dev, 0x52,
bcm43xx_radio_read16(dev, 0x52)
}
#if BCM43xx_DEBUG
-static int do_validate_loctl(struct bcm43xx_wldev *dev,
- struct bcm43xx_loctl *control)
+static void do_validate_loctl(struct bcm43xx_wldev *dev,
+ struct bcm43xx_loctl *control)
{
const int is_initializing = (bcm43xx_status(dev) == BCM43xx_STAT_INITIALIZING);
"(first: %d, second: %d, used %u)\n",
control->i, control->q, control->used);
}
- return 0;
}
static void validate_all_loctls(struct bcm43xx_wldev *dev)
{
bcm43xx_lo_write(dev, bcm43xx_lo_g_ctl_current(dev));
}
-static inline void fixup_rfatt_for_txctl1(struct bcm43xx_rfatt *rf,
- u16 txctl1)
+static inline void fixup_rfatt_for_txcontrol(struct bcm43xx_rfatt *rf,
+ u8 tx_control)
{
- if ((rf->att < 5) && (txctl1 & 0x0001))
- rf->att = 4;
+ if (tx_control & BCM43xx_TXCTL_TXMIX) {
+ if (rf->att < 5)
+ rf->att = 4;
+ }
}
void bcm43xx_lo_g_adjust_to(struct bcm43xx_wldev *dev,
- u16 rfatt, u16 bbatt, u16 txctl1)
+ u16 rfatt, u16 bbatt, u16 tx_control)
{
struct bcm43xx_rfatt rf;
struct bcm43xx_bbatt bb;
memset(&bb, 0, sizeof(bb));
rf.att = rfatt;
bb.att = bbatt;
- fixup_rfatt_for_txctl1(&rf, txctl1);
+ fixup_rfatt_for_txcontrol(&rf, tx_control);
loctl = bcm43xx_get_lo_g_ctl(dev, &rf, &bb);
bcm43xx_lo_write(dev, loctl);
}
struct bcm43xx_loctl * bcm43xx_lo_g_ctl_current(struct bcm43xx_wldev *dev)
{
struct bcm43xx_phy *phy = &dev->phy;
- struct bcm43xx_txpower_lo_control *lo = phy->lo_control;
struct bcm43xx_rfatt rf;
- memcpy(&rf, &lo->rfatt, sizeof(rf));
- fixup_rfatt_for_txctl1(&rf, phy->txctl1);
+ memcpy(&rf, &phy->rfatt, sizeof(rf));
+ fixup_rfatt_for_txcontrol(&rf, phy->tx_control);
- return bcm43xx_get_lo_g_ctl(dev, &rf, &lo->bbatt);
+ return bcm43xx_get_lo_g_ctl(dev, &rf, &phy->bbatt);
}
-static int do_mark_unused(struct bcm43xx_wldev *dev,
- struct bcm43xx_loctl *control)
+static void do_mark_unused(struct bcm43xx_wldev *dev,
+ struct bcm43xx_loctl *control)
{
control->used = 0;
- return 0;
}
void bcm43xx_lo_g_ctl_mark_all_unused(struct bcm43xx_wldev *dev)