diff options
Diffstat (limited to 'drivers/net')
-rw-r--r-- | drivers/net/wireless/p54/p54.h | 2 | ||||
-rw-r--r-- | drivers/net/wireless/p54/p54common.c | 158 |
2 files changed, 79 insertions, 81 deletions
diff --git a/drivers/net/wireless/p54/p54.h b/drivers/net/wireless/p54/p54.h index ebe1ea1f565b..d2dbb9e15d97 100644 --- a/drivers/net/wireless/p54/p54.h +++ b/drivers/net/wireless/p54/p54.h | |||
@@ -85,7 +85,6 @@ struct p54_common { | |||
85 | struct mutex conf_mutex; | 85 | struct mutex conf_mutex; |
86 | u8 mac_addr[ETH_ALEN]; | 86 | u8 mac_addr[ETH_ALEN]; |
87 | u8 bssid[ETH_ALEN]; | 87 | u8 bssid[ETH_ALEN]; |
88 | u16 mac_mode; | ||
89 | struct pda_iq_autocal_entry *iq_autocal; | 88 | struct pda_iq_autocal_entry *iq_autocal; |
90 | unsigned int iq_autocal_len; | 89 | unsigned int iq_autocal_len; |
91 | struct pda_channel_output_limit *output_limit; | 90 | struct pda_channel_output_limit *output_limit; |
@@ -95,7 +94,6 @@ struct p54_common { | |||
95 | bool use_short_slot; | 94 | bool use_short_slot; |
96 | u16 rxhw; | 95 | u16 rxhw; |
97 | u8 version; | 96 | u8 version; |
98 | u8 rx_antenna; | ||
99 | unsigned int tx_hdr_len; | 97 | unsigned int tx_hdr_len; |
100 | unsigned int fw_var; | 98 | unsigned int fw_var; |
101 | unsigned int fw_interface; | 99 | unsigned int fw_interface; |
diff --git a/drivers/net/wireless/p54/p54common.c b/drivers/net/wireless/p54/p54common.c index 1b5627d4062e..06c8321f4e3e 100644 --- a/drivers/net/wireless/p54/p54common.c +++ b/drivers/net/wireless/p54/p54common.c | |||
@@ -1334,11 +1334,12 @@ static int p54_tx(struct ieee80211_hw *dev, struct sk_buff *skb) | |||
1334 | return NETDEV_TX_BUSY; | 1334 | return NETDEV_TX_BUSY; |
1335 | } | 1335 | } |
1336 | 1336 | ||
1337 | static int p54_setup_mac(struct ieee80211_hw *dev, u16 mode, const u8 *bssid) | 1337 | static int p54_setup_mac(struct ieee80211_hw *dev) |
1338 | { | 1338 | { |
1339 | struct p54_common *priv = dev->priv; | 1339 | struct p54_common *priv = dev->priv; |
1340 | struct sk_buff *skb; | 1340 | struct sk_buff *skb; |
1341 | struct p54_setup_mac *setup; | 1341 | struct p54_setup_mac *setup; |
1342 | u16 mode; | ||
1342 | 1343 | ||
1343 | skb = p54_alloc_skb(dev, P54_HDR_FLAG_CONTROL_OPSET, sizeof(*setup) + | 1344 | skb = p54_alloc_skb(dev, P54_HDR_FLAG_CONTROL_OPSET, sizeof(*setup) + |
1344 | sizeof(struct p54_hdr), P54_CONTROL_TYPE_SETUP, | 1345 | sizeof(struct p54_hdr), P54_CONTROL_TYPE_SETUP, |
@@ -1347,14 +1348,31 @@ static int p54_setup_mac(struct ieee80211_hw *dev, u16 mode, const u8 *bssid) | |||
1347 | return -ENOMEM; | 1348 | return -ENOMEM; |
1348 | 1349 | ||
1349 | setup = (struct p54_setup_mac *) skb_put(skb, sizeof(*setup)); | 1350 | setup = (struct p54_setup_mac *) skb_put(skb, sizeof(*setup)); |
1350 | priv->mac_mode = mode; | 1351 | if (dev->conf.radio_enabled) { |
1352 | switch (priv->mode) { | ||
1353 | case NL80211_IFTYPE_STATION: | ||
1354 | mode = P54_FILTER_TYPE_STATION; | ||
1355 | break; | ||
1356 | case NL80211_IFTYPE_AP: | ||
1357 | mode = P54_FILTER_TYPE_AP; | ||
1358 | break; | ||
1359 | case NL80211_IFTYPE_ADHOC: | ||
1360 | case NL80211_IFTYPE_MESH_POINT: | ||
1361 | mode = P54_FILTER_TYPE_IBSS; | ||
1362 | break; | ||
1363 | default: | ||
1364 | mode = P54_FILTER_TYPE_NONE; | ||
1365 | break; | ||
1366 | } | ||
1367 | if (priv->filter_flags & FIF_PROMISC_IN_BSS) | ||
1368 | mode |= P54_FILTER_TYPE_TRANSPARENT; | ||
1369 | } else | ||
1370 | mode = P54_FILTER_TYPE_RX_DISABLED; | ||
1371 | |||
1351 | setup->mac_mode = cpu_to_le16(mode); | 1372 | setup->mac_mode = cpu_to_le16(mode); |
1352 | memcpy(setup->mac_addr, priv->mac_addr, ETH_ALEN); | 1373 | memcpy(setup->mac_addr, priv->mac_addr, ETH_ALEN); |
1353 | if (!bssid) | 1374 | memcpy(setup->bssid, priv->bssid, ETH_ALEN); |
1354 | memset(setup->bssid, ~0, ETH_ALEN); | 1375 | setup->rx_antenna = 2; /* automatic */ |
1355 | else | ||
1356 | memcpy(setup->bssid, bssid, ETH_ALEN); | ||
1357 | setup->rx_antenna = priv->rx_antenna; | ||
1358 | setup->rx_align = 0; | 1376 | setup->rx_align = 0; |
1359 | if (priv->fw_var < 0x500) { | 1377 | if (priv->fw_var < 0x500) { |
1360 | setup->v1.basic_rate_mask = cpu_to_le32(priv->basic_rate_mask); | 1378 | setup->v1.basic_rate_mask = cpu_to_le32(priv->basic_rate_mask); |
@@ -1383,7 +1401,8 @@ static int p54_setup_mac(struct ieee80211_hw *dev, u16 mode, const u8 *bssid) | |||
1383 | return 0; | 1401 | return 0; |
1384 | } | 1402 | } |
1385 | 1403 | ||
1386 | static int p54_set_freq(struct ieee80211_hw *dev, u16 frequency) | 1404 | static int p54_scan(struct ieee80211_hw *dev, u16 mode, u16 dwell, |
1405 | u16 frequency) | ||
1387 | { | 1406 | { |
1388 | struct p54_common *priv = dev->priv; | 1407 | struct p54_common *priv = dev->priv; |
1389 | struct sk_buff *skb; | 1408 | struct sk_buff *skb; |
@@ -1400,8 +1419,8 @@ static int p54_set_freq(struct ieee80211_hw *dev, u16 frequency) | |||
1400 | 1419 | ||
1401 | chan = (struct p54_scan *) skb_put(skb, sizeof(*chan)); | 1420 | chan = (struct p54_scan *) skb_put(skb, sizeof(*chan)); |
1402 | memset(chan->padding1, 0, sizeof(chan->padding1)); | 1421 | memset(chan->padding1, 0, sizeof(chan->padding1)); |
1403 | chan->mode = cpu_to_le16(P54_SCAN_EXIT); | 1422 | chan->mode = cpu_to_le16(mode); |
1404 | chan->dwell = cpu_to_le16(0x0); | 1423 | chan->dwell = cpu_to_le16(dwell); |
1405 | 1424 | ||
1406 | for (i = 0; i < priv->iq_autocal_len; i++) { | 1425 | for (i = 0; i < priv->iq_autocal_len; i++) { |
1407 | if (priv->iq_autocal[i].freq != freq) | 1426 | if (priv->iq_autocal[i].freq != freq) |
@@ -1644,10 +1663,14 @@ static int p54_start(struct ieee80211_hw *dev) | |||
1644 | err = p54_init_stats(dev); | 1663 | err = p54_init_stats(dev); |
1645 | if (err) | 1664 | if (err) |
1646 | goto out; | 1665 | goto out; |
1647 | err = p54_setup_mac(dev, P54_FILTER_TYPE_NONE, NULL); | 1666 | |
1648 | if (err) | 1667 | memset(priv->bssid, ~0, ETH_ALEN); |
1649 | goto out; | ||
1650 | priv->mode = NL80211_IFTYPE_MONITOR; | 1668 | priv->mode = NL80211_IFTYPE_MONITOR; |
1669 | err = p54_setup_mac(dev); | ||
1670 | if (err) { | ||
1671 | priv->mode = NL80211_IFTYPE_UNSPECIFIED; | ||
1672 | goto out; | ||
1673 | } | ||
1651 | 1674 | ||
1652 | out: | 1675 | out: |
1653 | mutex_unlock(&priv->conf_mutex); | 1676 | mutex_unlock(&priv->conf_mutex); |
@@ -1700,27 +1723,8 @@ static int p54_add_interface(struct ieee80211_hw *dev, | |||
1700 | } | 1723 | } |
1701 | 1724 | ||
1702 | memcpy(priv->mac_addr, conf->mac_addr, ETH_ALEN); | 1725 | memcpy(priv->mac_addr, conf->mac_addr, ETH_ALEN); |
1703 | 1726 | p54_setup_mac(dev); | |
1704 | p54_setup_mac(dev, P54_FILTER_TYPE_NONE, NULL); | ||
1705 | |||
1706 | switch (conf->type) { | ||
1707 | case NL80211_IFTYPE_STATION: | ||
1708 | p54_setup_mac(dev, P54_FILTER_TYPE_STATION, NULL); | ||
1709 | break; | ||
1710 | case NL80211_IFTYPE_AP: | ||
1711 | p54_setup_mac(dev, P54_FILTER_TYPE_AP, priv->mac_addr); | ||
1712 | break; | ||
1713 | case NL80211_IFTYPE_ADHOC: | ||
1714 | case NL80211_IFTYPE_MESH_POINT: | ||
1715 | p54_setup_mac(dev, P54_FILTER_TYPE_IBSS, NULL); | ||
1716 | break; | ||
1717 | default: | ||
1718 | BUG(); /* impossible */ | ||
1719 | break; | ||
1720 | } | ||
1721 | |||
1722 | p54_set_leds(dev, 1, 0, 0); | 1727 | p54_set_leds(dev, 1, 0, 0); |
1723 | |||
1724 | mutex_unlock(&priv->conf_mutex); | 1728 | mutex_unlock(&priv->conf_mutex); |
1725 | return 0; | 1729 | return 0; |
1726 | } | 1730 | } |
@@ -1733,9 +1737,10 @@ static void p54_remove_interface(struct ieee80211_hw *dev, | |||
1733 | mutex_lock(&priv->conf_mutex); | 1737 | mutex_lock(&priv->conf_mutex); |
1734 | if (priv->cached_beacon) | 1738 | if (priv->cached_beacon) |
1735 | p54_tx_cancel(dev, priv->cached_beacon); | 1739 | p54_tx_cancel(dev, priv->cached_beacon); |
1736 | p54_setup_mac(dev, P54_FILTER_TYPE_NONE, NULL); | ||
1737 | priv->mode = NL80211_IFTYPE_MONITOR; | 1740 | priv->mode = NL80211_IFTYPE_MONITOR; |
1738 | memset(priv->mac_addr, 0, ETH_ALEN); | 1741 | memset(priv->mac_addr, 0, ETH_ALEN); |
1742 | memset(priv->bssid, 0, ETH_ALEN); | ||
1743 | p54_setup_mac(dev); | ||
1739 | mutex_unlock(&priv->conf_mutex); | 1744 | mutex_unlock(&priv->conf_mutex); |
1740 | } | 1745 | } |
1741 | 1746 | ||
@@ -1746,11 +1751,21 @@ static int p54_config(struct ieee80211_hw *dev, u32 changed) | |||
1746 | struct ieee80211_conf *conf = &dev->conf; | 1751 | struct ieee80211_conf *conf = &dev->conf; |
1747 | 1752 | ||
1748 | mutex_lock(&priv->conf_mutex); | 1753 | mutex_lock(&priv->conf_mutex); |
1749 | priv->rx_antenna = 2; /* automatic */ | 1754 | if (changed & IEEE80211_CONF_CHANGE_POWER) |
1750 | priv->output_power = conf->power_level << 2; | 1755 | priv->output_power = conf->power_level << 2; |
1751 | ret = p54_set_freq(dev, conf->channel->center_freq); | 1756 | if (changed & IEEE80211_CONF_CHANGE_RADIO_ENABLED) { |
1752 | if (!ret) | 1757 | ret = p54_setup_mac(dev); |
1753 | ret = p54_set_edcf(dev); | 1758 | if (ret) |
1759 | goto out; | ||
1760 | } | ||
1761 | if (changed & IEEE80211_CONF_CHANGE_CHANNEL) { | ||
1762 | ret = p54_scan(dev, P54_SCAN_EXIT, 0, | ||
1763 | conf->channel->center_freq); | ||
1764 | if (ret) | ||
1765 | goto out; | ||
1766 | } | ||
1767 | |||
1768 | out: | ||
1754 | mutex_unlock(&priv->conf_mutex); | 1769 | mutex_unlock(&priv->conf_mutex); |
1755 | return ret; | 1770 | return ret; |
1756 | } | 1771 | } |
@@ -1763,36 +1778,31 @@ static int p54_config_interface(struct ieee80211_hw *dev, | |||
1763 | int ret = 0; | 1778 | int ret = 0; |
1764 | 1779 | ||
1765 | mutex_lock(&priv->conf_mutex); | 1780 | mutex_lock(&priv->conf_mutex); |
1766 | switch (priv->mode) { | 1781 | if (conf->changed & IEEE80211_IFCC_BSSID) { |
1767 | case NL80211_IFTYPE_STATION: | 1782 | memcpy(priv->bssid, conf->bssid, ETH_ALEN); |
1768 | ret = p54_setup_mac(dev, P54_FILTER_TYPE_STATION, conf->bssid); | 1783 | ret = p54_setup_mac(dev); |
1769 | if (ret) | 1784 | if (ret) |
1770 | goto out; | 1785 | goto out; |
1771 | ret = p54_set_leds(dev, 1, | 1786 | } |
1772 | !is_multicast_ether_addr(conf->bssid), 0); | 1787 | |
1788 | if (conf->changed & IEEE80211_IFCC_BEACON) { | ||
1789 | ret = p54_scan(dev, P54_SCAN_EXIT, 0, | ||
1790 | dev->conf.channel->center_freq); | ||
1773 | if (ret) | 1791 | if (ret) |
1774 | goto out; | 1792 | goto out; |
1775 | memcpy(priv->bssid, conf->bssid, ETH_ALEN); | 1793 | ret = p54_setup_mac(dev); |
1776 | break; | ||
1777 | case NL80211_IFTYPE_AP: | ||
1778 | case NL80211_IFTYPE_ADHOC: | ||
1779 | case NL80211_IFTYPE_MESH_POINT: | ||
1780 | memcpy(priv->bssid, conf->bssid, ETH_ALEN); | ||
1781 | ret = p54_set_freq(dev, dev->conf.channel->center_freq); | ||
1782 | if (ret) | 1794 | if (ret) |
1783 | goto out; | 1795 | goto out; |
1784 | ret = p54_setup_mac(dev, priv->mac_mode, priv->bssid); | 1796 | ret = p54_beacon_update(dev, vif); |
1797 | if (ret) | ||
1798 | goto out; | ||
1799 | ret = p54_set_edcf(dev); | ||
1785 | if (ret) | 1800 | if (ret) |
1786 | goto out; | 1801 | goto out; |
1787 | if (conf->changed & IEEE80211_IFCC_BEACON) { | ||
1788 | ret = p54_beacon_update(dev, vif); | ||
1789 | if (ret) | ||
1790 | goto out; | ||
1791 | ret = p54_set_edcf(dev); | ||
1792 | if (ret) | ||
1793 | goto out; | ||
1794 | } | ||
1795 | } | 1802 | } |
1803 | |||
1804 | ret = p54_set_leds(dev, 1, !is_multicast_ether_addr(priv->bssid), 0); | ||
1805 | |||
1796 | out: | 1806 | out: |
1797 | mutex_unlock(&priv->conf_mutex); | 1807 | mutex_unlock(&priv->conf_mutex); |
1798 | return ret; | 1808 | return ret; |
@@ -1805,25 +1815,14 @@ static void p54_configure_filter(struct ieee80211_hw *dev, | |||
1805 | { | 1815 | { |
1806 | struct p54_common *priv = dev->priv; | 1816 | struct p54_common *priv = dev->priv; |
1807 | 1817 | ||
1808 | *total_flags &= FIF_BCN_PRBRESP_PROMISC | | 1818 | *total_flags &= FIF_PROMISC_IN_BSS | |
1809 | FIF_PROMISC_IN_BSS | | 1819 | (*total_flags & FIF_PROMISC_IN_BSS) ? |
1810 | FIF_FCSFAIL; | 1820 | FIF_FCSFAIL : 0; |
1811 | 1821 | ||
1812 | priv->filter_flags = *total_flags; | 1822 | priv->filter_flags = *total_flags; |
1813 | 1823 | ||
1814 | if (changed_flags & FIF_BCN_PRBRESP_PROMISC) { | 1824 | if (changed_flags & FIF_PROMISC_IN_BSS) |
1815 | if (*total_flags & FIF_BCN_PRBRESP_PROMISC) | 1825 | p54_setup_mac(dev); |
1816 | p54_setup_mac(dev, priv->mac_mode, NULL); | ||
1817 | else | ||
1818 | p54_setup_mac(dev, priv->mac_mode, priv->bssid); | ||
1819 | } | ||
1820 | |||
1821 | if (changed_flags & FIF_PROMISC_IN_BSS) { | ||
1822 | if (*total_flags & FIF_PROMISC_IN_BSS) | ||
1823 | p54_setup_mac(dev, priv->mac_mode | 0x8, NULL); | ||
1824 | else | ||
1825 | p54_setup_mac(dev, priv->mac_mode & ~0x8, priv->bssid); | ||
1826 | } | ||
1827 | } | 1826 | } |
1828 | 1827 | ||
1829 | static int p54_conf_tx(struct ieee80211_hw *dev, u16 queue, | 1828 | static int p54_conf_tx(struct ieee80211_hw *dev, u16 queue, |
@@ -1920,16 +1919,17 @@ static void p54_bss_info_changed(struct ieee80211_hw *dev, | |||
1920 | priv->basic_rate_mask = (info->basic_rates << 4); | 1919 | priv->basic_rate_mask = (info->basic_rates << 4); |
1921 | else | 1920 | else |
1922 | priv->basic_rate_mask = info->basic_rates; | 1921 | priv->basic_rate_mask = info->basic_rates; |
1923 | p54_setup_mac(dev, priv->mac_mode, priv->bssid); | 1922 | p54_setup_mac(dev); |
1924 | if (priv->fw_var >= 0x500) | 1923 | if (priv->fw_var >= 0x500) |
1925 | p54_set_freq(dev, dev->conf.channel->center_freq); | 1924 | p54_scan(dev, P54_SCAN_EXIT, 0, |
1925 | dev->conf.channel->center_freq); | ||
1926 | } | 1926 | } |
1927 | if (changed & BSS_CHANGED_ASSOC) { | 1927 | if (changed & BSS_CHANGED_ASSOC) { |
1928 | if (info->assoc) { | 1928 | if (info->assoc) { |
1929 | priv->aid = info->aid; | 1929 | priv->aid = info->aid; |
1930 | priv->wakeup_timer = info->beacon_int * | 1930 | priv->wakeup_timer = info->beacon_int * |
1931 | info->dtim_period * 5; | 1931 | info->dtim_period * 5; |
1932 | p54_setup_mac(dev, priv->mac_mode, priv->bssid); | 1932 | p54_setup_mac(dev); |
1933 | } | 1933 | } |
1934 | } | 1934 | } |
1935 | 1935 | ||