diff options
-rw-r--r-- | drivers/net/wireless/orinoco/main.c | 140 |
1 files changed, 81 insertions, 59 deletions
diff --git a/drivers/net/wireless/orinoco/main.c b/drivers/net/wireless/orinoco/main.c index f49cabd70877..c5aad88e54d9 100644 --- a/drivers/net/wireless/orinoco/main.c +++ b/drivers/net/wireless/orinoco/main.c | |||
@@ -282,6 +282,33 @@ static inline void set_port_type(struct orinoco_private *priv) | |||
282 | } | 282 | } |
283 | } | 283 | } |
284 | 284 | ||
285 | static int orinoco_get_bitratemode(int bitrate, int automatic) | ||
286 | { | ||
287 | int ratemode = -1; | ||
288 | int i; | ||
289 | |||
290 | if ((bitrate != 10) && (bitrate != 20) && | ||
291 | (bitrate != 55) && (bitrate != 110)) | ||
292 | return ratemode; | ||
293 | |||
294 | for (i = 0; i < BITRATE_TABLE_SIZE; i++) { | ||
295 | if ((bitrate_table[i].bitrate == bitrate) && | ||
296 | (bitrate_table[i].automatic == automatic)) { | ||
297 | ratemode = i; | ||
298 | break; | ||
299 | } | ||
300 | } | ||
301 | return ratemode; | ||
302 | } | ||
303 | |||
304 | static void orinoco_get_ratemode_cfg(int ratemode, int *bitrate, int *automatic) | ||
305 | { | ||
306 | BUG_ON((ratemode < 0) || (ratemode >= BITRATE_TABLE_SIZE)); | ||
307 | |||
308 | *bitrate = bitrate_table[ratemode].bitrate * 100000; | ||
309 | *automatic = bitrate_table[ratemode].automatic; | ||
310 | } | ||
311 | |||
285 | static inline u8 *orinoco_get_ie(u8 *data, size_t len, | 312 | static inline u8 *orinoco_get_ie(u8 *data, size_t len, |
286 | enum ieee80211_eid eid) | 313 | enum ieee80211_eid eid) |
287 | { | 314 | { |
@@ -1794,6 +1821,50 @@ static int __orinoco_hw_set_bitrate(struct orinoco_private *priv) | |||
1794 | return err; | 1821 | return err; |
1795 | } | 1822 | } |
1796 | 1823 | ||
1824 | static int orinoco_hw_get_act_bitrate(struct orinoco_private *priv, | ||
1825 | int *bitrate) | ||
1826 | { | ||
1827 | hermes_t *hw = &priv->hw; | ||
1828 | int i; | ||
1829 | int err = 0; | ||
1830 | u16 val; | ||
1831 | |||
1832 | err = hermes_read_wordrec(hw, USER_BAP, | ||
1833 | HERMES_RID_CURRENTTXRATE, &val); | ||
1834 | if (err) | ||
1835 | return err; | ||
1836 | |||
1837 | switch (priv->firmware_type) { | ||
1838 | case FIRMWARE_TYPE_AGERE: /* Lucent style rate */ | ||
1839 | /* Note : in Lucent firmware, the return value of | ||
1840 | * HERMES_RID_CURRENTTXRATE is the bitrate in Mb/s, | ||
1841 | * and therefore is totally different from the | ||
1842 | * encoding of HERMES_RID_CNFTXRATECONTROL. | ||
1843 | * Don't forget that 6Mb/s is really 5.5Mb/s */ | ||
1844 | if (val == 6) | ||
1845 | *bitrate = 5500000; | ||
1846 | else | ||
1847 | *bitrate = val * 1000000; | ||
1848 | break; | ||
1849 | case FIRMWARE_TYPE_INTERSIL: /* Intersil style rate */ | ||
1850 | case FIRMWARE_TYPE_SYMBOL: /* Symbol style rate */ | ||
1851 | for (i = 0; i < BITRATE_TABLE_SIZE; i++) | ||
1852 | if (bitrate_table[i].intersil_txratectrl == val) | ||
1853 | break; | ||
1854 | |||
1855 | if (i >= BITRATE_TABLE_SIZE) | ||
1856 | printk(KERN_INFO "%s: Unable to determine current bitrate (0x%04hx)\n", | ||
1857 | priv->ndev->name, val); | ||
1858 | |||
1859 | *bitrate = bitrate_table[i].bitrate * 100000; | ||
1860 | break; | ||
1861 | default: | ||
1862 | BUG(); | ||
1863 | } | ||
1864 | |||
1865 | return err; | ||
1866 | } | ||
1867 | |||
1797 | /* Set fixed AP address */ | 1868 | /* Set fixed AP address */ |
1798 | static int __orinoco_hw_set_wap(struct orinoco_private *priv) | 1869 | static int __orinoco_hw_set_wap(struct orinoco_private *priv) |
1799 | { | 1870 | { |
@@ -3975,9 +4046,8 @@ static int orinoco_ioctl_setrate(struct net_device *dev, | |||
3975 | char *extra) | 4046 | char *extra) |
3976 | { | 4047 | { |
3977 | struct orinoco_private *priv = netdev_priv(dev); | 4048 | struct orinoco_private *priv = netdev_priv(dev); |
3978 | int ratemode = -1; | 4049 | int ratemode; |
3979 | int bitrate; /* 100s of kilobits */ | 4050 | int bitrate; /* 100s of kilobits */ |
3980 | int i; | ||
3981 | unsigned long flags; | 4051 | unsigned long flags; |
3982 | 4052 | ||
3983 | /* As the user space doesn't know our highest rate, it uses -1 | 4053 | /* As the user space doesn't know our highest rate, it uses -1 |
@@ -3991,16 +4061,7 @@ static int orinoco_ioctl_setrate(struct net_device *dev, | |||
3991 | bitrate = rrq->value / 100000; | 4061 | bitrate = rrq->value / 100000; |
3992 | } | 4062 | } |
3993 | 4063 | ||
3994 | if ((bitrate != 10) && (bitrate != 20) && | 4064 | ratemode = orinoco_get_bitratemode(bitrate, !rrq->fixed); |
3995 | (bitrate != 55) && (bitrate != 110)) | ||
3996 | return -EINVAL; | ||
3997 | |||
3998 | for (i = 0; i < BITRATE_TABLE_SIZE; i++) | ||
3999 | if ((bitrate_table[i].bitrate == bitrate) && | ||
4000 | (bitrate_table[i].automatic == !rrq->fixed)) { | ||
4001 | ratemode = i; | ||
4002 | break; | ||
4003 | } | ||
4004 | 4065 | ||
4005 | if (ratemode == -1) | 4066 | if (ratemode == -1) |
4006 | return -EINVAL; | 4067 | return -EINVAL; |
@@ -4019,65 +4080,26 @@ static int orinoco_ioctl_getrate(struct net_device *dev, | |||
4019 | char *extra) | 4080 | char *extra) |
4020 | { | 4081 | { |
4021 | struct orinoco_private *priv = netdev_priv(dev); | 4082 | struct orinoco_private *priv = netdev_priv(dev); |
4022 | hermes_t *hw = &priv->hw; | ||
4023 | int err = 0; | 4083 | int err = 0; |
4024 | int ratemode; | 4084 | int bitrate, automatic; |
4025 | int i; | ||
4026 | u16 val; | ||
4027 | unsigned long flags; | 4085 | unsigned long flags; |
4028 | 4086 | ||
4029 | if (orinoco_lock(priv, &flags) != 0) | 4087 | if (orinoco_lock(priv, &flags) != 0) |
4030 | return -EBUSY; | 4088 | return -EBUSY; |
4031 | 4089 | ||
4032 | ratemode = priv->bitratemode; | 4090 | orinoco_get_ratemode_cfg(priv->bitratemode, &bitrate, &automatic); |
4033 | |||
4034 | BUG_ON((ratemode < 0) || (ratemode >= BITRATE_TABLE_SIZE)); | ||
4035 | |||
4036 | rrq->value = bitrate_table[ratemode].bitrate * 100000; | ||
4037 | rrq->fixed = !bitrate_table[ratemode].automatic; | ||
4038 | rrq->disabled = 0; | ||
4039 | 4091 | ||
4040 | /* If the interface is running we try to find more about the | 4092 | /* If the interface is running we try to find more about the |
4041 | current mode */ | 4093 | current mode */ |
4042 | if (netif_running(dev)) { | 4094 | if (netif_running(dev)) |
4043 | err = hermes_read_wordrec(hw, USER_BAP, | 4095 | err = orinoco_hw_get_act_bitrate(priv, &bitrate); |
4044 | HERMES_RID_CURRENTTXRATE, &val); | ||
4045 | if (err) | ||
4046 | goto out; | ||
4047 | 4096 | ||
4048 | switch (priv->firmware_type) { | ||
4049 | case FIRMWARE_TYPE_AGERE: /* Lucent style rate */ | ||
4050 | /* Note : in Lucent firmware, the return value of | ||
4051 | * HERMES_RID_CURRENTTXRATE is the bitrate in Mb/s, | ||
4052 | * and therefore is totally different from the | ||
4053 | * encoding of HERMES_RID_CNFTXRATECONTROL. | ||
4054 | * Don't forget that 6Mb/s is really 5.5Mb/s */ | ||
4055 | if (val == 6) | ||
4056 | rrq->value = 5500000; | ||
4057 | else | ||
4058 | rrq->value = val * 1000000; | ||
4059 | break; | ||
4060 | case FIRMWARE_TYPE_INTERSIL: /* Intersil style rate */ | ||
4061 | case FIRMWARE_TYPE_SYMBOL: /* Symbol style rate */ | ||
4062 | for (i = 0; i < BITRATE_TABLE_SIZE; i++) | ||
4063 | if (bitrate_table[i].intersil_txratectrl == val) { | ||
4064 | ratemode = i; | ||
4065 | break; | ||
4066 | } | ||
4067 | if (i >= BITRATE_TABLE_SIZE) | ||
4068 | printk(KERN_INFO "%s: Unable to determine current bitrate (0x%04hx)\n", | ||
4069 | dev->name, val); | ||
4070 | |||
4071 | rrq->value = bitrate_table[ratemode].bitrate * 100000; | ||
4072 | break; | ||
4073 | default: | ||
4074 | BUG(); | ||
4075 | } | ||
4076 | } | ||
4077 | |||
4078 | out: | ||
4079 | orinoco_unlock(priv, &flags); | 4097 | orinoco_unlock(priv, &flags); |
4080 | 4098 | ||
4099 | rrq->value = bitrate; | ||
4100 | rrq->fixed = !automatic; | ||
4101 | rrq->disabled = 0; | ||
4102 | |||
4081 | return err; | 4103 | return err; |
4082 | } | 4104 | } |
4083 | 4105 | ||