Minix2.0/src/inet/generic/icmp.c

/*
icmp.c
*/

#include "inet.h"
#include "buf.h"
#include "type.h"

#include "assert.h"
#include "clock.h"
#include "icmp.h"
#include "icmp_lib.h"
#include "io.h"
#include "ip.h"
#include "ip_int.h"
#include "ipr.h"

INIT_PANIC();

typedef struct icmp_port
{
	int icp_flags;
	int icp_state;
	int icp_ipport;
	int icp_ipfd;
	acc_t *icp_head_queue;
	acc_t *icp_tail_queue;
	acc_t *icp_write_pack;
} icmp_port_t;

#define ICPF_EMPTY	0x0
#define ICPF_SUSPEND	0x1
#define ICPF_READ_IP	0x2
#define ICPF_READ_SP	0x4
#define ICPF_WRITE_IP	0x8
#define ICPF_WRITE_SP	0x10

#define ICPS_BEGIN	0
#define ICPS_IPOPT	1
#define ICPS_MAIN	2
#define ICPS_ERROR	3

#define ICMP_PORT_NR	IP_PORT_NR

PRIVATE  icmp_port_t icmp_port_table[ICMP_PORT_NR];

FORWARD void icmp_main ARGS(( icmp_port_t *icmp_port ));
FORWARD acc_t *icmp_getdata ARGS(( int port, size_t offset,
	size_t count, int for_ioctl ));
FORWARD int icmp_putdata ARGS(( int port, size_t offset,
	acc_t *data, int for_ioctl ));
FORWARD void icmp_read ARGS(( icmp_port_t *icmp_port ));
FORWARD void process_data ARGS(( icmp_port_t *icmp_port,
	acc_t *data ));
FORWARD u16_t icmp_pack_oneCsum ARGS(( acc_t *ip_pack ));
FORWARD void icmp_echo_request ARGS(( icmp_port_t *icmp_port,
	acc_t *ip_pack, int ip_hdr_len, ip_hdr_t *ip_hdr,
	acc_t *icmp_pack, int icmp_len, icmp_hdr_t *icmp_hdr ));
FORWARD void icmp_dst_unreach ARGS(( icmp_port_t *icmp_port,
	acc_t *ip_pack, int ip_hdr_len, ip_hdr_t *ip_hdr,
	acc_t *icmp_pack, int icmp_len, icmp_hdr_t *icmp_hdr ));
FORWARD void icmp_time_exceeded ARGS(( icmp_port_t *icmp_port,
	acc_t *ip_pack, int ip_hdr_len, ip_hdr_t *ip_hdr,
	acc_t *icmp_pack, int icmp_len, icmp_hdr_t *icmp_hdr ));
FORWARD void icmp_router_advertisement ARGS(( icmp_port_t *icmp_port,
	acc_t *icmp_pack, int icmp_len, icmp_hdr_t *icmp_hdr ));
FORWARD void icmp_redirect ARGS(( icmp_port_t *icmp_port,
	ip_hdr_t *ip_hdr, acc_t *icmp_pack, int icmp_len,
	icmp_hdr_t *icmp_hdr ));
FORWARD acc_t *make_repl_ip ARGS(( ip_hdr_t *ip_hdr,
	int ip_len ));
FORWARD void enqueue_pack ARGS(( icmp_port_t *icmp_port,
	acc_t *reply_ip_hdr ));
FORWARD void icmp_write ARGS(( icmp_port_t *icmp_port ));
FORWARD void icmp_buffree ARGS(( int priority, size_t reqsize ));

PUBLIC void icmp_init()
{
	int i;
	icmp_port_t *icmp_port;

	assert (BUF_S >= sizeof (nwio_ipopt_t));

	for (i= 0, icmp_port= icmp_port_table; i<ICMP_PORT_NR; i++,
		icmp_port++)
	{
		icmp_port->icp_flags= ICPF_EMPTY;
		icmp_port->icp_state= ICPS_BEGIN;
		icmp_port->icp_ipport= i;
	}

	bf_logon(icmp_buffree);

	for (i= 0, icmp_port= icmp_port_table; i<ICMP_PORT_NR; i++,
		icmp_port++)
	{
		icmp_main (icmp_port);
	}
}

