/*
* Copyright (c) 1998 Regents of the University of California.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
* This product includes software developed by the Network Research
* Group at Lawrence Berkeley National Laboratory.
* 4. Neither the name of the University nor of the Laboratory may be used
* to endorse or promote products derived from this software without
* specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/
#ifndef lint
static const char rcsid[] =
"@(#) $Header: /nfs/jade/vint/CVSROOT/ns-2/emulate/icmp.cc,v 1.5 2000/09/01 03:04:10 haoboy Exp $";
#endif
#include <stdio.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <netinet/in_systm.h>
#include <netinet/ip.h>
#include <netinet/ip_icmp.h>
#include <netinet/ip_icmp.h>
#include <arpa/inet.h>
#include "agent.h"
#include "scheduler.h"
#include "packet.h"
#include "emulate/net.h"
#include "emulate/internet.h"
#ifndef IPPROTO_GGP
#define IPPROTO_GGP 3
#endif // IPPROTO_GGP
//
// icmp.cc -- a very limited-functionality set of icmp routines
//
class IcmpAgent : public Agent {
public:
IcmpAgent();
void recv(Packet*, Handler*) { abort(); }
protected:
void sendredirect(in_addr& me, in_addr& target, in_addr& dest, in_addr& gw);
int command(int argc, const char*const* argv);
int ttl_;
};
static class IcmpAgentClass : public TclClass {
public:
IcmpAgentClass() : TclClass("Agent/IcmpAgent") {}
TclObject* create(int , const char*const*) {
return (new IcmpAgent());
}
} class_icmpsagent;
IcmpAgent::IcmpAgent() : Agent(PT_LIVE)
{
bind("ttl_", &ttl_);
}
/*
* sendredirect -- send a packet to "target" containing a redirect
* for the network specified by "dst", so that the gateway "gw" is used
* also, forge the source address so as to appear to come from "me"
*/
void
IcmpAgent::sendredirect(in_addr& me, in_addr& target, in_addr& dst, in_addr& gw)
{
// make a simulator packet to hold the IP packet, which in turn
// holds: ip header, icmp header, embedded ip header, plus 64 bits
// data
int iplen = sizeof(ip) + 8 + sizeof(ip) + 8;
Packet* p = allocpkt(iplen);
hdr_cmn* hc = HDR_CMN(p);
ip* iph = (ip*) p->accessdata();
hc->size() = iplen;
// make an IP packet ready to send to target
// size will be min icmp + a dummy'd-up IP header
Internet::makeip(iph, iplen, ttl_, IPPROTO_ICMP, me, target);
// make an ICMP host redirect, set the gwaddr field
icmp* icp = (icmp*) (iph + 1);
icp->icmp_gwaddr = gw;
// make a dummy IP packet to go in the ICMP data, which will
// be used to indicate to the end host which routing table
// entry to update
ip* dummyhdr = (ip*)((u_char*)icp + 8); // past icmp hdr
// deprecated protocol inside
Internet::makeip(dummyhdr, 20, 254, IPPROTO_GGP, target, dst);
u_short *port = (u_short*) (dummyhdr + 1); // past ip hdr
*port++ = htons(9); // discard port
*port = htons(9); // discard port
icp->icmp_cksum = 0;
icp->icmp_type = ICMP_REDIRECT;
icp->icmp_code = ICMP_REDIRECT_HOST;
icp->icmp_cksum = Internet::in_cksum((u_short*)icp,
8 + sizeof(ip) + 8);
send(p, 0);
return;
}
int
IcmpAgent::command(int argc, const char*const* argv)
{
if (argc > 5) {
// $obj send name src dst [...stuff...]
if (strcmp(argv[1], "send") == 0) {
if (strcmp(argv[2], "redirect") == 0 &&
argc == 7) {
// $obj send redirect src target dst gwaddr
// as src, send to targ, so that it changes
// its route to dst to use gwaddr
u_long s, t, d, g;
s = inet_addr(argv[3]);
t = inet_addr(argv[4]);
d = inet_addr(argv[5]);
g = inet_addr(argv[6]);
in_addr src, targ, dst, gw;
src.s_addr = s;
targ.s_addr = t;
dst.s_addr = d;
gw.s_addr = g;
sendredirect(src, targ, dst, gw);
return (TCL_OK);
}
}
}
return (Agent::command(argc, argv));
}
syntax highlighted by Code2HTML, v. 0.9.1