1537078d8f
Mainly focus on files that use BSD 2-Clause license, however the tool I was using misidentified many licenses so this was mostly a manual - error prone - task. The Software Package Data Exchange (SPDX) group provides a specification to make it easier for automated tools to detect and summarize well known opensource licenses. We are gradually adopting the specification, noting that the tags are considered only advisory and do not, in any way, superceed or replace the license texts.
2781 lines
71 KiB
C
2781 lines
71 KiB
C
/*-
|
|
* SPDX-License-Identifier: BSD-2-Clause-FreeBSD
|
|
*
|
|
* Copyright (c) 2007-2009 Sam Leffler, Errno Consulting
|
|
* Copyright (c) 2007-2009 Marvell Semiconductor, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer,
|
|
* without modification.
|
|
* 2. Redistributions in binary form must reproduce at minimum a disclaimer
|
|
* similar to the "NO WARRANTY" disclaimer below ("Disclaimer") and any
|
|
* redistribution must be conditioned upon including a substantially
|
|
* similar Disclaimer requirement for further binary redistribution.
|
|
*
|
|
* NO WARRANTY
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF NONINFRINGEMENT, MERCHANTIBILITY
|
|
* AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
|
* THE COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR SPECIAL, EXEMPLARY,
|
|
* OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
|
|
* IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
|
|
* THE POSSIBILITY OF SUCH DAMAGES.
|
|
*
|
|
* $FreeBSD$
|
|
*/
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/systm.h>
|
|
#include <sys/sysctl.h>
|
|
#include <sys/malloc.h>
|
|
#include <sys/lock.h>
|
|
#include <sys/mutex.h>
|
|
#include <sys/kernel.h>
|
|
#include <sys/errno.h>
|
|
#include <sys/bus.h>
|
|
#include <sys/endian.h>
|
|
|
|
#include <sys/linker.h>
|
|
#include <sys/firmware.h>
|
|
|
|
#include <machine/bus.h>
|
|
|
|
#include <dev/mwl/mwlhal.h>
|
|
#include <dev/mwl/mwlreg.h>
|
|
|
|
#include <sys/socket.h>
|
|
#include <sys/sockio.h>
|
|
#include <net/if.h>
|
|
#include <dev/mwl/mwldiag.h>
|
|
|
|
#define MWLHAL_DEBUG /* debug msgs */
|
|
|
|
typedef enum {
|
|
WL_ANTENNAMODE_RX = 0xffff,
|
|
WL_ANTENNAMODE_TX = 2,
|
|
} wlantennamode_e;
|
|
|
|
typedef enum {
|
|
WL_TX_POWERLEVEL_LOW = 5,
|
|
WL_TX_POWERLEVEL_MEDIUM = 10,
|
|
WL_TX_POWERLEVEL_HIGH = 15,
|
|
} wltxpowerlevel_e;
|
|
|
|
#define MWL_CMDBUF_SIZE 0x4000 /* size of f/w command buffer */
|
|
#define MWL_BASTREAMS_MAX 7 /* max BA streams (NB: fw >3.3.5.9) */
|
|
#define MWL_BAQID_MAX 8 /* max BA Q id's (NB: fw >3.3.5.9) */
|
|
#define MWL_MBSS_AP_MAX 8 /* max ap vap's */
|
|
#define MWL_MBSS_STA_MAX 24 /* max station/client vap's */
|
|
#define MWL_MBSS_MAX (MWL_MBSS_AP_MAX+MWL_MBSS_STA_MAX)
|
|
|
|
/*
|
|
* BA stream -> queue ID mapping
|
|
*
|
|
* The first 2 streams map to h/w; the remaining streams are
|
|
* implemented in firmware.
|
|
*/
|
|
static const int ba2qid[MWL_BASTREAMS_MAX] = {
|
|
5, 6 /* h/w supported */
|
|
#if MWL_BASTREAMS_MAX == 7
|
|
, 7, 0, 1, 2, 3 /* f/w supported */
|
|
#endif
|
|
};
|
|
static int qid2ba[MWL_BAQID_MAX];
|
|
|
|
#define IEEE80211_ADDR_LEN 6 /* XXX */
|
|
#define IEEE80211_ADDR_COPY(_dst, _src) \
|
|
memcpy(_dst, _src, IEEE80211_ADDR_LEN)
|
|
#define IEEE80211_ADDR_EQ(_dst, _src) \
|
|
(memcmp(_dst, _src, IEEE80211_ADDR_LEN) == 0)
|
|
|
|
#define _CMD_SETUP(pCmd, type, cmd) do { \
|
|
pCmd = (type *)&mh->mh_cmdbuf[0]; \
|
|
memset(pCmd, 0, sizeof(type)); \
|
|
pCmd->CmdHdr.Cmd = htole16(cmd); \
|
|
pCmd->CmdHdr.Length = htole16(sizeof(type)); \
|
|
} while (0)
|
|
|
|
#define _VCMD_SETUP(vap, pCmd, type, cmd) do { \
|
|
_CMD_SETUP(pCmd, type, cmd); \
|
|
pCmd->CmdHdr.MacId = vap->macid; \
|
|
} while (0)
|
|
|
|
#define PWTAGETRATETABLE20M 14*4
|
|
#define PWTAGETRATETABLE40M 9*4
|
|
#define PWTAGETRATETABLE20M_5G 35*4
|
|
#define PWTAGETRATETABLE40M_5G 16*4
|
|
|
|
struct mwl_hal_bastream {
|
|
MWL_HAL_BASTREAM public; /* public state */
|
|
uint8_t stream; /* stream # */
|
|
uint8_t setup; /* f/w cmd sent */
|
|
uint8_t ba_policy; /* direct/delayed BA policy */
|
|
uint8_t tid;
|
|
uint8_t paraminfo;
|
|
uint8_t macaddr[IEEE80211_ADDR_LEN];
|
|
};
|
|
|
|
struct mwl_hal_priv;
|
|
|
|
struct mwl_hal_vap {
|
|
struct mwl_hal_priv *mh; /* back pointer */
|
|
uint16_t bss_type; /* f/w type */
|
|
uint8_t vap_type; /* MWL_HAL_BSSTYPE */
|
|
uint8_t macid; /* for passing to f/w */
|
|
uint8_t flags;
|
|
#define MVF_RUNNING 0x01 /* BSS_START issued */
|
|
#define MVF_STATION 0x02 /* sta db entry created */
|
|
uint8_t mac[IEEE80211_ADDR_LEN];/* mac address */
|
|
};
|
|
#define MWLVAP(_vap) ((_vap)->mh)
|
|
|
|
/*
|
|
* Per-device state. We allocate a single cmd buffer for
|
|
* submitting operations to the firmware. Access to this
|
|
* buffer (and the f/w) are single-threaded. At present
|
|
* we spin waiting for cmds to complete which is bad. Not
|
|
* sure if it's possible to submit multiple requests or
|
|
* control when we get cmd done interrupts. There's no
|
|
* documentation and no example code to indicate what can
|
|
* or cannot be done so all we can do right now is follow the
|
|
* linux driver logic. This falls apart when the f/w fails;
|
|
* the system comes to a crawl as we spin waiting for operations
|
|
* to finish.
|
|
*/
|
|
struct mwl_hal_priv {
|
|
struct mwl_hal public; /* public area */
|
|
device_t mh_dev;
|
|
char mh_mtxname[12];
|
|
struct mtx mh_mtx;
|
|
bus_dma_tag_t mh_dmat; /* bus DMA tag for cmd buffer */
|
|
bus_dma_segment_t mh_seg; /* segment for cmd buffer */
|
|
bus_dmamap_t mh_dmamap; /* DMA map for cmd buffer */
|
|
uint16_t *mh_cmdbuf; /* f/w cmd buffer */
|
|
bus_addr_t mh_cmdaddr; /* physaddr of cmd buffer */
|
|
int mh_flags;
|
|
#define MHF_CALDATA 0x0001 /* cal data retrieved */
|
|
#define MHF_FWHANG 0x0002 /* fw appears hung */
|
|
#define MHF_MBSS 0x0004 /* mbss enabled */
|
|
struct mwl_hal_vap mh_vaps[MWL_MBSS_MAX+1];
|
|
int mh_bastreams; /* bit mask of available BA streams */
|
|
int mh_regioncode; /* XXX last region code sent to fw */
|
|
struct mwl_hal_bastream mh_streams[MWL_BASTREAMS_MAX];
|
|
int mh_debug;
|
|
MWL_HAL_CHANNELINFO mh_20M;
|
|
MWL_HAL_CHANNELINFO mh_40M;
|
|
MWL_HAL_CHANNELINFO mh_20M_5G;
|
|
MWL_HAL_CHANNELINFO mh_40M_5G;
|
|
int mh_SDRAMSIZE_Addr;
|
|
uint32_t mh_RTSSuccesses;/* cumulative stats for read-on-clear */
|
|
uint32_t mh_RTSFailures;
|
|
uint32_t mh_RxDuplicateFrames;
|
|
uint32_t mh_FCSErrorCount;
|
|
MWL_DIAG_REVS mh_revs;
|
|
};
|
|
#define MWLPRIV(_mh) ((struct mwl_hal_priv *)(_mh))
|
|
|
|
static int mwl_hal_setmac_locked(struct mwl_hal_vap *,
|
|
const uint8_t addr[IEEE80211_ADDR_LEN]);
|
|
static int mwlExecuteCmd(struct mwl_hal_priv *, unsigned short cmd);
|
|
static int mwlGetPwrCalTable(struct mwl_hal_priv *);
|
|
#ifdef MWLHAL_DEBUG
|
|
static const char *mwlcmdname(int cmd);
|
|
static void dumpresult(struct mwl_hal_priv *, int showresult);
|
|
#endif /* MWLHAL_DEBUG */
|
|
|
|
SYSCTL_DECL(_hw_mwl);
|
|
static SYSCTL_NODE(_hw_mwl, OID_AUTO, hal, CTLFLAG_RD, 0,
|
|
"Marvell HAL parameters");
|
|
|
|
static __inline void
|
|
MWL_HAL_LOCK(struct mwl_hal_priv *mh)
|
|
{
|
|
mtx_lock(&mh->mh_mtx);
|
|
}
|
|
|
|
static __inline void
|
|
MWL_HAL_LOCK_ASSERT(struct mwl_hal_priv *mh)
|
|
{
|
|
mtx_assert(&mh->mh_mtx, MA_OWNED);
|
|
}
|
|
|
|
static __inline void
|
|
MWL_HAL_UNLOCK(struct mwl_hal_priv *mh)
|
|
{
|
|
mtx_unlock(&mh->mh_mtx);
|
|
}
|
|
|
|
static __inline uint32_t
|
|
RD4(struct mwl_hal_priv *mh, bus_size_t off)
|
|
{
|
|
return bus_space_read_4(mh->public.mh_iot, mh->public.mh_ioh, off);
|
|
}
|
|
|
|
static __inline void
|
|
WR4(struct mwl_hal_priv *mh, bus_size_t off, uint32_t val)
|
|
{
|
|
bus_space_write_4(mh->public.mh_iot, mh->public.mh_ioh, off, val);
|
|
}
|
|
|
|
static void
|
|
mwl_hal_load_cb(void *arg, bus_dma_segment_t *segs, int nsegs, int error)
|
|
{
|
|
bus_addr_t *paddr = (bus_addr_t*) arg;
|
|
KASSERT(error == 0, ("error %u on bus_dma callback", error));
|
|
*paddr = segs->ds_addr;
|
|
}
|
|
|
|
/*
|
|
* Setup for communication with the device. We allocate
|
|
* a command buffer and map it for bus dma use. The pci
|
|
* device id is used to identify whether the device has
|
|
* SRAM on it (in which case f/w download must include a
|
|
* memory controller reset). All bus i/o operations happen
|
|
* in BAR 1; the driver passes in the tag and handle we need.
|
|
*/
|
|
struct mwl_hal *
|
|
mwl_hal_attach(device_t dev, uint16_t devid,
|
|
bus_space_handle_t ioh, bus_space_tag_t iot, bus_dma_tag_t tag)
|
|
{
|
|
struct mwl_hal_priv *mh;
|
|
struct mwl_hal_vap *hvap;
|
|
int error, i;
|
|
|
|
mh = malloc(sizeof(struct mwl_hal_priv), M_DEVBUF, M_NOWAIT | M_ZERO);
|
|
if (mh == NULL)
|
|
return NULL;
|
|
mh->mh_dev = dev;
|
|
mh->public.mh_ioh = ioh;
|
|
mh->public.mh_iot = iot;
|
|
for (i = 0; i < MWL_BASTREAMS_MAX; i++) {
|
|
mh->mh_streams[i].public.txq = ba2qid[i];
|
|
mh->mh_streams[i].stream = i;
|
|
/* construct back-mapping while we're at it */
|
|
if (mh->mh_streams[i].public.txq < MWL_BAQID_MAX)
|
|
qid2ba[mh->mh_streams[i].public.txq] = i;
|
|
else
|
|
device_printf(dev, "unexpected BA tx qid %d for "
|
|
"stream %d\n", mh->mh_streams[i].public.txq, i);
|
|
}
|
|
/* setup constant portion of vap state */
|
|
/* XXX should get max ap/client vap's from f/w */
|
|
i = 0;
|
|
hvap = &mh->mh_vaps[i];
|
|
hvap->vap_type = MWL_HAL_AP;
|
|
hvap->bss_type = htole16(WL_MAC_TYPE_PRIMARY_AP);
|
|
hvap->macid = 0;
|
|
for (i++; i < MWL_MBSS_AP_MAX; i++) {
|
|
hvap = &mh->mh_vaps[i];
|
|
hvap->vap_type = MWL_HAL_AP;
|
|
hvap->bss_type = htole16(WL_MAC_TYPE_SECONDARY_AP);
|
|
hvap->macid = i;
|
|
}
|
|
hvap = &mh->mh_vaps[i];
|
|
hvap->vap_type = MWL_HAL_STA;
|
|
hvap->bss_type = htole16(WL_MAC_TYPE_PRIMARY_CLIENT);
|
|
hvap->macid = i;
|
|
for (i++; i < MWL_MBSS_MAX; i++) {
|
|
hvap = &mh->mh_vaps[i];
|
|
hvap->vap_type = MWL_HAL_STA;
|
|
hvap->bss_type = htole16(WL_MAC_TYPE_SECONDARY_CLIENT);
|
|
hvap->macid = i;
|
|
}
|
|
mh->mh_revs.mh_devid = devid;
|
|
snprintf(mh->mh_mtxname, sizeof(mh->mh_mtxname),
|
|
"%s_hal", device_get_nameunit(dev));
|
|
mtx_init(&mh->mh_mtx, mh->mh_mtxname, NULL, MTX_DEF);
|
|
|
|
/*
|
|
* Allocate the command buffer and map into the address
|
|
* space of the h/w. We request "coherent" memory which
|
|
* will be uncached on some architectures.
|
|
*/
|
|
error = bus_dma_tag_create(tag, /* parent */
|
|
PAGE_SIZE, 0, /* alignment, bounds */
|
|
BUS_SPACE_MAXADDR_32BIT, /* lowaddr */
|
|
BUS_SPACE_MAXADDR, /* highaddr */
|
|
NULL, NULL, /* filter, filterarg */
|
|
MWL_CMDBUF_SIZE, /* maxsize */
|
|
1, /* nsegments */
|
|
MWL_CMDBUF_SIZE, /* maxsegsize */
|
|
BUS_DMA_ALLOCNOW, /* flags */
|
|
NULL, /* lockfunc */
|
|
NULL, /* lockarg */
|
|
&mh->mh_dmat);
|
|
if (error != 0) {
|
|
device_printf(dev, "unable to allocate memory for cmd tag, "
|
|
"error %u\n", error);
|
|
goto fail0;
|
|
}
|
|
|
|
/* allocate descriptors */
|
|
error = bus_dmamem_alloc(mh->mh_dmat, (void**) &mh->mh_cmdbuf,
|
|
BUS_DMA_NOWAIT | BUS_DMA_COHERENT,
|
|
&mh->mh_dmamap);
|
|
if (error != 0) {
|
|
device_printf(dev, "unable to allocate memory for cmd buffer, "
|
|
"error %u\n", error);
|
|
goto fail1;
|
|
}
|
|
|
|
error = bus_dmamap_load(mh->mh_dmat, mh->mh_dmamap,
|
|
mh->mh_cmdbuf, MWL_CMDBUF_SIZE,
|
|
mwl_hal_load_cb, &mh->mh_cmdaddr,
|
|
BUS_DMA_NOWAIT);
|
|
if (error != 0) {
|
|
device_printf(dev, "unable to load cmd buffer, error %u\n",
|
|
error);
|
|
goto fail2;
|
|
}
|
|
|
|
/*
|
|
* Some cards have SDRAM. When loading firmware we need
|
|
* to reset the SDRAM controller prior to doing this.
|
|
* When the SDRAMSIZE is non-zero we do that work in
|
|
* mwl_hal_fwload.
|
|
*/
|
|
switch (devid) {
|
|
case 0x2a02: /* CB82 */
|
|
case 0x2a03: /* CB85 */
|
|
case 0x2a08: /* MC85_B1 */
|
|
case 0x2a0b: /* CB85AP */
|
|
case 0x2a24:
|
|
mh->mh_SDRAMSIZE_Addr = 0x40fe70b7; /* 8M SDRAM */
|
|
break;
|
|
case 0x2a04: /* MC85 */
|
|
mh->mh_SDRAMSIZE_Addr = 0x40fc70b7; /* 16M SDRAM */
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
return &mh->public;
|
|
fail2:
|
|
bus_dmamem_free(mh->mh_dmat, mh->mh_cmdbuf, mh->mh_dmamap);
|
|
fail1:
|
|
bus_dma_tag_destroy(mh->mh_dmat);
|
|
fail0:
|
|
mtx_destroy(&mh->mh_mtx);
|
|
free(mh, M_DEVBUF);
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
mwl_hal_detach(struct mwl_hal *mh0)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
|
|
bus_dmamem_free(mh->mh_dmat, mh->mh_cmdbuf, mh->mh_dmamap);
|
|
bus_dma_tag_destroy(mh->mh_dmat);
|
|
mtx_destroy(&mh->mh_mtx);
|
|
free(mh, M_DEVBUF);
|
|
}
|
|
|
|
/*
|
|
* Reset internal state after a firmware download.
|
|
*/
|
|
static int
|
|
mwlResetHalState(struct mwl_hal_priv *mh)
|
|
{
|
|
int i;
|
|
|
|
/* XXX get from f/w */
|
|
mh->mh_bastreams = (1<<MWL_BASTREAMS_MAX)-1;
|
|
for (i = 0; i < MWL_MBSS_MAX; i++)
|
|
mh->mh_vaps[i].mh = NULL;
|
|
/*
|
|
* Clear cumulative stats.
|
|
*/
|
|
mh->mh_RTSSuccesses = 0;
|
|
mh->mh_RTSFailures = 0;
|
|
mh->mh_RxDuplicateFrames = 0;
|
|
mh->mh_FCSErrorCount = 0;
|
|
/*
|
|
* Fetch cal data for later use.
|
|
* XXX may want to fetch other stuff too.
|
|
*/
|
|
/* XXX check return */
|
|
if ((mh->mh_flags & MHF_CALDATA) == 0)
|
|
mwlGetPwrCalTable(mh);
|
|
return 0;
|
|
}
|
|
|
|
struct mwl_hal_vap *
|
|
mwl_hal_newvap(struct mwl_hal *mh0, MWL_HAL_BSSTYPE type,
|
|
const uint8_t mac[IEEE80211_ADDR_LEN])
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
struct mwl_hal_vap *vap;
|
|
int i;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
/* NB: could optimize but not worth it w/ max 32 bss */
|
|
for (i = 0; i < MWL_MBSS_MAX; i++) {
|
|
vap = &mh->mh_vaps[i];
|
|
if (vap->vap_type == type && vap->mh == NULL) {
|
|
vap->mh = mh;
|
|
mwl_hal_setmac_locked(vap, mac);
|
|
break;
|
|
}
|
|
}
|
|
MWL_HAL_UNLOCK(mh);
|
|
return (i < MWL_MBSS_MAX) ? vap : NULL;
|
|
}
|
|
|
|
void
|
|
mwl_hal_delvap(struct mwl_hal_vap *vap)
|
|
{
|
|
/* NB: locking not needed for single write */
|
|
vap->mh = NULL;
|
|
}
|
|
|
|
/*
|
|
* Manipulate the debug mask. Note debug
|
|
* msgs are only provided when this code is
|
|
* compiled with MWLHAL_DEBUG defined.
|
|
*/
|
|
|
|
void
|
|
mwl_hal_setdebug(struct mwl_hal *mh, int debug)
|
|
{
|
|
MWLPRIV(mh)->mh_debug = debug;
|
|
}
|
|
|
|
int
|
|
mwl_hal_getdebug(struct mwl_hal *mh)
|
|
{
|
|
return MWLPRIV(mh)->mh_debug;
|
|
}
|
|
|
|
void
|
|
mwl_hal_setbastreams(struct mwl_hal *mh, int mask)
|
|
{
|
|
MWLPRIV(mh)->mh_bastreams = mask & ((1<<MWL_BASTREAMS_MAX)-1);
|
|
}
|
|
|
|
int
|
|
mwl_hal_getbastreams(struct mwl_hal *mh)
|
|
{
|
|
return MWLPRIV(mh)->mh_bastreams;
|
|
}
|
|
|
|
int
|
|
mwl_hal_ismbsscapable(struct mwl_hal *mh)
|
|
{
|
|
return (MWLPRIV(mh)->mh_flags & MHF_MBSS) != 0;
|
|
}
|
|
|
|
#if 0
|
|
/* XXX inlined */
|
|
/*
|
|
* Return the current ISR setting and clear the cause.
|
|
* XXX maybe make inline
|
|
*/
|
|
void
|
|
mwl_hal_getisr(struct mwl_hal *mh0, uint32_t *status)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
uint32_t cause;
|
|
|
|
cause = RD4(mh, MACREG_REG_A2H_INTERRUPT_CAUSE);
|
|
if (cause == 0xffffffff) { /* card removed */
|
|
device_printf(mh->mh_dev, "%s: cause 0x%x\n", __func__, cause);
|
|
cause = 0;
|
|
} else if (cause != 0) {
|
|
/* clear cause bits */
|
|
WR4(mh, MACREG_REG_A2H_INTERRUPT_CAUSE,
|
|
cause &~ mh->public.mh_imask);
|
|
RD4(mh, MACREG_REG_INT_CODE); /* XXX flush write? */
|
|
}
|
|
*status = cause;
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Set the interrupt mask.
|
|
*/
|
|
void
|
|
mwl_hal_intrset(struct mwl_hal *mh0, uint32_t mask)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
|
|
WR4(mh, MACREG_REG_A2H_INTERRUPT_MASK, 0);
|
|
RD4(mh, MACREG_REG_INT_CODE);
|
|
|
|
mh->public.mh_imask = mask;
|
|
WR4(mh, MACREG_REG_A2H_INTERRUPT_MASK, mask);
|
|
RD4(mh, MACREG_REG_INT_CODE);
|
|
}
|
|
|
|
#if 0
|
|
/* XXX inlined */
|
|
/*
|
|
* Kick the firmware to tell it there are new tx descriptors
|
|
* for processing. The driver says what h/w q has work in
|
|
* case the f/w ever gets smarter.
|
|
*/
|
|
void
|
|
mwl_hal_txstart(struct mwl_hal *mh0, int qnum)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
uint32_t dummy;
|
|
|
|
WR4(mh, MACREG_REG_H2A_INTERRUPT_EVENTS, MACREG_H2ARIC_BIT_PPA_READY);
|
|
dummy = RD4(mh, MACREG_REG_INT_CODE);
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Callback from the driver on a cmd done interrupt.
|
|
* Nothing to do right now as we spin waiting for
|
|
* cmd completion.
|
|
*/
|
|
void
|
|
mwl_hal_cmddone(struct mwl_hal *mh0)
|
|
{
|
|
#if 0
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
|
|
if (mh->mh_debug & MWL_HAL_DEBUG_CMDDONE) {
|
|
device_printf(mh->mh_dev, "cmd done interrupt:\n");
|
|
dumpresult(mh, 1);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
/*
|
|
* Return "hw specs". Note this must be the first
|
|
* cmd MUST be done after a firmware download or the
|
|
* f/w will lockup.
|
|
* XXX move into the hal so driver doesn't need to be responsible
|
|
*/
|
|
int
|
|
mwl_hal_gethwspecs(struct mwl_hal *mh0, struct mwl_hal_hwspec *hw)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_DS_GET_HW_SPEC *pCmd;
|
|
int retval, minrev;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_DS_GET_HW_SPEC, HostCmd_CMD_GET_HW_SPEC);
|
|
memset(&pCmd->PermanentAddr[0], 0xff, IEEE80211_ADDR_LEN);
|
|
pCmd->ulFwAwakeCookie = htole32((unsigned int)mh->mh_cmdaddr+2048);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_GET_HW_SPEC);
|
|
if (retval == 0) {
|
|
IEEE80211_ADDR_COPY(hw->macAddr, pCmd->PermanentAddr);
|
|
hw->wcbBase[0] = le32toh(pCmd->WcbBase0) & 0x0000ffff;
|
|
hw->wcbBase[1] = le32toh(pCmd->WcbBase1[0]) & 0x0000ffff;
|
|
hw->wcbBase[2] = le32toh(pCmd->WcbBase1[1]) & 0x0000ffff;
|
|
hw->wcbBase[3] = le32toh(pCmd->WcbBase1[2]) & 0x0000ffff;
|
|
hw->rxDescRead = le32toh(pCmd->RxPdRdPtr)& 0x0000ffff;
|
|
hw->rxDescWrite = le32toh(pCmd->RxPdWrPtr)& 0x0000ffff;
|
|
hw->regionCode = le16toh(pCmd->RegionCode) & 0x00ff;
|
|
hw->fwReleaseNumber = le32toh(pCmd->FWReleaseNumber);
|
|
hw->maxNumWCB = le16toh(pCmd->NumOfWCB);
|
|
hw->maxNumMCAddr = le16toh(pCmd->NumOfMCastAddr);
|
|
hw->numAntennas = le16toh(pCmd->NumberOfAntenna);
|
|
hw->hwVersion = pCmd->Version;
|
|
hw->hostInterface = pCmd->HostIf;
|
|
|
|
mh->mh_revs.mh_macRev = hw->hwVersion; /* XXX */
|
|
mh->mh_revs.mh_phyRev = hw->hostInterface; /* XXX */
|
|
|
|
minrev = ((hw->fwReleaseNumber) >> 16) & 0xff;
|
|
if (minrev >= 4) {
|
|
/* starting with 3.4.x.x s/w BA streams supported */
|
|
mh->mh_bastreams &= (1<<MWL_BASTREAMS_MAX)-1;
|
|
} else
|
|
mh->mh_bastreams &= (1<<2)-1;
|
|
}
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Inform the f/w about location of the tx/rx dma data structures
|
|
* and related state. This cmd must be done immediately after a
|
|
* mwl_hal_gethwspecs call or the f/w will lockup.
|
|
*/
|
|
int
|
|
mwl_hal_sethwdma(struct mwl_hal *mh0, const struct mwl_hal_txrxdma *dma)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_DS_SET_HW_SPEC *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_DS_SET_HW_SPEC, HostCmd_CMD_SET_HW_SPEC);
|
|
pCmd->WcbBase[0] = htole32(dma->wcbBase[0]);
|
|
pCmd->WcbBase[1] = htole32(dma->wcbBase[1]);
|
|
pCmd->WcbBase[2] = htole32(dma->wcbBase[2]);
|
|
pCmd->WcbBase[3] = htole32(dma->wcbBase[3]);
|
|
pCmd->TxWcbNumPerQueue = htole32(dma->maxNumTxWcb);
|
|
pCmd->NumTxQueues = htole32(dma->maxNumWCB);
|
|
pCmd->TotalRxWcb = htole32(1); /* XXX */
|
|
pCmd->RxPdWrPtr = htole32(dma->rxDescRead);
|
|
pCmd->Flags = htole32(SET_HW_SPEC_HOSTFORM_BEACON
|
|
#ifdef MWL_HOST_PS_SUPPORT
|
|
| SET_HW_SPEC_HOST_POWERSAVE
|
|
#endif
|
|
| SET_HW_SPEC_HOSTFORM_PROBERESP);
|
|
/* disable multi-bss operation for A1-A4 parts */
|
|
if (mh->mh_revs.mh_macRev < 5)
|
|
pCmd->Flags |= htole32(SET_HW_SPEC_DISABLEMBSS);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_HW_SPEC);
|
|
if (retval == 0) {
|
|
if (pCmd->Flags & htole32(SET_HW_SPEC_DISABLEMBSS))
|
|
mh->mh_flags &= ~MHF_MBSS;
|
|
else
|
|
mh->mh_flags |= MHF_MBSS;
|
|
}
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Retrieve statistics from the f/w.
|
|
* XXX should be in memory shared w/ driver
|
|
*/
|
|
int
|
|
mwl_hal_gethwstats(struct mwl_hal *mh0, struct mwl_hal_hwstats *stats)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_DS_802_11_GET_STAT *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_DS_802_11_GET_STAT,
|
|
HostCmd_CMD_802_11_GET_STAT);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_802_11_GET_STAT);
|
|
if (retval == 0) {
|
|
const uint32_t *sp = (const uint32_t *)&pCmd->TxRetrySuccesses;
|
|
uint32_t *dp = (uint32_t *)&stats->TxRetrySuccesses;
|
|
int i;
|
|
|
|
for (i = 0; i < sizeof(*stats)/sizeof(uint32_t); i++)
|
|
dp[i] = le32toh(sp[i]);
|
|
/*
|
|
* Update stats not returned by f/w but available
|
|
* through public registers. Note these registers
|
|
* are "clear on read" so we maintain cumulative data.
|
|
* XXX register defines
|
|
*/
|
|
mh->mh_RTSSuccesses += RD4(mh, 0xa834);
|
|
mh->mh_RTSFailures += RD4(mh, 0xa830);
|
|
mh->mh_RxDuplicateFrames += RD4(mh, 0xa84c);
|
|
mh->mh_FCSErrorCount += RD4(mh, 0xa840);
|
|
}
|
|
MWL_HAL_UNLOCK(mh);
|
|
|
|
stats->RTSSuccesses = mh->mh_RTSSuccesses;
|
|
stats->RTSFailures = mh->mh_RTSFailures;
|
|
stats->RxDuplicateFrames = mh->mh_RxDuplicateFrames;
|
|
stats->FCSErrorCount = mh->mh_FCSErrorCount;
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Set HT guard interval handling.
|
|
* Takes effect immediately.
|
|
*/
|
|
int
|
|
mwl_hal_sethtgi(struct mwl_hal_vap *vap, int GIType)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_FW_HT_GUARD_INTERVAL *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_FW_HT_GUARD_INTERVAL,
|
|
HostCmd_CMD_HT_GUARD_INTERVAL);
|
|
pCmd->Action = htole32(HostCmd_ACT_GEN_SET);
|
|
|
|
if (GIType == 0) {
|
|
pCmd->GIType = htole32(GI_TYPE_LONG);
|
|
} else if (GIType == 1) {
|
|
pCmd->GIType = htole32(GI_TYPE_LONG | GI_TYPE_SHORT);
|
|
} else {
|
|
pCmd->GIType = htole32(GI_TYPE_LONG);
|
|
}
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_HT_GUARD_INTERVAL);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Configure radio.
|
|
* Takes effect immediately.
|
|
* XXX preamble installed after set fixed rate cmd
|
|
*/
|
|
int
|
|
mwl_hal_setradio(struct mwl_hal *mh0, int onoff, MWL_HAL_PREAMBLE preamble)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_DS_802_11_RADIO_CONTROL *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_DS_802_11_RADIO_CONTROL,
|
|
HostCmd_CMD_802_11_RADIO_CONTROL);
|
|
pCmd->Action = htole16(HostCmd_ACT_GEN_SET);
|
|
if (onoff == 0)
|
|
pCmd->Control = 0;
|
|
else
|
|
pCmd->Control = htole16(preamble);
|
|
pCmd->RadioOn = htole16(onoff);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_802_11_RADIO_CONTROL);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Configure antenna use.
|
|
* Takes effect immediately.
|
|
* XXX tx antenna setting ignored
|
|
* XXX rx antenna setting should always be 3 (for now)
|
|
*/
|
|
int
|
|
mwl_hal_setantenna(struct mwl_hal *mh0, MWL_HAL_ANTENNA dirSet, int ant)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_DS_802_11_RF_ANTENNA *pCmd;
|
|
int retval;
|
|
|
|
if (!(dirSet == WL_ANTENNATYPE_RX || dirSet == WL_ANTENNATYPE_TX))
|
|
return EINVAL;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_DS_802_11_RF_ANTENNA,
|
|
HostCmd_CMD_802_11_RF_ANTENNA);
|
|
pCmd->Action = htole16(dirSet);
|
|
if (ant == 0) /* default to all/both antennae */
|
|
ant = 3;
|
|
pCmd->AntennaMode = htole16(ant);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_802_11_RF_ANTENNA);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Set packet size threshold for implicit use of RTS.
|
|
* Takes effect immediately.
|
|
* XXX packet length > threshold =>'s RTS
|
|
*/
|
|
int
|
|
mwl_hal_setrtsthreshold(struct mwl_hal_vap *vap, int threshold)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_DS_802_11_RTS_THSD *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_DS_802_11_RTS_THSD,
|
|
HostCmd_CMD_802_11_RTS_THSD);
|
|
pCmd->Action = htole16(HostCmd_ACT_GEN_SET);
|
|
pCmd->Threshold = htole16(threshold);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_802_11_RTS_THSD);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Enable sta-mode operation (disables beacon frame xmit).
|
|
*/
|
|
int
|
|
mwl_hal_setinframode(struct mwl_hal_vap *vap)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_FW_SET_INFRA_MODE *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_FW_SET_INFRA_MODE,
|
|
HostCmd_CMD_SET_INFRA_MODE);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_INFRA_MODE);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Configure radar detection in support of 802.11h.
|
|
*/
|
|
int
|
|
mwl_hal_setradardetection(struct mwl_hal *mh0, MWL_HAL_RADAR action)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_802_11h_Detect_Radar *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_802_11h_Detect_Radar,
|
|
HostCmd_CMD_802_11H_DETECT_RADAR);
|
|
pCmd->CmdHdr.Length = htole16(sizeof(HostCmd_802_11h_Detect_Radar));
|
|
pCmd->Action = htole16(action);
|
|
if (mh->mh_regioncode == DOMAIN_CODE_ETSI_131)
|
|
pCmd->RadarTypeCode = htole16(131);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_802_11H_DETECT_RADAR);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Convert public channel flags definition to a
|
|
* value suitable for feeding to the firmware.
|
|
* Note this includes byte swapping.
|
|
*/
|
|
static uint32_t
|
|
cvtChannelFlags(const MWL_HAL_CHANNEL *chan)
|
|
{
|
|
uint32_t w;
|
|
|
|
/*
|
|
* NB: f/w only understands FREQ_BAND_5GHZ, supplying the more
|
|
* precise band info causes it to lockup (sometimes).
|
|
*/
|
|
w = (chan->channelFlags.FreqBand == MWL_FREQ_BAND_2DOT4GHZ) ?
|
|
FREQ_BAND_2DOT4GHZ : FREQ_BAND_5GHZ;
|
|
switch (chan->channelFlags.ChnlWidth) {
|
|
case MWL_CH_10_MHz_WIDTH:
|
|
w |= CH_10_MHz_WIDTH;
|
|
break;
|
|
case MWL_CH_20_MHz_WIDTH:
|
|
w |= CH_20_MHz_WIDTH;
|
|
break;
|
|
case MWL_CH_40_MHz_WIDTH:
|
|
default:
|
|
w |= CH_40_MHz_WIDTH;
|
|
break;
|
|
}
|
|
switch (chan->channelFlags.ExtChnlOffset) {
|
|
case MWL_EXT_CH_NONE:
|
|
w |= EXT_CH_NONE;
|
|
break;
|
|
case MWL_EXT_CH_ABOVE_CTRL_CH:
|
|
w |= EXT_CH_ABOVE_CTRL_CH;
|
|
break;
|
|
case MWL_EXT_CH_BELOW_CTRL_CH:
|
|
w |= EXT_CH_BELOW_CTRL_CH;
|
|
break;
|
|
}
|
|
return htole32(w);
|
|
}
|
|
|
|
/*
|
|
* Start a channel switch announcement countdown. The IE
|
|
* in the beacon frame is allowed to go out and the firmware
|
|
* counts down and notifies the host when it's time to switch
|
|
* channels.
|
|
*/
|
|
int
|
|
mwl_hal_setchannelswitchie(struct mwl_hal *mh0,
|
|
const MWL_HAL_CHANNEL *nextchan, uint32_t mode, uint32_t count)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_SET_SWITCH_CHANNEL *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_SET_SWITCH_CHANNEL,
|
|
HostCmd_CMD_SET_SWITCH_CHANNEL);
|
|
pCmd->Next11hChannel = htole32(nextchan->channel);
|
|
pCmd->Mode = htole32(mode);
|
|
pCmd->InitialCount = htole32(count+1);
|
|
pCmd->ChannelFlags = cvtChannelFlags(nextchan);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_SWITCH_CHANNEL);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Set the region code that selects the radar bin'ing agorithm.
|
|
*/
|
|
int
|
|
mwl_hal_setregioncode(struct mwl_hal *mh0, int regionCode)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_SET_REGIONCODE_INFO *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_SET_REGIONCODE_INFO,
|
|
HostCmd_CMD_SET_REGION_CODE);
|
|
/* XXX map pseudo-codes to fw codes */
|
|
switch (regionCode) {
|
|
case DOMAIN_CODE_ETSI_131:
|
|
pCmd->regionCode = htole16(DOMAIN_CODE_ETSI);
|
|
break;
|
|
default:
|
|
pCmd->regionCode = htole16(regionCode);
|
|
break;
|
|
}
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_REGION_CODE);
|
|
if (retval == 0)
|
|
mh->mh_regioncode = regionCode;
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
#define RATEVAL(r) ((r) &~ RATE_MCS)
|
|
#define RATETYPE(r) (((r) & RATE_MCS) ? HT_RATE_TYPE : LEGACY_RATE_TYPE)
|
|
|
|
int
|
|
mwl_hal_settxrate(struct mwl_hal_vap *vap, MWL_HAL_TXRATE_HANDLING handling,
|
|
const MWL_HAL_TXRATE *rate)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_FW_USE_FIXED_RATE *pCmd;
|
|
FIXED_RATE_ENTRY *fp;
|
|
int retval, i, n;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_FW_USE_FIXED_RATE,
|
|
HostCmd_CMD_SET_FIXED_RATE);
|
|
|
|
pCmd->MulticastRate = RATEVAL(rate->McastRate);
|
|
pCmd->MultiRateTxType = RATETYPE(rate->McastRate);
|
|
/* NB: no rate type field */
|
|
pCmd->ManagementRate = RATEVAL(rate->MgtRate);
|
|
memset(pCmd->FixedRateTable, 0, sizeof(pCmd->FixedRateTable));
|
|
if (handling == RATE_FIXED) {
|
|
pCmd->Action = htole32(HostCmd_ACT_GEN_SET);
|
|
pCmd->AllowRateDrop = htole32(FIXED_RATE_WITHOUT_AUTORATE_DROP);
|
|
fp = pCmd->FixedRateTable;
|
|
fp->FixedRate =
|
|
htole32(RATEVAL(rate->RateSeries[0].Rate));
|
|
fp->FixRateTypeFlags.FixRateType =
|
|
htole32(RATETYPE(rate->RateSeries[0].Rate));
|
|
pCmd->EntryCount = htole32(1);
|
|
} else if (handling == RATE_FIXED_DROP) {
|
|
pCmd->Action = htole32(HostCmd_ACT_GEN_SET);
|
|
pCmd->AllowRateDrop = htole32(FIXED_RATE_WITH_AUTO_RATE_DROP);
|
|
n = 0;
|
|
fp = pCmd->FixedRateTable;
|
|
for (i = 0; i < 4; i++) {
|
|
if (rate->RateSeries[0].TryCount == 0)
|
|
break;
|
|
fp->FixRateTypeFlags.FixRateType =
|
|
htole32(RATETYPE(rate->RateSeries[i].Rate));
|
|
fp->FixedRate =
|
|
htole32(RATEVAL(rate->RateSeries[i].Rate));
|
|
fp->FixRateTypeFlags.RetryCountValid =
|
|
htole32(RETRY_COUNT_VALID);
|
|
fp->RetryCount =
|
|
htole32(rate->RateSeries[i].TryCount-1);
|
|
n++;
|
|
}
|
|
pCmd->EntryCount = htole32(n);
|
|
} else
|
|
pCmd->Action = htole32(HostCmd_ACT_NOT_USE_FIXED_RATE);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_FIXED_RATE);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_settxrate_auto(struct mwl_hal *mh0, const MWL_HAL_TXRATE *rate)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_USE_FIXED_RATE *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_USE_FIXED_RATE,
|
|
HostCmd_CMD_SET_FIXED_RATE);
|
|
|
|
pCmd->MulticastRate = RATEVAL(rate->McastRate);
|
|
pCmd->MultiRateTxType = RATETYPE(rate->McastRate);
|
|
/* NB: no rate type field */
|
|
pCmd->ManagementRate = RATEVAL(rate->MgtRate);
|
|
memset(pCmd->FixedRateTable, 0, sizeof(pCmd->FixedRateTable));
|
|
pCmd->Action = htole32(HostCmd_ACT_NOT_USE_FIXED_RATE);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_FIXED_RATE);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
#undef RATEVAL
|
|
#undef RATETYPE
|
|
|
|
int
|
|
mwl_hal_setslottime(struct mwl_hal *mh0, int usecs)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_SET_SLOT *pCmd;
|
|
int retval;
|
|
|
|
if (usecs != 9 && usecs != 20)
|
|
return EINVAL;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_SET_SLOT,
|
|
HostCmd_CMD_802_11_SET_SLOT);
|
|
pCmd->Action = htole16(HostCmd_ACT_GEN_SET);
|
|
pCmd->Slot = (usecs == 9 ? 1 : 0);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_802_11_SET_SLOT);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_adjusttxpower(struct mwl_hal *mh0, uint32_t level)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_DS_802_11_RF_TX_POWER *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_DS_802_11_RF_TX_POWER,
|
|
HostCmd_CMD_802_11_RF_TX_POWER);
|
|
pCmd->Action = htole16(HostCmd_ACT_GEN_SET);
|
|
|
|
if (level < 30) {
|
|
pCmd->SupportTxPowerLevel = htole16(WL_TX_POWERLEVEL_LOW);
|
|
} else if (level >= 30 && level < 60) {
|
|
pCmd->SupportTxPowerLevel = htole16(WL_TX_POWERLEVEL_MEDIUM);
|
|
} else {
|
|
pCmd->SupportTxPowerLevel = htole16(WL_TX_POWERLEVEL_HIGH);
|
|
}
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_802_11_RF_TX_POWER);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
static const struct mwl_hal_channel *
|
|
findchannel(const struct mwl_hal_priv *mh, const MWL_HAL_CHANNEL *c)
|
|
{
|
|
const struct mwl_hal_channel *hc;
|
|
const MWL_HAL_CHANNELINFO *ci;
|
|
int chan = c->channel, i;
|
|
|
|
if (c->channelFlags.FreqBand == MWL_FREQ_BAND_2DOT4GHZ) {
|
|
i = chan - 1;
|
|
if (c->channelFlags.ChnlWidth == MWL_CH_40_MHz_WIDTH) {
|
|
ci = &mh->mh_40M;
|
|
if (c->channelFlags.ExtChnlOffset == MWL_EXT_CH_BELOW_CTRL_CH)
|
|
i -= 4;
|
|
} else
|
|
ci = &mh->mh_20M;
|
|
/* 2.4G channel table is directly indexed */
|
|
hc = ((unsigned)i < ci->nchannels) ? &ci->channels[i] : NULL;
|
|
} else if (c->channelFlags.FreqBand == MWL_FREQ_BAND_5GHZ) {
|
|
if (c->channelFlags.ChnlWidth == MWL_CH_40_MHz_WIDTH) {
|
|
ci = &mh->mh_40M_5G;
|
|
if (c->channelFlags.ExtChnlOffset == MWL_EXT_CH_BELOW_CTRL_CH)
|
|
chan -= 4;
|
|
} else
|
|
ci = &mh->mh_20M_5G;
|
|
/* 5GHz channel table is sparse and must be searched */
|
|
for (i = 0; i < ci->nchannels; i++)
|
|
if (ci->channels[i].ieee == chan)
|
|
break;
|
|
hc = (i < ci->nchannels) ? &ci->channels[i] : NULL;
|
|
} else
|
|
hc = NULL;
|
|
return hc;
|
|
}
|
|
|
|
int
|
|
mwl_hal_settxpower(struct mwl_hal *mh0, const MWL_HAL_CHANNEL *c, uint8_t maxtxpow)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_DS_802_11_RF_TX_POWER *pCmd;
|
|
const struct mwl_hal_channel *hc;
|
|
int i, retval;
|
|
|
|
hc = findchannel(mh, c);
|
|
if (hc == NULL) {
|
|
/* XXX temp while testing */
|
|
device_printf(mh->mh_dev,
|
|
"%s: no cal data for channel %u band %u width %u ext %u\n",
|
|
__func__, c->channel, c->channelFlags.FreqBand,
|
|
c->channelFlags.ChnlWidth, c->channelFlags.ExtChnlOffset);
|
|
return EINVAL;
|
|
}
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_DS_802_11_RF_TX_POWER,
|
|
HostCmd_CMD_802_11_RF_TX_POWER);
|
|
pCmd->Action = htole16(HostCmd_ACT_GEN_SET_LIST);
|
|
i = 0;
|
|
/* NB: 5Ghz cal data have the channel # in [0]; don't truncate */
|
|
if (c->channelFlags.FreqBand == MWL_FREQ_BAND_5GHZ)
|
|
pCmd->PowerLevelList[i++] = htole16(hc->targetPowers[0]);
|
|
for (; i < 4; i++) {
|
|
uint16_t pow = hc->targetPowers[i];
|
|
if (pow > maxtxpow)
|
|
pow = maxtxpow;
|
|
pCmd->PowerLevelList[i] = htole16(pow);
|
|
}
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_802_11_RF_TX_POWER);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_getchannelinfo(struct mwl_hal *mh0, int band, int chw,
|
|
const MWL_HAL_CHANNELINFO **ci)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
|
|
switch (band) {
|
|
case MWL_FREQ_BAND_2DOT4GHZ:
|
|
*ci = (chw == MWL_CH_20_MHz_WIDTH) ? &mh->mh_20M : &mh->mh_40M;
|
|
break;
|
|
case MWL_FREQ_BAND_5GHZ:
|
|
*ci = (chw == MWL_CH_20_MHz_WIDTH) ?
|
|
&mh->mh_20M_5G : &mh->mh_40M_5G;
|
|
break;
|
|
default:
|
|
return EINVAL;
|
|
}
|
|
return ((*ci)->freqLow == (*ci)->freqHigh) ? EINVAL : 0;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setmcast(struct mwl_hal *mh0, int nmc, const uint8_t macs[])
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_DS_MAC_MULTICAST_ADR *pCmd;
|
|
int retval;
|
|
|
|
if (nmc > MWL_HAL_MCAST_MAX)
|
|
return EINVAL;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_DS_MAC_MULTICAST_ADR,
|
|
HostCmd_CMD_MAC_MULTICAST_ADR);
|
|
memcpy(pCmd->MACList, macs, nmc*IEEE80211_ADDR_LEN);
|
|
pCmd->NumOfAdrs = htole16(nmc);
|
|
pCmd->Action = htole16(0xffff);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_MAC_MULTICAST_ADR);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_keyset(struct mwl_hal_vap *vap, const MWL_HAL_KEYVAL *kv,
|
|
const uint8_t mac[IEEE80211_ADDR_LEN])
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_FW_UPDATE_ENCRYPTION_SET_KEY *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_FW_UPDATE_ENCRYPTION_SET_KEY,
|
|
HostCmd_CMD_UPDATE_ENCRYPTION);
|
|
if (kv->keyFlags & (KEY_FLAG_TXGROUPKEY|KEY_FLAG_RXGROUPKEY))
|
|
pCmd->ActionType = htole32(EncrActionTypeSetGroupKey);
|
|
else
|
|
pCmd->ActionType = htole32(EncrActionTypeSetKey);
|
|
pCmd->KeyParam.Length = htole16(sizeof(pCmd->KeyParam));
|
|
pCmd->KeyParam.KeyTypeId = htole16(kv->keyTypeId);
|
|
pCmd->KeyParam.KeyInfo = htole32(kv->keyFlags);
|
|
pCmd->KeyParam.KeyIndex = htole32(kv->keyIndex);
|
|
/* NB: includes TKIP MIC keys */
|
|
memcpy(&pCmd->KeyParam.Key, &kv->key, kv->keyLen);
|
|
switch (kv->keyTypeId) {
|
|
case KEY_TYPE_ID_WEP:
|
|
pCmd->KeyParam.KeyLen = htole16(kv->keyLen);
|
|
break;
|
|
case KEY_TYPE_ID_TKIP:
|
|
pCmd->KeyParam.KeyLen = htole16(sizeof(TKIP_TYPE_KEY));
|
|
pCmd->KeyParam.Key.TkipKey.TkipRsc.low =
|
|
htole16(kv->key.tkip.rsc.low);
|
|
pCmd->KeyParam.Key.TkipKey.TkipRsc.high =
|
|
htole32(kv->key.tkip.rsc.high);
|
|
pCmd->KeyParam.Key.TkipKey.TkipTsc.low =
|
|
htole16(kv->key.tkip.tsc.low);
|
|
pCmd->KeyParam.Key.TkipKey.TkipTsc.high =
|
|
htole32(kv->key.tkip.tsc.high);
|
|
break;
|
|
case KEY_TYPE_ID_AES:
|
|
pCmd->KeyParam.KeyLen = htole16(sizeof(AES_TYPE_KEY));
|
|
break;
|
|
}
|
|
#ifdef MWL_MBSS_SUPPORT
|
|
IEEE80211_ADDR_COPY(pCmd->KeyParam.Macaddr, mac);
|
|
#else
|
|
IEEE80211_ADDR_COPY(pCmd->Macaddr, mac);
|
|
#endif
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_UPDATE_ENCRYPTION);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_keyreset(struct mwl_hal_vap *vap, const MWL_HAL_KEYVAL *kv, const uint8_t mac[IEEE80211_ADDR_LEN])
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_FW_UPDATE_ENCRYPTION_SET_KEY *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_FW_UPDATE_ENCRYPTION_SET_KEY,
|
|
HostCmd_CMD_UPDATE_ENCRYPTION);
|
|
pCmd->ActionType = htole16(EncrActionTypeRemoveKey);
|
|
pCmd->KeyParam.Length = htole16(sizeof(pCmd->KeyParam));
|
|
pCmd->KeyParam.KeyTypeId = htole16(kv->keyTypeId);
|
|
pCmd->KeyParam.KeyInfo = htole32(kv->keyFlags);
|
|
pCmd->KeyParam.KeyIndex = htole32(kv->keyIndex);
|
|
#ifdef MWL_MBSS_SUPPORT
|
|
IEEE80211_ADDR_COPY(pCmd->KeyParam.Macaddr, mac);
|
|
#else
|
|
IEEE80211_ADDR_COPY(pCmd->Macaddr, mac);
|
|
#endif
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_UPDATE_ENCRYPTION);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
static int
|
|
mwl_hal_setmac_locked(struct mwl_hal_vap *vap,
|
|
const uint8_t addr[IEEE80211_ADDR_LEN])
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_DS_SET_MAC *pCmd;
|
|
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_DS_SET_MAC, HostCmd_CMD_SET_MAC_ADDR);
|
|
IEEE80211_ADDR_COPY(&pCmd->MacAddr[0], addr);
|
|
#ifdef MWL_MBSS_SUPPORT
|
|
pCmd->MacType = vap->bss_type; /* NB: already byte swapped */
|
|
IEEE80211_ADDR_COPY(vap->mac, addr); /* XXX do only if success */
|
|
#endif
|
|
return mwlExecuteCmd(mh, HostCmd_CMD_SET_MAC_ADDR);
|
|
}
|
|
|
|
int
|
|
mwl_hal_setmac(struct mwl_hal_vap *vap, const uint8_t addr[IEEE80211_ADDR_LEN])
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
retval = mwl_hal_setmac_locked(vap, addr);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setbeacon(struct mwl_hal_vap *vap, const void *frame, size_t frameLen)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_DS_SET_BEACON *pCmd;
|
|
int retval;
|
|
|
|
/* XXX verify frameLen fits */
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_DS_SET_BEACON, HostCmd_CMD_SET_BEACON);
|
|
/* XXX override _VCMD_SETUP */
|
|
pCmd->CmdHdr.Length = htole16(sizeof(HostCmd_DS_SET_BEACON)-1+frameLen);
|
|
pCmd->FrmBodyLen = htole16(frameLen);
|
|
memcpy(pCmd->FrmBody, frame, frameLen);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_BEACON);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setpowersave_bss(struct mwl_hal_vap *vap, uint8_t nsta)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_SET_POWERSAVESTATION *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_SET_POWERSAVESTATION,
|
|
HostCmd_CMD_SET_POWERSAVESTATION);
|
|
pCmd->NumberOfPowersave = nsta;
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_POWERSAVESTATION);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setpowersave_sta(struct mwl_hal_vap *vap, uint16_t aid, int ena)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_SET_TIM *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_SET_TIM, HostCmd_CMD_SET_TIM);
|
|
pCmd->Aid = htole16(aid);
|
|
pCmd->Set = htole32(ena);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_TIM);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setassocid(struct mwl_hal_vap *vap,
|
|
const uint8_t bssId[IEEE80211_ADDR_LEN], uint16_t assocId)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_FW_SET_AID *pCmd = (HostCmd_FW_SET_AID *) &mh->mh_cmdbuf[0];
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_FW_SET_AID, HostCmd_CMD_SET_AID);
|
|
pCmd->AssocID = htole16(assocId);
|
|
IEEE80211_ADDR_COPY(&pCmd->MacAddr[0], bssId);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_AID);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setchannel(struct mwl_hal *mh0, const MWL_HAL_CHANNEL *chan)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_SET_RF_CHANNEL *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_SET_RF_CHANNEL, HostCmd_CMD_SET_RF_CHANNEL);
|
|
pCmd->Action = htole16(HostCmd_ACT_GEN_SET);
|
|
pCmd->CurrentChannel = chan->channel;
|
|
pCmd->ChannelFlags = cvtChannelFlags(chan); /* NB: byte-swapped */
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_RF_CHANNEL);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
static int
|
|
bastream_check_available(struct mwl_hal_vap *vap, int qid,
|
|
const uint8_t Macaddr[IEEE80211_ADDR_LEN],
|
|
uint8_t Tid, uint8_t ParamInfo)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_FW_BASTREAM *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK_ASSERT(mh);
|
|
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_FW_BASTREAM, HostCmd_CMD_BASTREAM);
|
|
pCmd->ActionType = htole32(BaCheckCreateStream);
|
|
pCmd->BaInfo.CreateParams.BarThrs = htole32(63);
|
|
pCmd->BaInfo.CreateParams.WindowSize = htole32(64);
|
|
pCmd->BaInfo.CreateParams.IdleThrs = htole32(0x22000);
|
|
IEEE80211_ADDR_COPY(&pCmd->BaInfo.CreateParams.PeerMacAddr[0], Macaddr);
|
|
pCmd->BaInfo.CreateParams.DialogToken = 10;
|
|
pCmd->BaInfo.CreateParams.Tid = Tid;
|
|
pCmd->BaInfo.CreateParams.QueueId = qid;
|
|
pCmd->BaInfo.CreateParams.ParamInfo = (uint8_t) ParamInfo;
|
|
#if 0
|
|
cvtBAFlags(&pCmd->BaInfo.CreateParams.Flags, sp->ba_policy, 0);
|
|
#else
|
|
pCmd->BaInfo.CreateParams.Flags =
|
|
htole32(BASTREAM_FLAG_IMMEDIATE_TYPE)
|
|
| htole32(BASTREAM_FLAG_DIRECTION_UPSTREAM)
|
|
;
|
|
#endif
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_BASTREAM);
|
|
if (retval == 0) {
|
|
/*
|
|
* NB: BA stream create may fail when the stream is
|
|
* h/w backed under some (as yet not understood) conditions.
|
|
* Check the result code to catch this.
|
|
*/
|
|
if (le16toh(pCmd->CmdHdr.Result) != HostCmd_RESULT_OK)
|
|
retval = EIO;
|
|
}
|
|
return retval;
|
|
}
|
|
|
|
const MWL_HAL_BASTREAM *
|
|
mwl_hal_bastream_alloc(struct mwl_hal_vap *vap, int ba_policy,
|
|
const uint8_t Macaddr[IEEE80211_ADDR_LEN],
|
|
uint8_t Tid, uint8_t ParamInfo, void *a1, void *a2)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
struct mwl_hal_bastream *sp;
|
|
int s;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
if (mh->mh_bastreams == 0) {
|
|
/* no streams available */
|
|
MWL_HAL_UNLOCK(mh);
|
|
return NULL;
|
|
}
|
|
for (s = 0; (mh->mh_bastreams & (1<<s)) == 0; s++)
|
|
;
|
|
if (bastream_check_available(vap, s, Macaddr, Tid, ParamInfo)) {
|
|
MWL_HAL_UNLOCK(mh);
|
|
return NULL;
|
|
}
|
|
sp = &mh->mh_streams[s];
|
|
mh->mh_bastreams &= ~(1<<s);
|
|
sp->public.data[0] = a1;
|
|
sp->public.data[1] = a2;
|
|
IEEE80211_ADDR_COPY(sp->macaddr, Macaddr);
|
|
sp->tid = Tid;
|
|
sp->paraminfo = ParamInfo;
|
|
sp->setup = 0;
|
|
sp->ba_policy = ba_policy;
|
|
MWL_HAL_UNLOCK(mh);
|
|
return &sp->public;
|
|
}
|
|
|
|
const MWL_HAL_BASTREAM *
|
|
mwl_hal_bastream_lookup(struct mwl_hal *mh0, int s)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
|
|
if (!(0 <= s && s < MWL_BASTREAMS_MAX))
|
|
return NULL;
|
|
if (mh->mh_bastreams & (1<<s))
|
|
return NULL;
|
|
return &mh->mh_streams[s].public;
|
|
}
|
|
|
|
#ifndef __DECONST
|
|
#define __DECONST(type, var) ((type)(uintptr_t)(const void *)(var))
|
|
#endif
|
|
|
|
int
|
|
mwl_hal_bastream_create(struct mwl_hal_vap *vap,
|
|
const MWL_HAL_BASTREAM *s, int BarThrs, int WindowSize, uint16_t seqno)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
struct mwl_hal_bastream *sp = __DECONST(struct mwl_hal_bastream *, s);
|
|
HostCmd_FW_BASTREAM *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_FW_BASTREAM, HostCmd_CMD_BASTREAM);
|
|
pCmd->ActionType = htole32(BaCreateStream);
|
|
pCmd->BaInfo.CreateParams.BarThrs = htole32(BarThrs);
|
|
pCmd->BaInfo.CreateParams.WindowSize = htole32(WindowSize);
|
|
pCmd->BaInfo.CreateParams.IdleThrs = htole32(0x22000);
|
|
IEEE80211_ADDR_COPY(&pCmd->BaInfo.CreateParams.PeerMacAddr[0],
|
|
sp->macaddr);
|
|
/* XXX proxy STA */
|
|
memset(&pCmd->BaInfo.CreateParams.StaSrcMacAddr, 0, IEEE80211_ADDR_LEN);
|
|
#if 0
|
|
pCmd->BaInfo.CreateParams.DialogToken = DialogToken;
|
|
#else
|
|
pCmd->BaInfo.CreateParams.DialogToken = 10;
|
|
#endif
|
|
pCmd->BaInfo.CreateParams.Tid = sp->tid;
|
|
pCmd->BaInfo.CreateParams.QueueId = sp->stream;
|
|
pCmd->BaInfo.CreateParams.ParamInfo = sp->paraminfo;
|
|
/* NB: ResetSeqNo known to be zero */
|
|
pCmd->BaInfo.CreateParams.StartSeqNo = htole16(seqno);
|
|
#if 0
|
|
cvtBAFlags(&pCmd->BaInfo.CreateParams.Flags, sp->ba_policy, 0);
|
|
#else
|
|
pCmd->BaInfo.CreateParams.Flags =
|
|
htole32(BASTREAM_FLAG_IMMEDIATE_TYPE)
|
|
| htole32(BASTREAM_FLAG_DIRECTION_UPSTREAM)
|
|
;
|
|
#endif
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_BASTREAM);
|
|
if (retval == 0) {
|
|
/*
|
|
* NB: BA stream create may fail when the stream is
|
|
* h/w backed under some (as yet not understood) conditions.
|
|
* Check the result code to catch this.
|
|
*/
|
|
if (le16toh(pCmd->CmdHdr.Result) != HostCmd_RESULT_OK)
|
|
retval = EIO;
|
|
else
|
|
sp->setup = 1;
|
|
}
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_bastream_destroy(struct mwl_hal *mh0, const MWL_HAL_BASTREAM *s)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
struct mwl_hal_bastream *sp = __DECONST(struct mwl_hal_bastream *, s);
|
|
HostCmd_FW_BASTREAM *pCmd;
|
|
int retval;
|
|
|
|
if (sp->stream >= MWL_BASTREAMS_MAX) {
|
|
/* XXX */
|
|
return EINVAL;
|
|
}
|
|
MWL_HAL_LOCK(mh);
|
|
if (sp->setup) {
|
|
_CMD_SETUP(pCmd, HostCmd_FW_BASTREAM, HostCmd_CMD_BASTREAM);
|
|
pCmd->ActionType = htole32(BaDestroyStream);
|
|
pCmd->BaInfo.DestroyParams.FwBaContext.Context =
|
|
htole32(sp->stream);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_BASTREAM);
|
|
} else
|
|
retval = 0;
|
|
/* NB: always reclaim stream */
|
|
mh->mh_bastreams |= 1<<sp->stream;
|
|
sp->public.data[0] = NULL;
|
|
sp->public.data[1] = NULL;
|
|
sp->setup = 0;
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_bastream_get_seqno(struct mwl_hal *mh0,
|
|
const MWL_HAL_BASTREAM *s, const uint8_t Macaddr[IEEE80211_ADDR_LEN],
|
|
uint16_t *pseqno)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
struct mwl_hal_bastream *sp = __DECONST(struct mwl_hal_bastream *, s);
|
|
HostCmd_GET_SEQNO *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_GET_SEQNO, HostCmd_CMD_GET_SEQNO);
|
|
IEEE80211_ADDR_COPY(pCmd->MacAddr, Macaddr);
|
|
pCmd->TID = sp->tid;
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_GET_SEQNO);
|
|
if (retval == 0)
|
|
*pseqno = le16toh(pCmd->SeqNo);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_getwatchdogbitmap(struct mwl_hal *mh0, uint8_t bitmap[1])
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_GET_WATCHDOG_BITMAP *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_GET_WATCHDOG_BITMAP,
|
|
HostCmd_CMD_GET_WATCHDOG_BITMAP);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_GET_WATCHDOG_BITMAP);
|
|
if (retval == 0) {
|
|
bitmap[0] = pCmd->Watchdogbitmap;
|
|
/* fw returns qid, map it to BA stream */
|
|
if (bitmap[0] < MWL_BAQID_MAX)
|
|
bitmap[0] = qid2ba[bitmap[0]];
|
|
}
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Configure aggressive Ampdu rate mode.
|
|
*/
|
|
int
|
|
mwl_hal_setaggampduratemode(struct mwl_hal *mh0, int mode, int threshold)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_AMPDU_RETRY_RATEDROP_MODE *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_AMPDU_RETRY_RATEDROP_MODE,
|
|
HostCmd_CMD_AMPDU_RETRY_RATEDROP_MODE);
|
|
pCmd->Action = htole16(1);
|
|
pCmd->Option = htole32(mode);
|
|
pCmd->Threshold = htole32(threshold);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_AMPDU_RETRY_RATEDROP_MODE);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_getaggampduratemode(struct mwl_hal *mh0, int *mode, int *threshold)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_AMPDU_RETRY_RATEDROP_MODE *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_AMPDU_RETRY_RATEDROP_MODE,
|
|
HostCmd_CMD_AMPDU_RETRY_RATEDROP_MODE);
|
|
pCmd->Action = htole16(0);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_AMPDU_RETRY_RATEDROP_MODE);
|
|
MWL_HAL_UNLOCK(mh);
|
|
*mode = le32toh(pCmd->Option);
|
|
*threshold = le32toh(pCmd->Threshold);
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Set CFEND status Enable/Disable
|
|
*/
|
|
int
|
|
mwl_hal_setcfend(struct mwl_hal *mh0, int ena)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_CFEND_ENABLE *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_CFEND_ENABLE,
|
|
HostCmd_CMD_CFEND_ENABLE);
|
|
pCmd->Enable = htole32(ena);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_CFEND_ENABLE);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setdwds(struct mwl_hal *mh0, int ena)
|
|
{
|
|
HostCmd_DWDS_ENABLE *pCmd;
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_DWDS_ENABLE, HostCmd_CMD_DWDS_ENABLE);
|
|
pCmd->Enable = htole32(ena);
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_DWDS_ENABLE);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
static void
|
|
cvtPeerInfo(PeerInfo_t *to, const MWL_HAL_PEERINFO *from)
|
|
{
|
|
to->LegacyRateBitMap = htole32(from->LegacyRateBitMap);
|
|
to->HTRateBitMap = htole32(from->HTRateBitMap);
|
|
to->CapInfo = htole16(from->CapInfo);
|
|
to->HTCapabilitiesInfo = htole16(from->HTCapabilitiesInfo);
|
|
to->MacHTParamInfo = from->MacHTParamInfo;
|
|
to->AddHtInfo.ControlChan = from->AddHtInfo.ControlChan;
|
|
to->AddHtInfo.AddChan = from->AddHtInfo.AddChan;
|
|
to->AddHtInfo.OpMode = htole16(from->AddHtInfo.OpMode);
|
|
to->AddHtInfo.stbc = htole16(from->AddHtInfo.stbc);
|
|
}
|
|
|
|
/* XXX station id must be in [0..63] */
|
|
int
|
|
mwl_hal_newstation(struct mwl_hal_vap *vap,
|
|
const uint8_t addr[IEEE80211_ADDR_LEN], uint16_t aid, uint16_t sid,
|
|
const MWL_HAL_PEERINFO *peer, int isQosSta, int wmeInfo)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_FW_SET_NEW_STN *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_FW_SET_NEW_STN, HostCmd_CMD_SET_NEW_STN);
|
|
pCmd->AID = htole16(aid);
|
|
pCmd->StnId = htole16(sid);
|
|
pCmd->Action = htole16(0); /* SET */
|
|
if (peer != NULL) {
|
|
/* NB: must fix up byte order */
|
|
cvtPeerInfo(&pCmd->PeerInfo, peer);
|
|
}
|
|
IEEE80211_ADDR_COPY(&pCmd->MacAddr[0], addr);
|
|
pCmd->Qosinfo = wmeInfo;
|
|
pCmd->isQosSta = (isQosSta != 0);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_NEW_STN);
|
|
if (retval == 0 && IEEE80211_ADDR_EQ(vap->mac, addr))
|
|
vap->flags |= MVF_STATION;
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_delstation(struct mwl_hal_vap *vap,
|
|
const uint8_t addr[IEEE80211_ADDR_LEN])
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_FW_SET_NEW_STN *pCmd;
|
|
int retval, islocal;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
islocal = IEEE80211_ADDR_EQ(vap->mac, addr);
|
|
if (!islocal || (vap->flags & MVF_STATION)) {
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_FW_SET_NEW_STN,
|
|
HostCmd_CMD_SET_NEW_STN);
|
|
pCmd->Action = htole16(2); /* REMOVE */
|
|
IEEE80211_ADDR_COPY(&pCmd->MacAddr[0], addr);
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_NEW_STN);
|
|
if (islocal)
|
|
vap->flags &= ~MVF_STATION;
|
|
} else
|
|
retval = 0;
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Prod the firmware to age packets on station power
|
|
* save queues and reap frames on the tx aggregation q's.
|
|
*/
|
|
int
|
|
mwl_hal_setkeepalive(struct mwl_hal *mh0)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_SET_KEEP_ALIVE_TICK *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_SET_KEEP_ALIVE_TICK,
|
|
HostCmd_CMD_SET_KEEP_ALIVE);
|
|
/*
|
|
* NB: tick must be 0 to prod the f/w;
|
|
* a non-zero value is a noop.
|
|
*/
|
|
pCmd->tick = 0;
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_KEEP_ALIVE);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setapmode(struct mwl_hal_vap *vap, MWL_HAL_APMODE ApMode)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_FW_SET_APMODE *pCmd;
|
|
int retval;
|
|
|
|
/* XXX validate ApMode? */
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_FW_SET_APMODE, HostCmd_CMD_SET_APMODE);
|
|
pCmd->ApMode = ApMode;
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_APMODE);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_stop(struct mwl_hal_vap *vap)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_DS_BSS_START *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
if (vap->flags & MVF_RUNNING) {
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_DS_BSS_START,
|
|
HostCmd_CMD_BSS_START);
|
|
pCmd->Enable = htole32(HostCmd_ACT_GEN_OFF);
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_BSS_START);
|
|
} else
|
|
retval = 0;
|
|
/* NB: mark !running regardless */
|
|
vap->flags &= ~MVF_RUNNING;
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_start(struct mwl_hal_vap *vap)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_DS_BSS_START *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_DS_BSS_START, HostCmd_CMD_BSS_START);
|
|
pCmd->Enable = htole32(HostCmd_ACT_GEN_ON);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_BSS_START);
|
|
if (retval == 0)
|
|
vap->flags |= MVF_RUNNING;
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setgprot(struct mwl_hal *mh0, int prot)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_SET_G_PROTECT_FLAG *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_SET_G_PROTECT_FLAG,
|
|
HostCmd_CMD_SET_G_PROTECT_FLAG);
|
|
pCmd->GProtectFlag = htole32(prot);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_G_PROTECT_FLAG);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setwmm(struct mwl_hal *mh0, int onoff)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_SetWMMMode *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_SetWMMMode,
|
|
HostCmd_CMD_SET_WMM_MODE);
|
|
pCmd->Action = htole16(onoff);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_WMM_MODE);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setedcaparams(struct mwl_hal *mh0, uint8_t qnum,
|
|
uint32_t CWmin, uint32_t CWmax, uint8_t AIFSN, uint16_t TXOPLimit)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_SET_EDCA_PARAMS *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_SET_EDCA_PARAMS,
|
|
HostCmd_CMD_SET_EDCA_PARAMS);
|
|
/*
|
|
* NB: CWmin and CWmax are always set.
|
|
* TxOpLimit is set if bit 0x2 is marked in Action
|
|
* AIFSN is set if bit 0x4 is marked in Action
|
|
*/
|
|
pCmd->Action = htole16(0xffff); /* NB: set everything */
|
|
pCmd->TxOP = htole16(TXOPLimit);
|
|
pCmd->CWMax = htole32(CWmax);
|
|
pCmd->CWMin = htole32(CWmin);
|
|
pCmd->AIFSN = AIFSN;
|
|
pCmd->TxQNum = qnum; /* XXX check */
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_EDCA_PARAMS);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
/* XXX 0 = indoor, 1 = outdoor */
|
|
int
|
|
mwl_hal_setrateadaptmode(struct mwl_hal *mh0, uint16_t mode)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_DS_SET_RATE_ADAPT_MODE *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_DS_SET_RATE_ADAPT_MODE,
|
|
HostCmd_CMD_SET_RATE_ADAPT_MODE);
|
|
pCmd->Action = htole16(HostCmd_ACT_GEN_SET);
|
|
pCmd->RateAdaptMode = htole16(mode);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_RATE_ADAPT_MODE);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setcsmode(struct mwl_hal *mh0, MWL_HAL_CSMODE csmode)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_DS_SET_LINKADAPT_CS_MODE *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_DS_SET_LINKADAPT_CS_MODE,
|
|
HostCmd_CMD_SET_LINKADAPT_CS_MODE);
|
|
pCmd->Action = htole16(HostCmd_ACT_GEN_SET);
|
|
pCmd->CSMode = htole16(csmode);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_LINKADAPT_CS_MODE);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setnprot(struct mwl_hal_vap *vap, MWL_HAL_HTPROTECT mode)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_FW_SET_N_PROTECT_FLAG *pCmd;
|
|
int retval;
|
|
|
|
/* XXX validate mode */
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_FW_SET_N_PROTECT_FLAG,
|
|
HostCmd_CMD_SET_N_PROTECT_FLAG);
|
|
pCmd->NProtectFlag = htole32(mode);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_N_PROTECT_FLAG);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setnprotmode(struct mwl_hal_vap *vap, uint8_t mode)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLVAP(vap);
|
|
HostCmd_FW_SET_N_PROTECT_OPMODE *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_VCMD_SETUP(vap, pCmd, HostCmd_FW_SET_N_PROTECT_OPMODE,
|
|
HostCmd_CMD_SET_N_PROTECT_OPMODE);
|
|
pCmd->NProtectOpMode = mode;
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_N_PROTECT_OPMODE);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setoptimizationlevel(struct mwl_hal *mh0, int level)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_SET_OPTIMIZATION_LEVEL *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_SET_OPTIMIZATION_LEVEL,
|
|
HostCmd_CMD_SET_OPTIMIZATION_LEVEL);
|
|
pCmd->OptLevel = level;
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_OPTIMIZATION_LEVEL);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setmimops(struct mwl_hal *mh0, const uint8_t addr[IEEE80211_ADDR_LEN],
|
|
uint8_t enable, uint8_t mode)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_SET_MIMOPSHT *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_SET_MIMOPSHT, HostCmd_CMD_SET_MIMOPSHT);
|
|
IEEE80211_ADDR_COPY(pCmd->Addr, addr);
|
|
pCmd->Enable = enable;
|
|
pCmd->Mode = mode;
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_MIMOPSHT);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
static int
|
|
mwlGetCalTable(struct mwl_hal_priv *mh, uint8_t annex, uint8_t index)
|
|
{
|
|
HostCmd_FW_GET_CALTABLE *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK_ASSERT(mh);
|
|
|
|
_CMD_SETUP(pCmd, HostCmd_FW_GET_CALTABLE, HostCmd_CMD_GET_CALTABLE);
|
|
pCmd->annex = annex;
|
|
pCmd->index = index;
|
|
memset(pCmd->calTbl, 0, sizeof(pCmd->calTbl));
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_GET_CALTABLE);
|
|
if (retval == 0 &&
|
|
pCmd->calTbl[0] != annex && annex != 0 && annex != 255)
|
|
retval = EIO;
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Calculate the max tx power from the channel's cal data.
|
|
*/
|
|
static void
|
|
setmaxtxpow(struct mwl_hal_channel *hc, int i, int maxix)
|
|
{
|
|
hc->maxTxPow = hc->targetPowers[i];
|
|
for (i++; i < maxix; i++)
|
|
if (hc->targetPowers[i] > hc->maxTxPow)
|
|
hc->maxTxPow = hc->targetPowers[i];
|
|
}
|
|
|
|
/*
|
|
* Construct channel info for 5GHz channels from cal data.
|
|
*/
|
|
static void
|
|
get5Ghz(MWL_HAL_CHANNELINFO *ci, const uint8_t table[], int len)
|
|
{
|
|
int i, j, f, l, h;
|
|
|
|
l = 32000;
|
|
h = 0;
|
|
j = 0;
|
|
for (i = 0; i < len; i += 4) {
|
|
struct mwl_hal_channel *hc;
|
|
|
|
if (table[i] == 0)
|
|
continue;
|
|
f = 5000 + 5*table[i];
|
|
if (f < l)
|
|
l = f;
|
|
if (f > h)
|
|
h = f;
|
|
hc = &ci->channels[j];
|
|
hc->freq = f;
|
|
hc->ieee = table[i];
|
|
memcpy(hc->targetPowers, &table[i], 4);
|
|
setmaxtxpow(hc, 1, 4); /* NB: col 1 is the freq, skip*/
|
|
j++;
|
|
}
|
|
ci->nchannels = j;
|
|
ci->freqLow = (l == 32000) ? 0 : l;
|
|
ci->freqHigh = h;
|
|
}
|
|
|
|
static uint16_t
|
|
ieee2mhz(int chan)
|
|
{
|
|
if (chan == 14)
|
|
return 2484;
|
|
if (chan < 14)
|
|
return 2407 + chan*5;
|
|
return 2512 + (chan-15)*20;
|
|
}
|
|
|
|
/*
|
|
* Construct channel info for 2.4GHz channels from cal data.
|
|
*/
|
|
static void
|
|
get2Ghz(MWL_HAL_CHANNELINFO *ci, const uint8_t table[], int len)
|
|
{
|
|
int i, j;
|
|
|
|
j = 0;
|
|
for (i = 0; i < len; i += 4) {
|
|
struct mwl_hal_channel *hc = &ci->channels[j];
|
|
hc->ieee = 1+j;
|
|
hc->freq = ieee2mhz(1+j);
|
|
memcpy(hc->targetPowers, &table[i], 4);
|
|
setmaxtxpow(hc, 0, 4);
|
|
j++;
|
|
}
|
|
ci->nchannels = j;
|
|
ci->freqLow = ieee2mhz(1);
|
|
ci->freqHigh = ieee2mhz(j);
|
|
}
|
|
|
|
#undef DUMPCALDATA
|
|
#ifdef DUMPCALDATA
|
|
static void
|
|
dumpcaldata(const char *name, const uint8_t *table, int n)
|
|
{
|
|
int i;
|
|
printf("\n%s:\n", name);
|
|
for (i = 0; i < n; i += 4)
|
|
printf("[%2d] %3d %3d %3d %3d\n", i/4, table[i+0], table[i+1], table[i+2], table[i+3]);
|
|
}
|
|
#endif
|
|
|
|
static int
|
|
mwlGetPwrCalTable(struct mwl_hal_priv *mh)
|
|
{
|
|
const uint8_t *data;
|
|
MWL_HAL_CHANNELINFO *ci;
|
|
int len;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
/* NB: we hold the lock so it's ok to use cmdbuf */
|
|
data = ((const HostCmd_FW_GET_CALTABLE *) mh->mh_cmdbuf)->calTbl;
|
|
if (mwlGetCalTable(mh, 33, 0) == 0) {
|
|
len = (data[2] | (data[3] << 8)) - 12;
|
|
if (len > PWTAGETRATETABLE20M)
|
|
len = PWTAGETRATETABLE20M;
|
|
#ifdef DUMPCALDATA
|
|
dumpcaldata("2.4G 20M", &data[12], len);/*XXX*/
|
|
#endif
|
|
get2Ghz(&mh->mh_20M, &data[12], len);
|
|
}
|
|
if (mwlGetCalTable(mh, 34, 0) == 0) {
|
|
len = (data[2] | (data[3] << 8)) - 12;
|
|
if (len > PWTAGETRATETABLE40M)
|
|
len = PWTAGETRATETABLE40M;
|
|
#ifdef DUMPCALDATA
|
|
dumpcaldata("2.4G 40M", &data[12], len);/*XXX*/
|
|
#endif
|
|
ci = &mh->mh_40M;
|
|
get2Ghz(ci, &data[12], len);
|
|
}
|
|
if (mwlGetCalTable(mh, 35, 0) == 0) {
|
|
len = (data[2] | (data[3] << 8)) - 20;
|
|
if (len > PWTAGETRATETABLE20M_5G)
|
|
len = PWTAGETRATETABLE20M_5G;
|
|
#ifdef DUMPCALDATA
|
|
dumpcaldata("5G 20M", &data[20], len);/*XXX*/
|
|
#endif
|
|
get5Ghz(&mh->mh_20M_5G, &data[20], len);
|
|
}
|
|
if (mwlGetCalTable(mh, 36, 0) == 0) {
|
|
len = (data[2] | (data[3] << 8)) - 20;
|
|
if (len > PWTAGETRATETABLE40M_5G)
|
|
len = PWTAGETRATETABLE40M_5G;
|
|
#ifdef DUMPCALDATA
|
|
dumpcaldata("5G 40M", &data[20], len);/*XXX*/
|
|
#endif
|
|
ci = &mh->mh_40M_5G;
|
|
get5Ghz(ci, &data[20], len);
|
|
}
|
|
mh->mh_flags |= MHF_CALDATA;
|
|
MWL_HAL_UNLOCK(mh);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
mwl_hal_getregioncode(struct mwl_hal *mh0, uint8_t *countryCode)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
retval = mwlGetCalTable(mh, 0, 0);
|
|
if (retval == 0) {
|
|
const HostCmd_FW_GET_CALTABLE *pCmd =
|
|
(const HostCmd_FW_GET_CALTABLE *) mh->mh_cmdbuf;
|
|
*countryCode = pCmd->calTbl[16];
|
|
}
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_setpromisc(struct mwl_hal *mh0, int ena)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
uint32_t v;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
v = RD4(mh, MACREG_REG_PROMISCUOUS);
|
|
WR4(mh, MACREG_REG_PROMISCUOUS, ena ? v | 1 : v &~ 1);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
mwl_hal_getpromisc(struct mwl_hal *mh0)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
uint32_t v;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
v = RD4(mh, MACREG_REG_PROMISCUOUS);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return (v & 1) != 0;
|
|
}
|
|
|
|
int
|
|
mwl_hal_GetBeacon(struct mwl_hal *mh0, uint8_t *pBcn, uint16_t *pLen)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_GET_BEACON *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_GET_BEACON, HostCmd_CMD_GET_BEACON);
|
|
pCmd->Bcnlen = htole16(0);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_GET_BEACON);
|
|
if (retval == 0) {
|
|
/* XXX bounds check */
|
|
memcpy(pBcn, &pCmd->Bcn, pCmd->Bcnlen);
|
|
*pLen = pCmd->Bcnlen;
|
|
}
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
int
|
|
mwl_hal_SetRifs(struct mwl_hal *mh0, uint8_t QNum)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
HostCmd_FW_SET_RIFS *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_FW_SET_RIFS, HostCmd_CMD_SET_RIFS);
|
|
pCmd->QNum = QNum;
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_SET_RIFS);
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
/*
|
|
* Diagnostic api's for set/get registers.
|
|
*/
|
|
|
|
static int
|
|
getRFReg(struct mwl_hal_priv *mh, int flag, uint32_t reg, uint32_t *val)
|
|
{
|
|
HostCmd_DS_RF_REG_ACCESS *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_DS_RF_REG_ACCESS, HostCmd_CMD_RF_REG_ACCESS);
|
|
pCmd->Offset = htole16(reg);
|
|
pCmd->Action = htole16(flag);
|
|
pCmd->Value = htole32(*val);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_RF_REG_ACCESS);
|
|
if (retval == 0)
|
|
*val = pCmd->Value;
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
static int
|
|
getBBReg(struct mwl_hal_priv *mh, int flag, uint32_t reg, uint32_t *val)
|
|
{
|
|
HostCmd_DS_BBP_REG_ACCESS *pCmd;
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
_CMD_SETUP(pCmd, HostCmd_DS_BBP_REG_ACCESS, HostCmd_CMD_BBP_REG_ACCESS);
|
|
pCmd->Offset = htole16(reg);
|
|
pCmd->Action = htole16(flag);
|
|
pCmd->Value = htole32(*val);
|
|
|
|
retval = mwlExecuteCmd(mh, HostCmd_CMD_BBP_REG_ACCESS);
|
|
if (retval == 0)
|
|
*val = pCmd->Value;
|
|
MWL_HAL_UNLOCK(mh);
|
|
return retval;
|
|
}
|
|
|
|
static u_int
|
|
mwl_hal_getregdump(struct mwl_hal_priv *mh, const MWL_DIAG_REGRANGE *regs,
|
|
void *dstbuf, int space)
|
|
{
|
|
uint32_t *dp = dstbuf;
|
|
int i;
|
|
|
|
for (i = 0; space >= 2*sizeof(uint32_t); i++) {
|
|
u_int r = regs[i].start;
|
|
u_int e = regs[i].end;
|
|
*dp++ = (r<<16) | e;
|
|
space -= sizeof(uint32_t);
|
|
do {
|
|
if (MWL_DIAG_ISMAC(r))
|
|
*dp = RD4(mh, r);
|
|
else if (MWL_DIAG_ISBB(r))
|
|
getBBReg(mh, HostCmd_ACT_GEN_READ,
|
|
r - MWL_DIAG_BASE_BB, dp);
|
|
else if (MWL_DIAG_ISRF(r))
|
|
getRFReg(mh, HostCmd_ACT_GEN_READ,
|
|
r - MWL_DIAG_BASE_RF, dp);
|
|
else if (r < 0x1000 || r == MACREG_REG_FW_PRESENT)
|
|
*dp = RD4(mh, r);
|
|
else
|
|
*dp = 0xffffffff;
|
|
dp++;
|
|
r += sizeof(uint32_t);
|
|
space -= sizeof(uint32_t);
|
|
} while (r <= e && space >= sizeof(uint32_t));
|
|
}
|
|
return (char *) dp - (char *) dstbuf;
|
|
}
|
|
|
|
int
|
|
mwl_hal_getdiagstate(struct mwl_hal *mh0, int request,
|
|
const void *args, uint32_t argsize,
|
|
void **result, uint32_t *resultsize)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
|
|
switch (request) {
|
|
case MWL_DIAG_CMD_REVS:
|
|
*result = &mh->mh_revs;
|
|
*resultsize = sizeof(mh->mh_revs);
|
|
return 1;
|
|
case MWL_DIAG_CMD_REGS:
|
|
*resultsize = mwl_hal_getregdump(mh, args, *result, *resultsize);
|
|
return 1;
|
|
case MWL_DIAG_CMD_HOSTCMD: {
|
|
FWCmdHdr *pCmd = (FWCmdHdr *) &mh->mh_cmdbuf[0];
|
|
int retval;
|
|
|
|
MWL_HAL_LOCK(mh);
|
|
memcpy(pCmd, args, argsize);
|
|
retval = mwlExecuteCmd(mh, le16toh(pCmd->Cmd));
|
|
*result = (*resultsize != 0) ? pCmd : NULL;
|
|
MWL_HAL_UNLOCK(mh);
|
|
return (retval == 0);
|
|
}
|
|
case MWL_DIAG_CMD_FWLOAD:
|
|
if (mwl_hal_fwload(mh0, __DECONST(void *, args))) {
|
|
device_printf(mh->mh_dev, "problem loading fw image\n");
|
|
return 0;
|
|
}
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Low level firmware cmd block handshake support.
|
|
*/
|
|
|
|
static void
|
|
mwlSendCmd(struct mwl_hal_priv *mh)
|
|
{
|
|
uint32_t dummy;
|
|
|
|
bus_dmamap_sync(mh->mh_dmat, mh->mh_dmamap,
|
|
BUS_DMASYNC_PREREAD | BUS_DMASYNC_PREWRITE);
|
|
|
|
WR4(mh, MACREG_REG_GEN_PTR, mh->mh_cmdaddr);
|
|
dummy = RD4(mh, MACREG_REG_INT_CODE);
|
|
|
|
WR4(mh, MACREG_REG_H2A_INTERRUPT_EVENTS, MACREG_H2ARIC_BIT_DOOR_BELL);
|
|
}
|
|
|
|
static int
|
|
mwlWaitForCmdComplete(struct mwl_hal_priv *mh, uint16_t cmdCode)
|
|
{
|
|
#define MAX_WAIT_FW_COMPLETE_ITERATIONS 10000
|
|
int i;
|
|
|
|
for (i = 0; i < MAX_WAIT_FW_COMPLETE_ITERATIONS; i++) {
|
|
if (mh->mh_cmdbuf[0] == le16toh(cmdCode))
|
|
return 1;
|
|
DELAY(1*1000);
|
|
}
|
|
return 0;
|
|
#undef MAX_WAIT_FW_COMPLETE_ITERATIONS
|
|
}
|
|
|
|
static int
|
|
mwlExecuteCmd(struct mwl_hal_priv *mh, unsigned short cmd)
|
|
{
|
|
|
|
MWL_HAL_LOCK_ASSERT(mh);
|
|
|
|
if ((mh->mh_flags & MHF_FWHANG) &&
|
|
(mh->mh_debug & MWL_HAL_DEBUG_IGNHANG) == 0) {
|
|
#ifdef MWLHAL_DEBUG
|
|
device_printf(mh->mh_dev, "firmware hung, skipping cmd %s\n",
|
|
mwlcmdname(cmd));
|
|
#else
|
|
device_printf(mh->mh_dev, "firmware hung, skipping cmd 0x%x\n",
|
|
cmd);
|
|
#endif
|
|
return ENXIO;
|
|
}
|
|
if (RD4(mh, MACREG_REG_INT_CODE) == 0xffffffff) {
|
|
device_printf(mh->mh_dev, "%s: device not present!\n",
|
|
__func__);
|
|
return EIO;
|
|
}
|
|
#ifdef MWLHAL_DEBUG
|
|
if (mh->mh_debug & MWL_HAL_DEBUG_SENDCMD)
|
|
dumpresult(mh, 0);
|
|
#endif
|
|
mwlSendCmd(mh);
|
|
if (!mwlWaitForCmdComplete(mh, 0x8000 | cmd)) {
|
|
#ifdef MWLHAL_DEBUG
|
|
device_printf(mh->mh_dev,
|
|
"timeout waiting for f/w cmd %s\n", mwlcmdname(cmd));
|
|
#else
|
|
device_printf(mh->mh_dev,
|
|
"timeout waiting for f/w cmd 0x%x\n", cmd);
|
|
#endif
|
|
mh->mh_flags |= MHF_FWHANG;
|
|
return ETIMEDOUT;
|
|
}
|
|
bus_dmamap_sync(mh->mh_dmat, mh->mh_dmamap,
|
|
BUS_DMASYNC_POSTREAD | BUS_DMASYNC_POSTWRITE);
|
|
#ifdef MWLHAL_DEBUG
|
|
if (mh->mh_debug & MWL_HAL_DEBUG_CMDDONE)
|
|
dumpresult(mh, 1);
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Firmware download support.
|
|
*/
|
|
#define FW_DOWNLOAD_BLOCK_SIZE 256
|
|
#define FW_CHECK_USECS (5*1000) /* 5ms */
|
|
#define FW_MAX_NUM_CHECKS 200
|
|
|
|
#if 0
|
|
/* XXX read f/w from file */
|
|
#include <dev/mwl/mwlbootfw.h>
|
|
#include <dev/mwl/mwl88W8363fw.h>
|
|
#endif
|
|
|
|
static void
|
|
mwlFwReset(struct mwl_hal_priv *mh)
|
|
{
|
|
if (RD4(mh, MACREG_REG_INT_CODE) == 0xffffffff) {
|
|
device_printf(mh->mh_dev, "%s: device not present!\n",
|
|
__func__);
|
|
return;
|
|
}
|
|
WR4(mh, MACREG_REG_H2A_INTERRUPT_EVENTS, ISR_RESET);
|
|
mh->mh_flags &= ~MHF_FWHANG;
|
|
}
|
|
|
|
static void
|
|
mwlTriggerPciCmd(struct mwl_hal_priv *mh)
|
|
{
|
|
uint32_t dummy;
|
|
|
|
bus_dmamap_sync(mh->mh_dmat, mh->mh_dmamap, BUS_DMASYNC_PREWRITE);
|
|
|
|
WR4(mh, MACREG_REG_GEN_PTR, mh->mh_cmdaddr);
|
|
dummy = RD4(mh, MACREG_REG_INT_CODE);
|
|
|
|
WR4(mh, MACREG_REG_INT_CODE, 0x00);
|
|
dummy = RD4(mh, MACREG_REG_INT_CODE);
|
|
|
|
WR4(mh, MACREG_REG_H2A_INTERRUPT_EVENTS, MACREG_H2ARIC_BIT_DOOR_BELL);
|
|
dummy = RD4(mh, MACREG_REG_INT_CODE);
|
|
}
|
|
|
|
static int
|
|
mwlWaitFor(struct mwl_hal_priv *mh, uint32_t val)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < FW_MAX_NUM_CHECKS; i++) {
|
|
DELAY(FW_CHECK_USECS);
|
|
if (RD4(mh, MACREG_REG_INT_CODE) == val)
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Firmware block xmit when talking to the boot-rom.
|
|
*/
|
|
static int
|
|
mwlSendBlock(struct mwl_hal_priv *mh, int bsize, const void *data, size_t dsize)
|
|
{
|
|
mh->mh_cmdbuf[0] = htole16(HostCmd_CMD_CODE_DNLD);
|
|
mh->mh_cmdbuf[1] = htole16(bsize);
|
|
memcpy(&mh->mh_cmdbuf[4], data , dsize);
|
|
mwlTriggerPciCmd(mh);
|
|
/* XXX 2000 vs 200 */
|
|
if (mwlWaitFor(mh, MACREG_INT_CODE_CMD_FINISHED)) {
|
|
WR4(mh, MACREG_REG_INT_CODE, 0);
|
|
return 1;
|
|
}
|
|
device_printf(mh->mh_dev,
|
|
"%s: timeout waiting for CMD_FINISHED, INT_CODE 0x%x\n",
|
|
__func__, RD4(mh, MACREG_REG_INT_CODE));
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Firmware block xmit when talking to the 1st-stage loader.
|
|
*/
|
|
static int
|
|
mwlSendBlock2(struct mwl_hal_priv *mh, const void *data, size_t dsize)
|
|
{
|
|
memcpy(&mh->mh_cmdbuf[0], data, dsize);
|
|
mwlTriggerPciCmd(mh);
|
|
if (mwlWaitFor(mh, MACREG_INT_CODE_CMD_FINISHED)) {
|
|
WR4(mh, MACREG_REG_INT_CODE, 0);
|
|
return 1;
|
|
}
|
|
device_printf(mh->mh_dev,
|
|
"%s: timeout waiting for CMD_FINISHED, INT_CODE 0x%x\n",
|
|
__func__, RD4(mh, MACREG_REG_INT_CODE));
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
mwlPokeSdramController(struct mwl_hal_priv *mh, int SDRAMSIZE_Addr)
|
|
{
|
|
/** Set up sdram controller for superflyv2 **/
|
|
WR4(mh, 0x00006014, 0x33);
|
|
WR4(mh, 0x00006018, 0xa3a2632);
|
|
WR4(mh, 0x00006010, SDRAMSIZE_Addr);
|
|
}
|
|
|
|
int
|
|
mwl_hal_fwload(struct mwl_hal *mh0, void *fwargs)
|
|
{
|
|
struct mwl_hal_priv *mh = MWLPRIV(mh0);
|
|
const char *fwname = "mw88W8363fw";
|
|
const char *fwbootname = "mwlboot";
|
|
const struct firmware *fwboot = NULL;
|
|
const struct firmware *fw;
|
|
/* XXX get from firmware header */
|
|
uint32_t FwReadySignature = HostCmd_SOFTAP_FWRDY_SIGNATURE;
|
|
uint32_t OpMode = HostCmd_SOFTAP_MODE;
|
|
const uint8_t *fp, *ep;
|
|
const uint8_t *fmdata;
|
|
uint32_t blocksize, nbytes, fmsize;
|
|
int i, error, ntries;
|
|
|
|
fw = firmware_get(fwname);
|
|
if (fw == NULL) {
|
|
device_printf(mh->mh_dev,
|
|
"could not load firmware image %s\n", fwname);
|
|
return ENXIO;
|
|
}
|
|
fmdata = fw->data;
|
|
fmsize = fw->datasize;
|
|
if (fmsize < 4) {
|
|
device_printf(mh->mh_dev, "firmware image %s too small\n",
|
|
fwname);
|
|
error = ENXIO;
|
|
goto bad2;
|
|
}
|
|
if (fmdata[0] == 0x01 && fmdata[1] == 0x00 &&
|
|
fmdata[2] == 0x00 && fmdata[3] == 0x00) {
|
|
/*
|
|
* 2-stage load, get the boot firmware.
|
|
*/
|
|
fwboot = firmware_get(fwbootname);
|
|
if (fwboot == NULL) {
|
|
device_printf(mh->mh_dev,
|
|
"could not load firmware image %s\n", fwbootname);
|
|
error = ENXIO;
|
|
goto bad2;
|
|
}
|
|
} else
|
|
fwboot = NULL;
|
|
|
|
mwlFwReset(mh);
|
|
|
|
WR4(mh, MACREG_REG_A2H_INTERRUPT_CLEAR_SEL, MACREG_A2HRIC_BIT_MASK);
|
|
WR4(mh, MACREG_REG_A2H_INTERRUPT_CAUSE, 0x00);
|
|
WR4(mh, MACREG_REG_A2H_INTERRUPT_MASK, 0x00);
|
|
WR4(mh, MACREG_REG_A2H_INTERRUPT_STATUS_MASK, MACREG_A2HRIC_BIT_MASK);
|
|
if (mh->mh_SDRAMSIZE_Addr != 0) {
|
|
/** Set up sdram controller for superflyv2 **/
|
|
mwlPokeSdramController(mh, mh->mh_SDRAMSIZE_Addr);
|
|
}
|
|
device_printf(mh->mh_dev, "load %s firmware image (%u bytes)\n",
|
|
fwname, fmsize);
|
|
if (fwboot != NULL) {
|
|
/*
|
|
* Do 2-stage load. The 1st stage loader is setup
|
|
* with the bootrom loader then we load the real
|
|
* image using a different handshake. With this
|
|
* mechanism the firmware is segmented into chunks
|
|
* that have a CRC. If a chunk is incorrect we'll
|
|
* be told to retransmit.
|
|
*/
|
|
/* XXX assumes hlpimage fits in a block */
|
|
/* NB: zero size block indicates download is finished */
|
|
if (!mwlSendBlock(mh, fwboot->datasize, fwboot->data, fwboot->datasize) ||
|
|
!mwlSendBlock(mh, 0, NULL, 0)) {
|
|
error = ETIMEDOUT;
|
|
goto bad;
|
|
}
|
|
DELAY(200*FW_CHECK_USECS);
|
|
if (mh->mh_SDRAMSIZE_Addr != 0) {
|
|
/** Set up sdram controller for superflyv2 **/
|
|
mwlPokeSdramController(mh, mh->mh_SDRAMSIZE_Addr);
|
|
}
|
|
nbytes = ntries = 0; /* NB: silence compiler */
|
|
for (fp = fmdata, ep = fp + fmsize; fp < ep; ) {
|
|
WR4(mh, MACREG_REG_INT_CODE, 0);
|
|
blocksize = RD4(mh, MACREG_REG_SCRATCH);
|
|
if (blocksize == 0) /* download complete */
|
|
break;
|
|
if (blocksize > 0x00000c00) {
|
|
error = EINVAL;
|
|
goto bad;
|
|
}
|
|
if ((blocksize & 0x1) == 0) {
|
|
/* block successfully downloaded, advance */
|
|
fp += nbytes;
|
|
ntries = 0;
|
|
} else {
|
|
if (++ntries > 2) {
|
|
/*
|
|
* Guard against f/w telling us to
|
|
* retry infinitely.
|
|
*/
|
|
error = ELOOP;
|
|
goto bad;
|
|
}
|
|
/* clear NAK bit/flag */
|
|
blocksize &= ~0x1;
|
|
}
|
|
if (blocksize > ep - fp) {
|
|
/* XXX this should not happen, what to do? */
|
|
blocksize = ep - fp;
|
|
}
|
|
nbytes = blocksize;
|
|
if (!mwlSendBlock2(mh, fp, nbytes)) {
|
|
error = ETIMEDOUT;
|
|
goto bad;
|
|
}
|
|
}
|
|
} else {
|
|
for (fp = fmdata, ep = fp + fmsize; fp < ep;) {
|
|
nbytes = ep - fp;
|
|
if (nbytes > FW_DOWNLOAD_BLOCK_SIZE)
|
|
nbytes = FW_DOWNLOAD_BLOCK_SIZE;
|
|
if (!mwlSendBlock(mh, FW_DOWNLOAD_BLOCK_SIZE, fp, nbytes)) {
|
|
error = EIO;
|
|
goto bad;
|
|
}
|
|
fp += nbytes;
|
|
}
|
|
}
|
|
/* done with firmware... */
|
|
if (fwboot != NULL)
|
|
firmware_put(fwboot, FIRMWARE_UNLOAD);
|
|
firmware_put(fw, FIRMWARE_UNLOAD);
|
|
/*
|
|
* Wait for firmware to startup; we monitor the
|
|
* INT_CODE register waiting for a signature to
|
|
* written back indicating it's ready to go.
|
|
*/
|
|
mh->mh_cmdbuf[1] = 0;
|
|
/*
|
|
* XXX WAR for mfg fw download
|
|
*/
|
|
if (OpMode != HostCmd_STA_MODE)
|
|
mwlTriggerPciCmd(mh);
|
|
for (i = 0; i < FW_MAX_NUM_CHECKS; i++) {
|
|
WR4(mh, MACREG_REG_GEN_PTR, OpMode);
|
|
DELAY(FW_CHECK_USECS);
|
|
if (RD4(mh, MACREG_REG_INT_CODE) == FwReadySignature) {
|
|
WR4(mh, MACREG_REG_INT_CODE, 0x00);
|
|
return mwlResetHalState(mh);
|
|
}
|
|
}
|
|
return ETIMEDOUT;
|
|
bad:
|
|
mwlFwReset(mh);
|
|
bad2:
|
|
/* done with firmware... */
|
|
if (fwboot != NULL)
|
|
firmware_put(fwboot, FIRMWARE_UNLOAD);
|
|
firmware_put(fw, FIRMWARE_UNLOAD);
|
|
return error;
|
|
}
|
|
|
|
#ifdef MWLHAL_DEBUG
|
|
static const char *
|
|
mwlcmdname(int cmd)
|
|
{
|
|
static char buf[12];
|
|
#define CMD(x) case HostCmd_CMD_##x: return #x
|
|
switch (cmd) {
|
|
CMD(CODE_DNLD);
|
|
CMD(GET_HW_SPEC);
|
|
CMD(SET_HW_SPEC);
|
|
CMD(MAC_MULTICAST_ADR);
|
|
CMD(802_11_GET_STAT);
|
|
CMD(MAC_REG_ACCESS);
|
|
CMD(BBP_REG_ACCESS);
|
|
CMD(RF_REG_ACCESS);
|
|
CMD(802_11_RADIO_CONTROL);
|
|
CMD(802_11_RF_TX_POWER);
|
|
CMD(802_11_RF_ANTENNA);
|
|
CMD(SET_BEACON);
|
|
CMD(SET_RF_CHANNEL);
|
|
CMD(SET_AID);
|
|
CMD(SET_INFRA_MODE);
|
|
CMD(SET_G_PROTECT_FLAG);
|
|
CMD(802_11_RTS_THSD);
|
|
CMD(802_11_SET_SLOT);
|
|
CMD(SET_EDCA_PARAMS);
|
|
CMD(802_11H_DETECT_RADAR);
|
|
CMD(SET_WMM_MODE);
|
|
CMD(HT_GUARD_INTERVAL);
|
|
CMD(SET_FIXED_RATE);
|
|
CMD(SET_LINKADAPT_CS_MODE);
|
|
CMD(SET_MAC_ADDR);
|
|
CMD(SET_RATE_ADAPT_MODE);
|
|
CMD(BSS_START);
|
|
CMD(SET_NEW_STN);
|
|
CMD(SET_KEEP_ALIVE);
|
|
CMD(SET_APMODE);
|
|
CMD(SET_SWITCH_CHANNEL);
|
|
CMD(UPDATE_ENCRYPTION);
|
|
CMD(BASTREAM);
|
|
CMD(SET_RIFS);
|
|
CMD(SET_N_PROTECT_FLAG);
|
|
CMD(SET_N_PROTECT_OPMODE);
|
|
CMD(SET_OPTIMIZATION_LEVEL);
|
|
CMD(GET_CALTABLE);
|
|
CMD(SET_MIMOPSHT);
|
|
CMD(GET_BEACON);
|
|
CMD(SET_REGION_CODE);
|
|
CMD(SET_POWERSAVESTATION);
|
|
CMD(SET_TIM);
|
|
CMD(GET_TIM);
|
|
CMD(GET_SEQNO);
|
|
CMD(DWDS_ENABLE);
|
|
CMD(AMPDU_RETRY_RATEDROP_MODE);
|
|
CMD(CFEND_ENABLE);
|
|
}
|
|
snprintf(buf, sizeof(buf), "0x%x", cmd);
|
|
return buf;
|
|
#undef CMD
|
|
}
|
|
|
|
static void
|
|
dumpresult(struct mwl_hal_priv *mh, int showresult)
|
|
{
|
|
const FWCmdHdr *h = (const FWCmdHdr *)mh->mh_cmdbuf;
|
|
const uint8_t *cp;
|
|
int len, i;
|
|
|
|
len = le16toh(h->Length);
|
|
#ifdef MWL_MBSS_SUPPORT
|
|
device_printf(mh->mh_dev, "Cmd %s Length %d SeqNum %d MacId %d",
|
|
mwlcmdname(le16toh(h->Cmd) &~ 0x8000), len, h->SeqNum, h->MacId);
|
|
#else
|
|
device_printf(mh->mh_dev, "Cmd %s Length %d SeqNum %d",
|
|
mwlcmdname(le16toh(h->Cmd) &~ 0x8000), len, le16toh(h->SeqNum));
|
|
#endif
|
|
if (showresult) {
|
|
const char *results[] =
|
|
{ "OK", "ERROR", "NOT_SUPPORT", "PENDING", "BUSY",
|
|
"PARTIAL_DATA" };
|
|
int result = le16toh(h->Result);
|
|
|
|
if (result <= HostCmd_RESULT_PARTIAL_DATA)
|
|
printf(" Result %s", results[result]);
|
|
else
|
|
printf(" Result %d", result);
|
|
}
|
|
cp = (const uint8_t *)h;
|
|
for (i = 0; i < len; i++) {
|
|
if ((i % 16) == 0)
|
|
printf("\n%02x", cp[i]);
|
|
else
|
|
printf(" %02x", cp[i]);
|
|
}
|
|
printf("\n");
|
|
}
|
|
#endif /* MWLHAL_DEBUG */
|