Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
cf99172b
Commit
cf99172b
authored
Aug 23, 2002
by
Andras Kis-Szabo
Committed by
David S. Miller
Aug 23, 2002
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
net/ipv6/netfilter/ip6_tables.c: Fix extension header parsing bugs.
parent
a3d7813b
Changes
1
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
69 additions
and
43 deletions
+69
-43
net/ipv6/netfilter/ip6_tables.c
net/ipv6/netfilter/ip6_tables.c
+69
-43
No files found.
net/ipv6/netfilter/ip6_tables.c
View file @
cf99172b
...
@@ -7,6 +7,8 @@
...
@@ -7,6 +7,8 @@
* 19 Jan 2002 Harald Welte <laforge@gnumonks.org>
* 19 Jan 2002 Harald Welte <laforge@gnumonks.org>
* - increase module usage count as soon as we have rules inside
* - increase module usage count as soon as we have rules inside
* a table
* a table
* 06 Jun 2002 Andras Kis-Szabo <kisza@sch.bme.hu>
* - new extension header parser code
*/
*/
#include <linux/config.h>
#include <linux/config.h>
#include <linux/skbuff.h>
#include <linux/skbuff.h>
...
@@ -25,6 +27,7 @@
...
@@ -25,6 +27,7 @@
#include <linux/netfilter_ipv6/ip6_tables.h>
#include <linux/netfilter_ipv6/ip6_tables.h>
#define IPV6_HDR_LEN (sizeof(struct ipv6hdr))
#define IPV6_HDR_LEN (sizeof(struct ipv6hdr))
#define IPV6_OPTHDR_LEN (sizeof(struct ipv6_opt_hdr))
/*#define DEBUG_IP_FIREWALL*/
/*#define DEBUG_IP_FIREWALL*/
/*#define DEBUG_ALLOW_ALL*/
/* Useful for remote debugging */
/*#define DEBUG_ALLOW_ALL*/
/* Useful for remote debugging */
...
@@ -133,43 +136,23 @@ static int ip6_masked_addrcmp(struct in6_addr addr1, struct in6_addr mask,
...
@@ -133,43 +136,23 @@ static int ip6_masked_addrcmp(struct in6_addr addr1, struct in6_addr mask,
return
0
;
return
0
;
}
}
/* takes in current header and pointer to the header */
/* Check for an extension */
/* if another header exists, sets hdrptr to the next header
int
and returns the new header value, else returns 0 */
ip6t_ext_hdr
(
u8
nexthdr
)
static
u_int8_t
ip6_nexthdr
(
u_int8_t
currenthdr
,
u_int8_t
*
hdrptr
)
{
{
int
i
;
return
(
(
nexthdr
==
IPPROTO_HOPOPTS
)
||
u_int8_t
hdrlen
,
nexthdr
=
0
;
(
nexthdr
==
IPPROTO_ROUTING
)
||
switch
(
currenthdr
){
(
nexthdr
==
IPPROTO_FRAGMENT
)
||
case
IPPROTO_AH
:
(
nexthdr
==
IPPROTO_ESP
)
||
/* whoever decided to do the length of AUTH for ipv6
(
nexthdr
==
IPPROTO_AH
)
||
in 32bit units unlike other headers should be beaten...
(
nexthdr
==
IPPROTO_NONE
)
||
repeatedly...with a large stick...no, an even LARGER
(
nexthdr
==
IPPROTO_DSTOPTS
)
);
stick...no, you're still not thinking big enough */
nexthdr
=
*
hdrptr
;
hdrlen
=
hdrptr
[
i
]
*
4
+
8
;
hdrptr
=
hdrptr
+
hdrlen
;
break
;
/*stupid rfc2402 */
case
IPPROTO_DSTOPTS
:
case
IPPROTO_ROUTING
:
case
IPPROTO_HOPOPTS
:
nexthdr
=
*
hdrptr
;
hdrlen
=
hdrptr
[
1
]
*
8
+
8
;
hdrptr
=
hdrptr
+
hdrlen
;
break
;
case
IPPROTO_FRAGMENT
:
nexthdr
=
*
hdrptr
;
hdrptr
=
hdrptr
+
8
;
break
;
}
return
nexthdr
;
}
}
/* Returns whether matches rule or not. */
/* Returns whether matches rule or not. */
static
inline
int
static
inline
int
ip6_packet_match
(
const
struct
ipv6hdr
*
ipv6
,
ip6_packet_match
(
const
struct
sk_buff
*
skb
,
const
struct
ipv6hdr
*
ipv6
,
const
char
*
indev
,
const
char
*
indev
,
const
char
*
outdev
,
const
char
*
outdev
,
const
struct
ip6t_ip6
*
ip6info
,
const
struct
ip6t_ip6
*
ip6info
,
...
@@ -227,17 +210,58 @@ ip6_packet_match(const struct ipv6hdr *ipv6,
...
@@ -227,17 +210,58 @@ ip6_packet_match(const struct ipv6hdr *ipv6,
/* look for the desired protocol header */
/* look for the desired protocol header */
if
((
ip6info
->
flags
&
IP6T_F_PROTO
))
{
if
((
ip6info
->
flags
&
IP6T_F_PROTO
))
{
u_int8_t
currenthdr
=
ipv6
->
nexthdr
;
u_int8_t
currenthdr
=
ipv6
->
nexthdr
;
u_int8_t
*
hdrptr
;
struct
ipv6_opt_hdr
*
hdrptr
;
hdrptr
=
(
u_int8_t
*
)(
ipv6
+
1
);
u_int16_t
ptr
;
/* Header offset in skb */
do
{
u_int16_t
hdrlen
;
/* Header */
ptr
=
IPV6_HDR_LEN
;
while
(
ip6t_ext_hdr
(
currenthdr
))
{
/* Is there enough space for the next ext header? */
if
(
skb
->
len
-
ptr
<
IPV6_OPTHDR_LEN
)
return
0
;
/* NONE or ESP: there isn't protocol part */
/* If we want to count these packets in '-p all',
* we will change the return 0 to 1*/
if
((
currenthdr
==
IPPROTO_NONE
)
||
(
currenthdr
==
IPPROTO_ESP
))
return
0
;
hdrptr
=
(
struct
ipv6_opt_hdr
*
)(
skb
->
data
+
ptr
);
/* Size calculation */
if
(
currenthdr
==
IPPROTO_FRAGMENT
)
{
hdrlen
=
8
;
}
else
if
(
currenthdr
==
IPPROTO_AH
)
hdrlen
=
(
hdrptr
->
hdrlen
+
2
)
<<
2
;
else
hdrlen
=
ipv6_optlen
(
hdrptr
);
currenthdr
=
hdrptr
->
nexthdr
;
ptr
+=
hdrlen
;
/* ptr is too large */
if
(
ptr
>
skb
->
len
)
return
0
;
}
/* currenthdr contains the protocol header */
dprintf
(
"Packet protocol %hi ?= %s%hi.
\n
"
,
currenthdr
,
ip6info
->
invflags
&
IP6T_INV_PROTO
?
"!"
:
""
,
ip6info
->
proto
);
if
(
ip6info
->
proto
==
currenthdr
)
{
if
(
ip6info
->
proto
==
currenthdr
)
{
if
(
ip6info
->
invflags
&
IP6T_INV_PROTO
)
if
(
ip6info
->
invflags
&
IP6T_INV_PROTO
)
{
return
0
;
return
0
;
}
return
1
;
return
1
;
}
}
currenthdr
=
ip6_nexthdr
(
currenthdr
,
hdrptr
);
}
while
(
currenthdr
);
/* We need match for the '-p all', too! */
if
(
!
(
ip6info
->
invflags
&
IP6T_INV_PROTO
))
if
((
ip6info
->
proto
!=
0
)
&&
!
(
ip6info
->
invflags
&
IP6T_INV_PROTO
))
return
0
;
return
0
;
}
}
return
1
;
return
1
;
...
@@ -359,7 +383,8 @@ ip6t_do_table(struct sk_buff **pskb,
...
@@ -359,7 +383,8 @@ ip6t_do_table(struct sk_buff **pskb,
IP_NF_ASSERT
(
e
);
IP_NF_ASSERT
(
e
);
IP_NF_ASSERT
(
back
);
IP_NF_ASSERT
(
back
);
(
*
pskb
)
->
nfcache
|=
e
->
nfcache
;
(
*
pskb
)
->
nfcache
|=
e
->
nfcache
;
if
(
ip6_packet_match
(
ipv6
,
indev
,
outdev
,
&
e
->
ipv6
,
offset
))
{
if
(
ip6_packet_match
(
*
pskb
,
ipv6
,
indev
,
outdev
,
&
e
->
ipv6
,
offset
))
{
struct
ip6t_entry_target
*
t
;
struct
ip6t_entry_target
*
t
;
if
(
IP6T_MATCH_ITERATE
(
e
,
do_match
,
if
(
IP6T_MATCH_ITERATE
(
e
,
do_match
,
...
@@ -1826,6 +1851,7 @@ EXPORT_SYMBOL(ip6t_register_match);
...
@@ -1826,6 +1851,7 @@ EXPORT_SYMBOL(ip6t_register_match);
EXPORT_SYMBOL
(
ip6t_unregister_match
);
EXPORT_SYMBOL
(
ip6t_unregister_match
);
EXPORT_SYMBOL
(
ip6t_register_target
);
EXPORT_SYMBOL
(
ip6t_register_target
);
EXPORT_SYMBOL
(
ip6t_unregister_target
);
EXPORT_SYMBOL
(
ip6t_unregister_target
);
EXPORT_SYMBOL
(
ip6t_ext_hdr
);
module_init
(
init
);
module_init
(
init
);
module_exit
(
fini
);
module_exit
(
fini
);
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment