www.pudn.com > e041tcpipc51.rar > ip.c
#include#include #include "ethernet.h" #include "ip.h" #define IP_TTL 128 /* Time To Live for an outgoing IP datagram */ extern NODE locnode; //本机的节点信息结构(mac,ip,mask,port) /* Check ARP packet, swap bytes, return -1, 0 if not ARP */ int is_arp(ETHERFRAME *efp, int len) { ARPKT *arp; int dlen=0; if (efp->e.ptype==PCOL_ARP && len>=sizeof(ARPKT)) { /* If protocol OK.. */ arp = (ARPKT *)(efp->edata); if (arp->hrd==HTYPE && arp->pro==ARPPRO) dlen = -1; /* Return non-zero if OK */ else dlen = 0; } return(dlen); } /* Make an ARP packet, return its total length */ int make_arp(ETHERFRAME *efp, NODE *srcep, NODE *destp, WORD codetype) { ARPKT *arp; arp = (ARPKT *)(efp->edata); memcpy(arp->smac, srcep->mac, MACLEN); /* Srce ARP ether addr */ memcpy(arp->dmac, destp->mac, MACLEN); /* Dest ARP ether addr */ arp->hrd = HTYPE; /* Hware & protocol types */ arp->pro = ARPPRO; arp->hln = MACLEN; /* Hardware addr len */ arp->pln = sizeof(LWORD); /* IP addr len */ arp->op = codetype; /* ARP opcode */ arp->dip = gate_ip(destp, srcep); /* Dest ip addr (maybe gateway) */ arp->sip = srcep->ip; /* Source IP addr */ return(make_frame(efp,srcep->mac,destp->mac, PCOL_ARP, sizeof(ARPKT))); } /* Check frame is IP, checksum & byte-swap, return data len */ int is_ip(ETHERFRAME *efp, int len) { int ver, dlen=0, hlen; WORD sum; IPKT *ip; if (efp->e.ptype==PCOL_IP && len>=sizeof(IPHDR)) { ip = (IPKT *)(efp->edata); /* Get pointer to IP frame */ ver = ip->i.vhl >> 4; /* Get IP version & hdr len */ hlen = (ip->i.vhl & 0xf) << 2; sum = ~csum(&ip->i, (WORD)hlen); /* Do checksum */ if (ver==4 && len>=hlen && sum==0) /* If OK.. */ { dlen = min(ip->i.len,len); dlen-=hlen; if (hlen > sizeof(IPHDR)) /* If IP options present.. */ { /* ..delete them, move data down */ memmove(ip->ipdata, &ip->ipdata[hlen-sizeof(IPHDR)], len); dlen -= hlen-sizeof(IPHDR); } } } return(dlen); } /* Make an IP packet, if greater than the MTU, also make fragment (subframe) in ** this frame. Return total length of frame and subframes (if any) */ int make_ip(ETHERFRAME *efp, NODE *srcep, NODE *destp, BYTE pcol, WORD dlen) { IPKT *ip; static WORD ident=1; ip = (IPKT *)(efp->edata); /* Get pointer to IP datagram */ ip->i.ident = ident; /* Set datagram ident */ ip->i.frags = 0; /* Frag offset in units of 8 bytes */ ip->i.vhl = 0x40+(sizeof(IPHDR)>>2); /* Version 4, header len 5 LWORDs */ ip->i.service = 0; /* Routine message */ ip->i.ttl = IP_TTL; /* Time To Live */ ip->i.pcol = pcol; /* Set IP protocol */ ip->i.sip = srcep->ip; /* Srce, dest IP addrs */ ip->i.dip = destp->ip; ip->i.len = dlen + sizeof(IPHDR); /* Data length */ ip->i.check = 0; /* Clear checksum */ ip->i.check = ~csum(ip, sizeof(IPHDR)); /* ..then set to calc value */ ident++; /* Increment datagram ident */ return(make_frame(efp, srcep->mac,destp->mac, PCOL_IP, (WORD)dlen+sizeof(IPHDR))); } /* Get the frame driver type, source IP and Ethernet addresses ** Returned data does not include port number, netmask or gateway addr */ void getip_srce(ETHERFRAME *efp, NODE *np) { IPHDR *iph; memcpy(np->mac,efp->e.srce,6); iph = (IPHDR *)(efp->edata); np->ip = iph->sip; } /* Get local node data corresponding to a frame destination IP address ** Data does not include port number. Return 0 if no matching local node */ int getip_locdest(ETHERFRAME *efp,NODE *np) { IPHDR *iph; int ok=0; iph =(IPHDR *)(efp->edata); if (iph->dip==locnode.ip) { *np=locnode; ok=1; } return (ok); } /* Check a remote address to see if it is on the local subnet. ** If so (or no gateway), return it. If not, return the gateway IP address */ LWORD gate_ip(NODE *remp, NODE *locp) { return((locp->gate==0||on_subnet(remp->ip, locp)) ? remp->ip : locp->gate); } /* Check an IP address to see if it is on a subnet, return 0 if not */ int on_subnet(LWORD remip, NODE *locp) { return(((remip ^ locp->ip) & locp->mask) == 0); } /* Return ICMP data length (-1 if no data), 0 if not ICMP */ int is_icmp(IPKT *ip, int len) { ICMPKT *icmp; WORD sum; int dlen=0; if (ip->i.pcol==PICMP && len>=sizeof(ICMPHDR)) { icmp = (ICMPKT *)ip; if ((sum=csum(&icmp->c, (WORD)len)) == 0xffff) dlen = len>sizeof(ICMPHDR) ? len-sizeof(ICMPHDR) : -1; // else // printf("\r\nICMP checksum error: %04X\r\n", sum); } return(dlen); } /* Make an ICMP packet */ int make_icmp(ETHERFRAME *efp, NODE *srcep, NODE *destp, BYTE type, BYTE codetype,WORD dlen) { ICMPKT *icmp; WORD len; icmp = (ICMPKT *)(efp->edata); icmp->c.type = type; icmp->c.codetype = codetype; icmp->c.check = 0; len = (WORD)(dlen + sizeof(ICMPHDR)); icmp->c.check = ~csum(&icmp->c, len); return(make_ip(efp, srcep, destp, PICMP, len)); } /* Make ICMP 'destination unreachable' for incoming frame */ /* int icmp_unreach(ETHERFRAME *efp, NODE *srcep, NODE *destp, BYTE codetype) { int len; ICMPKT *icmp; icmp = (ICMPKT *)(efp->edata); len = ((icmp->i.vhl & 0xf) << 2) + 8; memmove(icmp->icmpdata, icmp, len); return(make_icmp(efp, srcep, destp, ICUNREACH, codetype, (WORD)len)); } */