1*7453645fSAndriy Voskoboinyk /*-
2*7453645fSAndriy Voskoboinyk * Copyright (c) 2015-2016 Andriy Voskoboinyk <avos@FreeBSD.org>
3*7453645fSAndriy Voskoboinyk *
4*7453645fSAndriy Voskoboinyk * Permission to use, copy, modify, and distribute this software for any
5*7453645fSAndriy Voskoboinyk * purpose with or without fee is hereby granted, provided that the above
6*7453645fSAndriy Voskoboinyk * copyright notice and this permission notice appear in all copies.
7*7453645fSAndriy Voskoboinyk *
8*7453645fSAndriy Voskoboinyk * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
9*7453645fSAndriy Voskoboinyk * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
10*7453645fSAndriy Voskoboinyk * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
11*7453645fSAndriy Voskoboinyk * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
12*7453645fSAndriy Voskoboinyk * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
13*7453645fSAndriy Voskoboinyk * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
14*7453645fSAndriy Voskoboinyk * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
15*7453645fSAndriy Voskoboinyk */
16*7453645fSAndriy Voskoboinyk
17*7453645fSAndriy Voskoboinyk #include <sys/cdefs.h>
18*7453645fSAndriy Voskoboinyk #include "opt_wlan.h"
19*7453645fSAndriy Voskoboinyk
20*7453645fSAndriy Voskoboinyk #include <sys/param.h>
21*7453645fSAndriy Voskoboinyk #include <sys/lock.h>
22*7453645fSAndriy Voskoboinyk #include <sys/mutex.h>
23*7453645fSAndriy Voskoboinyk #include <sys/mbuf.h>
24*7453645fSAndriy Voskoboinyk #include <sys/kernel.h>
25*7453645fSAndriy Voskoboinyk #include <sys/socket.h>
26*7453645fSAndriy Voskoboinyk #include <sys/systm.h>
27*7453645fSAndriy Voskoboinyk #include <sys/malloc.h>
28*7453645fSAndriy Voskoboinyk #include <sys/queue.h>
29*7453645fSAndriy Voskoboinyk #include <sys/taskqueue.h>
30*7453645fSAndriy Voskoboinyk #include <sys/bus.h>
31*7453645fSAndriy Voskoboinyk #include <sys/endian.h>
32*7453645fSAndriy Voskoboinyk
33*7453645fSAndriy Voskoboinyk #include <net/if.h>
34*7453645fSAndriy Voskoboinyk #include <net/ethernet.h>
35*7453645fSAndriy Voskoboinyk #include <net/if_media.h>
36*7453645fSAndriy Voskoboinyk
37*7453645fSAndriy Voskoboinyk #include <net80211/ieee80211_var.h>
38*7453645fSAndriy Voskoboinyk #include <net80211/ieee80211_radiotap.h>
39*7453645fSAndriy Voskoboinyk
40*7453645fSAndriy Voskoboinyk #include <dev/rtwn/if_rtwnvar.h>
41*7453645fSAndriy Voskoboinyk
42*7453645fSAndriy Voskoboinyk #include <dev/rtwn/if_rtwn_task.h>
43*7453645fSAndriy Voskoboinyk
44*7453645fSAndriy Voskoboinyk static void
rtwn_cmdq_cb(void * arg,int pending)45*7453645fSAndriy Voskoboinyk rtwn_cmdq_cb(void *arg, int pending)
46*7453645fSAndriy Voskoboinyk {
47*7453645fSAndriy Voskoboinyk struct rtwn_softc *sc = arg;
48*7453645fSAndriy Voskoboinyk struct rtwn_cmdq *item;
49*7453645fSAndriy Voskoboinyk
50*7453645fSAndriy Voskoboinyk /*
51*7453645fSAndriy Voskoboinyk * Device must be powered on (via rtwn_power_on())
52*7453645fSAndriy Voskoboinyk * before any command may be sent.
53*7453645fSAndriy Voskoboinyk */
54*7453645fSAndriy Voskoboinyk RTWN_LOCK(sc);
55*7453645fSAndriy Voskoboinyk if (!(sc->sc_flags & RTWN_RUNNING)) {
56*7453645fSAndriy Voskoboinyk RTWN_UNLOCK(sc);
57*7453645fSAndriy Voskoboinyk return;
58*7453645fSAndriy Voskoboinyk }
59*7453645fSAndriy Voskoboinyk
60*7453645fSAndriy Voskoboinyk RTWN_CMDQ_LOCK(sc);
61*7453645fSAndriy Voskoboinyk while (sc->cmdq[sc->cmdq_first].func != NULL) {
62*7453645fSAndriy Voskoboinyk item = &sc->cmdq[sc->cmdq_first];
63*7453645fSAndriy Voskoboinyk sc->cmdq_first = (sc->cmdq_first + 1) % RTWN_CMDQ_SIZE;
64*7453645fSAndriy Voskoboinyk RTWN_CMDQ_UNLOCK(sc);
65*7453645fSAndriy Voskoboinyk
66*7453645fSAndriy Voskoboinyk item->func(sc, &item->data);
67*7453645fSAndriy Voskoboinyk
68*7453645fSAndriy Voskoboinyk RTWN_CMDQ_LOCK(sc);
69*7453645fSAndriy Voskoboinyk memset(item, 0, sizeof (*item));
70*7453645fSAndriy Voskoboinyk }
71*7453645fSAndriy Voskoboinyk RTWN_CMDQ_UNLOCK(sc);
72*7453645fSAndriy Voskoboinyk RTWN_UNLOCK(sc);
73*7453645fSAndriy Voskoboinyk }
74*7453645fSAndriy Voskoboinyk
75*7453645fSAndriy Voskoboinyk void
rtwn_cmdq_init(struct rtwn_softc * sc)76*7453645fSAndriy Voskoboinyk rtwn_cmdq_init(struct rtwn_softc *sc)
77*7453645fSAndriy Voskoboinyk {
78*7453645fSAndriy Voskoboinyk RTWN_CMDQ_LOCK_INIT(sc);
79*7453645fSAndriy Voskoboinyk TASK_INIT(&sc->cmdq_task, 0, rtwn_cmdq_cb, sc);
80*7453645fSAndriy Voskoboinyk }
81*7453645fSAndriy Voskoboinyk
82*7453645fSAndriy Voskoboinyk void
rtwn_cmdq_destroy(struct rtwn_softc * sc)83*7453645fSAndriy Voskoboinyk rtwn_cmdq_destroy(struct rtwn_softc *sc)
84*7453645fSAndriy Voskoboinyk {
85*7453645fSAndriy Voskoboinyk if (RTWN_CMDQ_LOCK_INITIALIZED(sc))
86*7453645fSAndriy Voskoboinyk RTWN_CMDQ_LOCK_DESTROY(sc);
87*7453645fSAndriy Voskoboinyk }
88*7453645fSAndriy Voskoboinyk
89*7453645fSAndriy Voskoboinyk int
rtwn_cmd_sleepable(struct rtwn_softc * sc,const void * ptr,size_t len,CMD_FUNC_PROTO)90*7453645fSAndriy Voskoboinyk rtwn_cmd_sleepable(struct rtwn_softc *sc, const void *ptr, size_t len,
91*7453645fSAndriy Voskoboinyk CMD_FUNC_PROTO)
92*7453645fSAndriy Voskoboinyk {
93*7453645fSAndriy Voskoboinyk struct ieee80211com *ic = &sc->sc_ic;
94*7453645fSAndriy Voskoboinyk
95*7453645fSAndriy Voskoboinyk KASSERT(len <= sizeof(union sec_param), ("buffer overflow"));
96*7453645fSAndriy Voskoboinyk
97*7453645fSAndriy Voskoboinyk RTWN_CMDQ_LOCK(sc);
98*7453645fSAndriy Voskoboinyk if (sc->sc_detached) {
99*7453645fSAndriy Voskoboinyk RTWN_CMDQ_UNLOCK(sc);
100*7453645fSAndriy Voskoboinyk return (ESHUTDOWN);
101*7453645fSAndriy Voskoboinyk }
102*7453645fSAndriy Voskoboinyk
103*7453645fSAndriy Voskoboinyk if (sc->cmdq[sc->cmdq_last].func != NULL) {
104*7453645fSAndriy Voskoboinyk device_printf(sc->sc_dev, "%s: cmdq overflow\n", __func__);
105*7453645fSAndriy Voskoboinyk RTWN_CMDQ_UNLOCK(sc);
106*7453645fSAndriy Voskoboinyk
107*7453645fSAndriy Voskoboinyk return (EAGAIN);
108*7453645fSAndriy Voskoboinyk }
109*7453645fSAndriy Voskoboinyk
110*7453645fSAndriy Voskoboinyk if (ptr != NULL)
111*7453645fSAndriy Voskoboinyk memcpy(&sc->cmdq[sc->cmdq_last].data, ptr, len);
112*7453645fSAndriy Voskoboinyk sc->cmdq[sc->cmdq_last].func = func;
113*7453645fSAndriy Voskoboinyk sc->cmdq_last = (sc->cmdq_last + 1) % RTWN_CMDQ_SIZE;
114*7453645fSAndriy Voskoboinyk RTWN_CMDQ_UNLOCK(sc);
115*7453645fSAndriy Voskoboinyk
116*7453645fSAndriy Voskoboinyk ieee80211_runtask(ic, &sc->cmdq_task);
117*7453645fSAndriy Voskoboinyk
118*7453645fSAndriy Voskoboinyk return (0);
119*7453645fSAndriy Voskoboinyk }
120