⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 proberesponder.cc

📁 Click is a modular router toolkit. To use it you ll need to know how to compile and install the sof
💻 CC
字号:
/* * proberesponder.{cc,hh} -- sends 802.11 probe responses from requests. * John Bicket * * Copyright (c) 2004 Massachusetts Institute of Technology * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, subject to the conditions * listed in the Click LICENSE file. These conditions include: you must * preserve this copyright notice, and you cannot mention the copyright * holders in advertising related to the Software without their permission. * The Software is provided WITHOUT ANY WARRANTY, EXPRESS OR IMPLIED. This * notice is a summary of the Click LICENSE file; the license in that file is * legally binding. */#include <click/config.h>#include <clicknet/wifi.h>#include <click/etheraddress.hh>#include <click/confparse.hh>#include <click/error.hh>#include <click/glue.hh>#include <clicknet/llc.h>#include <click/straccum.hh>#include <click/vector.hh>#include <click/hashmap.hh>#include <click/packet_anno.hh>#include <click/error.hh>#include "proberesponder.hh"#include <elements/wifi/availablerates.hh>CLICK_DECLS#define min(x,y)      ((x)<(y) ? (x) : (y))#define max(x,y)      ((x)>(y) ? (x) : (y))ProbeResponder::ProbeResponder()  : _rtable(0){}ProbeResponder::~ProbeResponder(){}intProbeResponder::configure(Vector<String> &conf, ErrorHandler *errh){  _debug = false;  _channel = 0;  _ssid = String();  _interval_ms = 0;  if (cp_va_kparse(conf, this, errh,		   "DEBUG", 0, cpBool, &_debug,		   "CHANNEL", 0, cpInteger, &_channel,		   "SSID", 0, cpString, &_ssid,		   "BSSID", 0, cpEthernetAddress, &_bssid,		   "INTERVAL", 0, cpInteger, &_interval_ms,		   "RT", 0, cpElement, &_rtable,		   cpEnd) < 0)    return -1;  if (!_rtable || _rtable->cast("AvailableRates") == 0)    return errh->error("AvailableRates element is not provided or not a AvailableRates");  if (_interval_ms <= 0) {    return errh->error("INTERVAL must be >0\n");  }  return 0;}voidProbeResponder::push(int, Packet *p){  uint8_t dir;  uint8_t type;  uint8_t subtype;  if (p->length() < sizeof(struct click_wifi)) {    click_chatter("%{element}: packet too small: %d vs %d\n",		  this,		  p->length(),		  sizeof(struct click_wifi));    p->kill();    return;  }  struct click_wifi *w = (struct click_wifi *) p->data();  dir = w->i_fc[1] & WIFI_FC1_DIR_MASK;  type = w->i_fc[0] & WIFI_FC0_TYPE_MASK;  subtype = w->i_fc[0] & WIFI_FC0_SUBTYPE_MASK;  if (type != WIFI_FC0_TYPE_MGT) {    click_chatter("%{element}: received non-management packet\n",		  this);    p->kill();    return;  }  if (subtype != WIFI_FC0_SUBTYPE_PROBE_REQ) {    click_chatter("%{element}: received non-probe-req packet\n",		  this);    p->kill();    return;  }  uint8_t *ptr;  ptr = (uint8_t *) p->data() + sizeof(struct click_wifi);  uint8_t *end  = (uint8_t *) p->data() + p->length();  uint8_t *ssid_l = NULL;  uint8_t *rates_l = NULL;  while (ptr < end) {    switch (*ptr) {    case WIFI_ELEMID_SSID:      ssid_l = ptr;      break;    case WIFI_ELEMID_RATES:      rates_l = ptr;      break;    default:      if (_debug) {	click_chatter("%{element}: ignored element id %u %u \n",		      this,		      *ptr,		      ptr[1]);      }    }    ptr += ptr[1] + 2;  }  StringAccum sa;  String ssid = "";  if (ssid_l && ssid_l[1]) {    ssid = String((char *) ssid_l + 2, min((int)ssid_l[1], WIFI_NWID_MAXSIZE));  }  /* respond to blank ssid probes also */  if (ssid != "" && ssid != _ssid) {    if (_debug) {      click_chatter("%{element}: other ssid %s wanted %s\n",		    this,		    ssid.c_str(),		    _ssid.c_str());    }    p->kill();    return;  }  EtherAddress src = EtherAddress(w->i_addr2);  sa << "ProbeReq: " << src << " ssid " << ssid << " ";  sa << "rates {";  if (rates_l) {    for (int x = 0; x < min((int)rates_l[1], WIFI_RATES_MAXSIZE); x++) {      uint8_t rate = rates_l[x + 2];      if (rate & WIFI_RATE_BASIC) {	sa << " * " << (int) (rate ^ WIFI_RATE_BASIC);      } else {	sa << " " << (int) rate;      }    }  }  sa << " }";  if (_debug) {    click_chatter("%{element}: %s\n",		  this,		  sa.take_string().c_str());  }  send_probe_response(src);  p->kill();  return;}voidProbeResponder::send_probe_response(EtherAddress dst){  Vector<int> rates = _rtable->lookup(_bssid);  int len = sizeof (struct click_wifi) +    8 +                  /* timestamp */    2 +                  /* beacon interval */    2 +                  /* cap_info */    2 + _ssid.length() + /* ssid */    2 + min(WIFI_RATES_MAXSIZE, rates.size()) +  /* rates */    2 + 1 +              /* ds parms */    2 + 4 +              /* tim */    0;  WritablePacket *p = Packet::make(len);  if (p == 0)    return;  struct click_wifi *w = (struct click_wifi *) p->data();  w->i_fc[0] = WIFI_FC0_VERSION_0 | WIFI_FC0_TYPE_MGT | WIFI_FC0_SUBTYPE_PROBE_RESP;  w->i_fc[1] = WIFI_FC1_DIR_NODS;  memcpy(w->i_addr1, dst.data(), 6);  memcpy(w->i_addr2, _bssid.data(), 6);  memcpy(w->i_addr3, _bssid.data(), 6);  *(uint16_t *) w->i_dur = 0;  *(uint16_t *) w->i_seq = 0;  uint8_t *ptr;  ptr = (uint8_t *) p->data() + sizeof(struct click_wifi);  /* timestamp is set in the hal. ??? */  memset(ptr, 0, 8);  ptr += 8;  uint16_t beacon_int = (uint16_t) _interval_ms;  *(uint16_t *)ptr = cpu_to_le16(beacon_int);  ptr += 2;  uint16_t cap_info = 0;  cap_info |= WIFI_CAPINFO_ESS;  *(uint16_t *)ptr = cpu_to_le16(cap_info);  ptr += 2;  /* ssid */  ptr[0] = WIFI_ELEMID_SSID;  ptr[1] = _ssid.length();  memcpy(ptr + 2, _ssid.data(), _ssid.length());  ptr += 2 + _ssid.length();  /* rates */  ptr[0] = WIFI_ELEMID_RATES;  ptr[1] = min(WIFI_RATES_MAXSIZE, rates.size());  for (int x = 0; x < min (WIFI_RATES_MAXSIZE, rates.size()); x++) {    ptr[2 + x] = (uint8_t) rates[x];    if (rates[x] == 2) {      ptr [2 + x] |= WIFI_RATE_BASIC;    }  }  ptr += 2 + rates.size();  /* channel */  ptr[0] = WIFI_ELEMID_DSPARMS;  ptr[1] = 1;  ptr[2] = (uint8_t) _channel;  ptr += 2 + 1;  /* tim */  ptr[0] = WIFI_ELEMID_TIM;  ptr[1] = 4;  ptr[2] = 0; //count  ptr[3] = 1; //period  ptr[4] = 0; //bitmap control  ptr[5] = 0; //paritial virtual bitmap  ptr += 2 + 4;  output(0).push(p);}enum {H_DEBUG, H_BSSID, H_SSID, H_CHANNEL, H_INTERVAL};static StringProbeResponder_read_param(Element *e, void *thunk){  ProbeResponder *td = (ProbeResponder *)e;  switch ((uintptr_t) thunk) {  case H_DEBUG:    return String(td->_debug) + "\n";  case H_BSSID:    return td->_bssid.unparse() + "\n";  case H_SSID:    return td->_ssid + "\n";  case H_CHANNEL:    return String(td->_channel) + "\n";  case H_INTERVAL:    return String(td->_interval_ms) + "\n";  default:    return String();  }}static intProbeResponder_write_param(const String &in_s, Element *e, void *vparam,		      ErrorHandler *errh){  ProbeResponder *f = (ProbeResponder *)e;  String s = cp_uncomment(in_s);  switch((intptr_t)vparam) {  case H_DEBUG: {    //debug    bool debug;    if (!cp_bool(s, &debug))      return errh->error("debug parameter must be boolean");    f->_debug = debug;    break;  }  case H_BSSID: {    //debug    EtherAddress e;    if (!cp_ethernet_address(s, &e))      return errh->error("bssid parameter must be ethernet address");    f->_bssid = e;    break;  }  case H_SSID: {    //debug    f->_ssid = s;    break;  }  case H_CHANNEL: {    //channel    int channel;    if (!cp_integer(s, &channel))      return errh->error("channel parameter must be int");    f->_channel = channel;    break;  }  case H_INTERVAL: {    //mode    int m;    if (!cp_integer(s, &m))      return errh->error("interval parameter must be int");    f->_interval_ms = m;    break;  }  }  return 0;}voidProbeResponder::add_handlers(){  add_read_handler("debug", ProbeResponder_read_param, (void *) H_DEBUG);  add_read_handler("bssid", ProbeResponder_read_param, (void *) H_BSSID);  add_read_handler("ssid", ProbeResponder_read_param, (void *) H_SSID);  add_read_handler("channel", ProbeResponder_read_param, (void *) H_CHANNEL);  add_read_handler("interval", ProbeResponder_read_param, (void *) H_INTERVAL);  add_write_handler("debug", ProbeResponder_write_param, (void *) H_DEBUG);  add_write_handler("bssid", ProbeResponder_write_param, (void *) H_BSSID);  add_write_handler("ssid", ProbeResponder_write_param, (void *) H_SSID);  add_write_handler("channel", ProbeResponder_write_param, (void *) H_CHANNEL);  add_write_handler("interval", ProbeResponder_write_param, (void *) H_INTERVAL);}CLICK_ENDDECLSEXPORT_ELEMENT(ProbeResponder)

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -