1 #include <linux/ipv6.h>
2 #include <linux/version.h>
5 #if LINUX_VERSION_CODE < KERNEL_VERSION(3,3,0)
6 int rpl_ipv6_skip_exthdr(const struct sk_buff *skb, int start,
7 u8 *nexthdrp, __be16 *frag_offp)
9 u8 nexthdr = *nexthdrp;
13 while (ipv6_ext_hdr(nexthdr)) {
14 struct ipv6_opt_hdr _hdr, *hp;
17 if (nexthdr == NEXTHDR_NONE)
19 hp = skb_header_pointer(skb, start, sizeof(_hdr), &_hdr);
22 if (nexthdr == NEXTHDR_FRAGMENT) {
23 __be16 _frag_off, *fp;
24 fp = skb_header_pointer(skb,
25 start+offsetof(struct frag_hdr,
33 if (ntohs(*frag_offp) & ~0x7)
36 } else if (nexthdr == NEXTHDR_AUTH)
37 hdrlen = (hp->hdrlen+2)<<2;
39 hdrlen = ipv6_optlen(hp);
41 nexthdr = hp->nexthdr;
48 EXPORT_SYMBOL_GPL(rpl_ipv6_skip_exthdr);
49 #endif /* Kernel version < 3.3 */
51 #ifndef HAVE_IP6_FH_F_SKIP_RH
53 * find the offset to specified header or the protocol number of last header
54 * if target < 0. "last header" is transport protocol header, ESP, or
57 * Note that *offset is used as input/output parameter. an if it is not zero,
58 * then it must be a valid offset to an inner IPv6 header. This can be used
59 * to explore inner IPv6 header, eg. ICMPv6 error messages.
61 * If target header is found, its offset is set in *offset and return protocol
62 * number. Otherwise, return -1.
64 * If the first fragment doesn't contain the final protocol header or
65 * NEXTHDR_NONE it is considered invalid.
67 * Note that non-1st fragment is special case that "the protocol number
68 * of last header" is "next header" field in Fragment header. In this case,
69 * *offset is meaningless and fragment offset is stored in *fragoff if fragoff
72 * if flags is not NULL and it's a fragment, then the frag flag
73 * IP6_FH_F_FRAG will be set. If it's an AH header, the
74 * IP6_FH_F_AUTH flag is set and target < 0, then this function will
75 * stop at the AH header. If IP6_FH_F_SKIP_RH flag was passed, then this
76 * function will skip all those routing headers, where segements_left was 0.
78 int rpl_ipv6_find_hdr(const struct sk_buff *skb, unsigned int *offset,
79 int target, unsigned short *fragoff, int *flags)
81 unsigned int start = skb_network_offset(skb) + sizeof(struct ipv6hdr);
82 u8 nexthdr = ipv6_hdr(skb)->nexthdr;
90 struct ipv6hdr _ip6, *ip6;
92 ip6 = skb_header_pointer(skb, *offset, sizeof(_ip6), &_ip6);
93 if (!ip6 || (ip6->version != 6)) {
94 printk(KERN_ERR "IPv6 header not found\n");
97 start = *offset + sizeof(struct ipv6hdr);
98 nexthdr = ip6->nexthdr;
100 len = skb->len - start;
103 struct ipv6_opt_hdr _hdr, *hp;
105 found = (nexthdr == target);
107 if ((!ipv6_ext_hdr(nexthdr)) || nexthdr == NEXTHDR_NONE) {
108 if (target < 0 || found)
113 hp = skb_header_pointer(skb, start, sizeof(_hdr), &_hdr);
117 if (nexthdr == NEXTHDR_ROUTING) {
118 struct ipv6_rt_hdr _rh, *rh;
120 rh = skb_header_pointer(skb, start, sizeof(_rh),
125 if (flags && (*flags & IP6_FH_F_SKIP_RH) &&
126 rh->segments_left == 0)
130 if (nexthdr == NEXTHDR_FRAGMENT) {
131 unsigned short _frag_off;
134 if (flags) /* Indicate that this is a fragment */
135 *flags |= IP6_FH_F_FRAG;
136 fp = skb_header_pointer(skb,
137 start+offsetof(struct frag_hdr,
144 _frag_off = ntohs(*fp) & ~0x7;
147 ((!ipv6_ext_hdr(hp->nexthdr)) ||
148 hp->nexthdr == NEXTHDR_NONE)) {
150 *fragoff = _frag_off;
156 } else if (nexthdr == NEXTHDR_AUTH) {
157 if (flags && (*flags & IP6_FH_F_AUTH) && (target < 0))
159 hdrlen = (hp->hdrlen + 2) << 2;
161 hdrlen = ipv6_optlen(hp);
164 nexthdr = hp->nexthdr;
173 EXPORT_SYMBOL_GPL(rpl_ipv6_find_hdr);