2007-06-13 09:11:42 +07:00
|
|
|
/*
|
|
|
|
* Definitions for RTL8187 hardware
|
|
|
|
*
|
|
|
|
* Copyright 2007 Michael Wu <flamingice@sourmilk.net>
|
|
|
|
* Copyright 2007 Andrea Merello <andreamrl@tiscali.it>
|
|
|
|
*
|
|
|
|
* Based on the r8187 driver, which is:
|
|
|
|
* Copyright 2005 Andrea Merello <andreamrl@tiscali.it>, et al.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*/
|
|
|
|
|
2007-05-14 12:41:02 +07:00
|
|
|
#ifndef RTL8187_H
|
|
|
|
#define RTL8187_H
|
|
|
|
|
|
|
|
#include "rtl818x.h"
|
2009-04-17 07:56:38 +07:00
|
|
|
#include "rtl8187_leds.h"
|
2007-05-14 12:41:02 +07:00
|
|
|
|
|
|
|
#define RTL8187_EEPROM_TXPWR_BASE 0x05
|
|
|
|
#define RTL8187_EEPROM_MAC_ADDR 0x07
|
|
|
|
#define RTL8187_EEPROM_TXPWR_CHAN_1 0x16 /* 3 channels */
|
|
|
|
#define RTL8187_EEPROM_TXPWR_CHAN_6 0x1B /* 2 channels */
|
|
|
|
#define RTL8187_EEPROM_TXPWR_CHAN_4 0x3D /* 2 channels */
|
2009-12-06 08:25:22 +07:00
|
|
|
#define RTL8187_EEPROM_SELECT_GPIO 0x3B
|
2007-05-14 12:41:02 +07:00
|
|
|
|
|
|
|
#define RTL8187_REQT_READ 0xC0
|
|
|
|
#define RTL8187_REQT_WRITE 0x40
|
|
|
|
#define RTL8187_REQ_GET_REG 0x05
|
|
|
|
#define RTL8187_REQ_SET_REG 0x05
|
|
|
|
|
|
|
|
#define RTL8187_MAX_RX 0x9C4
|
|
|
|
|
2009-12-06 08:25:22 +07:00
|
|
|
#define RFKILL_MASK_8187_89_97 0x2
|
|
|
|
#define RFKILL_MASK_8198 0x4
|
|
|
|
|
2007-05-14 12:41:02 +07:00
|
|
|
struct rtl8187_rx_info {
|
|
|
|
struct urb *urb;
|
|
|
|
struct ieee80211_hw *dev;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct rtl8187_rx_hdr {
|
[PATCH] mac80211: revamp interface and filter configuration
Drivers are currently supposed to keep track of monitor
interfaces if they allow so-called "hard" monitor, and
they are also supposed to keep track of multicast etc.
This patch changes that, replaces the set_multicast_list()
callback with a new configure_filter() callback that takes
filter flags (FIF_*) instead of interface flags (IFF_*).
For a driver, this means it should open the filter as much
as necessary to get all frames requested by the filter flags.
Accordingly, the filter flags are named "positively", e.g.
FIF_ALLMULTI.
Multicast filtering is a bit special in that drivers that
have no multicast address filters need to allow multicast
frames through when either the FIF_ALLMULTI flag is set or
when the mc_count value is positive.
At the same time, drivers are no longer notified about
monitor interfaces at all, this means they now need to
implement the start() and stop() callbacks and the new
change_filter_flags() callback. Also, the start()/stop()
ordering changed, start() is now called *before* any
add_interface() as it really should be, and stop() after
any remove_interface().
The patch also changes the behaviour of setting the bssid
to multicast for scanning when IEEE80211_HW_NO_PROBE_FILTERING
is set; the IEEE80211_HW_NO_PROBE_FILTERING flag is removed
and the filter flag FIF_BCN_PRBRESP_PROMISC introduced.
This is a lot more efficient for hardware like b43 that
supports it and other hardware can still set the BSSID
to all-ones.
Driver modifications by Johannes Berg (b43 & iwlwifi), Michael Wu
(rtl8187, adm8211, and p54), Larry Finger (b43legacy), and
Ivo van Doorn (rt2x00).
Signed-off-by: Johannes Berg <johannes@sipsolutions.net>
Signed-off-by: Michael Wu <flamingice@sourmilk.net>
Signed-off-by: Larry Finger <Larry.Finger@lwfinger.net>
Signed-off-by: Ivo van Doorn <IvDoorn@gmail.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2007-09-17 12:29:23 +07:00
|
|
|
__le32 flags;
|
2007-05-14 12:41:02 +07:00
|
|
|
u8 noise;
|
|
|
|
u8 signal;
|
|
|
|
u8 agc;
|
|
|
|
u8 reserved;
|
|
|
|
__le64 mac_time;
|
2010-06-03 01:10:09 +07:00
|
|
|
} __packed;
|
2007-05-14 12:41:02 +07:00
|
|
|
|
2008-07-08 18:27:54 +07:00
|
|
|
struct rtl8187b_rx_hdr {
|
2007-05-14 12:41:02 +07:00
|
|
|
__le32 flags;
|
2008-07-08 18:27:54 +07:00
|
|
|
__le64 mac_time;
|
2008-07-29 10:25:08 +07:00
|
|
|
u8 sq;
|
|
|
|
u8 rssi;
|
2008-07-08 18:27:54 +07:00
|
|
|
u8 agc;
|
2008-07-29 10:25:08 +07:00
|
|
|
u8 flags2;
|
|
|
|
__le16 snr_long2end;
|
|
|
|
s8 pwdb_g12;
|
|
|
|
u8 fot;
|
2010-06-03 01:10:09 +07:00
|
|
|
} __packed;
|
2008-07-08 18:27:54 +07:00
|
|
|
|
|
|
|
/* {rtl8187,rtl8187b}_tx_info is in skb */
|
|
|
|
|
|
|
|
struct rtl8187_tx_hdr {
|
|
|
|
__le32 flags;
|
2007-05-14 12:41:02 +07:00
|
|
|
__le16 rts_duration;
|
|
|
|
__le16 len;
|
|
|
|
__le32 retry;
|
2010-06-03 01:10:09 +07:00
|
|
|
} __packed;
|
2007-05-14 12:41:02 +07:00
|
|
|
|
2008-07-08 18:27:54 +07:00
|
|
|
struct rtl8187b_tx_hdr {
|
|
|
|
__le32 flags;
|
|
|
|
__le16 rts_duration;
|
|
|
|
__le16 len;
|
|
|
|
__le32 unused_1;
|
|
|
|
__le16 unused_2;
|
|
|
|
__le16 tx_duration;
|
|
|
|
__le32 unused_3;
|
|
|
|
__le32 retry;
|
|
|
|
__le32 unused_4[2];
|
2010-06-03 01:10:09 +07:00
|
|
|
} __packed;
|
2008-07-08 18:27:54 +07:00
|
|
|
|
|
|
|
enum {
|
|
|
|
DEVICE_RTL8187,
|
|
|
|
DEVICE_RTL8187B
|
|
|
|
};
|
|
|
|
|
2007-05-14 12:41:02 +07:00
|
|
|
struct rtl8187_priv {
|
|
|
|
/* common between rtl818x drivers */
|
|
|
|
struct rtl818x_csr *map;
|
2007-10-15 01:43:16 +07:00
|
|
|
const struct rtl818x_rf_ops *rf;
|
2007-12-19 07:31:26 +07:00
|
|
|
struct ieee80211_vif *vif;
|
2009-12-23 06:13:05 +07:00
|
|
|
|
2008-08-01 07:30:48 +07:00
|
|
|
/* The mutex protects the TX loopback state.
|
|
|
|
* Any attempt to set channels concurrently locks the device.
|
|
|
|
*/
|
|
|
|
struct mutex conf_mutex;
|
2007-05-14 12:41:02 +07:00
|
|
|
|
|
|
|
/* rtl8187 specific */
|
|
|
|
struct ieee80211_channel channels[14];
|
|
|
|
struct ieee80211_rate rates[12];
|
2008-01-25 01:38:38 +07:00
|
|
|
struct ieee80211_supported_band band;
|
2007-05-14 12:41:02 +07:00
|
|
|
struct usb_device *udev;
|
[PATCH] mac80211: revamp interface and filter configuration
Drivers are currently supposed to keep track of monitor
interfaces if they allow so-called "hard" monitor, and
they are also supposed to keep track of multicast etc.
This patch changes that, replaces the set_multicast_list()
callback with a new configure_filter() callback that takes
filter flags (FIF_*) instead of interface flags (IFF_*).
For a driver, this means it should open the filter as much
as necessary to get all frames requested by the filter flags.
Accordingly, the filter flags are named "positively", e.g.
FIF_ALLMULTI.
Multicast filtering is a bit special in that drivers that
have no multicast address filters need to allow multicast
frames through when either the FIF_ALLMULTI flag is set or
when the mc_count value is positive.
At the same time, drivers are no longer notified about
monitor interfaces at all, this means they now need to
implement the start() and stop() callbacks and the new
change_filter_flags() callback. Also, the start()/stop()
ordering changed, start() is now called *before* any
add_interface() as it really should be, and stop() after
any remove_interface().
The patch also changes the behaviour of setting the bssid
to multicast for scanning when IEEE80211_HW_NO_PROBE_FILTERING
is set; the IEEE80211_HW_NO_PROBE_FILTERING flag is removed
and the filter flag FIF_BCN_PRBRESP_PROMISC introduced.
This is a lot more efficient for hardware like b43 that
supports it and other hardware can still set the BSSID
to all-ones.
Driver modifications by Johannes Berg (b43 & iwlwifi), Michael Wu
(rtl8187, adm8211, and p54), Larry Finger (b43legacy), and
Ivo van Doorn (rt2x00).
Signed-off-by: Johannes Berg <johannes@sipsolutions.net>
Signed-off-by: Michael Wu <flamingice@sourmilk.net>
Signed-off-by: Larry Finger <Larry.Finger@lwfinger.net>
Signed-off-by: Ivo van Doorn <IvDoorn@gmail.com>
Signed-off-by: John W. Linville <linville@tuxdriver.com>
2007-09-17 12:29:23 +07:00
|
|
|
u32 rx_conf;
|
2008-12-10 12:34:27 +07:00
|
|
|
struct usb_anchor anchored;
|
2009-01-24 00:40:22 +07:00
|
|
|
struct delayed_work work;
|
|
|
|
struct ieee80211_hw *dev;
|
2009-04-17 07:56:38 +07:00
|
|
|
#ifdef CONFIG_RTL8187_LEDS
|
2009-12-09 23:56:13 +07:00
|
|
|
struct rtl8187_led led_radio;
|
2009-04-17 07:56:38 +07:00
|
|
|
struct rtl8187_led led_tx;
|
|
|
|
struct rtl8187_led led_rx;
|
|
|
|
struct delayed_work led_on;
|
|
|
|
struct delayed_work led_off;
|
|
|
|
#endif
|
2007-05-14 12:41:02 +07:00
|
|
|
u16 txpwr_base;
|
|
|
|
u8 asic_rev;
|
2008-07-08 18:27:54 +07:00
|
|
|
u8 is_rtl8187b;
|
|
|
|
enum {
|
|
|
|
RTL8187BvB,
|
|
|
|
RTL8187BvD,
|
|
|
|
RTL8187BvE
|
|
|
|
} hw_rev;
|
2007-05-14 12:41:02 +07:00
|
|
|
struct sk_buff_head rx_queue;
|
2008-07-08 18:27:54 +07:00
|
|
|
u8 signal;
|
|
|
|
u8 noise;
|
2008-11-13 22:39:15 +07:00
|
|
|
u8 slot_time;
|
|
|
|
u8 aifsn[4];
|
2009-12-06 08:25:22 +07:00
|
|
|
u8 rfkill_mask;
|
2008-11-13 22:39:16 +07:00
|
|
|
struct {
|
|
|
|
__le64 buf;
|
|
|
|
struct sk_buff_head queue;
|
2009-01-24 00:40:22 +07:00
|
|
|
} b_tx_status; /* This queue is used by both -b and non-b devices */
|
2009-05-07 00:57:27 +07:00
|
|
|
struct mutex io_mutex;
|
|
|
|
union {
|
|
|
|
u8 bits8;
|
|
|
|
__le16 bits16;
|
|
|
|
__le32 bits32;
|
|
|
|
} *io_dmabuf;
|
2009-08-26 23:54:09 +07:00
|
|
|
bool rfkill_off;
|
2007-05-14 12:41:02 +07:00
|
|
|
};
|
|
|
|
|
|
|
|
void rtl8187_write_phy(struct ieee80211_hw *dev, u8 addr, u32 data);
|
|
|
|
|
2008-07-08 18:27:54 +07:00
|
|
|
static inline u8 rtl818x_ioread8_idx(struct rtl8187_priv *priv,
|
|
|
|
u8 *addr, u8 idx)
|
2007-05-14 12:41:02 +07:00
|
|
|
{
|
|
|
|
u8 val;
|
|
|
|
|
2009-05-07 00:57:27 +07:00
|
|
|
mutex_lock(&priv->io_mutex);
|
2007-05-14 12:41:02 +07:00
|
|
|
usb_control_msg(priv->udev, usb_rcvctrlpipe(priv->udev, 0),
|
|
|
|
RTL8187_REQ_GET_REG, RTL8187_REQT_READ,
|
2009-05-07 00:57:27 +07:00
|
|
|
(unsigned long)addr, idx & 0x03,
|
|
|
|
&priv->io_dmabuf->bits8, sizeof(val), HZ / 2);
|
|
|
|
|
|
|
|
val = priv->io_dmabuf->bits8;
|
|
|
|
mutex_unlock(&priv->io_mutex);
|
2007-05-14 12:41:02 +07:00
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2008-07-08 18:27:54 +07:00
|
|
|
static inline u8 rtl818x_ioread8(struct rtl8187_priv *priv, u8 *addr)
|
|
|
|
{
|
|
|
|
return rtl818x_ioread8_idx(priv, addr, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline u16 rtl818x_ioread16_idx(struct rtl8187_priv *priv,
|
|
|
|
__le16 *addr, u8 idx)
|
2007-05-14 12:41:02 +07:00
|
|
|
{
|
|
|
|
__le16 val;
|
|
|
|
|
2009-05-07 00:57:27 +07:00
|
|
|
mutex_lock(&priv->io_mutex);
|
2007-05-14 12:41:02 +07:00
|
|
|
usb_control_msg(priv->udev, usb_rcvctrlpipe(priv->udev, 0),
|
|
|
|
RTL8187_REQ_GET_REG, RTL8187_REQT_READ,
|
2009-05-07 00:57:27 +07:00
|
|
|
(unsigned long)addr, idx & 0x03,
|
|
|
|
&priv->io_dmabuf->bits16, sizeof(val), HZ / 2);
|
|
|
|
|
|
|
|
val = priv->io_dmabuf->bits16;
|
|
|
|
mutex_unlock(&priv->io_mutex);
|
2007-05-14 12:41:02 +07:00
|
|
|
|
|
|
|
return le16_to_cpu(val);
|
|
|
|
}
|
|
|
|
|
2008-07-08 18:27:54 +07:00
|
|
|
static inline u16 rtl818x_ioread16(struct rtl8187_priv *priv, __le16 *addr)
|
|
|
|
{
|
|
|
|
return rtl818x_ioread16_idx(priv, addr, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline u32 rtl818x_ioread32_idx(struct rtl8187_priv *priv,
|
|
|
|
__le32 *addr, u8 idx)
|
2007-05-14 12:41:02 +07:00
|
|
|
{
|
|
|
|
__le32 val;
|
|
|
|
|
2009-05-07 00:57:27 +07:00
|
|
|
mutex_lock(&priv->io_mutex);
|
2007-05-14 12:41:02 +07:00
|
|
|
usb_control_msg(priv->udev, usb_rcvctrlpipe(priv->udev, 0),
|
|
|
|
RTL8187_REQ_GET_REG, RTL8187_REQT_READ,
|
2009-05-07 00:57:27 +07:00
|
|
|
(unsigned long)addr, idx & 0x03,
|
|
|
|
&priv->io_dmabuf->bits32, sizeof(val), HZ / 2);
|
|
|
|
|
|
|
|
val = priv->io_dmabuf->bits32;
|
|
|
|
mutex_unlock(&priv->io_mutex);
|
2007-05-14 12:41:02 +07:00
|
|
|
|
|
|
|
return le32_to_cpu(val);
|
|
|
|
}
|
|
|
|
|
2008-07-08 18:27:54 +07:00
|
|
|
static inline u32 rtl818x_ioread32(struct rtl8187_priv *priv, __le32 *addr)
|
|
|
|
{
|
|
|
|
return rtl818x_ioread32_idx(priv, addr, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void rtl818x_iowrite8_idx(struct rtl8187_priv *priv,
|
|
|
|
u8 *addr, u8 val, u8 idx)
|
2007-05-14 12:41:02 +07:00
|
|
|
{
|
2009-05-07 00:57:27 +07:00
|
|
|
mutex_lock(&priv->io_mutex);
|
|
|
|
|
|
|
|
priv->io_dmabuf->bits8 = val;
|
2007-05-14 12:41:02 +07:00
|
|
|
usb_control_msg(priv->udev, usb_sndctrlpipe(priv->udev, 0),
|
|
|
|
RTL8187_REQ_SET_REG, RTL8187_REQT_WRITE,
|
2009-05-07 00:57:27 +07:00
|
|
|
(unsigned long)addr, idx & 0x03,
|
|
|
|
&priv->io_dmabuf->bits8, sizeof(val), HZ / 2);
|
|
|
|
|
|
|
|
mutex_unlock(&priv->io_mutex);
|
2008-07-08 18:27:54 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void rtl818x_iowrite8(struct rtl8187_priv *priv, u8 *addr, u8 val)
|
|
|
|
{
|
|
|
|
rtl818x_iowrite8_idx(priv, addr, val, 0);
|
2007-05-14 12:41:02 +07:00
|
|
|
}
|
|
|
|
|
2008-07-08 18:27:54 +07:00
|
|
|
static inline void rtl818x_iowrite16_idx(struct rtl8187_priv *priv,
|
|
|
|
__le16 *addr, u16 val, u8 idx)
|
2007-05-14 12:41:02 +07:00
|
|
|
{
|
2009-05-07 00:57:27 +07:00
|
|
|
mutex_lock(&priv->io_mutex);
|
2007-05-14 12:41:02 +07:00
|
|
|
|
2009-05-07 00:57:27 +07:00
|
|
|
priv->io_dmabuf->bits16 = cpu_to_le16(val);
|
2007-05-14 12:41:02 +07:00
|
|
|
usb_control_msg(priv->udev, usb_sndctrlpipe(priv->udev, 0),
|
|
|
|
RTL8187_REQ_SET_REG, RTL8187_REQT_WRITE,
|
2009-05-07 00:57:27 +07:00
|
|
|
(unsigned long)addr, idx & 0x03,
|
|
|
|
&priv->io_dmabuf->bits16, sizeof(val), HZ / 2);
|
|
|
|
|
|
|
|
mutex_unlock(&priv->io_mutex);
|
2007-05-14 12:41:02 +07:00
|
|
|
}
|
|
|
|
|
2008-07-08 18:27:54 +07:00
|
|
|
static inline void rtl818x_iowrite16(struct rtl8187_priv *priv, __le16 *addr,
|
|
|
|
u16 val)
|
|
|
|
{
|
|
|
|
rtl818x_iowrite16_idx(priv, addr, val, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void rtl818x_iowrite32_idx(struct rtl8187_priv *priv,
|
|
|
|
__le32 *addr, u32 val, u8 idx)
|
2007-05-14 12:41:02 +07:00
|
|
|
{
|
2009-05-07 00:57:27 +07:00
|
|
|
mutex_lock(&priv->io_mutex);
|
2007-05-14 12:41:02 +07:00
|
|
|
|
2009-05-07 00:57:27 +07:00
|
|
|
priv->io_dmabuf->bits32 = cpu_to_le32(val);
|
2007-05-14 12:41:02 +07:00
|
|
|
usb_control_msg(priv->udev, usb_sndctrlpipe(priv->udev, 0),
|
|
|
|
RTL8187_REQ_SET_REG, RTL8187_REQT_WRITE,
|
2009-05-07 00:57:27 +07:00
|
|
|
(unsigned long)addr, idx & 0x03,
|
|
|
|
&priv->io_dmabuf->bits32, sizeof(val), HZ / 2);
|
|
|
|
|
|
|
|
mutex_unlock(&priv->io_mutex);
|
2008-07-08 18:27:54 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void rtl818x_iowrite32(struct rtl8187_priv *priv, __le32 *addr,
|
|
|
|
u32 val)
|
|
|
|
{
|
|
|
|
rtl818x_iowrite32_idx(priv, addr, val, 0);
|
2007-05-14 12:41:02 +07:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* RTL8187_H */
|