PRIVATE void icmp_main(icmp_port)
icmp_port_t *icmp_port;
{
	int result;
	switch (icmp_port->icp_state)
	{
	case ICPS_BEGIN:
		icmp_port->icp_head_queue= 0;
		icmp_port->icp_ipfd= ip_open (icmp_port->icp_ipport,
			icmp_port-icmp_port_table, icmp_getdata,
			icmp_putdata);
		if (icmp_port->icp_ipfd<0)
		{
			where();
			printf("unable to open ip_port %d\n",
				icmp_port->icp_ipport);
			break;
		}
		icmp_port->icp_state= ICPS_IPOPT;
		icmp_port->icp_flags &= ~ICPF_SUSPEND;
		result= ip_ioctl (icmp_port->icp_ipfd, NWIOSIPOPT);
		if (result == NW_SUSPEND)
		{
			icmp_port->icp_flags |= ICPF_SUSPEND;
			break;
		}
		else if (result<0)
		{
			where();
			printf("ip_ioctl (.., NWIOSIPOPT)= %d\n",
				result);
			break;
		}
		/* falls through */
	case ICPS_IPOPT:
		icmp_port->icp_state= ICPS_MAIN;
		icmp_port->icp_flags &= ~ICPF_SUSPEND;
		icmp_read(icmp_port);
		break;
	default:
		where();
		printf("unknown state %d\n", icmp_port->icp_state);
		break;
	}
}

PRIVATE acc_t *icmp_getdata(port, offset, count, for_ioctl)
int port;
size_t offset, count;
int for_ioctl;
{
	icmp_port_t *icmp_port;
	nwio_ipopt_t *ipopt;
	acc_t *data;
	int result;

	icmp_port= &icmp_port_table[port];

	if (icmp_port->icp_flags & ICPF_WRITE_IP)
	{
		if (!count)
		{
			bf_afree(icmp_port->icp_write_pack);
			icmp_port->icp_write_pack= 0;

			result= (int)offset;
			if (result<0)
			{
				where();
				printf("got write error %d\n", result);
			}
#if DEBUG & 256
 { where(); printf("ip_write completed\n"); }
#endif
			if (icmp_port->icp_flags & ICPF_WRITE_SP)
			{
				icmp_port->icp_flags &=
					~(ICPF_WRITE_IP|ICPF_WRITE_SP);
				icmp_write (icmp_port);
			}
			return NW_OK;
		}
		return bf_cut(icmp_port->icp_write_pack, offset, count);
	}
	switch (icmp_port->icp_state)
	{
	case ICPS_IPOPT:
		if (!count)
		{
			result= (int)offset;
			if (result < 0)
			{
				icmp_port->icp_state= ICPS_ERROR;
				break;
			}
			if (icmp_port->icp_flags & ICPF_SUSPEND)
				icmp_main(icmp_port);
			return NW_OK;
		}

assert (count == sizeof (*ipopt));
		data= bf_memreq (sizeof (*ipopt));
assert (data->acc_length == sizeof(*ipopt));
		ipopt= (nwio_ipopt_t *)ptr2acc_data(data);
		ipopt->nwio_flags= NWIO_COPY | NWIO_EN_LOC |
			NWIO_EN_BROAD | NWIO_REMANY | NWIO_PROTOSPEC |
			NWIO_HDR_O_ANY | NWIO_RWDATALL;
		ipopt->nwio_proto= IPPROTO_ICMP;
		return data;
	default:
		where();
		printf("unknown state %d\n", icmp_port->icp_state);
		return 0;
	}
}

PRIVATE int icmp_putdata(port, offset, data, for_ioctl)
int port;
size_t offset;
acc_t *data;
int for_ioctl;
{
	icmp_port_t *icmp_port;
	int result;

	icmp_port= &icmp_port_table[port];

	if (icmp_port->icp_flags & ICPF_READ_IP)
	{
assert (!for_ioctl);
		if (!data)
		{
			result= (int)offset;
			if (result<0)
			{
				where();
				printf("got read error %d\n", result);
			}
			if (icmp_port->icp_flags & ICPF_READ_SP)
			{
				icmp_port->icp_flags &=
					~(ICPF_READ_IP|ICPF_READ_SP);
				icmp_read (icmp_port);
			}
			return NW_OK;
		}
		process_data(icmp_port, data);
		return NW_OK;
	}
	switch (icmp_port->icp_state)
	{
	default:
		where();
		printf("unknown state %d\n", icmp_port->icp_state);
		return 0;
	}
}

