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

📄 checkipheader.cc

📁 COPE the first practical network coding scheme which is developped on click
💻 CC
字号:
/* * checkipheader.{cc,hh} -- element checks IP header for correctness * (checksums, lengths, source addresses) * Robert Morris, Eddie Kohler * * Copyright (c) 1999-2000 Massachusetts Institute of Technology * Copyright (c) 2003 International Computer Science Institute * * 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 "checkipheader.hh"#include <clicknet/ip.h>#include <click/glue.hh>#include <click/confparse.hh>#include <click/straccum.hh>#include <click/error.hh>#include <click/standard/alignmentinfo.hh>#ifdef CLICK_LINUXMODULE# include <net/checksum.h>#endifCLICK_DECLSconst char * const CheckIPHeader::reason_texts[NREASONS] = {  "tiny packet", "bad IP version", "bad IP header length",  "bad IP length", "bad IP checksum", "bad source address"};#define IPADDR_LIST_INTERFACES		((void *)0)#define IPADDR_LIST_BADSRC		((void *)1)#define IPADDR_LIST_BADSRC_OLD		((void *)2)static voidipaddr_list_parse(cp_value *v, const String &arg, ErrorHandler *errh, const char *argname, Element *context){  Vector<String> words;  cp_spacevec(arg, words);  StringAccum sa;  IPAddress addr[2];  if (v->argtype->user_data == IPADDR_LIST_INTERFACES) {    for (int i = 0; i < words.size(); i++)      if (cp_ip_prefix(words[i], &addr[0], &addr[1], true, context))	memcpy(sa.extend(8), &addr[0], 8);      else {	errh->error("%s takes list of IP prefixes (%s)", argname, v->description);	return;      }  } else {    for (int i = 0; i < words.size(); i++)      if (cp_ip_address(words[i], &addr[0], context))	memcpy(sa.extend(4), &addr[0], 4);      else {	errh->error("%s takes list of IP addresses (%s) [%s]", argname, v->description, words[i].cc());	return;      }    if (v->argtype->user_data == IPADDR_LIST_BADSRC_OLD)      memcpy(sa.extend(8), "\x00\x00\x00\x00\xFF\xFF\xFF\xFF", 8);  }  v->v_string = sa.take_string();}static voidipaddr_list_store(cp_value *v, Element *){  IPAddressList *l1 = (IPAddressList *)v->store;  const uint32_t *vec = reinterpret_cast<const uint32_t *>(v->v_string.data());  int len = v->v_string.length() / 4;  if (v->argtype->user_data == IPADDR_LIST_INTERFACES) {    uint32_t *l = new uint32_t[len/2 + 2];    for (int i = 0; i < len/2; i++)      l[i] = (vec[i*2] & vec[i*2 + 1]) | ~vec[i*2 + 1];    l[len/2] = 0x00000000U;    l[len/2+1] = 0xFFFFFFFFU;    l1->assign(len/2 + 2, l);    IPAddressList *l2 = (IPAddressList *)v->store2;    l = new uint32_t[len/2];    for (int i = 0; i < len/2; i++)      l[i] = vec[i*2];    l2->assign(len/2, l);  } else {    uint32_t *l = new uint32_t[len];    memcpy(l, vec, len * sizeof(uint32_t));    l1->assign(len, l);  }}voidCheckIPHeader::static_initialize(){  cp_register_argtype("CheckIPHeader.INTERFACES", "list of router IP addresses and prefixes", cpArgStore2, ipaddr_list_parse, ipaddr_list_store, IPADDR_LIST_INTERFACES);  cp_register_argtype("CheckIPHeader.BADSRC", "list of IP addresses", cpArgNormal, ipaddr_list_parse, ipaddr_list_store, IPADDR_LIST_BADSRC);  cp_register_argtype("CheckIPHeader.BADSRC_OLD", "list of IP addresses", cpArgNormal, ipaddr_list_parse, ipaddr_list_store, IPADDR_LIST_BADSRC_OLD);}voidCheckIPHeader::static_cleanup(){  cp_unregister_argtype("CheckIPHeader.INTERFACES");  cp_unregister_argtype("CheckIPHeader.BADSRC");  cp_unregister_argtype("CheckIPHeader.BADSRC_OLD");}CheckIPHeader::CheckIPHeader()  : Element(1, 1), _checksum(true), _reason_drops(0){  _drops = 0;}CheckIPHeader::~CheckIPHeader(){  delete[] _reason_drops;}voidCheckIPHeader::notify_noutputs(int n){  set_noutputs(n < 2 ? 1 : 2);}intCheckIPHeader::configure(Vector<String> &conf, ErrorHandler *errh){  _offset = 0;  bool verbose = false;  bool details = false;  if (cp_va_parse_remove_keywords(conf, 0, this, errh,		"INTERFACES", "CheckIPHeader.INTERFACES", "router interface addresses", &_bad_src, &_good_dst,		"BADSRC", "CheckIPHeader.BADSRC", "bad source addresses", &_bad_src,		"GOODDST", "CheckIPHeader.BADSRC", "good destination addresses", &_good_dst,		"OFFSET", cpUnsigned, "IP header offset", &_offset,		"VERBOSE", cpBool, "be verbose?", &verbose,		"DETAILS", cpBool, "keep detailed counts?", &details,		"CHECKSUM", cpBool, "check checksum?", &_checksum,		cpEnd) < 0)    return -1;  if (conf.size() == 1 && cp_unsigned(conf[0], &_offset))    /* nada */;  else if (cp_va_parse(conf, this, errh, cpOptional,		       "CheckIPHeader.BADSRC_OLD", "bad source addresses", &_bad_src,		       cpUnsigned, "IP header offset", &_offset,		       cpEnd) < 0)    return -1;  _verbose = verbose;  if (details)    _reason_drops = new atomic_uint32_t[NREASONS];#if HAVE_FAST_CHECKSUM && FAST_CHECKSUM_ALIGNED  // check alignment  if (_checksum) {    int ans, c, o;    ans = AlignmentInfo::query(this, 0, c, o);    o = (o + 4 - (_offset % 4)) % 4;    _aligned = (ans && c == 4 && o == 0);    if (!_aligned)      errh->warning("IP header unaligned, cannot use fast IP checksum");    if (!ans)      errh->message("(Try passing the configuration through `click-align'.)");  }#endif  //for (int i = 0; i < _bad_src.n; i++)  //  click_chatter("bad: %s", IPAddress(_bad_src.vec[i]).s().cc());  //for (int i = 0; i < _good_dst.n; i++)  //  click_chatter("good: %s", IPAddress(_good_dst.vec[i]).s().cc());    return 0;}Packet *CheckIPHeader::drop(Reason reason, Packet *p){  if (_drops == 0 || _verbose)    click_chatter("IP header check failed: %s", reason_texts[reason]);  _drops++;  if (_reason_drops)    _reason_drops[reason]++;    if (noutputs() == 2)    output(1).push(p);  else    p->kill();  return 0;}Packet *CheckIPHeader::simple_action(Packet *p){  const click_ip *ip = reinterpret_cast<const click_ip *>(p->data() + _offset);  unsigned plen = p->length() - _offset;  unsigned hlen, len;  // cast to int so very large plen is interpreted as negative   if ((int)plen < (int)sizeof(click_ip))    return drop(MINISCULE_PACKET, p);  if (ip->ip_v != 4)    return drop(BAD_VERSION, p);    hlen = ip->ip_hl << 2;  if (hlen < sizeof(click_ip))    return drop(BAD_HLEN, p);    len = ntohs(ip->ip_len);  if (len > plen || len < hlen)    return drop(BAD_IP_LEN, p);  if (_checksum) {    int val;#if HAVE_FAST_CHECKSUM && FAST_CHECKSUM_ALIGNED    if (_aligned)      val = ip_fast_csum((unsigned char *)ip, ip->ip_hl);    else      val = click_in_cksum((const unsigned char *)ip, hlen);#elif HAVE_FAST_CHECKSUM    val = ip_fast_csum((unsigned char *)ip, ip->ip_hl);#else    val = click_in_cksum((const unsigned char *)ip, hlen);#endif    if (val != 0)      return drop(BAD_CHECKSUM, p);  }  /*   * RFC1812 5.3.7 and 4.2.2.11: discard illegal source addresses.   * Configuration string should have listed all subnet   * broadcast addresses known to this router.   */  if (_bad_src.contains(ip->ip_src)      && !_good_dst.contains(ip->ip_dst))    return drop(BAD_SADDR, p);  /*   * RFC1812 4.2.3.1: discard illegal destinations.   * We now do this in the IP routing table.   */    p->set_ip_header(ip, hlen);  // shorten packet according to IP length field -- 7/28/2000  if (plen > len)    p->take(plen - len);  // set destination IP address annotation if it doesn't exist already --  // 9/26/2001  // always set destination IP address annotation; linuxmodule problem  // reported by Parveen Kumar Patel at Utah -- 4/3/2002  p->set_dst_ip_anno(ip->ip_dst);    return(p);}StringCheckIPHeader::read_handler(Element *e, void *thunk){  CheckIPHeader *c = reinterpret_cast<CheckIPHeader *>(e);  switch ((intptr_t)thunk) {   case 0:			// drops    return String(c->_drops) + "\n";   case 1: {			// drop_details     StringAccum sa;     for (int i = 0; i < NREASONS; i++)       sa << c->_reason_drops[i] << '\t' << reason_texts[i] << '\n';     return sa.take_string();   }   default:    return String("<error>\n");  }}voidCheckIPHeader::add_handlers(){  add_read_handler("drops", read_handler, (void *)0);  if (_reason_drops)    add_read_handler("drop_details", read_handler, (void *)1);}CLICK_ENDDECLSEXPORT_ELEMENT(CheckIPHeader)

⌨️ 快捷键说明

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