ip_options.c

来自「linux 内核源代码」· C语言 代码 · 共 659 行 · 第 1/2 页

C
659
字号
			if (optlen < 3) {				pp_ptr = optptr + 1;				goto error;			}			if (optptr[2] < 4) {				pp_ptr = optptr + 2;				goto error;			}			if (optptr[2] <= optlen) {				if (optptr[2]+3 > optlen) {					pp_ptr = optptr + 2;					goto error;				}				if (skb) {					memcpy(&optptr[optptr[2]-1], &rt->rt_spec_dst, 4);					opt->is_changed = 1;				}				optptr[2] += 4;				opt->rr_needaddr = 1;			}			opt->rr = optptr - iph;			break;		      case IPOPT_TIMESTAMP:			if (opt->ts) {				pp_ptr = optptr;				goto error;			}			if (optlen < 4) {				pp_ptr = optptr + 1;				goto error;			}			if (optptr[2] < 5) {				pp_ptr = optptr + 2;				goto error;			}			if (optptr[2] <= optlen) {				__be32 *timeptr = NULL;				if (optptr[2]+3 > optptr[1]) {					pp_ptr = optptr + 2;					goto error;				}				switch (optptr[3]&0xF) {				      case IPOPT_TS_TSONLY:					opt->ts = optptr - iph;					if (skb)						timeptr = (__be32*)&optptr[optptr[2]-1];					opt->ts_needtime = 1;					optptr[2] += 4;					break;				      case IPOPT_TS_TSANDADDR:					if (optptr[2]+7 > optptr[1]) {						pp_ptr = optptr + 2;						goto error;					}					opt->ts = optptr - iph;					if (skb) {						memcpy(&optptr[optptr[2]-1], &rt->rt_spec_dst, 4);						timeptr = (__be32*)&optptr[optptr[2]+3];					}					opt->ts_needaddr = 1;					opt->ts_needtime = 1;					optptr[2] += 8;					break;				      case IPOPT_TS_PRESPEC:					if (optptr[2]+7 > optptr[1]) {						pp_ptr = optptr + 2;						goto error;					}					opt->ts = optptr - iph;					{						__be32 addr;						memcpy(&addr, &optptr[optptr[2]-1], 4);						if (inet_addr_type(addr) == RTN_UNICAST)							break;						if (skb)							timeptr = (__be32*)&optptr[optptr[2]+3];					}					opt->ts_needtime = 1;					optptr[2] += 8;					break;				      default:					if (!skb && !capable(CAP_NET_RAW)) {						pp_ptr = optptr + 3;						goto error;					}					break;				}				if (timeptr) {					struct timeval tv;					__be32  midtime;					do_gettimeofday(&tv);					midtime = htonl((tv.tv_sec % 86400) * 1000 + tv.tv_usec / 1000);					memcpy(timeptr, &midtime, sizeof(__be32));					opt->is_changed = 1;				}			} else {				unsigned overflow = optptr[3]>>4;				if (overflow == 15) {					pp_ptr = optptr + 3;					goto error;				}				opt->ts = optptr - iph;				if (skb) {					optptr[3] = (optptr[3]&0xF)|((overflow+1)<<4);					opt->is_changed = 1;				}			}			break;		      case IPOPT_RA:			if (optlen < 4) {				pp_ptr = optptr + 1;				goto error;			}			if (optptr[2] == 0 && optptr[3] == 0)				opt->router_alert = optptr - iph;			break;		      case IPOPT_CIPSO:			if ((!skb && !capable(CAP_NET_RAW)) || opt->cipso) {				pp_ptr = optptr;				goto error;			}			opt->cipso = optptr - iph;			if (cipso_v4_validate(&optptr)) {				pp_ptr = optptr;				goto error;			}			break;		      case IPOPT_SEC:		      case IPOPT_SID:		      default:			if (!skb && !capable(CAP_NET_RAW)) {				pp_ptr = optptr;				goto error;			}			break;		}		l -= optlen;		optptr += optlen;	}eol:	if (!pp_ptr)		return 0;error:	if (skb) {		icmp_send(skb, ICMP_PARAMETERPROB, 0, htonl((pp_ptr-iph)<<24));	}	return -EINVAL;}/* *	Undo all the changes done by ip_options_compile(). */void ip_options_undo(struct ip_options * opt){	if (opt->srr) {		unsigned  char * optptr = opt->__data+opt->srr-sizeof(struct  iphdr);		memmove(optptr+7, optptr+3, optptr[1]-7);		memcpy(optptr+3, &opt->faddr, 4);	}	if (opt->rr_needaddr) {		unsigned  char * optptr = opt->__data+opt->rr-sizeof(struct  iphdr);		optptr[2] -= 4;		memset(&optptr[optptr[2]-1], 0, 4);	}	if (opt->ts) {		unsigned  char * optptr = opt->__data+opt->ts-sizeof(struct  iphdr);		if (opt->ts_needtime) {			optptr[2] -= 4;			memset(&optptr[optptr[2]-1], 0, 4);			if ((optptr[3]&0xF) == IPOPT_TS_PRESPEC)				optptr[2] -= 4;		}		if (opt->ts_needaddr) {			optptr[2] -= 4;			memset(&optptr[optptr[2]-1], 0, 4);		}	}}static struct ip_options *ip_options_get_alloc(const int optlen){	return kzalloc(sizeof(struct ip_options) + ((optlen + 3) & ~3),		       GFP_KERNEL);}static int ip_options_get_finish(struct ip_options **optp,				 struct ip_options *opt, int optlen){	while (optlen & 3)		opt->__data[optlen++] = IPOPT_END;	opt->optlen = optlen;	opt->is_data = 1;	if (optlen && ip_options_compile(opt, NULL)) {		kfree(opt);		return -EINVAL;	}	kfree(*optp);	*optp = opt;	return 0;}int ip_options_get_from_user(struct ip_options **optp, unsigned char __user *data, int optlen){	struct ip_options *opt = ip_options_get_alloc(optlen);	if (!opt)		return -ENOMEM;	if (optlen && copy_from_user(opt->__data, data, optlen)) {		kfree(opt);		return -EFAULT;	}	return ip_options_get_finish(optp, opt, optlen);}int ip_options_get(struct ip_options **optp, unsigned char *data, int optlen){	struct ip_options *opt = ip_options_get_alloc(optlen);	if (!opt)		return -ENOMEM;	if (optlen)		memcpy(opt->__data, data, optlen);	return ip_options_get_finish(optp, opt, optlen);}void ip_forward_options(struct sk_buff *skb){	struct   ip_options * opt	= &(IPCB(skb)->opt);	unsigned char * optptr;	struct rtable *rt = (struct rtable*)skb->dst;	unsigned char *raw = skb_network_header(skb);	if (opt->rr_needaddr) {		optptr = (unsigned char *)raw + opt->rr;		ip_rt_get_source(&optptr[optptr[2]-5], rt);		opt->is_changed = 1;	}	if (opt->srr_is_hit) {		int srrptr, srrspace;		optptr = raw + opt->srr;		for ( srrptr=optptr[2], srrspace = optptr[1];		     srrptr <= srrspace;		     srrptr += 4		     ) {			if (srrptr + 3 > srrspace)				break;			if (memcmp(&rt->rt_dst, &optptr[srrptr-1], 4) == 0)				break;		}		if (srrptr + 3 <= srrspace) {			opt->is_changed = 1;			ip_rt_get_source(&optptr[srrptr-1], rt);			ip_hdr(skb)->daddr = rt->rt_dst;			optptr[2] = srrptr+4;		} else if (net_ratelimit())			printk(KERN_CRIT "ip_forward(): Argh! Destination lost!\n");		if (opt->ts_needaddr) {			optptr = raw + opt->ts;			ip_rt_get_source(&optptr[optptr[2]-9], rt);			opt->is_changed = 1;		}	}	if (opt->is_changed) {		opt->is_changed = 0;		ip_send_check(ip_hdr(skb));	}}int ip_options_rcv_srr(struct sk_buff *skb){	struct ip_options *opt = &(IPCB(skb)->opt);	int srrspace, srrptr;	__be32 nexthop;	struct iphdr *iph = ip_hdr(skb);	unsigned char *optptr = skb_network_header(skb) + opt->srr;	struct rtable *rt = (struct rtable*)skb->dst;	struct rtable *rt2;	int err;	if (!opt->srr)		return 0;	if (skb->pkt_type != PACKET_HOST)		return -EINVAL;	if (rt->rt_type == RTN_UNICAST) {		if (!opt->is_strictroute)			return 0;		icmp_send(skb, ICMP_PARAMETERPROB, 0, htonl(16<<24));		return -EINVAL;	}	if (rt->rt_type != RTN_LOCAL)		return -EINVAL;	for (srrptr=optptr[2], srrspace = optptr[1]; srrptr <= srrspace; srrptr += 4) {		if (srrptr + 3 > srrspace) {			icmp_send(skb, ICMP_PARAMETERPROB, 0, htonl((opt->srr+2)<<24));			return -EINVAL;		}		memcpy(&nexthop, &optptr[srrptr-1], 4);		rt = (struct rtable*)skb->dst;		skb->dst = NULL;		err = ip_route_input(skb, nexthop, iph->saddr, iph->tos, skb->dev);		rt2 = (struct rtable*)skb->dst;		if (err || (rt2->rt_type != RTN_UNICAST && rt2->rt_type != RTN_LOCAL)) {			ip_rt_put(rt2);			skb->dst = &rt->u.dst;			return -EINVAL;		}		ip_rt_put(rt);		if (rt2->rt_type != RTN_LOCAL)			break;		/* Superfast 8) loopback forward */		memcpy(&iph->daddr, &optptr[srrptr-1], 4);		opt->is_changed = 1;	}	if (srrptr <= srrspace) {		opt->srr_is_hit = 1;		opt->is_changed = 1;	}	return 0;}

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?