📄 ipencap.cc
字号:
/* * ipencap.{cc,hh} -- element encapsulates packet in IP header * Robert Morris, Eddie Kohler, Alex Snoeren * * Copyright (c) 1999-2000 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 "ipencap.hh"#include <click/nameinfo.hh>#include <click/confparse.hh>#include <click/error.hh>#include <click/glue.hh>CLICK_DECLSIPEncap::IPEncap(){}IPEncap::~IPEncap(){}intIPEncap::configure(Vector<String> &conf, ErrorHandler *errh){ click_ip iph; memset(&iph, 0, sizeof(click_ip)); iph.ip_v = 4; iph.ip_hl = sizeof(click_ip) >> 2; iph.ip_ttl = 250; int proto, tos = -1, dscp = -1; bool ce = false, df = false; String ect_str, dst_str; if (cp_va_kparse(conf, this, errh, "PROTO", cpkP+cpkM, cpNamedInteger, NameInfo::T_IP_PROTO, &proto, "SRC", cpkP+cpkM, cpIPAddress, &iph.ip_src, "DST", cpkP+cpkM, cpArgument, &dst_str, "TOS", 0, cpUnsigned, &tos, "TTL", 0, cpByte, &iph.ip_ttl, "DSCP", 0, cpUnsigned, &dscp, "ECT", 0, cpKeyword, &ect_str, "CE", 0, cpBool, &ce, "DF", 0, cpBool, &df, cpEnd) < 0) return -1; if (proto < 0 || proto > 255) return errh->error("bad IP protocol"); iph.ip_p = proto; bool use_dst_anno = dst_str == "DST_ANNO"; if (use_dst_anno) iph.ip_dst.s_addr = 0; else if (!cp_ip_address(dst_str, &iph.ip_dst, this)) return errh->error("DST argument should be IP address or 'DST_ANNO'"); int ect = 0; if (ect_str) { bool x; if (cp_bool(ect_str, &x)) ect = x; else if (ect_str == "2") ect = 2; else return errh->error("bad ECT value '%s'", ect_str.c_str()); } if (tos >= 0 && dscp >= 0) return errh->error("cannot set both TOS and DSCP"); else if (tos >= 0 && (ect || ce)) return errh->error("cannot set both TOS and ECN bits"); if (tos >= 256 || tos < -1) return errh->error("TOS too large; max 255"); else if (dscp >= 63 || dscp < -1) return errh->error("DSCP too large; max 63"); if (ect && ce) return errh->error("can set at most one ECN option"); if (tos >= 0) iph.ip_tos = tos; else if (dscp >= 0) iph.ip_tos = (dscp << 2); if (ect) iph.ip_tos |= (ect == 1 ? IP_ECN_ECT1 : IP_ECN_ECT2); if (ce) iph.ip_tos |= IP_ECN_CE; if (df) iph.ip_off |= htons(IP_DF); _iph = iph; // set the checksum field so we can calculate the checksum incrementally#if HAVE_FAST_CHECKSUM _iph.ip_sum = ip_fast_csum((unsigned char *) &_iph, sizeof(click_ip) >> 2);#else _iph.ip_sum = click_in_cksum((unsigned char *) &_iph, sizeof(click_ip));#endif // store information about use_dst_anno in the otherwise useless // _iph.ip_len field _iph.ip_len = (use_dst_anno ? 1 : 0); return 0;}intIPEncap::initialize(ErrorHandler *){ _id = 0; return 0;}inline voidIPEncap::update_cksum(click_ip *ip, int off) const{#if HAVE_INDIFFERENT_ALIGNMENT click_update_in_cksum(&ip->ip_sum, 0, ((uint16_t *) ip)[off/2]);#else const uint8_t *u = (const uint8_t *) ip;# if CLICK_BYTE_ORDER == CLICK_BIG_ENDIAN click_update_in_cksum(&ip->ip_sum, 0, u[off]*256 + u[off+1]);# else click_update_in_cksum(&ip->ip_sum, 0, u[off] + u[off+1]*256);# endif#endif}Packet *IPEncap::simple_action(Packet *p_in){ WritablePacket *p = p_in->push(sizeof(click_ip)); if (!p) return 0; click_ip *ip = reinterpret_cast<click_ip *>(p->data()); memcpy(ip, &_iph, sizeof(click_ip)); if (ip->ip_len) { // use_dst_anno ip->ip_dst = p->dst_ip_anno(); update_cksum(ip, 16); update_cksum(ip, 18); } else p->set_dst_ip_anno(IPAddress(ip->ip_dst)); ip->ip_len = htons(p->length()); ip->ip_id = htons(_id.fetch_and_add(1)); update_cksum(ip, 2); update_cksum(ip, 4); p->set_ip_header(ip, sizeof(click_ip)); return p;}StringIPEncap::read_handler(Element *e, void *thunk){ IPEncap *ipe = static_cast<IPEncap *>(e); switch ((intptr_t)thunk) { case 0: return IPAddress(ipe->_iph.ip_src).unparse(); case 1: if (ipe->_iph.ip_len == 1) return "DST_ANNO"; else return IPAddress(ipe->_iph.ip_dst).unparse(); default: return "<error>"; }}voidIPEncap::add_handlers(){ add_read_handler("src", read_handler, 0, Handler::CALM); add_write_handler("src", reconfigure_keyword_handler, "1 SRC"); add_read_handler("dst", read_handler, 1, Handler::CALM); add_write_handler("dst", reconfigure_keyword_handler, "2 DST");}CLICK_ENDDECLSEXPORT_ELEMENT(IPEncap)ELEMENT_MT_SAFE(IPEncap)
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -