blob: b94ca50bad29357dbdc5d0dea3e3ff1730d1ae25 [file] [log] [blame]
Yi Tsengbe342052017-11-03 10:21:23 -07001/*
2 * Copyright 2017-present Open Networking Foundation
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#ifndef __PARSER__
18#define __PARSER__
19
20#include "define.p4"
21
22parser FabricParser (
23packet_in packet,
24out parsed_headers_t hdr,
25inout fabric_metadata_t fabric_metadata,
26inout standard_metadata_t standard_metadata) {
Yi Tseng1d842672017-11-28 16:06:52 -080027
Yi Tsengbe342052017-11-03 10:21:23 -070028 state start {
29 transition select(standard_metadata.ingress_port) {
30 CPU_PORT: parse_packet_out;
31 default: parse_ethernet;
32 }
33 }
34
35 state parse_packet_out {
36 packet.extract(hdr.packet_out);
37 transition parse_ethernet;
38 }
39
40 state parse_ethernet {
41 packet.extract(hdr.ethernet);
42 transition select(hdr.ethernet.ether_type){
Yi Tsengbe342052017-11-03 10:21:23 -070043 ETHERTYPE_VLAN: parse_vlan_tag;
44 ETHERTYPE_MPLS: parse_mpls;
45 ETHERTYPE_ARP: parse_arp;
46 ETHERTYPE_IPV4: parse_ipv4;
Carmelo Casconeed88f2b2018-01-26 17:36:34 -080047#ifdef WITH_IPV6
Yi Tsengbe342052017-11-03 10:21:23 -070048 ETHERTYPE_IPV6: parse_ipv6;
Carmelo Casconeed88f2b2018-01-26 17:36:34 -080049#endif // WITH_IPV6
Yi Tsengbe342052017-11-03 10:21:23 -070050 default: accept;
51 }
52 }
53
54 state parse_vlan_tag {
55 packet.extract(hdr.vlan_tag);
56 transition select(hdr.vlan_tag.ether_type){
Yi Tsengbe342052017-11-03 10:21:23 -070057 ETHERTYPE_ARP: parse_arp;
58 ETHERTYPE_IPV4: parse_ipv4;
Carmelo Casconeed88f2b2018-01-26 17:36:34 -080059#ifdef WITH_IPV6
Yi Tsengbe342052017-11-03 10:21:23 -070060 ETHERTYPE_IPV6: parse_ipv6;
Carmelo Casconeed88f2b2018-01-26 17:36:34 -080061#endif // WITH_IPV6
Yi Tsengbd46d052018-01-22 17:18:16 -080062 ETHERTYPE_MPLS: parse_mpls;
Yi Tsengbe342052017-11-03 10:21:23 -070063 default: accept;
64 }
65 }
66
67 state parse_mpls {
68 packet.extract(hdr.mpls);
Yi Tsengc6844f52017-12-19 11:58:25 -080069 // There is only one MPLS label for this fabric.
70 // Assume header after MPLS header is IP/IPv6
71 // Lookup first 4 bits for version
Yi Tseng3d3956d2018-01-31 17:28:05 -080072 transition select(packet.lookahead<bit<IP_VER_LENGTH>>()) {
Yi Tsengbe342052017-11-03 10:21:23 -070073 //The packet should be either IPv4 or IPv6.
74 IP_VERSION_4: parse_ipv4;
Carmelo Casconeed88f2b2018-01-26 17:36:34 -080075#ifdef WITH_IPV6
Yi Tsengbe342052017-11-03 10:21:23 -070076 IP_VERSION_6: parse_ipv6;
Carmelo Casconeed88f2b2018-01-26 17:36:34 -080077#endif // WITH_IPV6
Yi Tsengbe342052017-11-03 10:21:23 -070078 default: parse_ethernet;
79 }
80 }
81
82 state parse_ipv4 {
83 packet.extract(hdr.ipv4);
84 fabric_metadata.ip_proto = hdr.ipv4.protocol;
85 //Need header verification?
86 transition select(hdr.ipv4.protocol) {
87 PROTO_TCP: parse_tcp;
88 PROTO_UDP: parse_udp;
89 PROTO_ICMP: parse_icmp;
90 default: accept;
91 }
92 }
93
Carmelo Casconeed88f2b2018-01-26 17:36:34 -080094#ifdef WITH_IPV6
Yi Tsengbe342052017-11-03 10:21:23 -070095 state parse_ipv6 {
96 packet.extract(hdr.ipv6);
97 fabric_metadata.ip_proto = hdr.ipv6.next_hdr;
98 transition select(hdr.ipv6.next_hdr) {
99 PROTO_TCP: parse_tcp;
100 PROTO_UDP: parse_udp;
101 PROTO_ICMPV6: parse_icmp;
102 default: accept;
103 }
104 }
Carmelo Casconeed88f2b2018-01-26 17:36:34 -0800105#endif // WITH_IPV6
Yi Tsengbe342052017-11-03 10:21:23 -0700106
107 state parse_arp {
108 packet.extract(hdr.arp);
109 transition accept;
110 }
111
112 state parse_tcp {
113 packet.extract(hdr.tcp);
114 fabric_metadata.l4_src_port = hdr.tcp.src_port;
115 fabric_metadata.l4_dst_port = hdr.tcp.dst_port;
Jonghwan Hyuned478dc2018-08-06 15:35:18 +0900116#ifdef WITH_INT
117 transition select(hdr.ipv4.isValid() && ((hdr.ipv4.dscp & INT_DSCP) == INT_DSCP)) {
118 true: parse_intl4_shim;
119 default: accept;
120 }
121#else
Yi Tsengbe342052017-11-03 10:21:23 -0700122 transition accept;
Jonghwan Hyuned478dc2018-08-06 15:35:18 +0900123#endif // WITH_INT
Yi Tsengbe342052017-11-03 10:21:23 -0700124 }
125
126 state parse_udp {
127 packet.extract(hdr.udp);
128 fabric_metadata.l4_src_port = hdr.udp.src_port;
129 fabric_metadata.l4_dst_port = hdr.udp.dst_port;
Carmelo Casconeb81f4be2018-01-16 23:24:01 -0800130#ifdef WITH_SPGW
131 transition select(hdr.udp.dst_port) {
132 UDP_PORT_GTPU: parse_gtpu;
133 default: accept;
134 }
Jonghwan Hyuned478dc2018-08-06 15:35:18 +0900135#elif WITH_INT
136 transition select(hdr.ipv4.isValid() && (hdr.ipv4.dscp & INT_DSCP) == INT_DSCP) {
137 true: parse_intl4_shim;
138 default: accept;
139 }
Carmelo Casconeb81f4be2018-01-16 23:24:01 -0800140#else
Yi Tsengbe342052017-11-03 10:21:23 -0700141 transition accept;
Jonghwan Hyuned478dc2018-08-06 15:35:18 +0900142#endif // WITH_SPGW, WITH_INT
Yi Tsengbe342052017-11-03 10:21:23 -0700143 }
144
145 state parse_icmp {
146 packet.extract(hdr.icmp);
147 transition accept;
148 }
Carmelo Casconeb81f4be2018-01-16 23:24:01 -0800149
Jonghwan Hyuned478dc2018-08-06 15:35:18 +0900150#ifdef WITH_INT
151 state parse_intl4_shim {
152 packet.extract(hdr.intl4_shim);
153 transition parse_int_header;
154 }
155
156 state parse_int_header {
157 packet.extract(hdr.int_header);
158 // If there is no INT metadata but the INT header (and corresponding shim header
159 // and tail header) exists, default value of length field in shim header
160 // should be INT_HEADER_LEN_WORD.
161 fabric_metadata.int_meta.metadata_len = hdr.intl4_shim.len - INT_HEADER_LEN_WORD;
162 transition select (fabric_metadata.int_meta.metadata_len) {
163 0: parse_intl4_tail;
164 default: parse_int_data;
165 }
166 }
167
168 state parse_int_data {
169 // Parse INT metadata, not INT header, INT shim header and INT tail header
170 packet.extract(hdr.int_data, (bit<32>) ((hdr.intl4_shim.len - INT_HEADER_LEN_WORD) << 5));
171 transition parse_intl4_tail;
172 }
173
174 state parse_intl4_tail {
175 packet.extract(hdr.intl4_tail);
176 transition accept;
177 }
178#endif // WITH_INT
179
Carmelo Casconeb81f4be2018-01-16 23:24:01 -0800180#ifdef WITH_SPGW
181 state parse_gtpu {
182 packet.extract(hdr.gtpu);
183 transition parse_ipv4_inner;
184 }
185
186 state parse_ipv4_inner {
Carmelo Casconeb757dbc2018-01-25 17:53:17 -0800187 packet.extract(hdr.gtpu_ipv4);
188 transition select(hdr.gtpu_ipv4.protocol) {
Carmelo Casconeb81f4be2018-01-16 23:24:01 -0800189 PROTO_TCP: parse_tcp;
190 PROTO_UDP: parse_udp_inner;
191 PROTO_ICMP: parse_icmp;
192 default: accept;
193 }
194 }
195
196 state parse_udp_inner {
Carmelo Casconeb757dbc2018-01-25 17:53:17 -0800197 packet.extract(hdr.gtpu_udp);
198 fabric_metadata.l4_src_port = hdr.gtpu_udp.src_port;
199 fabric_metadata.l4_dst_port = hdr.gtpu_udp.dst_port;
Jonghwan Hyuned478dc2018-08-06 15:35:18 +0900200#ifdef WITH_INT
201 transition select(hdr.ipv4.isValid() && (hdr.ipv4.dscp & INT_DSCP) == INT_DSCP) {
202 true: parse_intl4_shim;
203 default: accept;
204 }
205#else
Carmelo Casconeb81f4be2018-01-16 23:24:01 -0800206 transition accept;
Jonghwan Hyuned478dc2018-08-06 15:35:18 +0900207#endif // WITH_INT
Carmelo Casconeb81f4be2018-01-16 23:24:01 -0800208 }
209#endif // WITH_SPGW
Yi Tsengbe342052017-11-03 10:21:23 -0700210}
211
212control FabricDeparser(packet_out packet, in parsed_headers_t hdr) {
Yi Tseng3d3956d2018-01-31 17:28:05 -0800213 apply {
Yi Tsengbe342052017-11-03 10:21:23 -0700214 packet.emit(hdr.packet_in);
Jonghwan Hyuned478dc2018-08-06 15:35:18 +0900215#ifdef WITH_INT
216 packet.emit(hdr.report_ethernet);
217 packet.emit(hdr.report_ipv4);
218 packet.emit(hdr.report_udp);
219 packet.emit(hdr.report_fixed_header);
220#endif // WITH_INT
Yi Tsengbe342052017-11-03 10:21:23 -0700221 packet.emit(hdr.ethernet);
222 packet.emit(hdr.vlan_tag);
Yi Tsengbe342052017-11-03 10:21:23 -0700223 packet.emit(hdr.mpls);
224 packet.emit(hdr.arp);
Carmelo Casconeb81f4be2018-01-16 23:24:01 -0800225#ifdef WITH_SPGW
226 packet.emit(hdr.gtpu_ipv4);
227 packet.emit(hdr.gtpu_udp);
228 packet.emit(hdr.gtpu);
229#endif // WITH_SPGW
Yi Tsengbe342052017-11-03 10:21:23 -0700230 packet.emit(hdr.ipv4);
Carmelo Casconeed88f2b2018-01-26 17:36:34 -0800231#ifdef WITH_IPV6
Yi Tsengbe342052017-11-03 10:21:23 -0700232 packet.emit(hdr.ipv6);
Carmelo Casconeed88f2b2018-01-26 17:36:34 -0800233#endif // WITH_IPV6
Yi Tsengbe342052017-11-03 10:21:23 -0700234 packet.emit(hdr.tcp);
235 packet.emit(hdr.udp);
Yi Tsengf73a5532017-11-17 15:58:57 -0800236 packet.emit(hdr.icmp);
Jonghwan Hyuned478dc2018-08-06 15:35:18 +0900237#ifdef WITH_INT
238 packet.emit(hdr.intl4_shim);
239 packet.emit(hdr.int_header);
240 packet.emit(hdr.int_switch_id);
241 packet.emit(hdr.int_port_ids);
242 packet.emit(hdr.int_hop_latency);
243 packet.emit(hdr.int_q_occupancy);
244 packet.emit(hdr.int_ingress_tstamp);
245 packet.emit(hdr.int_egress_tstamp);
246 packet.emit(hdr.int_q_congestion);
247 packet.emit(hdr.int_egress_tx_util);
248 packet.emit(hdr.int_data);
249 packet.emit(hdr.intl4_tail);
250#endif // WITH_INT
Yi Tsengbe342052017-11-03 10:21:23 -0700251 }
252}
253
254#endif