1f13f9fadSAlexander Langer /*
259a7c613SBrian Somers * natd - Network Address Translation Daemon for FreeBSD.
359a7c613SBrian Somers *
4a228025eSAlexander Langer * This software is provided free of charge, with no
5a228025eSAlexander Langer * warranty of any kind, either expressed or implied.
6a228025eSAlexander Langer * Use at your own risk.
7a228025eSAlexander Langer *
859a7c613SBrian Somers * You may copy, modify and distribute this software (icmp.c) freely.
9a228025eSAlexander Langer *
10a228025eSAlexander Langer * Ari Suutari <suutari@iki.fi>
11f13f9fadSAlexander Langer */
12f13f9fadSAlexander Langer
1324084f9bSBrian Somers #include <stdlib.h>
1424084f9bSBrian Somers #include <stdio.h>
1524084f9bSBrian Somers #include <unistd.h>
1624084f9bSBrian Somers #include <string.h>
1724084f9bSBrian Somers #include <ctype.h>
1824084f9bSBrian Somers
1924084f9bSBrian Somers #include <sys/types.h>
2024084f9bSBrian Somers #include <sys/socket.h>
2124084f9bSBrian Somers #include <sys/time.h>
2224084f9bSBrian Somers #include <errno.h>
2324084f9bSBrian Somers #include <signal.h>
2424084f9bSBrian Somers
2524084f9bSBrian Somers #include <netdb.h>
2624084f9bSBrian Somers
2724084f9bSBrian Somers #include <netinet/in.h>
2824084f9bSBrian Somers #include <netinet/in_systm.h>
2924084f9bSBrian Somers #include <netinet/ip.h>
3024084f9bSBrian Somers #include <netinet/ip_icmp.h>
3124084f9bSBrian Somers
3224084f9bSBrian Somers #include <alias.h>
3324084f9bSBrian Somers
3424084f9bSBrian Somers #include "natd.h"
3524084f9bSBrian Somers
SendNeedFragIcmp(int sock,struct ip * failedDgram,int mtu)3624084f9bSBrian Somers int SendNeedFragIcmp (int sock, struct ip* failedDgram, int mtu)
3724084f9bSBrian Somers {
3824084f9bSBrian Somers char icmpBuf[IP_MAXPACKET];
3924084f9bSBrian Somers struct ip* ip;
4024084f9bSBrian Somers struct icmp* icmp;
4124084f9bSBrian Somers int icmpLen;
4224084f9bSBrian Somers int failBytes;
4324084f9bSBrian Somers int failHdrLen;
4424084f9bSBrian Somers struct sockaddr_in addr;
4524084f9bSBrian Somers int wrote;
4624084f9bSBrian Somers struct in_addr swap;
4724084f9bSBrian Somers /*
4824084f9bSBrian Somers * Don't send error if packet is
4924084f9bSBrian Somers * not the first fragment.
5024084f9bSBrian Somers */
5124084f9bSBrian Somers if (ntohs (failedDgram->ip_off) & ~(IP_MF | IP_DF))
5224084f9bSBrian Somers return 0;
5324084f9bSBrian Somers /*
5424084f9bSBrian Somers * Dont respond if failed datagram is ICMP.
5524084f9bSBrian Somers */
5624084f9bSBrian Somers if (failedDgram->ip_p == IPPROTO_ICMP)
5724084f9bSBrian Somers return 0;
5824084f9bSBrian Somers /*
5924084f9bSBrian Somers * Start building the message.
6024084f9bSBrian Somers */
6124084f9bSBrian Somers ip = (struct ip*) icmpBuf;
6224084f9bSBrian Somers icmp = (struct icmp*) (icmpBuf + sizeof (struct ip));
6324084f9bSBrian Somers /*
6424084f9bSBrian Somers * Complete ICMP part.
6524084f9bSBrian Somers */
6624084f9bSBrian Somers icmp->icmp_type = ICMP_UNREACH;
6724084f9bSBrian Somers icmp->icmp_code = ICMP_UNREACH_NEEDFRAG;
6824084f9bSBrian Somers icmp->icmp_cksum = 0;
6924084f9bSBrian Somers icmp->icmp_void = 0;
7024084f9bSBrian Somers icmp->icmp_nextmtu = htons (mtu);
7124084f9bSBrian Somers /*
7224084f9bSBrian Somers * Copy header + 64 bits of original datagram.
7324084f9bSBrian Somers */
7424084f9bSBrian Somers failHdrLen = (failedDgram->ip_hl << 2);
7524084f9bSBrian Somers failBytes = failedDgram->ip_len - failHdrLen;
7624084f9bSBrian Somers if (failBytes > 8)
7724084f9bSBrian Somers failBytes = 8;
7824084f9bSBrian Somers
7924084f9bSBrian Somers failBytes += failHdrLen;
8024084f9bSBrian Somers icmpLen = ICMP_MINLEN + failBytes;
8124084f9bSBrian Somers
8224084f9bSBrian Somers memcpy (&icmp->icmp_ip, failedDgram, failBytes);
8324084f9bSBrian Somers /*
8424084f9bSBrian Somers * Calculate checksum.
8524084f9bSBrian Somers */
8622c62477SPoul-Henning Kamp icmp->icmp_cksum = LibAliasInternetChecksum (mla, (u_short*) icmp,
87fb994b07SBrian Somers icmpLen);
8824084f9bSBrian Somers /*
8924084f9bSBrian Somers * Add IP header using old IP header as template.
9024084f9bSBrian Somers */
9124084f9bSBrian Somers memcpy (ip, failedDgram, sizeof (struct ip));
9224084f9bSBrian Somers
9324084f9bSBrian Somers ip->ip_v = 4;
9424084f9bSBrian Somers ip->ip_hl = 5;
9524084f9bSBrian Somers ip->ip_len = htons (sizeof (struct ip) + icmpLen);
9624084f9bSBrian Somers ip->ip_p = IPPROTO_ICMP;
9724084f9bSBrian Somers ip->ip_tos = 0;
9824084f9bSBrian Somers
9924084f9bSBrian Somers swap = ip->ip_dst;
10024084f9bSBrian Somers ip->ip_dst = ip->ip_src;
10124084f9bSBrian Somers ip->ip_src = swap;
10224084f9bSBrian Somers
10322c62477SPoul-Henning Kamp LibAliasIn (mla, (char*) ip, IP_MAXPACKET);
10424084f9bSBrian Somers
10524084f9bSBrian Somers addr.sin_family = AF_INET;
10624084f9bSBrian Somers addr.sin_addr = ip->ip_dst;
10724084f9bSBrian Somers addr.sin_port = 0;
10824084f9bSBrian Somers /*
10924084f9bSBrian Somers * Put packet into processing queue.
11024084f9bSBrian Somers */
11124084f9bSBrian Somers wrote = sendto (sock,
11224084f9bSBrian Somers icmp,
11324084f9bSBrian Somers icmpLen,
11424084f9bSBrian Somers 0,
11524084f9bSBrian Somers (struct sockaddr*) &addr,
11624084f9bSBrian Somers sizeof addr);
11724084f9bSBrian Somers
11824084f9bSBrian Somers if (wrote != icmpLen)
11924084f9bSBrian Somers Warn ("Cannot send ICMP message.");
12024084f9bSBrian Somers
12124084f9bSBrian Somers return 1;
12224084f9bSBrian Somers }
12324084f9bSBrian Somers
12424084f9bSBrian Somers
125