0001 #ifndef __LINUX_ERSPAN_H
0002 #define __LINUX_ERSPAN_H
0003
0004
0005
0006
0007
0008
0009
0010
0011
0012
0013
0014
0015
0016
0017
0018
0019
0020
0021
0022
0023
0024
0025
0026
0027
0028
0029
0030
0031
0032
0033
0034
0035
0036
0037
0038
0039
0040
0041
0042
0043
0044
0045
0046
0047
0048
0049
0050
0051
0052
0053
0054
0055
0056
0057
0058
0059
0060
0061 #include <linux/ip.h>
0062 #include <linux/ipv6.h>
0063 #include <linux/skbuff.h>
0064 #include <uapi/linux/erspan.h>
0065
0066 #define ERSPAN_VERSION 0x1
0067 #define VER_MASK 0xf000
0068 #define VLAN_MASK 0x0fff
0069 #define COS_MASK 0xe000
0070 #define EN_MASK 0x1800
0071 #define T_MASK 0x0400
0072 #define ID_MASK 0x03ff
0073 #define INDEX_MASK 0xfffff
0074
0075 #define ERSPAN_VERSION2 0x2
0076 #define BSO_MASK EN_MASK
0077 #define SGT_MASK 0xffff0000
0078 #define P_MASK 0x8000
0079 #define FT_MASK 0x7c00
0080 #define HWID_MASK 0x03f0
0081 #define DIR_MASK 0x0008
0082 #define GRA_MASK 0x0006
0083 #define O_MASK 0x0001
0084
0085 #define HWID_OFFSET 4
0086 #define DIR_OFFSET 3
0087
0088 enum erspan_encap_type {
0089 ERSPAN_ENCAP_NOVLAN = 0x0,
0090 ERSPAN_ENCAP_ISL = 0x1,
0091 ERSPAN_ENCAP_8021Q = 0x2,
0092 ERSPAN_ENCAP_INFRAME = 0x3,
0093 };
0094
0095 #define ERSPAN_V1_MDSIZE 4
0096 #define ERSPAN_V2_MDSIZE 8
0097
0098 struct erspan_base_hdr {
0099 #if defined(__LITTLE_ENDIAN_BITFIELD)
0100 __u8 vlan_upper:4,
0101 ver:4;
0102 __u8 vlan:8;
0103 __u8 session_id_upper:2,
0104 t:1,
0105 en:2,
0106 cos:3;
0107 __u8 session_id:8;
0108 #elif defined(__BIG_ENDIAN_BITFIELD)
0109 __u8 ver: 4,
0110 vlan_upper:4;
0111 __u8 vlan:8;
0112 __u8 cos:3,
0113 en:2,
0114 t:1,
0115 session_id_upper:2;
0116 __u8 session_id:8;
0117 #else
0118 #error "Please fix <asm/byteorder.h>"
0119 #endif
0120 };
0121
0122 static inline void set_session_id(struct erspan_base_hdr *ershdr, u16 id)
0123 {
0124 ershdr->session_id = id & 0xff;
0125 ershdr->session_id_upper = (id >> 8) & 0x3;
0126 }
0127
0128 static inline u16 get_session_id(const struct erspan_base_hdr *ershdr)
0129 {
0130 return (ershdr->session_id_upper << 8) + ershdr->session_id;
0131 }
0132
0133 static inline void set_vlan(struct erspan_base_hdr *ershdr, u16 vlan)
0134 {
0135 ershdr->vlan = vlan & 0xff;
0136 ershdr->vlan_upper = (vlan >> 8) & 0xf;
0137 }
0138
0139 static inline u16 get_vlan(const struct erspan_base_hdr *ershdr)
0140 {
0141 return (ershdr->vlan_upper << 8) + ershdr->vlan;
0142 }
0143
0144 static inline void set_hwid(struct erspan_md2 *md2, u8 hwid)
0145 {
0146 md2->hwid = hwid & 0xf;
0147 md2->hwid_upper = (hwid >> 4) & 0x3;
0148 }
0149
0150 static inline u8 get_hwid(const struct erspan_md2 *md2)
0151 {
0152 return (md2->hwid_upper << 4) + md2->hwid;
0153 }
0154
0155 static inline int erspan_hdr_len(int version)
0156 {
0157 if (version == 0)
0158 return 0;
0159
0160 return sizeof(struct erspan_base_hdr) +
0161 (version == 1 ? ERSPAN_V1_MDSIZE : ERSPAN_V2_MDSIZE);
0162 }
0163
0164 static inline u8 tos_to_cos(u8 tos)
0165 {
0166 u8 dscp, cos;
0167
0168 dscp = tos >> 2;
0169 cos = dscp >> 3;
0170 return cos;
0171 }
0172
0173 static inline void erspan_build_header(struct sk_buff *skb,
0174 u32 id, u32 index,
0175 bool truncate, bool is_ipv4)
0176 {
0177 struct ethhdr *eth = (struct ethhdr *)skb->data;
0178 enum erspan_encap_type enc_type;
0179 struct erspan_base_hdr *ershdr;
0180 struct qtag_prefix {
0181 __be16 eth_type;
0182 __be16 tci;
0183 } *qp;
0184 u16 vlan_tci = 0;
0185 u8 tos;
0186 __be32 *idx;
0187
0188 tos = is_ipv4 ? ip_hdr(skb)->tos :
0189 (ipv6_hdr(skb)->priority << 4) +
0190 (ipv6_hdr(skb)->flow_lbl[0] >> 4);
0191
0192 enc_type = ERSPAN_ENCAP_NOVLAN;
0193
0194
0195
0196
0197 if (eth->h_proto == htons(ETH_P_8021Q)) {
0198 qp = (struct qtag_prefix *)(skb->data + 2 * ETH_ALEN);
0199 vlan_tci = ntohs(qp->tci);
0200 enc_type = ERSPAN_ENCAP_INFRAME;
0201 }
0202
0203 skb_push(skb, sizeof(*ershdr) + ERSPAN_V1_MDSIZE);
0204 ershdr = (struct erspan_base_hdr *)skb->data;
0205 memset(ershdr, 0, sizeof(*ershdr) + ERSPAN_V1_MDSIZE);
0206
0207
0208 ershdr->ver = ERSPAN_VERSION;
0209 ershdr->cos = tos_to_cos(tos);
0210 ershdr->en = enc_type;
0211 ershdr->t = truncate;
0212 set_vlan(ershdr, vlan_tci);
0213 set_session_id(ershdr, id);
0214
0215
0216 idx = (__be32 *)(ershdr + 1);
0217 *idx = htonl(index & INDEX_MASK);
0218 }
0219
0220
0221
0222
0223
0224
0225
0226 static inline __be32 erspan_get_timestamp(void)
0227 {
0228 u64 h_usecs;
0229 ktime_t kt;
0230
0231 kt = ktime_get_real();
0232 h_usecs = ktime_divns(kt, 100 * NSEC_PER_USEC);
0233
0234
0235
0236
0237 return htonl((u32)h_usecs);
0238 }
0239
0240
0241
0242
0243
0244
0245
0246 enum erspan_bso {
0247 BSO_NOERROR = 0x0,
0248 BSO_SHORT = 0x1,
0249 BSO_OVERSIZED = 0x2,
0250 BSO_BAD = 0x3,
0251 };
0252
0253 static inline u8 erspan_detect_bso(struct sk_buff *skb)
0254 {
0255
0256
0257
0258 if (skb->len < ETH_ZLEN)
0259 return BSO_SHORT;
0260
0261 if (skb->len > ETH_FRAME_LEN)
0262 return BSO_OVERSIZED;
0263
0264 return BSO_NOERROR;
0265 }
0266
0267 static inline void erspan_build_header_v2(struct sk_buff *skb,
0268 u32 id, u8 direction, u16 hwid,
0269 bool truncate, bool is_ipv4)
0270 {
0271 struct ethhdr *eth = (struct ethhdr *)skb->data;
0272 struct erspan_base_hdr *ershdr;
0273 struct erspan_md2 *md2;
0274 struct qtag_prefix {
0275 __be16 eth_type;
0276 __be16 tci;
0277 } *qp;
0278 u16 vlan_tci = 0;
0279 u8 gra = 0;
0280 u8 bso = 0;
0281 u8 sgt = 0;
0282 u8 tos;
0283
0284 tos = is_ipv4 ? ip_hdr(skb)->tos :
0285 (ipv6_hdr(skb)->priority << 4) +
0286 (ipv6_hdr(skb)->flow_lbl[0] >> 4);
0287
0288
0289
0290
0291 if (eth->h_proto == htons(ETH_P_8021Q)) {
0292 qp = (struct qtag_prefix *)(skb->data + 2 * ETH_ALEN);
0293 vlan_tci = ntohs(qp->tci);
0294 }
0295
0296 bso = erspan_detect_bso(skb);
0297 skb_push(skb, sizeof(*ershdr) + ERSPAN_V2_MDSIZE);
0298 ershdr = (struct erspan_base_hdr *)skb->data;
0299 memset(ershdr, 0, sizeof(*ershdr) + ERSPAN_V2_MDSIZE);
0300
0301
0302 ershdr->ver = ERSPAN_VERSION2;
0303 ershdr->cos = tos_to_cos(tos);
0304 ershdr->en = bso;
0305 ershdr->t = truncate;
0306 set_vlan(ershdr, vlan_tci);
0307 set_session_id(ershdr, id);
0308
0309
0310 md2 = (struct erspan_md2 *)(ershdr + 1);
0311 md2->timestamp = erspan_get_timestamp();
0312 md2->sgt = htons(sgt);
0313 md2->p = 1;
0314 md2->ft = 0;
0315 md2->dir = direction;
0316 md2->gra = gra;
0317 md2->o = 0;
0318 set_hwid(md2, hwid);
0319 }
0320
0321 #endif