PRIVATE void icmp_read(icmp_port)
icmp_port_t *icmp_port;
{
	int result;

assert (!(icmp_port->icp_flags & (ICPF_READ_IP|ICPF_READ_SP) || 
	(icmp_port->icp_flags & (ICPF_READ_IP|ICPF_READ_SP)) ==
	(ICPF_READ_IP|ICPF_READ_SP)));

	for (;;)
	{
		icmp_port->icp_flags |= ICPF_READ_IP;
		icmp_port->icp_flags &= ~ICPF_READ_SP;

		result= ip_read(icmp_port->icp_ipfd, ICMP_MAX_DATAGRAM);
		if (result == NW_SUSPEND)
		{
			icmp_port->icp_flags |= ICPF_READ_SP;
			return;
		}
	}
}

PUBLIC void icmp_frag_ass_tim(pack)
acc_t *pack;
{
	ip_warning(( "icmp_frag_ass() called" ));
	bf_afree(pack);
}

PUBLIC void icmp_getnetmask(ip_port)
int ip_port;
{
	ip_port_t *port;

#if DEBUG & 256
 { where(); printf("icmp.c: icmp_getnetmask(ip_port= %d)\n", ip_port); }
#endif
	port= &ip_port_table[ip_port];
#if DEBUG & 2
 { where(); printf ("icmp_getnetmask() NOT implemented\n"); }
#endif

	port->ip_netmask= HTONL(0xffffff00L);
	port->ip_flags |= IPF_NETMASKSET;
#if DEBUG & 256
 { where(); printf("icmp.c: setting netmask to "); 
   writeIpAddr(port->ip_netmask); printf("\n"); }
#endif
}

PUBLIC void icmp_dont_frag(pack)
acc_t *pack;
{
printf ("icmp_dont_frag() called\n");
	bf_afree(pack);
}

PUBLIC void icmp_ttl_exceded(pack)
acc_t *pack;
{
printf ("icmp_ttl_execeded() called\n");
	bf_afree(pack);
}

PRIVATE void process_data(icmp_port, data)
icmp_port_t *icmp_port;
acc_t *data;
{
	ip_hdr_t *ip_hdr;
	icmp_hdr_t *icmp_hdr;
	acc_t *icmp_data;
	int ip_hdr_len;
	size_t pack_len;

#if DEBUG & 256
 { where(); printf("got an icmp packet\n"); }
#endif
	data= bf_packIffLess(data, IP_MIN_HDR_SIZE);
	ip_hdr= (ip_hdr_t *)ptr2acc_data(data);
	ip_hdr_len= (ip_hdr->ih_vers_ihl & IH_IHL_MASK) << 2;

	if (ip_hdr_len>IP_MIN_HDR_SIZE)
	{
		data= bf_packIffLess(data, ip_hdr_len);
		ip_hdr= (ip_hdr_t *)ptr2acc_data(data);
	}

	pack_len= bf_bufsize(data);
	pack_len -= ip_hdr_len;
	if (pack_len < ICMP_MIN_HDR_LEN)
	{
#if DEBUG
 { where(); printf("got an incomplete icmp packet\n"); }
#endif
		bf_afree(data);
		return;
	}

	icmp_data= bf_cut(data, ip_hdr_len, pack_len);

	icmp_data= bf_packIffLess (icmp_data, ICMP_MIN_HDR_LEN);
	icmp_hdr= (icmp_hdr_t *)ptr2acc_data(icmp_data);


	if ((u16_t)~icmp_pack_oneCsum(icmp_data))
	{
#if DEBUG
 { where(); printf("got packet with bad checksum (= 0x%x)\n",
	(u16_t)~icmp_pack_oneCsum(icmp_data)); }
#endif
		bf_afree(data);
		bf_afree(icmp_data);
		return;
	}
	switch (icmp_hdr->ih_type)
	{
	case ICMP_TYPE_ECHO_REPL:
#if DEBUG
 { where(); printf("got an icmp echo reply\n"); }
#endif
		break;
	case ICMP_TYPE_DST_UNRCH:
		icmp_dst_unreach (icmp_port, data, ip_hdr_len, ip_hdr,
			icmp_data, pack_len, icmp_hdr);
		break;
	case ICMP_TYPE_REDIRECT:
		icmp_redirect (icmp_port, ip_hdr, icmp_data, pack_len,
			icmp_hdr);
		break;
	case ICMP_TYPE_ECHO_REQ:
		icmp_echo_request(icmp_port, data, ip_hdr_len, ip_hdr,
			icmp_data, pack_len, icmp_hdr);
		return;
	case ICMP_TYPE_ROUTER_ADVER:
		icmp_router_advertisement(icmp_port, icmp_data, pack_len, 
			icmp_hdr);
		break;
	case ICMP_TYPE_TIME_EXCEEDED:
		icmp_time_exceeded (icmp_port, data, ip_hdr_len, ip_hdr,
			icmp_data, pack_len, icmp_hdr);
		break;
	default:
#if DEBUG
 { where(); printf("got an unknown icmp (%d) from ", icmp_hdr->ih_type); 
	writeIpAddr(ip_hdr->ih_src); printf("\n"); }
#endif
		break;
	}
	bf_afree(data);
	bf_afree(icmp_data);
}

PRIVATE void icmp_echo_request(icmp_port, ip_data, ip_len, ip_hdr,
	icmp_data, icmp_len, icmp_hdr)
icmp_port_t *icmp_port;
acc_t *ip_data, *icmp_data;
int ip_len, icmp_len;
ip_hdr_t *ip_hdr;
icmp_hdr_t *icmp_hdr;
{
	acc_t *repl_ip_hdr, *repl_icmp;
	icmp_hdr_t *repl_icmp_hdr;
	u32_t tmp_chksum;
	u16_t u16;

	if (icmp_hdr->ih_code != 0)
	{
#if DEBUG
 { where(); printf("got an icmp echo request with unknown code (%d)\n",
		icmp_hdr->ih_code); }
#endif
		bf_afree(ip_data);
		bf_afree(icmp_data);
		return;
	}
	if (icmp_len < ICMP_MIN_HDR_LEN + sizeof(icmp_id_seq_t))
	{
#if DEBUG
 { where(); printf("got an incomplete icmp echo request\n"); }
#endif
		bf_afree(ip_data);
		bf_afree(icmp_data);
		return;
	}
#if DEBUG & 256
 { where(); printf("got an icmp echo request, ident= %u, seq= %u\n",
	ntohs(icmp_hdr->ih_hun.ihh_idseq.iis_id),
	ntohs(icmp_hdr->ih_hun.ihh_idseq.iis_seq)); }
#endif
	repl_ip_hdr= make_repl_ip(ip_hdr, ip_len);
	repl_icmp= bf_memreq (ICMP_MIN_HDR_LEN);
assert (repl_icmp->acc_length == ICMP_MIN_HDR_LEN);
	repl_icmp_hdr= (icmp_hdr_t *)ptr2acc_data(repl_icmp);
	repl_icmp_hdr->ih_type= ICMP_TYPE_ECHO_REPL;
	repl_icmp_hdr->ih_code= 0;

	tmp_chksum= ~icmp_hdr->ih_chksum - *(u16_t *)&icmp_hdr->ih_type+
		*(u16_t *)&repl_icmp_hdr->ih_type;
	tmp_chksum= (tmp_chksum >> 16) + (tmp_chksum & 0xffff);
	tmp_chksum= (tmp_chksum >> 16) + (tmp_chksum & 0xffff);
	repl_icmp_hdr->ih_chksum= ~tmp_chksum;

	repl_ip_hdr->acc_next= repl_icmp;
	repl_icmp->acc_next= bf_cut (icmp_data, ICMP_MIN_HDR_LEN,
		icmp_len - ICMP_MIN_HDR_LEN);

	bf_afree(ip_data);
	bf_afree(icmp_data);

	enqueue_pack(icmp_port, repl_ip_hdr);
}

PRIVATE u16_t icmp_pack_oneCsum(icmp_pack)
acc_t *icmp_pack;
{
	u16_t prev;
	int odd_byte;
	char *data_ptr;
	int length;
	char byte_buf[2];

	assert (icmp_pack);

	prev= 0;

	odd_byte= FALSE;
	for (; icmp_pack; icmp_pack= icmp_pack->acc_next)
	{
		
		data_ptr= ptr2acc_data(icmp_pack);
		length= icmp_pack->acc_length;

		if (!length)
			continue;
		if (odd_byte)
		{
			byte_buf[1]= *data_ptr;
			prev= oneC_sum(prev, (u16_t *)byte_buf, 2);
			data_ptr++;
			length--;
			odd_byte= FALSE;
		}
		if (length & 1)
		{
			odd_byte= TRUE;
			length--;
			byte_buf[0]= data_ptr[length];
		}
		if (!length)
			continue;
		prev= oneC_sum (prev, (u16_t *)data_ptr, length);
	}
	if (odd_byte)
		prev= oneC_sum (prev, (u16_t *)byte_buf, 1);
	return prev;
}

PRIVATE acc_t *make_repl_ip(ip_hdr, ip_len)
ip_hdr_t *ip_hdr;
int ip_len;
{
	ip_hdr_t *repl_ip_hdr;
	acc_t *repl;
	int repl_hdr_len;

	if (ip_len>IP_MIN_HDR_SIZE)
	{
#if DEBUG
 { where(); printf("ip_hdr options NOT supported (yet?)\n"); }
#endif
		ip_len= IP_MIN_HDR_SIZE;
	}

	repl_hdr_len= IP_MIN_HDR_SIZE;

	repl= bf_memreq(repl_hdr_len);
assert (repl->acc_length == repl_hdr_len);

	repl_ip_hdr= (ip_hdr_t *)ptr2acc_data(repl);

	repl_ip_hdr->ih_vers_ihl= repl_hdr_len >> 2;
	repl_ip_hdr->ih_tos= ip_hdr->ih_tos;
	repl_ip_hdr->ih_ttl= ICMP_DEF_TTL;
	repl_ip_hdr->ih_proto= IPPROTO_ICMP;
	repl_ip_hdr->ih_dst= ip_hdr->ih_src;
	repl_ip_hdr->ih_flags_fragoff= 0;

	return repl;
}

PRIVATE void enqueue_pack(icmp_port, reply_ip_hdr)
icmp_port_t *icmp_port;
acc_t *reply_ip_hdr;
{
	reply_ip_hdr->acc_ext_link= 0;

	if (icmp_port->icp_head_queue)
	{
		icmp_port->icp_tail_queue->acc_ext_link=
			reply_ip_hdr;
	}
	else
	{
		icmp_port->icp_head_queue= reply_ip_hdr;
		icmp_port->icp_tail_queue= reply_ip_hdr;
	}

	if (!(icmp_port->icp_flags & ICPF_WRITE_IP))
		icmp_write(icmp_port);
}

PRIVATE void icmp_write(icmp_port)
icmp_port_t *icmp_port;
{
	int result;

assert (!(icmp_port->icp_flags & (ICPF_WRITE_IP|ICPF_WRITE_SP) || 
	(icmp_port->icp_flags & (ICPF_WRITE_IP|ICPF_WRITE_SP)) ==
	(ICPF_WRITE_IP|ICPF_WRITE_SP)));

	for (;icmp_port->icp_head_queue;)
	{
		icmp_port->icp_write_pack= icmp_port->icp_head_queue;
		icmp_port->icp_head_queue= icmp_port->icp_head_queue->
			acc_ext_link;

		icmp_port->icp_flags |= ICPF_WRITE_IP;
		icmp_port->icp_flags &= ~ICPF_WRITE_SP;

#if DEBUG & 256
 { where(); printf("calling ip_write\n"); }
#endif
		result= ip_write(icmp_port->icp_ipfd,
			bf_bufsize(icmp_port->icp_write_pack));
		if (result == NW_SUSPEND)
		{
#if DEBUG & 256
 { where(); printf("ip_write replied NW_SUSPEND\n"); }
#endif
			icmp_port->icp_flags |= ICPF_WRITE_SP;
			return;
		}
#if DEBUG & 256
 { where(); printf("ip_write done\n"); }
#endif
	}
	icmp_port->icp_flags &= ~ICPF_WRITE_IP;
}

