Commit 4d4eae2c authored by Herbert Xu's avatar Herbert Xu Committed by David S. Miller

[INET]: Create enum of ECN bits

This patch is a preparation for an update of the ECN encap/decap
code with respect to RFC3168.

It creates an enum of the four code-points defined by RFC3168
and uses them throughout the inet_ecn.h file.

The only non-trivial bit is in IP_ECN_set_ce/IP6_ECN_set_ce where
the patch uses INET_ECN_CE instead of 1.  This is OK as those
functions assume that the ECT bit is already set.
Signed-off-by: default avatarHerbert Xu <herbert@gondor.apana.org.au>
Signed-off-by: default avatarDavid S. Miller <davem@redhat.com>
parent 84d6ce44
...@@ -3,39 +3,48 @@ ...@@ -3,39 +3,48 @@
#include <linux/ip.h> #include <linux/ip.h>
enum {
INET_ECN_NOT_ECT = 0,
INET_ECN_ECT_1 = 1,
INET_ECN_ECT_0 = 2,
INET_ECN_CE = 3,
INET_ECN_MASK = 3,
};
static inline int INET_ECN_is_ce(__u8 dsfield) static inline int INET_ECN_is_ce(__u8 dsfield)
{ {
return (dsfield&3) == 3; return (dsfield & INET_ECN_MASK) == INET_ECN_CE;
} }
static inline int INET_ECN_is_not_ce(__u8 dsfield) static inline int INET_ECN_is_not_ce(__u8 dsfield)
{ {
return (dsfield&3) == 2; return (dsfield & INET_ECN_MASK) == INET_ECN_ECT_0;
} }
static inline int INET_ECN_is_capable(__u8 dsfield) static inline int INET_ECN_is_capable(__u8 dsfield)
{ {
return (dsfield&2); return (dsfield & INET_ECN_ECT_0);
} }
static inline __u8 INET_ECN_encapsulate(__u8 outer, __u8 inner) static inline __u8 INET_ECN_encapsulate(__u8 outer, __u8 inner)
{ {
outer &= ~3; outer &= ~INET_ECN_MASK;
if (INET_ECN_is_capable(inner)) if (INET_ECN_is_capable(inner))
outer |= (inner & 3); outer |= (inner & INET_ECN_MASK);
return outer; return outer;
} }
#define INET_ECN_xmit(sk) do { inet_sk(sk)->tos |= 2; } while (0) #define INET_ECN_xmit(sk) do { inet_sk(sk)->tos |= INET_ECN_ECT_0; } while (0)
#define INET_ECN_dontxmit(sk) do { inet_sk(sk)->tos &= ~3; } while (0) #define INET_ECN_dontxmit(sk) \
do { inet_sk(sk)->tos &= ~INET_ECN_MASK; } while (0)
#define IP6_ECN_flow_init(label) do { \ #define IP6_ECN_flow_init(label) do { \
(label) &= ~htonl(3<<20); \ (label) &= ~htonl(INET_ECN_MASK << 20); \
} while (0) } while (0)
#define IP6_ECN_flow_xmit(sk, label) do { \ #define IP6_ECN_flow_xmit(sk, label) do { \
if (INET_ECN_is_capable(inet_sk(sk)->tos)) \ if (INET_ECN_is_capable(inet_sk(sk)->tos)) \
(label) |= __constant_htons(2 << 4); \ (label) |= __constant_htons(INET_ECN_ECT_0 << 4); \
} while (0) } while (0)
static inline void IP_ECN_set_ce(struct iphdr *iph) static inline void IP_ECN_set_ce(struct iphdr *iph)
...@@ -43,24 +52,24 @@ static inline void IP_ECN_set_ce(struct iphdr *iph) ...@@ -43,24 +52,24 @@ static inline void IP_ECN_set_ce(struct iphdr *iph)
u32 check = iph->check; u32 check = iph->check;
check += __constant_htons(0xFFFE); check += __constant_htons(0xFFFE);
iph->check = check + (check>=0xFFFF); iph->check = check + (check>=0xFFFF);
iph->tos |= 1; iph->tos |= INET_ECN_CE;
} }
static inline void IP_ECN_clear(struct iphdr *iph) static inline void IP_ECN_clear(struct iphdr *iph)
{ {
iph->tos &= ~3; iph->tos &= ~INET_ECN_MASK;
} }
struct ipv6hdr; struct ipv6hdr;
static inline void IP6_ECN_set_ce(struct ipv6hdr *iph) static inline void IP6_ECN_set_ce(struct ipv6hdr *iph)
{ {
*(u32*)iph |= htonl(1<<20); *(u32*)iph |= htonl(INET_ECN_CE << 20);
} }
static inline void IP6_ECN_clear(struct ipv6hdr *iph) static inline void IP6_ECN_clear(struct ipv6hdr *iph)
{ {
*(u32*)iph &= ~htonl(3<<20); *(u32*)iph &= ~htonl(INET_ECN_MASK << 20);
} }
#define ip6_get_dsfield(iph) ((ntohs(*(u16*)(iph)) >> 4) & 0xFF) #define ip6_get_dsfield(iph) ((ntohs(*(u16*)(iph)) >> 4) & 0xFF)
......
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