📄 radio.c
字号:
phy_stacksave(0x04A3); phy_stacksave(0x04A9); phy_stacksave(0x04AA); phy_stacksave(0x04AC); phy_stacksave(0x0493); phy_stacksave(0x04A1); phy_stacksave(0x04A0); phy_stacksave(0x04A2); phy_stacksave(0x048A); phy_stacksave(0x04A8); phy_stacksave(0x04AB); if (phy->rev == 2) { phy_stacksave(0x04AD); phy_stacksave(0x04AE); } else if (phy->rev >= 3) { phy_stacksave(0x04AD); phy_stacksave(0x0415); phy_stacksave(0x0416); phy_stacksave(0x0417); ilt_stacksave(0x1A00 + 0x2); ilt_stacksave(0x1A00 + 0x3); } phy_stacksave(0x042B); phy_stacksave(0x048C); b43legacy_phy_write(dev, B43legacy_PHY_RADIO_BITFIELD, b43legacy_phy_read(dev, B43legacy_PHY_RADIO_BITFIELD) & ~0x1000); b43legacy_phy_write(dev, B43legacy_PHY_G_CRS, (b43legacy_phy_read(dev, B43legacy_PHY_G_CRS) & 0xFFFC) | 0x0002); b43legacy_phy_write(dev, 0x0033, 0x0800); b43legacy_phy_write(dev, 0x04A3, 0x2027); b43legacy_phy_write(dev, 0x04A9, 0x1CA8); b43legacy_phy_write(dev, 0x0493, 0x287A); b43legacy_phy_write(dev, 0x04AA, 0x1CA8); b43legacy_phy_write(dev, 0x04AC, 0x287A); b43legacy_phy_write(dev, 0x04A0, (b43legacy_phy_read(dev, 0x04A0) & 0xFFC0) | 0x001A); b43legacy_phy_write(dev, 0x04A7, 0x000D); if (phy->rev < 2) b43legacy_phy_write(dev, 0x0406, 0xFF0D); else if (phy->rev == 2) { b43legacy_phy_write(dev, 0x04C0, 0xFFFF); b43legacy_phy_write(dev, 0x04C1, 0x00A9); } else { b43legacy_phy_write(dev, 0x04C0, 0x00C1); b43legacy_phy_write(dev, 0x04C1, 0x0059); } b43legacy_phy_write(dev, 0x04A1, (b43legacy_phy_read(dev, 0x04A1) & 0xC0FF) | 0x1800); b43legacy_phy_write(dev, 0x04A1, (b43legacy_phy_read(dev, 0x04A1) & 0xFFC0) | 0x0015); b43legacy_phy_write(dev, 0x04A8, (b43legacy_phy_read(dev, 0x04A8) & 0xCFFF) | 0x1000); b43legacy_phy_write(dev, 0x04A8, (b43legacy_phy_read(dev, 0x04A8) & 0xF0FF) | 0x0A00); b43legacy_phy_write(dev, 0x04AB, (b43legacy_phy_read(dev, 0x04AB) & 0xCFFF) | 0x1000); b43legacy_phy_write(dev, 0x04AB, (b43legacy_phy_read(dev, 0x04AB) & 0xF0FF) | 0x0800); b43legacy_phy_write(dev, 0x04AB, (b43legacy_phy_read(dev, 0x04AB) & 0xFFCF) | 0x0010); b43legacy_phy_write(dev, 0x04AB, (b43legacy_phy_read(dev, 0x04AB) & 0xFFF0) | 0x0005); b43legacy_phy_write(dev, 0x04A8, (b43legacy_phy_read(dev, 0x04A8) & 0xFFCF) | 0x0010); b43legacy_phy_write(dev, 0x04A8, (b43legacy_phy_read(dev, 0x04A8) & 0xFFF0) | 0x0006); b43legacy_phy_write(dev, 0x04A2, (b43legacy_phy_read(dev, 0x04A2) & 0xF0FF) | 0x0800); b43legacy_phy_write(dev, 0x04A0, (b43legacy_phy_read(dev, 0x04A0) & 0xF0FF) | 0x0500); b43legacy_phy_write(dev, 0x04A2, (b43legacy_phy_read(dev, 0x04A2) & 0xFFF0) | 0x000B); if (phy->rev >= 3) { b43legacy_phy_write(dev, 0x048A, b43legacy_phy_read(dev, 0x048A) & ~0x8000); b43legacy_phy_write(dev, 0x0415, (b43legacy_phy_read(dev, 0x0415) & 0x8000) | 0x36D8); b43legacy_phy_write(dev, 0x0416, (b43legacy_phy_read(dev, 0x0416) & 0x8000) | 0x36D8); b43legacy_phy_write(dev, 0x0417, (b43legacy_phy_read(dev, 0x0417) & 0xFE00) | 0x016D); } else { b43legacy_phy_write(dev, 0x048A, b43legacy_phy_read(dev, 0x048A) | 0x1000); b43legacy_phy_write(dev, 0x048A, (b43legacy_phy_read(dev, 0x048A) & 0x9FFF) | 0x2000); tmp32 = b43legacy_shm_read32(dev, B43legacy_SHM_SHARED, B43legacy_UCODEFLAGS_OFFSET); if (!(tmp32 & 0x800)) { tmp32 |= 0x800; b43legacy_shm_write32(dev, B43legacy_SHM_SHARED, B43legacy_UCODEFLAGS_OFFSET, tmp32); } } if (phy->rev >= 2) b43legacy_phy_write(dev, 0x042B, b43legacy_phy_read(dev, 0x042B) | 0x0800); b43legacy_phy_write(dev, 0x048C, (b43legacy_phy_read(dev, 0x048C) & 0xF0FF) | 0x0200); if (phy->rev == 2) { b43legacy_phy_write(dev, 0x04AE, (b43legacy_phy_read(dev, 0x04AE) & 0xFF00) | 0x007F); b43legacy_phy_write(dev, 0x04AD, (b43legacy_phy_read(dev, 0x04AD) & 0x00FF) | 0x1300); } else if (phy->rev >= 6) { b43legacy_ilt_write(dev, 0x1A00 + 0x3, 0x007F); b43legacy_ilt_write(dev, 0x1A00 + 0x2, 0x007F); b43legacy_phy_write(dev, 0x04AD, b43legacy_phy_read(dev, 0x04AD) & 0x00FF); } b43legacy_calc_nrssi_slope(dev); break; default: B43legacy_BUG_ON(1); }}static voidb43legacy_radio_interference_mitigation_disable(struct b43legacy_wldev *dev, int mode){ struct b43legacy_phy *phy = &dev->phy; u32 tmp32; u32 *stack = phy->interfstack; switch (mode) { case B43legacy_RADIO_INTERFMODE_NONWLAN: if (phy->rev != 1) { b43legacy_phy_write(dev, 0x042B, b43legacy_phy_read(dev, 0x042B) & ~0x0800); b43legacy_phy_write(dev, B43legacy_PHY_G_CRS, b43legacy_phy_read(dev, B43legacy_PHY_G_CRS) | 0x4000); break; } phy_stackrestore(0x0078); b43legacy_calc_nrssi_threshold(dev); phy_stackrestore(0x0406); b43legacy_phy_write(dev, 0x042B, b43legacy_phy_read(dev, 0x042B) & ~0x0800); if (!dev->bad_frames_preempt) b43legacy_phy_write(dev, B43legacy_PHY_RADIO_BITFIELD, b43legacy_phy_read(dev, B43legacy_PHY_RADIO_BITFIELD) & ~(1 << 11)); b43legacy_phy_write(dev, B43legacy_PHY_G_CRS, b43legacy_phy_read(dev, B43legacy_PHY_G_CRS) | 0x4000); phy_stackrestore(0x04A0); phy_stackrestore(0x04A1); phy_stackrestore(0x04A2); phy_stackrestore(0x04A8); phy_stackrestore(0x04AB); phy_stackrestore(0x04A7); phy_stackrestore(0x04A3); phy_stackrestore(0x04A9); phy_stackrestore(0x0493); phy_stackrestore(0x04AA); phy_stackrestore(0x04AC); break; case B43legacy_RADIO_INTERFMODE_MANUALWLAN: if (!(b43legacy_phy_read(dev, 0x0033) & 0x0800)) break; phy->aci_enable = 0; phy_stackrestore(B43legacy_PHY_RADIO_BITFIELD); phy_stackrestore(B43legacy_PHY_G_CRS); phy_stackrestore(0x0033); phy_stackrestore(0x04A3); phy_stackrestore(0x04A9); phy_stackrestore(0x0493); phy_stackrestore(0x04AA); phy_stackrestore(0x04AC); phy_stackrestore(0x04A0); phy_stackrestore(0x04A7); if (phy->rev >= 2) { phy_stackrestore(0x04C0); phy_stackrestore(0x04C1); } else phy_stackrestore(0x0406); phy_stackrestore(0x04A1); phy_stackrestore(0x04AB); phy_stackrestore(0x04A8); if (phy->rev == 2) { phy_stackrestore(0x04AD); phy_stackrestore(0x04AE); } else if (phy->rev >= 3) { phy_stackrestore(0x04AD); phy_stackrestore(0x0415); phy_stackrestore(0x0416); phy_stackrestore(0x0417); ilt_stackrestore(0x1A00 + 0x2); ilt_stackrestore(0x1A00 + 0x3); } phy_stackrestore(0x04A2); phy_stackrestore(0x04A8); phy_stackrestore(0x042B); phy_stackrestore(0x048C); tmp32 = b43legacy_shm_read32(dev, B43legacy_SHM_SHARED, B43legacy_UCODEFLAGS_OFFSET); if (tmp32 & 0x800) { tmp32 &= ~0x800; b43legacy_shm_write32(dev, B43legacy_SHM_SHARED, B43legacy_UCODEFLAGS_OFFSET, tmp32); } b43legacy_calc_nrssi_slope(dev); break; default: B43legacy_BUG_ON(1); }}#undef phy_stacksave#undef phy_stackrestore#undef radio_stacksave#undef radio_stackrestore#undef ilt_stacksave#undef ilt_stackrestoreint b43legacy_radio_set_interference_mitigation(struct b43legacy_wldev *dev, int mode){ struct b43legacy_phy *phy = &dev->phy; int currentmode; if ((phy->type != B43legacy_PHYTYPE_G) || (phy->rev == 0) || (!phy->gmode)) return -ENODEV; phy->aci_wlan_automatic = 0; switch (mode) { case B43legacy_RADIO_INTERFMODE_AUTOWLAN: phy->aci_wlan_automatic = 1; if (phy->aci_enable) mode = B43legacy_RADIO_INTERFMODE_MANUALWLAN; else mode = B43legacy_RADIO_INTERFMODE_NONE; break; case B43legacy_RADIO_INTERFMODE_NONE: case B43legacy_RADIO_INTERFMODE_NONWLAN: case B43legacy_RADIO_INTERFMODE_MANUALWLAN: break; default: return -EINVAL; } currentmode = phy->interfmode; if (currentmode == mode) return 0; if (currentmode != B43legacy_RADIO_INTERFMODE_NONE) b43legacy_radio_interference_mitigation_disable(dev, currentmode); if (mode == B43legacy_RADIO_INTERFMODE_NONE) { phy->aci_enable = 0; phy->aci_hw_rssi = 0; } else b43legacy_radio_interference_mitigation_enable(dev, mode); phy->interfmode = mode; return 0;}u16 b43legacy_radio_calibrationvalue(struct b43legacy_wldev *dev){ u16 reg; u16 index; u16 ret; reg = b43legacy_radio_read16(dev, 0x0060); index = (reg & 0x001E) >> 1; ret = rcc_table[index] << 1; ret |= (reg & 0x0001); ret |= 0x0020; return ret;}#define LPD(L, P, D) (((L) << 2) | ((P) << 1) | ((D) << 0))static u16 b43legacy_get_812_value(struct b43legacy_wldev *dev, u8 lpd){ struct b43legacy_phy *phy = &dev->phy; u16 loop_or = 0; u16 adj_loopback_gain = phy->loopback_gain[0]; u8 loop; u16 extern_lna_control; if (!phy->gmode) return 0; if (!has_loopback_gain(phy)) { if (phy->rev < 7 || !(dev->dev->bus->sprom.r1.boardflags_lo & B43legacy_BFL_EXTLNA)) { switch (lpd) { case LPD(0, 1, 1): return 0x0FB2; case LPD(0, 0, 1): return 0x00B2; case LPD(1, 0, 1): return 0x30B2; case LPD(1, 0, 0): return 0x30B3; default: B43legacy_BUG_ON(1); } } else { switch (lpd) { case LPD(0, 1, 1): return 0x8FB2; case LPD(0, 0, 1): return 0x80B2; case LPD(1, 0, 1): return 0x20B2; case LPD(1, 0, 0): return 0x20B3; default: B43legacy_BUG_ON(1); } } } else { if (phy->radio_rev == 8) adj_loopback_gain += 0x003E; else adj_loopback_gain += 0x0026; if (adj_loopback_gain >= 0x46) { adj_loopback_gain -= 0x46; extern_lna_control = 0x3000; } else if (adj_loopback_gain >= 0x3A) { adj_loopback_gain -= 0x3A; extern_lna_control = 0x2000; } else if (adj_loopback_gain >= 0x2E) { adj_loopback_gain -= 0x2E; extern_lna_control = 0x1000; } else { adj_loopback_gain -= 0x10; extern_lna_control = 0x0000; } for (loop = 0; loop < 16; loop++) { u16 tmp = adj_loopback_gain - 6 * loop; if (tmp < 6) break; } loop_or = (loop << 8) | extern_lna_control; if (phy->rev >= 7 && dev->dev->bus->sprom.r1.boardflags_lo & B43legacy_BFL_EXTLNA) { if (extern_lna_control) loop_or |= 0x8000; switch (lpd) { case LPD(0, 1, 1): return 0x8F92; case LPD(0, 0, 1): return (0x8092 | loop_or); case LPD(1, 0, 1): return (0x2092 | loop_or); case LPD(1, 0, 0): return (0x2093 | loop_or); default: B43legacy_BUG_ON(1); } } else { switch (lpd) { case LPD(0, 1, 1): return 0x0F92; case LPD(0, 0, 1): case LPD(1, 0, 1): return (0x0092 | loop_or); case LPD(1, 0, 0): return (0x0093 | loop_or); default: B43legacy_BUG_ON(1); } } } return 0;}u16 b43legacy_radio_init2050(struct b43legacy_wldev *dev){ struct b43legacy_phy *phy = &dev->phy; u16 backup[21] = { 0 }; u16 ret; u16 i; u16 j; u32 tmp1 = 0; u32 tmp2 = 0; backup[0] = b43legacy_radio_read16(dev, 0x0043); backup[14] = b43legacy_radio_read16(dev, 0x0051); backup[15] = b43legacy_radio_read16(dev, 0x0052); backup[1] = b43legacy_phy_read(dev, 0x0015); backup[16] = b43legacy_phy_read(dev, 0x005A); backup[17] = b43legacy_phy_read(dev, 0x0059); backup[18] = b43legacy_phy_read(dev, 0x0058); if (phy->type == B43legacy_PHYTYPE_B) { backup[2] = b43legacy_phy_read(dev, 0x0030); backup[3] = b43legacy_read16(dev, 0x03EC); b43legacy_phy_write(dev, 0x0030, 0x00FF); b43legacy_write16(dev, 0x03EC, 0x3F3F); } else { if (phy->gmode) { backup[4] = b43legacy_phy_read(dev, 0x0811); backup[5] = b43legacy_phy_read(dev, 0x0812); backup[6] = b43legacy_phy_read(dev, 0x0814); backup[7] = b43legacy_phy_read(dev, 0x0815); backup[8] = b43legacy_phy_read(dev, B43legacy_PHY_G_CRS); backup[9] = b43legacy_phy_read(dev, 0x0802); b43legacy_phy_write(dev, 0x0814, (b43legacy_phy_read(dev, 0x0814) | 0x0003)); b43legacy_phy_write(dev, 0x0815, (b43legacy_phy_read(dev, 0x0815) & 0xFFFC)); b43legacy_phy_write(dev, B43legacy_PHY_G_CRS, (b43legacy_phy_read(dev, B43legacy_PHY_G_CRS) & 0x7FFF)); b43legacy_phy_write(dev, 0x0802, (b43legacy_phy_read(dev, 0x0802) & 0xFFFC)); if (phy->rev > 1) { /* loopback gain enabled */ backup[19] = b43legacy_phy_read(dev, 0x080F); backup[20] = b43legacy_phy_read(dev, 0x0810); if (phy->rev >= 3) b43legacy_phy_write(dev, 0x080F, 0xC020); else b43legacy_phy_write(dev, 0x080F, 0x8020); b43legacy_phy_write(dev, 0x0810, 0x0000); } b43legacy_phy_write(dev, 0x0812, b43legacy_get_812_value(dev, LPD(0, 1, 1))); if (phy->rev < 7 || !(dev->dev->bus->sprom.r1.boardflags_lo & B43legacy_BFL_EXTLNA)) b43legacy_phy_write(dev, 0x0811, 0x01B3); else b43legacy_phy_write(dev, 0x0811, 0x09B3); } } b43legacy_write16(dev, B43legacy_MMIO_PHY_RADIO, (b43legacy_read16(dev, B43legacy_MMIO_PHY_RADIO) | 0x8000)); backup[10] = b43legacy_phy_read(dev, 0x0035); b43legacy_phy_write(dev, 0x0035, (b43legacy_phy_read(dev, 0x0035) & 0xFF7F)); backup[11] = b43legacy_read16(dev, 0x03E6); backup[12] = b43legacy_read16(dev, B43legacy_MMIO_CHANNEL_EXT); /* Initialization */ if (phy->analog == 0) b43legacy_write16(dev, 0x03E6, 0x0122); else { if (phy->analog >= 2) b43legacy_phy_write(dev, 0x0003, (b43legacy_phy_read(dev, 0x0003) & 0xFFBF) | 0x0040); b43legacy_write16(dev, B43legacy_MMIO_CHANNEL_EXT, (b43legacy_read16(dev, B43legacy_MMIO_CHANNEL_EXT) | 0x2000)); } ret = b43legacy_radio_calibrationvalue(dev); if (phy->type == B43legacy_PHYTYPE_B) b43legacy_radio_write16(dev, 0x0078, 0x0026); if (phy->gmode) b43legacy_phy_write(dev, 0x0812, b43legacy_get_812_value(dev, LPD(0, 1, 1))); b43legacy_phy_write(dev, 0x0015, 0xBFAF); b43legacy_phy_write(dev, 0x002B, 0x1403); if (phy->gmode) b43legacy_phy_write(dev, 0x0812, b43legacy_get_812_value(dev, LPD(0, 0, 1))); b43legacy_phy_write(dev, 0x0015, 0xBFA0); b43legacy_radio_write16(dev, 0x0051, (b43legacy_radio_read16(dev, 0x0051) | 0x0004)); if (phy->radio_rev == 8) b43legacy_radio_write16(dev, 0x0043, 0x001F); else { b43legacy_radio_write16(dev, 0x0052, 0x0000); b43legacy_radio_write16(dev, 0x0043, (b43legacy_radio_read16(dev, 0x0043) & 0xFFF0) | 0x0009); } b43legacy_phy_write(dev, 0x0058, 0x0000); for (i = 0; i < 16; i++) { b43legacy_phy_write(dev, 0x005A, 0x0480); b43legacy_phy_write(dev, 0x0059, 0xC810); b43legacy_phy_write(dev, 0x0058, 0x000D); if (phy->gmode) b43legacy_phy_write(dev, 0x0812, b43legacy_get_812_value(dev, LPD(1, 0, 1))); b43legacy_phy_write(dev, 0x0015, 0xAFB0); udelay(10);
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -