xref: /freebsd/sys/dev/rtwn/if_rtwn_task.c (revision 685dc743dc3b5645e34836464128e1c0558b404b)
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