PRIVATE void icmp_buffree(priority, reqsize)
int priority;
size_t reqsize;
{
	acc_t *tmp_acc;
	int donesomething,i;
	icmp_port_t *icmp_port;

	donesomething= 0;

	if (priority < ICMP_PRI_QUEUE)
		return;

	while (bf_free_buffsize < reqsize)
	{
		for (i=0, icmp_port= icmp_port_table; i<ICMP_PORT_NR;
			i++, icmp_port++)
		{
			if (icmp_port->icp_head_queue)
			{
				tmp_acc= icmp_port->icp_head_queue;
				icmp_port->icp_head_queue= tmp_acc->
					acc_ext_link;
				bf_afree(tmp_acc);
				if (bf_free_buffsize >= reqsize)
					break;
				donesomething= 1;
			}
		}
		if (!donesomething)
			break;
	}
}

static void icmp_dst_unreach(icmp_port, ip_pack, ip_hdr_len, ip_hdr, icmp_pack,
	icmp_len, icmp_hdr)
icmp_port_t *icmp_port;
acc_t *ip_pack;
int ip_hdr_len;
ip_hdr_t *ip_hdr;
acc_t *icmp_pack;
int icmp_len;
icmp_hdr_t *icmp_hdr;
{
	acc_t *old_ip_pack;
	ip_hdr_t *old_ip_hdr;

	if (icmp_len < 8 + IP_MIN_HDR_SIZE)
	{
#if DEBUG
 { where(); printf("dest unrch with wrong size\n"); }
#endif
		return;
	}
	old_ip_pack= bf_cut (icmp_pack, 8, icmp_len-8);
	old_ip_pack= bf_packIffLess(old_ip_pack, IP_MIN_HDR_SIZE);
	old_ip_hdr= (ip_hdr_t *)ptr2acc_data(old_ip_pack);

	if (old_ip_hdr->ih_src != ip_hdr->ih_dst)
	{
#if DEBUG
 { where(); printf("dest unrch based on wrong packet\n"); }
#endif
		bf_afree(old_ip_pack);
		return;
	}

	switch(icmp_hdr->ih_code)
	{
	case ICMP_NET_UNRCH:
		ipr_destunrch (old_ip_hdr->ih_dst,
			ip_get_netmask(old_ip_hdr->ih_dst), IPR_UNRCH_TIMEOUT);
		break;
	case ICMP_HOST_UNRCH:
		ipr_destunrch (old_ip_hdr->ih_dst, (ipaddr_t)-1,
			IPR_UNRCH_TIMEOUT);
		break;
	default:
#if DEBUG
 { where(); printf("got strange code: %d\n", icmp_hdr->ih_code); }
#endif
		break;
	}
	bf_afree(old_ip_pack);
}

static void icmp_time_exceeded(icmp_port, ip_pack, ip_hdr_len, ip_hdr,
	icmp_pack, icmp_len, icmp_hdr)
icmp_port_t *icmp_port;
acc_t *ip_pack;
int ip_hdr_len;
ip_hdr_t *ip_hdr;
acc_t *icmp_pack;
int icmp_len;
icmp_hdr_t *icmp_hdr;
{
	acc_t *old_ip_pack;
	ip_hdr_t *old_ip_hdr;

	if (icmp_len < 8 + IP_MIN_HDR_SIZE)
	{
#if DEBUG
 { where(); printf("time exceeded with wrong size\n"); }
#endif
		return;
	}
	old_ip_pack= bf_cut (icmp_pack, 8, icmp_len-8);
	old_ip_pack= bf_packIffLess(old_ip_pack, IP_MIN_HDR_SIZE);
	old_ip_hdr= (ip_hdr_t *)ptr2acc_data(old_ip_pack);

	if (old_ip_hdr->ih_src != ip_hdr->ih_dst)
	{
#if DEBUG
 { where(); printf("time exceeded based on wrong packet\n"); }
#endif
		bf_afree(old_ip_pack);
		return;
	}

	switch(icmp_hdr->ih_code)
	{
	case ICMP_TTL_EXC:
		ipr_ttl_exc (old_ip_hdr->ih_dst, (ipaddr_t)-1,
			IPR_TTL_TIMEOUT);
		break;
	default:
		where();
		printf("got strange code: %d\n", icmp_hdr->ih_code);
		break;
	}
	bf_afree(old_ip_pack);
}

