blob: 086925bd3052d4cf542d7486fd4ebad6e42ff28e [file] [log] [blame]
Ed Warnickecb9cada2015-12-08 15:45:58 -07001/*
2 * Copyright (c) 2015 Cisco and/or its affiliates.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at:
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15/*
16 * llc_node.c: llc packet processing
17 *
18 * Copyright (c) 2010 Eliot Dresselhaus
19 *
20 * Permission is hereby granted, free of charge, to any person obtaining
21 * a copy of this software and associated documentation files (the
22 * "Software"), to deal in the Software without restriction, including
23 * without limitation the rights to use, copy, modify, merge, publish,
24 * distribute, sublicense, and/or sell copies of the Software, and to
25 * permit persons to whom the Software is furnished to do so, subject to
26 * the following conditions:
27 *
28 * The above copyright notice and this permission notice shall be
29 * included in all copies or substantial portions of the Software.
30 *
31 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
32 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
33 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
34 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
35 * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
36 * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
37 * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
38 */
39
40#include <vlib/vlib.h>
Neale Ranns68d48d92021-06-03 14:59:47 +000041#include <vnet/pg/pg.h>
Ed Warnickecb9cada2015-12-08 15:45:58 -070042#include <vnet/llc/llc.h>
43
44#define foreach_llc_input_next \
45 _ (PUNT, "error-punt") \
46 _ (DROP, "error-drop")
47
Calvin151fb722016-08-24 10:35:59 -040048typedef enum
49{
Ed Warnickecb9cada2015-12-08 15:45:58 -070050#define _(s,n) LLC_INPUT_NEXT_##s,
51 foreach_llc_input_next
52#undef _
Calvin151fb722016-08-24 10:35:59 -040053 LLC_INPUT_N_NEXT,
Ed Warnickecb9cada2015-12-08 15:45:58 -070054} llc_input_next_t;
55
Calvin151fb722016-08-24 10:35:59 -040056typedef struct
57{
Ed Warnickecb9cada2015-12-08 15:45:58 -070058 u8 packet_data[32];
59} llc_input_trace_t;
60
Calvin151fb722016-08-24 10:35:59 -040061static u8 *
62format_llc_input_trace (u8 * s, va_list * va)
Ed Warnickecb9cada2015-12-08 15:45:58 -070063{
64 CLIB_UNUSED (vlib_main_t * vm) = va_arg (*va, vlib_main_t *);
65 CLIB_UNUSED (vlib_node_t * node) = va_arg (*va, vlib_node_t *);
Calvin151fb722016-08-24 10:35:59 -040066 llc_input_trace_t *t = va_arg (*va, llc_input_trace_t *);
Ed Warnickecb9cada2015-12-08 15:45:58 -070067
68 s = format (s, "%U", format_llc_header, t->packet_data);
69
70 return s;
71}
72
73static uword
74llc_input (vlib_main_t * vm,
Calvin151fb722016-08-24 10:35:59 -040075 vlib_node_runtime_t * node, vlib_frame_t * from_frame)
Ed Warnickecb9cada2015-12-08 15:45:58 -070076{
Calvin151fb722016-08-24 10:35:59 -040077 llc_main_t *lm = &llc_main;
78 u32 n_left_from, next_index, *from, *to_next;
Ed Warnickecb9cada2015-12-08 15:45:58 -070079
80 from = vlib_frame_vector_args (from_frame);
81 n_left_from = from_frame->n_vectors;
82
83 if (node->flags & VLIB_NODE_FLAG_TRACE)
84 vlib_trace_frame_buffers_only (vm, node,
85 from,
86 n_left_from,
87 sizeof (from[0]),
88 sizeof (llc_input_trace_t));
89
90 next_index = node->cached_next_index;
91
92 while (n_left_from > 0)
93 {
94 u32 n_left_to_next;
95
Calvin151fb722016-08-24 10:35:59 -040096 vlib_get_next_frame (vm, node, next_index, to_next, n_left_to_next);
Ed Warnickecb9cada2015-12-08 15:45:58 -070097
98 while (n_left_from >= 4 && n_left_to_next >= 2)
99 {
100 u32 bi0, bi1;
Calvin151fb722016-08-24 10:35:59 -0400101 vlib_buffer_t *b0, *b1;
102 llc_header_t *h0, *h1;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700103 u8 next0, next1, len0, len1, enqueue_code;
104
105 /* Prefetch next iteration. */
106 {
Calvin151fb722016-08-24 10:35:59 -0400107 vlib_buffer_t *b2, *b3;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700108
109 b2 = vlib_get_buffer (vm, from[2]);
110 b3 = vlib_get_buffer (vm, from[3]);
111
112 vlib_prefetch_buffer_header (b2, LOAD);
113 vlib_prefetch_buffer_header (b3, LOAD);
114
115 CLIB_PREFETCH (b2->data, sizeof (h0[0]), LOAD);
116 CLIB_PREFETCH (b3->data, sizeof (h1[0]), LOAD);
117 }
118
119 bi0 = from[0];
120 bi1 = from[1];
121 to_next[0] = bi0;
122 to_next[1] = bi1;
123 from += 2;
124 to_next += 2;
125 n_left_to_next -= 2;
126 n_left_from -= 2;
127
128 b0 = vlib_get_buffer (vm, bi0);
129 b1 = vlib_get_buffer (vm, bi1);
130
Zhiyong Yangd7d24992019-05-28 00:57:39 -0400131 h0 = vlib_buffer_get_current (b0);
132 h1 = vlib_buffer_get_current (b1);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700133
134 len0 = llc_header_length (h0);
135 len1 = llc_header_length (h1);
136
Zhiyong Yangd7d24992019-05-28 00:57:39 -0400137 vlib_buffer_advance (b0, len0);
138 vlib_buffer_advance (b1, len1);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700139
140 next0 = lm->input_next_by_protocol[h0->dst_sap];
141 next1 = lm->input_next_by_protocol[h1->dst_sap];
142
Calvin151fb722016-08-24 10:35:59 -0400143 b0->error =
144 node->errors[next0 ==
145 LLC_INPUT_NEXT_DROP ? LLC_ERROR_UNKNOWN_PROTOCOL :
146 LLC_ERROR_NONE];
147 b1->error =
148 node->errors[next1 ==
149 LLC_INPUT_NEXT_DROP ? LLC_ERROR_UNKNOWN_PROTOCOL :
150 LLC_ERROR_NONE];
Ed Warnickecb9cada2015-12-08 15:45:58 -0700151
Calvin151fb722016-08-24 10:35:59 -0400152 enqueue_code = (next0 != next_index) + 2 * (next1 != next_index);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700153
154 if (PREDICT_FALSE (enqueue_code != 0))
155 {
156 switch (enqueue_code)
157 {
158 case 1:
159 /* A B A */
160 to_next[-2] = bi1;
161 to_next -= 1;
162 n_left_to_next += 1;
163 vlib_set_next_frame_buffer (vm, node, next0, bi0);
164 break;
165
166 case 2:
167 /* A A B */
168 to_next -= 1;
169 n_left_to_next += 1;
170 vlib_set_next_frame_buffer (vm, node, next1, bi1);
171 break;
172
173 case 3:
174 /* A B B or A B C */
175 to_next -= 2;
176 n_left_to_next += 2;
177 vlib_set_next_frame_buffer (vm, node, next0, bi0);
178 vlib_set_next_frame_buffer (vm, node, next1, bi1);
179 if (next0 == next1)
180 {
181 vlib_put_next_frame (vm, node, next_index,
182 n_left_to_next);
183 next_index = next1;
Calvin151fb722016-08-24 10:35:59 -0400184 vlib_get_next_frame (vm, node, next_index, to_next,
185 n_left_to_next);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700186 }
187 }
188 }
189 }
Calvin151fb722016-08-24 10:35:59 -0400190
Ed Warnickecb9cada2015-12-08 15:45:58 -0700191 while (n_left_from > 0 && n_left_to_next > 0)
192 {
193 u32 bi0;
Calvin151fb722016-08-24 10:35:59 -0400194 vlib_buffer_t *b0;
195 llc_header_t *h0;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700196 u8 next0, len0;
197
198 bi0 = from[0];
199 to_next[0] = bi0;
200 from += 1;
201 to_next += 1;
202 n_left_from -= 1;
203 n_left_to_next -= 1;
204
205 b0 = vlib_get_buffer (vm, bi0);
206
Zhiyong Yangd7d24992019-05-28 00:57:39 -0400207 h0 = vlib_buffer_get_current (b0);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700208
209 len0 = llc_header_length (h0);
210
Zhiyong Yangd7d24992019-05-28 00:57:39 -0400211 vlib_buffer_advance (b0, len0);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700212
213 next0 = lm->input_next_by_protocol[h0->dst_sap];
214
Calvin151fb722016-08-24 10:35:59 -0400215 b0->error =
216 node->errors[next0 ==
217 LLC_INPUT_NEXT_DROP ? LLC_ERROR_UNKNOWN_PROTOCOL :
218 LLC_ERROR_NONE];
Ed Warnickecb9cada2015-12-08 15:45:58 -0700219
220 /* Sent packet to wrong next? */
221 if (PREDICT_FALSE (next0 != next_index))
222 {
223 /* Return old frame; remove incorrectly enqueued packet. */
224 vlib_put_next_frame (vm, node, next_index, n_left_to_next + 1);
225
226 /* Send to correct next. */
227 next_index = next0;
228 vlib_get_next_frame (vm, node, next_index,
229 to_next, n_left_to_next);
230
231 to_next[0] = bi0;
232 to_next += 1;
233 n_left_to_next -= 1;
234 }
235 }
236
237 vlib_put_next_frame (vm, node, next_index, n_left_to_next);
238 }
239
240 return from_frame->n_vectors;
241}
242
Calvin151fb722016-08-24 10:35:59 -0400243static char *llc_error_strings[] = {
Ed Warnickecb9cada2015-12-08 15:45:58 -0700244#define _(f,s) s,
245 foreach_llc_error
246#undef _
247};
248
Calvin151fb722016-08-24 10:35:59 -0400249/* *INDENT-OFF* */
Ed Warnickecb9cada2015-12-08 15:45:58 -0700250VLIB_REGISTER_NODE (llc_input_node) = {
251 .function = llc_input,
252 .name = "llc-input",
253 /* Takes a vector of packets. */
254 .vector_size = sizeof (u32),
255
256 .n_errors = LLC_N_ERROR,
257 .error_strings = llc_error_strings,
258
259 .n_next_nodes = LLC_INPUT_N_NEXT,
260 .next_nodes = {
261#define _(s,n) [LLC_INPUT_NEXT_##s] = n,
262 foreach_llc_input_next
263#undef _
264 },
265
266 .format_buffer = format_llc_header_with_length,
267 .format_trace = format_llc_input_trace,
268 .unformat_buffer = unformat_llc_header,
269};
Calvin151fb722016-08-24 10:35:59 -0400270/* *INDENT-ON* */
Ed Warnickecb9cada2015-12-08 15:45:58 -0700271
Neale Ranns68d48d92021-06-03 14:59:47 +0000272static void
273llc_setup_node (vlib_main_t *vm, u32 node_index)
274{
275 vlib_node_t *n = vlib_get_node (vm, node_index);
276 pg_node_t *pn = pg_get_node (node_index);
277
278 n->format_buffer = format_llc_header_with_length;
279 n->unformat_buffer = unformat_llc_header;
280 pn->unformat_edit = unformat_pg_llc_header;
281}
282
Calvin151fb722016-08-24 10:35:59 -0400283static clib_error_t *
284llc_input_init (vlib_main_t * vm)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700285{
Calvin151fb722016-08-24 10:35:59 -0400286 llc_main_t *lm = &llc_main;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700287
288 {
Calvin151fb722016-08-24 10:35:59 -0400289 clib_error_t *error = vlib_call_init_function (vm, llc_init);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700290 if (error)
291 clib_error_report (error);
292 }
293
294 llc_setup_node (vm, llc_input_node.index);
295
296 {
297 int i;
298 for (i = 0; i < ARRAY_LEN (lm->input_next_by_protocol); i++)
299 lm->input_next_by_protocol[i] = LLC_INPUT_NEXT_DROP;
300 }
301
302 return 0;
303}
304
305VLIB_INIT_FUNCTION (llc_input_init);
306
307void
308llc_register_input_protocol (vlib_main_t * vm,
Calvin151fb722016-08-24 10:35:59 -0400309 llc_protocol_t protocol, u32 node_index)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700310{
Calvin151fb722016-08-24 10:35:59 -0400311 llc_main_t *lm = &llc_main;
312 llc_protocol_info_t *pi;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700313
314 {
Calvin151fb722016-08-24 10:35:59 -0400315 clib_error_t *error = vlib_call_init_function (vm, llc_input_init);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700316 if (error)
317 clib_error_report (error);
Dave Barachced48e72016-02-08 15:57:35 -0500318 /* Otherwise, osi_input_init will wipe out e.g. the snap init */
319 error = vlib_call_init_function (vm, osi_input_init);
320 if (error)
321 clib_error_report (error);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700322 }
323
324 pi = llc_get_protocol_info (lm, protocol);
325 pi->node_index = node_index;
Calvin151fb722016-08-24 10:35:59 -0400326 pi->next_index = vlib_node_add_next (vm, llc_input_node.index, node_index);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700327
328 lm->input_next_by_protocol[protocol] = pi->next_index;
329}
Calvin151fb722016-08-24 10:35:59 -0400330
331/*
332 * fd.io coding-style-patch-verification: ON
333 *
334 * Local Variables:
335 * eval: (c-set-style "gnu")
336 * End:
337 */