1/* 2 * Copyright (C) 2015, Broadcom Corporation. All Rights Reserved. 3 * 4 * Permission to use, copy, modify, and/or distribute this software for any 5 * purpose with or without fee is hereby granted, provided that the above 6 * copyright notice and this permission notice appear in all copies. 7 * 8 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES 9 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF 10 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY 11 * SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES 12 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION 13 * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN 14 * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. 15 * 16 * Fundamental constants relating to IP Protocol 17 * 18 * $Id: bcmip.h 449391 2014-01-16 23:23:52Z $ 19 */ 20 21#ifndef _bcmip_h_ 22#define _bcmip_h_ 23 24#ifndef _TYPEDEFS_H_ 25#include <typedefs.h> 26#endif 27 28/* This marks the start of a packed structure section. */ 29#include <packed_section_start.h> 30 31 32/* IPV4 and IPV6 common */ 33#define IP_VER_OFFSET 0x0 /* offset to version field */ 34#define IP_VER_MASK 0xf0 /* version mask */ 35#define IP_VER_SHIFT 4 /* version shift */ 36#define IP_VER_4 4 /* version number for IPV4 */ 37#define IP_VER_6 6 /* version number for IPV6 */ 38 39#define IP_VER(ip_body) \ 40 ((((uint8 *)(ip_body))[IP_VER_OFFSET] & IP_VER_MASK) >> IP_VER_SHIFT) 41 42#define IP_PROT_ICMP 0x1 /* ICMP protocol */ 43#define IP_PROT_IGMP 0x2 /* IGMP protocol */ 44#define IP_PROT_TCP 0x6 /* TCP protocol */ 45#define IP_PROT_UDP 0x11 /* UDP protocol type */ 46#define IP_PROT_ICMP6 0x3a /* ICMPv6 protocol type */ 47 48/* IPV4 field offsets */ 49#define IPV4_VER_HL_OFFSET 0 /* version and ihl byte offset */ 50#define IPV4_TOS_OFFSET 1 /* type of service offset */ 51#define IPV4_PKTLEN_OFFSET 2 /* packet length offset */ 52#define IPV4_PKTFLAG_OFFSET 6 /* more-frag,dont-frag flag offset */ 53#define IPV4_PROT_OFFSET 9 /* protocol type offset */ 54#define IPV4_CHKSUM_OFFSET 10 /* IP header checksum offset */ 55#define IPV4_SRC_IP_OFFSET 12 /* src IP addr offset */ 56#define IPV4_DEST_IP_OFFSET 16 /* dest IP addr offset */ 57#define IPV4_OPTIONS_OFFSET 20 /* IP options offset */ 58#define IPV4_MIN_HEADER_LEN 20 /* Minimum size for an IP header (no options) */ 59 60/* IPV4 field decodes */ 61#define IPV4_VER_MASK 0xf0 /* IPV4 version mask */ 62#define IPV4_VER_SHIFT 4 /* IPV4 version shift */ 63 64#define IPV4_HLEN_MASK 0x0f /* IPV4 header length mask */ 65#define IPV4_HLEN(ipv4_body) (4 * (((uint8 *)(ipv4_body))[IPV4_VER_HL_OFFSET] & IPV4_HLEN_MASK)) 66 67#define IPV4_ADDR_LEN 4 /* IPV4 address length */ 68 69#define IPV4_ADDR_NULL(a) ((((uint8 *)(a))[0] | ((uint8 *)(a))[1] | \ 70 ((uint8 *)(a))[2] | ((uint8 *)(a))[3]) == 0) 71 72#define IPV4_ADDR_BCAST(a) ((((uint8 *)(a))[0] & ((uint8 *)(a))[1] & \ 73 ((uint8 *)(a))[2] & ((uint8 *)(a))[3]) == 0xff) 74 75#define IPV4_TOS_DSCP_MASK 0xfc /* DiffServ codepoint mask */ 76#define IPV4_TOS_DSCP_SHIFT 2 /* DiffServ codepoint shift */ 77 78#define IPV4_TOS(ipv4_body) (((uint8 *)(ipv4_body))[IPV4_TOS_OFFSET]) 79 80#define IPV4_TOS_PREC_MASK 0xe0 /* Historical precedence mask */ 81#define IPV4_TOS_PREC_SHIFT 5 /* Historical precedence shift */ 82 83#define IPV4_TOS_LOWDELAY 0x10 /* Lowest delay requested */ 84#define IPV4_TOS_THROUGHPUT 0x8 /* Best throughput requested */ 85#define IPV4_TOS_RELIABILITY 0x4 /* Most reliable delivery requested */ 86 87#define IPV4_TOS_ROUTINE 0 88#define IPV4_TOS_PRIORITY 1 89#define IPV4_TOS_IMMEDIATE 2 90#define IPV4_TOS_FLASH 3 91#define IPV4_TOS_FLASHOVERRIDE 4 92#define IPV4_TOS_CRITICAL 5 93#define IPV4_TOS_INETWORK_CTRL 6 94#define IPV4_TOS_NETWORK_CTRL 7 95 96#define IPV4_PROT(ipv4_body) (((uint8 *)(ipv4_body))[IPV4_PROT_OFFSET]) 97 98#define IPV4_FRAG_RESV 0x8000 /* Reserved */ 99#define IPV4_FRAG_DONT 0x4000 /* Don't fragment */ 100#define IPV4_FRAG_MORE 0x2000 /* More fragments */ 101#define IPV4_FRAG_OFFSET_MASK 0x1fff /* Fragment offset */ 102 103#define IPV4_ADDR_STR_LEN 16 /* Max IP address length in string format */ 104 105/*borg DTM QoS*/ 106/* IPv4, no options only. */ 107#define IPV4_NO_OPTIONS_HDR_LEN 20 108#define IPV4_NO_OPTIONS_PAYLOAD(ip_hdr) (&(((uint8 *)(ip_hdr))[IPV4_NO_OPTIONS_HDR_LEN])) 109 110#define IPV4_PAYLOAD_LEN(ip_body) \ 111 (((int)(((uint8 *)(ip_body))[IPV4_PKTLEN_OFFSET + 0]) << 8) | \ 112 ((uint8 *)(ip_body))[IPV4_PKTLEN_OFFSET + 1]) 113 114/* IPV4 packet formats */ 115BWL_PRE_PACKED_STRUCT struct ipv4_addr { 116 uint8 addr[IPV4_ADDR_LEN]; 117} BWL_POST_PACKED_STRUCT; 118 119BWL_PRE_PACKED_STRUCT struct ipv4_hdr { 120 uint8 version_ihl; /* Version and Internet Header Length */ 121 uint8 tos; /* Type Of Service */ 122 uint16 tot_len; /* Number of bytes in packet (max 65535) */ 123 uint16 id; 124 uint16 frag; /* 3 flag bits and fragment offset */ 125 uint8 ttl; /* Time To Live */ 126 uint8 prot; /* Protocol */ 127 uint16 hdr_chksum; /* IP header checksum */ 128 uint8 src_ip[IPV4_ADDR_LEN]; /* Source IP Address */ 129 uint8 dst_ip[IPV4_ADDR_LEN]; /* Destination IP Address */ 130} BWL_POST_PACKED_STRUCT; 131 132/* IPV6 field offsets */ 133#define IPV6_PAYLOAD_LEN_OFFSET 4 /* payload length offset */ 134#define IPV6_NEXT_HDR_OFFSET 6 /* next header/protocol offset */ 135#define IPV6_HOP_LIMIT_OFFSET 7 /* hop limit offset */ 136#define IPV6_SRC_IP_OFFSET 8 /* src IP addr offset */ 137#define IPV6_DEST_IP_OFFSET 24 /* dst IP addr offset */ 138 139/* IPV6 field decodes */ 140#define IPV6_TRAFFIC_CLASS(ipv6_body) \ 141 (((((uint8 *)(ipv6_body))[0] & 0x0f) << 4) | \ 142 ((((uint8 *)(ipv6_body))[1] & 0xf0) >> 4)) 143 144#define IPV6_FLOW_LABEL(ipv6_body) \ 145 (((((uint8 *)(ipv6_body))[1] & 0x0f) << 16) | \ 146 (((uint8 *)(ipv6_body))[2] << 8) | \ 147 (((uint8 *)(ipv6_body))[3])) 148 149#define IPV6_PAYLOAD_LEN(ipv6_body) \ 150 ((((uint8 *)(ipv6_body))[IPV6_PAYLOAD_LEN_OFFSET + 0] << 8) | \ 151 ((uint8 *)(ipv6_body))[IPV6_PAYLOAD_LEN_OFFSET + 1]) 152 153#define IPV6_NEXT_HDR(ipv6_body) \ 154 (((uint8 *)(ipv6_body))[IPV6_NEXT_HDR_OFFSET]) 155 156#define IPV6_PROT(ipv6_body) IPV6_NEXT_HDR(ipv6_body) 157 158#define IPV6_ADDR_LEN 16 /* IPV6 address length */ 159 160/* IPV4 TOS or IPV6 Traffic Classifier or 0 */ 161#define IP_TOS46(ip_body) \ 162 (IP_VER(ip_body) == IP_VER_4 ? IPV4_TOS(ip_body) : \ 163 IP_VER(ip_body) == IP_VER_6 ? IPV6_TRAFFIC_CLASS(ip_body) : 0) 164 165#define IP_DSCP46(ip_body) (IP_TOS46(ip_body) >> IPV4_TOS_DSCP_SHIFT); 166 167/* IPV4 or IPV6 Protocol Classifier or 0 */ 168#define IP_PROT46(ip_body) \ 169 (IP_VER(ip_body) == IP_VER_4 ? IPV4_PROT(ip_body) : \ 170 IP_VER(ip_body) == IP_VER_6 ? IPV6_PROT(ip_body) : 0) 171 172/* IPV6 extension headers (options) */ 173#define IPV6_EXTHDR_HOP 0 174#define IPV6_EXTHDR_ROUTING 43 175#define IPV6_EXTHDR_FRAGMENT 44 176#define IPV6_EXTHDR_AUTH 51 177#define IPV6_EXTHDR_NONE 59 178#define IPV6_EXTHDR_DEST 60 179 180#define IPV6_EXTHDR(prot) (((prot) == IPV6_EXTHDR_HOP) || \ 181 ((prot) == IPV6_EXTHDR_ROUTING) || \ 182 ((prot) == IPV6_EXTHDR_FRAGMENT) || \ 183 ((prot) == IPV6_EXTHDR_AUTH) || \ 184 ((prot) == IPV6_EXTHDR_NONE) || \ 185 ((prot) == IPV6_EXTHDR_DEST)) 186 187#define IPV6_MIN_HLEN 40 188 189#define IPV6_EXTHDR_LEN(eh) ((((struct ipv6_exthdr *)(eh))->hdrlen + 1) << 3) 190 191BWL_PRE_PACKED_STRUCT struct ipv6_exthdr { 192 uint8 nexthdr; 193 uint8 hdrlen; 194} BWL_POST_PACKED_STRUCT; 195 196BWL_PRE_PACKED_STRUCT struct ipv6_exthdr_frag { 197 uint8 nexthdr; 198 uint8 rsvd; 199 uint16 frag_off; 200 uint32 ident; 201} BWL_POST_PACKED_STRUCT; 202 203static INLINE int32 204ipv6_exthdr_len(uint8 *h, uint8 *proto) 205{ 206 uint16 len = 0, hlen; 207 struct ipv6_exthdr *eh = (struct ipv6_exthdr *)h; 208 209 while (IPV6_EXTHDR(eh->nexthdr)) { 210 if (eh->nexthdr == IPV6_EXTHDR_NONE) 211 return -1; 212 else if (eh->nexthdr == IPV6_EXTHDR_FRAGMENT) 213 hlen = 8; 214 else if (eh->nexthdr == IPV6_EXTHDR_AUTH) 215 hlen = (eh->hdrlen + 2) << 2; 216 else 217 hlen = IPV6_EXTHDR_LEN(eh); 218 219 len += hlen; 220 eh = (struct ipv6_exthdr *)(h + len); 221 } 222 223 *proto = eh->nexthdr; 224 return len; 225} 226 227#define IPV4_ISMULTI(a) (((a) & 0xf0000000) == 0xe0000000) 228 229#define IPV4_MCAST_TO_ETHER_MCAST(ipv4, ether) \ 230{ \ 231 ether[0] = 0x01; \ 232 ether[1] = 0x00; \ 233 ether[2] = 0x5E; \ 234 ether[3] = (ipv4 & 0x7f0000) >> 16; \ 235 ether[4] = (ipv4 & 0xff00) >> 8; \ 236 ether[5] = (ipv4 & 0xff); \ 237} 238 239/* This marks the end of a packed structure section. */ 240#include <packed_section_end.h> 241 242#endif /* _bcmip_h_ */ 243