static void icmp_router_advertisement(icmp_port, icmp_pack, icmp_len, icmp_hdr)
icmp_port_t *icmp_port;
acc_t *icmp_pack;
int icmp_len;
icmp_hdr_t *icmp_hdr;
{
	int entries;
	int entry_size;
	u16_t lifetime;
	int i;
	char *bufp;

	if (icmp_len < 8)
	{
#if DEBUG
 { where(); printf("router advertisement with wrong size (%d)\n", icmp_len); }
#endif
		return;
	}
	if (icmp_hdr->ih_code != 0)
	{
#if DEBUG
 { where(); printf("router advertisement with wrong code (%d)\n", 
							icmp_hdr->ih_code); }
#endif
		return;
	}
	entries= icmp_hdr->ih_hun.ihh_ram.iram_na;
	entry_size= icmp_hdr->ih_hun.ihh_ram.iram_aes * 4;
	if (entries < 1)
	{
#if DEBUG
 { where(); printf("router advertisement with wrong number of entries (%d)\n", 
							entries); }
#endif
		return;
	}
	if (entry_size < 8)
	{
#if DEBUG
 { where(); printf("router advertisement with wrong entry size (%d)\n", 
							entry_size); }
#endif
		return;
	}
	if (icmp_len < 8 + entries * entry_size)
	{
#if DEBUG
 { where(); printf("router advertisement with wrong size\n"); 
	printf("\t(entries= %d, entry_size= %d, icmp_len= %d)\n", entries,
						entry_size, icmp_len); }
#endif
		return;
	}
	lifetime= ntohs(icmp_hdr->ih_hun.ihh_ram.iram_lt);
	if (lifetime > 9000)
	{
#if DEBUG
 { where(); printf("router advertisement with wrong lifetime (%d)\n",
								lifetime); }
#endif
		return;
	}
	for (i= 0, bufp= (char *)&icmp_hdr->ih_dun.uhd_data[0]; i< entries; i++,
		bufp += entry_size)
	{
		ipr_add_route(HTONL(0L), HTONL(0L), *(ipaddr_t *)bufp,
			icmp_port->icp_ipport, lifetime * HZ, 1, 0, 
			ntohl(*(i32_t *)(bufp+4)));
	}
}
		
static void icmp_redirect(icmp_port, ip_hdr, icmp_pack, icmp_len, icmp_hdr)
icmp_port_t *icmp_port;
ip_hdr_t *ip_hdr;
acc_t *icmp_pack;
int icmp_len;
icmp_hdr_t *icmp_hdr;
{
	acc_t *old_ip_pack;
	ip_hdr_t *old_ip_hdr;
	int port;

	if (icmp_len < 8 + IP_MIN_HDR_SIZE)
	{
#if DEBUG
 { where(); printf("redirect with wrong size\n"); }
#endif
		return;
	}
	old_ip_pack= bf_cut (icmp_pack, 8, icmp_len-8);
	old_ip_pack= bf_packIffLess(old_ip_pack, IP_MIN_HDR_SIZE);
	old_ip_hdr= (ip_hdr_t *)ptr2acc_data(old_ip_pack);

	port= icmp_port->icp_ipport;

	switch(icmp_hdr->ih_code)
	{
	case ICMP_REDIRECT_NET:
		ipr_redirect (old_ip_hdr->ih_dst,
			ip_get_netmask(old_ip_hdr->ih_dst),
			ip_hdr->ih_src, icmp_hdr->ih_hun.ihh_gateway, port, 
			IPR_REDIRECT_TIMEOUT);
		break;
	case ICMP_REDIRECT_HOST:
		ipr_redirect (old_ip_hdr->ih_dst, (ipaddr_t)-1,
			ip_hdr->ih_src, icmp_hdr->ih_hun.ihh_gateway, port, 
			IPR_REDIRECT_TIMEOUT);
		break;
	default:
		where();
		printf("got strange code: %d\n", icmp_hdr->ih_code);
		break;
	}
	bf_afree(old_ip_pack);
}