Commit 0bbdd42b authored by Julius Volz's avatar Julius Volz Committed by Simon Horman

IPVS: Extend protocol DNAT/SNAT and state handlers

Extend protocol DNAT/SNAT and state handlers to work with IPv6. Also
change/introduce new checksumming helper functions for this.
Signed-off-by: default avatarJulius Volz <juliusv@google.com>
Signed-off-by: default avatarSimon Horman <horms@verge.net.au>
parent 3b047d9d
...@@ -904,6 +904,17 @@ static inline __wsum ip_vs_check_diff4(__be32 old, __be32 new, __wsum oldsum) ...@@ -904,6 +904,17 @@ static inline __wsum ip_vs_check_diff4(__be32 old, __be32 new, __wsum oldsum)
return csum_partial((char *) diff, sizeof(diff), oldsum); return csum_partial((char *) diff, sizeof(diff), oldsum);
} }
#ifdef CONFIG_IP_VS_IPV6
static inline __wsum ip_vs_check_diff16(const __be32 *old, const __be32 *new,
__wsum oldsum)
{
__be32 diff[8] = { ~old[3], ~old[2], ~old[1], ~old[0],
new[3], new[2], new[1], new[0] };
return csum_partial((char *) diff, sizeof(diff), oldsum);
}
#endif
static inline __wsum ip_vs_check_diff2(__be16 old, __be16 new, __wsum oldsum) static inline __wsum ip_vs_check_diff2(__be16 old, __be16 new, __wsum oldsum)
{ {
__be16 diff[2] = { ~old, new }; __be16 diff[2] = { ~old, new };
......
...@@ -114,11 +114,21 @@ tcp_conn_schedule(int af, struct sk_buff *skb, struct ip_vs_protocol *pp, ...@@ -114,11 +114,21 @@ tcp_conn_schedule(int af, struct sk_buff *skb, struct ip_vs_protocol *pp,
static inline void static inline void
tcp_fast_csum_update(struct tcphdr *tcph, __be32 oldip, __be32 newip, tcp_fast_csum_update(int af, struct tcphdr *tcph,
const union nf_inet_addr *oldip,
const union nf_inet_addr *newip,
__be16 oldport, __be16 newport) __be16 oldport, __be16 newport)
{ {
#ifdef CONFIG_IP_VS_IPV6
if (af == AF_INET6)
tcph->check = tcph->check =
csum_fold(ip_vs_check_diff4(oldip, newip, csum_fold(ip_vs_check_diff16(oldip->ip6, newip->ip6,
ip_vs_check_diff2(oldport, newport,
~csum_unfold(tcph->check))));
else
#endif
tcph->check =
csum_fold(ip_vs_check_diff4(oldip->ip, newip->ip,
ip_vs_check_diff2(oldport, newport, ip_vs_check_diff2(oldport, newport,
~csum_unfold(tcph->check)))); ~csum_unfold(tcph->check))));
} }
...@@ -129,7 +139,14 @@ tcp_snat_handler(struct sk_buff *skb, ...@@ -129,7 +139,14 @@ tcp_snat_handler(struct sk_buff *skb,
struct ip_vs_protocol *pp, struct ip_vs_conn *cp) struct ip_vs_protocol *pp, struct ip_vs_conn *cp)
{ {
struct tcphdr *tcph; struct tcphdr *tcph;
const unsigned int tcphoff = ip_hdrlen(skb); unsigned int tcphoff;
#ifdef CONFIG_IP_VS_IPV6
if (cp->af == AF_INET6)
tcphoff = sizeof(struct ipv6hdr);
else
#endif
tcphoff = ip_hdrlen(skb);
/* csum_check requires unshared skb */ /* csum_check requires unshared skb */
if (!skb_make_writable(skb, tcphoff+sizeof(*tcph))) if (!skb_make_writable(skb, tcphoff+sizeof(*tcph)))
...@@ -137,7 +154,7 @@ tcp_snat_handler(struct sk_buff *skb, ...@@ -137,7 +154,7 @@ tcp_snat_handler(struct sk_buff *skb,
if (unlikely(cp->app != NULL)) { if (unlikely(cp->app != NULL)) {
/* Some checks before mangling */ /* Some checks before mangling */
if (pp->csum_check && !pp->csum_check(AF_INET, skb, pp)) if (pp->csum_check && !pp->csum_check(cp->af, skb, pp))
return 0; return 0;
/* Call application helper if needed */ /* Call application helper if needed */
...@@ -145,13 +162,13 @@ tcp_snat_handler(struct sk_buff *skb, ...@@ -145,13 +162,13 @@ tcp_snat_handler(struct sk_buff *skb,
return 0; return 0;
} }
tcph = (void *)ip_hdr(skb) + tcphoff; tcph = (void *)skb_network_header(skb) + tcphoff;
tcph->source = cp->vport; tcph->source = cp->vport;
/* Adjust TCP checksums */ /* Adjust TCP checksums */
if (!cp->app) { if (!cp->app) {
/* Only port and addr are changed, do fast csum update */ /* Only port and addr are changed, do fast csum update */
tcp_fast_csum_update(tcph, cp->daddr.ip, cp->vaddr.ip, tcp_fast_csum_update(cp->af, tcph, &cp->daddr, &cp->vaddr,
cp->dport, cp->vport); cp->dport, cp->vport);
if (skb->ip_summed == CHECKSUM_COMPLETE) if (skb->ip_summed == CHECKSUM_COMPLETE)
skb->ip_summed = CHECKSUM_NONE; skb->ip_summed = CHECKSUM_NONE;
...@@ -159,9 +176,20 @@ tcp_snat_handler(struct sk_buff *skb, ...@@ -159,9 +176,20 @@ tcp_snat_handler(struct sk_buff *skb,
/* full checksum calculation */ /* full checksum calculation */
tcph->check = 0; tcph->check = 0;
skb->csum = skb_checksum(skb, tcphoff, skb->len - tcphoff, 0); skb->csum = skb_checksum(skb, tcphoff, skb->len - tcphoff, 0);
tcph->check = csum_tcpudp_magic(cp->vaddr.ip, cp->caddr.ip, #ifdef CONFIG_IP_VS_IPV6
if (cp->af == AF_INET6)
tcph->check = csum_ipv6_magic(&cp->vaddr.in6,
&cp->caddr.in6,
skb->len - tcphoff, skb->len - tcphoff,
cp->protocol, skb->csum); cp->protocol, skb->csum);
else
#endif
tcph->check = csum_tcpudp_magic(cp->vaddr.ip,
cp->caddr.ip,
skb->len - tcphoff,
cp->protocol,
skb->csum);
IP_VS_DBG(11, "O-pkt: %s O-csum=%d (+%zd)\n", IP_VS_DBG(11, "O-pkt: %s O-csum=%d (+%zd)\n",
pp->name, tcph->check, pp->name, tcph->check,
(char*)&(tcph->check) - (char*)tcph); (char*)&(tcph->check) - (char*)tcph);
...@@ -175,7 +203,14 @@ tcp_dnat_handler(struct sk_buff *skb, ...@@ -175,7 +203,14 @@ tcp_dnat_handler(struct sk_buff *skb,
struct ip_vs_protocol *pp, struct ip_vs_conn *cp) struct ip_vs_protocol *pp, struct ip_vs_conn *cp)
{ {
struct tcphdr *tcph; struct tcphdr *tcph;
const unsigned int tcphoff = ip_hdrlen(skb); unsigned int tcphoff;
#ifdef CONFIG_IP_VS_IPV6
if (cp->af == AF_INET6)
tcphoff = sizeof(struct ipv6hdr);
else
#endif
tcphoff = ip_hdrlen(skb);
/* csum_check requires unshared skb */ /* csum_check requires unshared skb */
if (!skb_make_writable(skb, tcphoff+sizeof(*tcph))) if (!skb_make_writable(skb, tcphoff+sizeof(*tcph)))
...@@ -183,7 +218,7 @@ tcp_dnat_handler(struct sk_buff *skb, ...@@ -183,7 +218,7 @@ tcp_dnat_handler(struct sk_buff *skb,
if (unlikely(cp->app != NULL)) { if (unlikely(cp->app != NULL)) {
/* Some checks before mangling */ /* Some checks before mangling */
if (pp->csum_check && !pp->csum_check(AF_INET, skb, pp)) if (pp->csum_check && !pp->csum_check(cp->af, skb, pp))
return 0; return 0;
/* /*
...@@ -194,7 +229,7 @@ tcp_dnat_handler(struct sk_buff *skb, ...@@ -194,7 +229,7 @@ tcp_dnat_handler(struct sk_buff *skb,
return 0; return 0;
} }
tcph = (void *)ip_hdr(skb) + tcphoff; tcph = (void *)skb_network_header(skb) + tcphoff;
tcph->dest = cp->dport; tcph->dest = cp->dport;
/* /*
...@@ -202,7 +237,7 @@ tcp_dnat_handler(struct sk_buff *skb, ...@@ -202,7 +237,7 @@ tcp_dnat_handler(struct sk_buff *skb,
*/ */
if (!cp->app) { if (!cp->app) {
/* Only port and addr are changed, do fast csum update */ /* Only port and addr are changed, do fast csum update */
tcp_fast_csum_update(tcph, cp->vaddr.ip, cp->daddr.ip, tcp_fast_csum_update(cp->af, tcph, &cp->vaddr, &cp->daddr,
cp->vport, cp->dport); cp->vport, cp->dport);
if (skb->ip_summed == CHECKSUM_COMPLETE) if (skb->ip_summed == CHECKSUM_COMPLETE)
skb->ip_summed = CHECKSUM_NONE; skb->ip_summed = CHECKSUM_NONE;
...@@ -210,9 +245,19 @@ tcp_dnat_handler(struct sk_buff *skb, ...@@ -210,9 +245,19 @@ tcp_dnat_handler(struct sk_buff *skb,
/* full checksum calculation */ /* full checksum calculation */
tcph->check = 0; tcph->check = 0;
skb->csum = skb_checksum(skb, tcphoff, skb->len - tcphoff, 0); skb->csum = skb_checksum(skb, tcphoff, skb->len - tcphoff, 0);
tcph->check = csum_tcpudp_magic(cp->caddr.ip, cp->daddr.ip, #ifdef CONFIG_IP_VS_IPV6
if (cp->af == AF_INET6)
tcph->check = csum_ipv6_magic(&cp->caddr.in6,
&cp->daddr.in6,
skb->len - tcphoff, skb->len - tcphoff,
cp->protocol, skb->csum); cp->protocol, skb->csum);
else
#endif
tcph->check = csum_tcpudp_magic(cp->caddr.ip,
cp->daddr.ip,
skb->len - tcphoff,
cp->protocol,
skb->csum);
skb->ip_summed = CHECKSUM_UNNECESSARY; skb->ip_summed = CHECKSUM_UNNECESSARY;
} }
return 1; return 1;
...@@ -487,7 +532,13 @@ tcp_state_transition(struct ip_vs_conn *cp, int direction, ...@@ -487,7 +532,13 @@ tcp_state_transition(struct ip_vs_conn *cp, int direction,
{ {
struct tcphdr _tcph, *th; struct tcphdr _tcph, *th;
th = skb_header_pointer(skb, ip_hdrlen(skb), sizeof(_tcph), &_tcph); #ifdef CONFIG_IP_VS_IPV6
int ihl = cp->af == AF_INET ? ip_hdrlen(skb) : sizeof(struct ipv6hdr);
#else
int ihl = ip_hdrlen(skb);
#endif
th = skb_header_pointer(skb, ihl, sizeof(_tcph), &_tcph);
if (th == NULL) if (th == NULL)
return 0; return 0;
......
...@@ -120,11 +120,21 @@ udp_conn_schedule(int af, struct sk_buff *skb, struct ip_vs_protocol *pp, ...@@ -120,11 +120,21 @@ udp_conn_schedule(int af, struct sk_buff *skb, struct ip_vs_protocol *pp,
static inline void static inline void
udp_fast_csum_update(struct udphdr *uhdr, __be32 oldip, __be32 newip, udp_fast_csum_update(int af, struct udphdr *uhdr,
const union nf_inet_addr *oldip,
const union nf_inet_addr *newip,
__be16 oldport, __be16 newport) __be16 oldport, __be16 newport)
{ {
#ifdef CONFIG_IP_VS_IPV6
if (af == AF_INET6)
uhdr->check =
csum_fold(ip_vs_check_diff16(oldip->ip6, newip->ip6,
ip_vs_check_diff2(oldport, newport,
~csum_unfold(uhdr->check))));
else
#endif
uhdr->check = uhdr->check =
csum_fold(ip_vs_check_diff4(oldip, newip, csum_fold(ip_vs_check_diff4(oldip->ip, newip->ip,
ip_vs_check_diff2(oldport, newport, ip_vs_check_diff2(oldport, newport,
~csum_unfold(uhdr->check)))); ~csum_unfold(uhdr->check))));
if (!uhdr->check) if (!uhdr->check)
...@@ -136,7 +146,14 @@ udp_snat_handler(struct sk_buff *skb, ...@@ -136,7 +146,14 @@ udp_snat_handler(struct sk_buff *skb,
struct ip_vs_protocol *pp, struct ip_vs_conn *cp) struct ip_vs_protocol *pp, struct ip_vs_conn *cp)
{ {
struct udphdr *udph; struct udphdr *udph;
const unsigned int udphoff = ip_hdrlen(skb); unsigned int udphoff;
#ifdef CONFIG_IP_VS_IPV6
if (cp->af == AF_INET6)
udphoff = sizeof(struct ipv6hdr);
else
#endif
udphoff = ip_hdrlen(skb);
/* csum_check requires unshared skb */ /* csum_check requires unshared skb */
if (!skb_make_writable(skb, udphoff+sizeof(*udph))) if (!skb_make_writable(skb, udphoff+sizeof(*udph)))
...@@ -144,7 +161,7 @@ udp_snat_handler(struct sk_buff *skb, ...@@ -144,7 +161,7 @@ udp_snat_handler(struct sk_buff *skb,
if (unlikely(cp->app != NULL)) { if (unlikely(cp->app != NULL)) {
/* Some checks before mangling */ /* Some checks before mangling */
if (pp->csum_check && !pp->csum_check(AF_INET, skb, pp)) if (pp->csum_check && !pp->csum_check(cp->af, skb, pp))
return 0; return 0;
/* /*
...@@ -154,7 +171,7 @@ udp_snat_handler(struct sk_buff *skb, ...@@ -154,7 +171,7 @@ udp_snat_handler(struct sk_buff *skb,
return 0; return 0;
} }
udph = (void *)ip_hdr(skb) + udphoff; udph = (void *)skb_network_header(skb) + udphoff;
udph->source = cp->vport; udph->source = cp->vport;
/* /*
...@@ -162,7 +179,7 @@ udp_snat_handler(struct sk_buff *skb, ...@@ -162,7 +179,7 @@ udp_snat_handler(struct sk_buff *skb,
*/ */
if (!cp->app && (udph->check != 0)) { if (!cp->app && (udph->check != 0)) {
/* Only port and addr are changed, do fast csum update */ /* Only port and addr are changed, do fast csum update */
udp_fast_csum_update(udph, cp->daddr.ip, cp->vaddr.ip, udp_fast_csum_update(cp->af, udph, &cp->daddr, &cp->vaddr,
cp->dport, cp->vport); cp->dport, cp->vport);
if (skb->ip_summed == CHECKSUM_COMPLETE) if (skb->ip_summed == CHECKSUM_COMPLETE)
skb->ip_summed = CHECKSUM_NONE; skb->ip_summed = CHECKSUM_NONE;
...@@ -170,9 +187,19 @@ udp_snat_handler(struct sk_buff *skb, ...@@ -170,9 +187,19 @@ udp_snat_handler(struct sk_buff *skb,
/* full checksum calculation */ /* full checksum calculation */
udph->check = 0; udph->check = 0;
skb->csum = skb_checksum(skb, udphoff, skb->len - udphoff, 0); skb->csum = skb_checksum(skb, udphoff, skb->len - udphoff, 0);
udph->check = csum_tcpudp_magic(cp->vaddr.ip, cp->caddr.ip, #ifdef CONFIG_IP_VS_IPV6
if (cp->af == AF_INET6)
udph->check = csum_ipv6_magic(&cp->vaddr.in6,
&cp->caddr.in6,
skb->len - udphoff, skb->len - udphoff,
cp->protocol, skb->csum); cp->protocol, skb->csum);
else
#endif
udph->check = csum_tcpudp_magic(cp->vaddr.ip,
cp->caddr.ip,
skb->len - udphoff,
cp->protocol,
skb->csum);
if (udph->check == 0) if (udph->check == 0)
udph->check = CSUM_MANGLED_0; udph->check = CSUM_MANGLED_0;
IP_VS_DBG(11, "O-pkt: %s O-csum=%d (+%zd)\n", IP_VS_DBG(11, "O-pkt: %s O-csum=%d (+%zd)\n",
...@@ -188,7 +215,14 @@ udp_dnat_handler(struct sk_buff *skb, ...@@ -188,7 +215,14 @@ udp_dnat_handler(struct sk_buff *skb,
struct ip_vs_protocol *pp, struct ip_vs_conn *cp) struct ip_vs_protocol *pp, struct ip_vs_conn *cp)
{ {
struct udphdr *udph; struct udphdr *udph;
unsigned int udphoff = ip_hdrlen(skb); unsigned int udphoff;
#ifdef CONFIG_IP_VS_IPV6
if (cp->af == AF_INET6)
udphoff = sizeof(struct ipv6hdr);
else
#endif
udphoff = ip_hdrlen(skb);
/* csum_check requires unshared skb */ /* csum_check requires unshared skb */
if (!skb_make_writable(skb, udphoff+sizeof(*udph))) if (!skb_make_writable(skb, udphoff+sizeof(*udph)))
...@@ -196,7 +230,7 @@ udp_dnat_handler(struct sk_buff *skb, ...@@ -196,7 +230,7 @@ udp_dnat_handler(struct sk_buff *skb,
if (unlikely(cp->app != NULL)) { if (unlikely(cp->app != NULL)) {
/* Some checks before mangling */ /* Some checks before mangling */
if (pp->csum_check && !pp->csum_check(AF_INET, skb, pp)) if (pp->csum_check && !pp->csum_check(cp->af, skb, pp))
return 0; return 0;
/* /*
...@@ -207,7 +241,7 @@ udp_dnat_handler(struct sk_buff *skb, ...@@ -207,7 +241,7 @@ udp_dnat_handler(struct sk_buff *skb,
return 0; return 0;
} }
udph = (void *)ip_hdr(skb) + udphoff; udph = (void *)skb_network_header(skb) + udphoff;
udph->dest = cp->dport; udph->dest = cp->dport;
/* /*
...@@ -215,7 +249,7 @@ udp_dnat_handler(struct sk_buff *skb, ...@@ -215,7 +249,7 @@ udp_dnat_handler(struct sk_buff *skb,
*/ */
if (!cp->app && (udph->check != 0)) { if (!cp->app && (udph->check != 0)) {
/* Only port and addr are changed, do fast csum update */ /* Only port and addr are changed, do fast csum update */
udp_fast_csum_update(udph, cp->vaddr.ip, cp->daddr.ip, udp_fast_csum_update(cp->af, udph, &cp->vaddr, &cp->daddr,
cp->vport, cp->dport); cp->vport, cp->dport);
if (skb->ip_summed == CHECKSUM_COMPLETE) if (skb->ip_summed == CHECKSUM_COMPLETE)
skb->ip_summed = CHECKSUM_NONE; skb->ip_summed = CHECKSUM_NONE;
...@@ -223,9 +257,19 @@ udp_dnat_handler(struct sk_buff *skb, ...@@ -223,9 +257,19 @@ udp_dnat_handler(struct sk_buff *skb,
/* full checksum calculation */ /* full checksum calculation */
udph->check = 0; udph->check = 0;
skb->csum = skb_checksum(skb, udphoff, skb->len - udphoff, 0); skb->csum = skb_checksum(skb, udphoff, skb->len - udphoff, 0);
udph->check = csum_tcpudp_magic(cp->caddr.ip, cp->daddr.ip, #ifdef CONFIG_IP_VS_IPV6
if (cp->af == AF_INET6)
udph->check = csum_ipv6_magic(&cp->caddr.in6,
&cp->daddr.in6,
skb->len - udphoff, skb->len - udphoff,
cp->protocol, skb->csum); cp->protocol, skb->csum);
else
#endif
udph->check = csum_tcpudp_magic(cp->caddr.ip,
cp->daddr.ip,
skb->len - udphoff,
cp->protocol,
skb->csum);
if (udph->check == 0) if (udph->check == 0)
udph->check = CSUM_MANGLED_0; udph->check = CSUM_MANGLED_0;
skb->ip_summed = CHECKSUM_UNNECESSARY; skb->ip_summed = CHECKSUM_UNNECESSARY;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment