nexmon – Rev 1
?pathlinks?
/***************************************************************************
* *
* ########### ########### ########## ########## *
* ############ ############ ############ ############ *
* ## ## ## ## ## ## ## *
* ## ## ## ## ## ## ## *
* ########### #### ###### ## ## ## ## ###### *
* ########### #### # ## ## ## ## # # *
* ## ## ###### ## ## ## ## # # *
* ## ## # ## ## ## ## # # *
* ############ ##### ###### ## ## ## ##### ###### *
* ########### ########### ## ## ## ########## *
* *
* S E C U R E M O B I L E N E T W O R K I N G *
* *
* This file is part of NexMon. *
* *
* Copyright (c) 2016 NexMon Team *
* *
* NexMon is free software: you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation, either version 3 of the License, or *
* (at your option) any later version. *
* *
* NexMon is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License *
* along with NexMon. If not, see <http://www.gnu.org/licenses/>. *
* *
**************************************************************************/
#pragma NEXMON targetregion "patch"
#include <firmware_version.h> // definition of firmware version macros
#include <debug.h> // contains macros to access the debug hardware
#include <wrapper.h> // wrapper definitions for functions that already exist in the firmware
#include <structs.h> // structures that are used by the code in the firmware
#include <helper.h> // useful helper functions
#include <patcher.h> // macros used to craete patches such as BLPatch, BPatch, ...
#include <rates.h> // rates used to build the ratespec for frame injection
#include <bcmwifi_channels.h>
#include <monitormode.h> // defitionons such as MONITOR_...
#define RADIOTAP_MCS
#define RADIOTAP_VENDOR
#include <ieee80211_radiotap.h>
// plcp length in bytes
#define PLCP_LEN 6
extern void prepend_ethernet_ipv4_udp_header(struct sk_buff *p);
static int
channel2freq(struct wl_info *wl, unsigned int channel)
{
int freq = 0;
void *ci = 0;
wlc_phy_chan2freq_acphy(wl->wlc->band->pi, channel, &freq, &ci);
return freq;
}
static void
wl_monitor_radiotap(struct wl_info *wl, struct wl_rxsts *sts, struct sk_buff *p)
{
struct osl_info *osh = wl->wlc->osh;
unsigned int p_len_new;
struct sk_buff *p_new;
p_len_new = p->len + sizeof(struct nexmon_radiotap_header);
// We figured out that frames larger than 2032 will not arrive in user space
if (p_len_new > 2032) {
printf("ERR: frame too large\n");
return;
} else {
p_new = pkt_buf_get_skb(osh, p_len_new);
}
if (!p_new) {
printf("ERR: no free sk_buff\n");
return;
}
struct nexmon_radiotap_header *frame = (struct nexmon_radiotap_header *) p_new->data;
memset(p_new->data, 0, sizeof(struct nexmon_radiotap_header));
frame->header.it_version = 0;
frame->header.it_pad = 0;
frame->header.it_len = sizeof(struct nexmon_radiotap_header) + PLCP_LEN;
frame->header.it_present =
(1<<IEEE80211_RADIOTAP_TSFT)
| (1<<IEEE80211_RADIOTAP_FLAGS)
| (1<<IEEE80211_RADIOTAP_RATE)
| (1<<IEEE80211_RADIOTAP_CHANNEL)
| (1<<IEEE80211_RADIOTAP_DBM_ANTSIGNAL)
| (1<<IEEE80211_RADIOTAP_DBM_ANTNOISE)
| (1<<IEEE80211_RADIOTAP_MCS)
| (1<<IEEE80211_RADIOTAP_VENDOR_NAMESPACE);
frame->tsf.tsf_l = sts->mactime;
frame->tsf.tsf_h = 0;
frame->flags = IEEE80211_RADIOTAP_F_FCS;
frame->chan_freq = channel2freq(wl, CHSPEC_CHANNEL(sts->chanspec));
if (frame->chan_freq > 3000)
frame->chan_flags |= IEEE80211_CHAN_5GHZ;
else
frame->chan_flags |= IEEE80211_CHAN_2GHZ;
if (sts->encoding == WL_RXS_ENCODING_OFDM)
frame->chan_flags |= IEEE80211_CHAN_OFDM;
if (sts->encoding == WL_RXS_ENCODING_DSSS_CCK)
frame->chan_flags |= IEEE80211_CHAN_CCK;
frame->data_rate = sts->datarate;
frame->dbm_antsignal = sts->signal;
frame->dbm_antnoise = sts->noise;
if (sts->encoding == WL_RXS_ENCODING_HT) {
frame->mcs[0] =
IEEE80211_RADIOTAP_MCS_HAVE_BW
| IEEE80211_RADIOTAP_MCS_HAVE_MCS
| IEEE80211_RADIOTAP_MCS_HAVE_GI
| IEEE80211_RADIOTAP_MCS_HAVE_FMT
| IEEE80211_RADIOTAP_MCS_HAVE_FEC
| IEEE80211_RADIOTAP_MCS_HAVE_STBC;
switch(sts->htflags) {
case WL_RXS_HTF_40:
frame->mcs[1] |= IEEE80211_RADIOTAP_MCS_BW_40;
break;
case WL_RXS_HTF_20L:
frame->mcs[1] |= IEEE80211_RADIOTAP_MCS_BW_20L;
break;
case WL_RXS_HTF_20U:
frame->mcs[1] |= IEEE80211_RADIOTAP_MCS_BW_20U;
break;
case WL_RXS_HTF_SGI:
frame->mcs[1] |= IEEE80211_RADIOTAP_MCS_SGI;
break;
case WL_RXS_HTF_STBC_MASK:
frame->mcs[1] |= ((sts->htflags & WL_RXS_HTF_STBC_MASK) >> WL_RXS_HTF_STBC_SHIFT) << IEEE80211_RADIOTAP_MCS_STBC_SHIFT;
break;
case WL_RXS_HTF_LDPC:
frame->mcs[1] |= IEEE80211_RADIOTAP_MCS_FEC_LDPC;
break;
}
frame->mcs[2] = sts->mcs;
}
frame->vendor_oui[0] = 'N';
frame->vendor_oui[1] = 'E';
frame->vendor_oui[2] = 'X';
frame->vendor_sub_namespace = 0;
frame->vendor_skip_length = PLCP_LEN;
memcpy(p_new->data + sizeof(struct nexmon_radiotap_header), p->data, p->len);
//wl_sendup(wl, 0, p_new);
wl->dev->chained->funcs->xmit(wl->dev, wl->dev->chained, p_new);
}
void
wl_monitor_hook(struct wl_info *wl, struct wl_rxsts *sts, struct sk_buff *p) {
unsigned char monitor = wl->wlc->monitor & 0xFF;
if (monitor & MONITOR_RADIOTAP) {
wl_monitor_radiotap(wl, sts, p);
}
if (monitor & MONITOR_IEEE80211) {
wl_monitor(wl, sts, p);
}
if (monitor & MONITOR_LOG_ONLY) {
printf("frame received\n");
}
if (monitor & MONITOR_DROP_FRM) {
;
}
if (monitor & MONITOR_IPV4_UDP) {
printf("MONITOR over udp is not supported!\n");
}
}
// Hook the call to wl_monitor in wlc_monitor
__attribute__((at(0x209570, "", CHIP_VER_BCM43455, FW_VER_7_120_7_1_sta_C0)))
BLPatch(wl_monitor_hook, wl_monitor_hook);
// activate badfcs, if MONITOR_ACTIVATE_BADFCS is set
void
wlc_mctrl_hook(struct wlc_info *wlc, uint32 mask, uint32 val)
{
if (wlc->monitor & MONITOR_ACTIVATE_BADFCS)
wlc_mctrl(wlc, MCTL_PROMISC | MCTL_KEEPBADFCS | MCTL_KEEPCONTROL, MCTL_PROMISC | MCTL_KEEPBADFCS | MCTL_KEEPCONTROL);
else
wlc_mctrl(wlc, mask, val);
}
__attribute__((at(0x2FB9A, "flashpatch", CHIP_VER_BCM43455, FW_VER_ALL)))
BLPatch(wlc_mctrl_hook, wlc_mctrl_hook);