diff options
-rw-r--r-- | net/ipv4/netfilter/ip_nat_standalone.c | 34 |
1 files changed, 19 insertions, 15 deletions
diff --git a/net/ipv4/netfilter/ip_nat_standalone.c b/net/ipv4/netfilter/ip_nat_standalone.c index f04111f74e09..1bb50897a5a2 100644 --- a/net/ipv4/netfilter/ip_nat_standalone.c +++ b/net/ipv4/netfilter/ip_nat_standalone.c | |||
@@ -162,18 +162,20 @@ ip_nat_in(unsigned int hooknum, | |||
162 | const struct net_device *out, | 162 | const struct net_device *out, |
163 | int (*okfn)(struct sk_buff *)) | 163 | int (*okfn)(struct sk_buff *)) |
164 | { | 164 | { |
165 | u_int32_t saddr, daddr; | 165 | struct ip_conntrack *ct; |
166 | enum ip_conntrack_info ctinfo; | ||
166 | unsigned int ret; | 167 | unsigned int ret; |
167 | 168 | ||
168 | saddr = (*pskb)->nh.iph->saddr; | ||
169 | daddr = (*pskb)->nh.iph->daddr; | ||
170 | |||
171 | ret = ip_nat_fn(hooknum, pskb, in, out, okfn); | 169 | ret = ip_nat_fn(hooknum, pskb, in, out, okfn); |
172 | if (ret != NF_DROP && ret != NF_STOLEN | 170 | if (ret != NF_DROP && ret != NF_STOLEN |
173 | && ((*pskb)->nh.iph->saddr != saddr | 171 | && (ct = ip_conntrack_get(*pskb, &ctinfo)) != NULL) { |
174 | || (*pskb)->nh.iph->daddr != daddr)) { | 172 | enum ip_conntrack_dir dir = CTINFO2DIR(ctinfo); |
175 | dst_release((*pskb)->dst); | 173 | |
176 | (*pskb)->dst = NULL; | 174 | if (ct->tuplehash[dir].tuple.src.ip != |
175 | ct->tuplehash[!dir].tuple.dst.ip) { | ||
176 | dst_release((*pskb)->dst); | ||
177 | (*pskb)->dst = NULL; | ||
178 | } | ||
177 | } | 179 | } |
178 | return ret; | 180 | return ret; |
179 | } | 181 | } |
@@ -200,7 +202,8 @@ ip_nat_local_fn(unsigned int hooknum, | |||
200 | const struct net_device *out, | 202 | const struct net_device *out, |
201 | int (*okfn)(struct sk_buff *)) | 203 | int (*okfn)(struct sk_buff *)) |
202 | { | 204 | { |
203 | u_int32_t saddr, daddr; | 205 | struct ip_conntrack *ct; |
206 | enum ip_conntrack_info ctinfo; | ||
204 | unsigned int ret; | 207 | unsigned int ret; |
205 | 208 | ||
206 | /* root is playing with raw sockets. */ | 209 | /* root is playing with raw sockets. */ |
@@ -208,14 +211,15 @@ ip_nat_local_fn(unsigned int hooknum, | |||
208 | || (*pskb)->nh.iph->ihl * 4 < sizeof(struct iphdr)) | 211 | || (*pskb)->nh.iph->ihl * 4 < sizeof(struct iphdr)) |
209 | return NF_ACCEPT; | 212 | return NF_ACCEPT; |
210 | 213 | ||
211 | saddr = (*pskb)->nh.iph->saddr; | ||
212 | daddr = (*pskb)->nh.iph->daddr; | ||
213 | |||
214 | ret = ip_nat_fn(hooknum, pskb, in, out, okfn); | 214 | ret = ip_nat_fn(hooknum, pskb, in, out, okfn); |
215 | if (ret != NF_DROP && ret != NF_STOLEN | 215 | if (ret != NF_DROP && ret != NF_STOLEN |
216 | && ((*pskb)->nh.iph->saddr != saddr | 216 | && (ct = ip_conntrack_get(*pskb, &ctinfo)) != NULL) { |
217 | || (*pskb)->nh.iph->daddr != daddr)) | 217 | enum ip_conntrack_dir dir = CTINFO2DIR(ctinfo); |
218 | return ip_route_me_harder(pskb) == 0 ? ret : NF_DROP; | 218 | |
219 | if (ct->tuplehash[dir].tuple.dst.ip != | ||
220 | ct->tuplehash[!dir].tuple.src.ip) | ||
221 | return ip_route_me_harder(pskb) == 0 ? ret : NF_DROP; | ||
222 | } | ||
219 | return ret; | 223 | return ret; |
220 | } | 224 | } |
221 | 225 | ||