summaryrefslogtreecommitdiff
path: root/sys/dev/ic/if_wi.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/ic/if_wi.c')
-rw-r--r--sys/dev/ic/if_wi.c109
1 files changed, 107 insertions, 2 deletions
diff --git a/sys/dev/ic/if_wi.c b/sys/dev/ic/if_wi.c
index add84ce8de6..95851e093c4 100644
--- a/sys/dev/ic/if_wi.c
+++ b/sys/dev/ic/if_wi.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: if_wi.c,v 1.115 2005/01/15 05:24:11 brad Exp $ */
+/* $OpenBSD: if_wi.c,v 1.116 2005/02/15 19:44:15 reyk Exp $ */
/*
* Copyright (c) 1997, 1998, 1999
@@ -127,7 +127,7 @@ u_int32_t widebug = WIDEBUG;
#if !defined(lint) && !defined(__OpenBSD__)
static const char rcsid[] =
- "$OpenBSD: if_wi.c,v 1.115 2005/01/15 05:24:11 brad Exp $";
+ "$OpenBSD: if_wi.c,v 1.116 2005/02/15 19:44:15 reyk Exp $";
#endif /* lint */
#ifdef foo
@@ -167,6 +167,8 @@ STATIC int wi_get_nwkey(struct wi_softc *, struct ieee80211_nwkey *);
STATIC int wi_sync_media(struct wi_softc *, int, int);
STATIC int wi_set_pm(struct wi_softc *, struct ieee80211_power *);
STATIC int wi_get_pm(struct wi_softc *, struct ieee80211_power *);
+STATIC int wi_set_txpower(struct wi_softc *, struct ieee80211_txpower *);
+STATIC int wi_get_txpower(struct wi_softc *, struct ieee80211_txpower *);
STATIC int wi_get_debug(struct wi_softc *, struct wi_req *);
STATIC int wi_set_debug(struct wi_softc *, struct wi_req *);
@@ -1595,6 +1597,7 @@ wi_ioctl(ifp, command, data)
case SIOCS80211NWID:
case SIOCS80211NWKEY:
case SIOCS80211POWER:
+ case SIOCS80211TXPOWER:
error = suser(p, 0);
if (error) {
splx(s);
@@ -1856,6 +1859,12 @@ wi_ioctl(ifp, command, data)
case SIOCG80211POWER:
error = wi_get_pm(sc, (struct ieee80211_power *)data);
break;
+ case SIOCS80211TXPOWER:
+ error = wi_set_txpower(sc, (struct ieee80211_txpower *)data);
+ break;
+ case SIOCG80211TXPOWER:
+ error = wi_get_txpower(sc, (struct ieee80211_txpower *)data);
+ break;
case SIOCHOSTAP_ADD:
case SIOCHOSTAP_DEL:
case SIOCHOSTAP_GET:
@@ -2008,6 +2017,10 @@ wi_init_io(sc)
WI_PRT_ARG(sc));
sc->wi_tx_mgmt_id = id;
+ /* Set txpower */
+ if (sc->wi_flags & WI_FLAGS_TXPOWER)
+ wi_set_txpower(sc, NULL);
+
/* enable interrupts */
wi_intr_enable(sc, WI_INTRS);
@@ -2878,6 +2891,98 @@ wi_get_pm(struct wi_softc *sc, struct ieee80211_power *power)
}
STATIC int
+wi_set_txpower(struct wi_softc *sc, struct ieee80211_txpower *txpower)
+{
+ u_int16_t cmd;
+ u_int16_t power;
+ int8_t tmp;
+ int error;
+ int alc;
+
+ if (txpower == NULL) {
+ if (!(sc->wi_flags & WI_FLAGS_TXPOWER))
+ return (EINVAL);
+ alc = 0; /* disable ALC */
+ } else {
+ if (txpower->i_mode == IEEE80211_TXPOWER_MODE_AUTO) {
+ alc = 1; /* enable ALC */
+ sc->wi_flags &= ~WI_FLAGS_TXPOWER;
+ } else {
+ alc = 0; /* disable ALC */
+ sc->wi_flags |= WI_FLAGS_TXPOWER;
+ sc->wi_txpower = txpower->i_val;
+ }
+ }
+
+ /* Set ALC */
+ cmd = WI_CMD_DEBUG | (WI_DEBUG_CONFBITS << 8);
+ if ((error = wi_cmd(sc, cmd, alc, 0x8, 0)) != 0)
+ return (error);
+
+ /* No need to set the TX power value if ALC is enabled */
+ if (alc)
+ return (0);
+
+ /* Convert dBM to internal TX power value */
+ if (sc->wi_txpower > 20)
+ power = 128;
+ else if (sc->wi_txpower < -43)
+ power = 127;
+ else {
+ tmp = sc->wi_txpower;
+ tmp = -12 - tmp;
+ tmp <<= 2;
+
+ power = (u_int16_t)tmp;
+ }
+
+ /* Set manual TX power */
+ cmd = WI_CMD_WRITE_MIF;
+ if ((error = wi_cmd(sc, cmd,
+ WI_HFA384X_CR_MANUAL_TX_POWER, power, 0)) != 0)
+ return (error);
+
+ if (sc->sc_arpcom.ac_if.if_flags & IFF_DEBUG)
+ printf("%s: %u (%d dBm)\n", sc->sc_dev.dv_xname, power,
+ sc->wi_txpower);
+
+ return (0);
+}
+
+STATIC int
+wi_get_txpower(struct wi_softc *sc, struct ieee80211_txpower *txpower)
+{
+ u_int16_t cmd;
+ u_int16_t power;
+ int8_t tmp;
+ int error;
+
+ /* Get manual TX power */
+ cmd = WI_CMD_READ_MIF;
+ if ((error = wi_cmd(sc, cmd,
+ WI_HFA384X_CR_MANUAL_TX_POWER, 0, 0)) != 0)
+ return (error);
+
+ power = CSR_READ_2(sc, WI_RESP0);
+
+ /* Convert internal TX power value to dBM */
+ if (power > 255)
+ txpower->i_val = 255;
+ else {
+ tmp = power;
+ tmp >>= 2;
+ txpower->i_val = (u_int16_t)(-12 - tmp);
+ }
+
+ if (sc->wi_flags & WI_FLAGS_TXPOWER)
+ txpower->i_mode = IEEE80211_TXPOWER_MODE_FIXED;
+ else
+ txpower->i_mode = IEEE80211_TXPOWER_MODE_AUTO;
+
+ return (0);
+}
+
+STATIC int
wi_set_ssid(ws, id, len)
struct ieee80211_nwid *ws;
u_int8_t *id;