| /* |
| * Copyright (c) 2015 Cisco and/or its affiliates. |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at: |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #ifndef __IP_PUNT_DROP_H__ |
| #define __IP_PUNT_DROP_H__ |
| |
| #include <vnet/ip/ip.h> |
| #include <vnet/policer/policer.h> |
| #include <vnet/policer/police_inlines.h> |
| |
| /** |
| * IP4 punt policer configuration |
| * we police the punt rate to prevent overloading the host |
| */ |
| typedef struct ip_punt_policer_t_ |
| { |
| u32 policer_index; |
| u32 fq_index; |
| } ip_punt_policer_t; |
| |
| typedef enum ip_punt_policer_next_t_ |
| { |
| IP_PUNT_POLICER_NEXT_DROP, |
| IP_PUNT_POLICER_NEXT_HANDOFF, |
| IP_PUNT_POLICER_N_NEXT, |
| } ip_punt_policer_next_t; |
| |
| typedef struct ip_punt_policer_trace_t_ |
| { |
| u32 policer_index; |
| u32 next; |
| } ip_punt_policer_trace_t; |
| |
| #define foreach_ip_punt_policer_error \ |
| _(DROP, "ip punt policer drop") |
| |
| typedef enum |
| { |
| #define _(sym,str) IP_PUNT_POLICER_ERROR_##sym, |
| foreach_ip_punt_policer_error |
| #undef _ |
| IP4_PUNT_POLICER_N_ERROR, |
| } ip_punt_policer_error_t; |
| |
| extern u8 *format_ip_punt_policer_trace (u8 * s, va_list * args); |
| extern vlib_node_registration_t ip4_punt_policer_node; |
| extern ip_punt_policer_t ip4_punt_policer_cfg; |
| extern vlib_node_registration_t ip6_punt_policer_node; |
| extern ip_punt_policer_t ip6_punt_policer_cfg; |
| |
| /** |
| * IP punt policing node function |
| */ |
| always_inline uword |
| ip_punt_policer (vlib_main_t * vm, |
| vlib_node_runtime_t * node, |
| vlib_frame_t * frame, u8 arc_index, u32 policer_index) |
| { |
| u32 *from, *to_next, n_left_from, n_left_to_next, next_index; |
| u64 time_in_policer_periods; |
| vnet_feature_main_t *fm = &feature_main; |
| vnet_feature_config_main_t *cm = &fm->feature_config_mains[arc_index]; |
| vnet_policer_main_t *pm = &vnet_policer_main; |
| policer_read_response_type_st *pol = &pm->policers[policer_index]; |
| u32 pol_thread_index = pol->thread_index; |
| u32 this_thread_index = vm->thread_index; |
| |
| time_in_policer_periods = |
| clib_cpu_time_now () >> POLICER_TICKS_PER_PERIOD_SHIFT; |
| |
| from = vlib_frame_vector_args (frame); |
| n_left_from = frame->n_vectors; |
| next_index = node->cached_next_index; |
| |
| if (PREDICT_FALSE (pol_thread_index == ~0)) |
| { |
| /* |
| * This is the first packet to use this policer. Set the |
| * thread index in the policer to this thread and any |
| * packets seen by this node on other threads will |
| * be handed off to this one. |
| * |
| * This could happen simultaneously on another thread. |
| */ |
| clib_atomic_cmp_and_swap (&pol->thread_index, ~0, this_thread_index); |
| pol_thread_index = this_thread_index; |
| } |
| |
| while (n_left_from > 0) |
| { |
| vlib_get_next_frame (vm, node, next_index, to_next, n_left_to_next); |
| |
| while (n_left_from >= 4 && n_left_to_next >= 2) |
| { |
| vlib_buffer_t *b0, *b1; |
| u32 next0, next1; |
| u8 act0, act1; |
| u32 bi0, bi1; |
| |
| next0 = next1 = 0; |
| bi0 = to_next[0] = from[0]; |
| bi1 = to_next[1] = from[1]; |
| |
| from += 2; |
| n_left_from -= 2; |
| to_next += 2; |
| n_left_to_next -= 2; |
| |
| b0 = vlib_get_buffer (vm, bi0); |
| b1 = vlib_get_buffer (vm, bi1); |
| |
| if (PREDICT_FALSE (this_thread_index != pol_thread_index)) |
| { |
| next0 = next1 = IP_PUNT_POLICER_NEXT_HANDOFF; |
| } |
| else |
| { |
| |
| vnet_get_config_data (&cm->config_main, |
| &b0->current_config_index, &next0, 0); |
| vnet_get_config_data (&cm->config_main, |
| &b1->current_config_index, &next1, 0); |
| |
| act0 = |
| vnet_policer_police (vm, b0, policer_index, |
| time_in_policer_periods, POLICE_CONFORM); |
| act1 = |
| vnet_policer_police (vm, b1, policer_index, |
| time_in_policer_periods, POLICE_CONFORM); |
| |
| if (PREDICT_FALSE (act0 == SSE2_QOS_ACTION_DROP)) |
| { |
| next0 = IP_PUNT_POLICER_NEXT_DROP; |
| b0->error = node->errors[IP_PUNT_POLICER_ERROR_DROP]; |
| } |
| if (PREDICT_FALSE (act1 == SSE2_QOS_ACTION_DROP)) |
| { |
| next1 = IP_PUNT_POLICER_NEXT_DROP; |
| b1->error = node->errors[IP_PUNT_POLICER_ERROR_DROP]; |
| } |
| |
| if (PREDICT_FALSE (b0->flags & VLIB_BUFFER_IS_TRACED)) |
| { |
| ip_punt_policer_trace_t *t = |
| vlib_add_trace (vm, node, b0, sizeof (*t)); |
| t->next = next0; |
| t->policer_index = policer_index; |
| } |
| if (PREDICT_FALSE (b1->flags & VLIB_BUFFER_IS_TRACED)) |
| { |
| ip_punt_policer_trace_t *t = |
| vlib_add_trace (vm, node, b1, sizeof (*t)); |
| t->next = next1; |
| t->policer_index = policer_index; |
| } |
| } |
| |
| vlib_validate_buffer_enqueue_x2 (vm, node, next_index, to_next, |
| n_left_to_next, |
| bi0, bi1, next0, next1); |
| } |
| while (n_left_from > 0 && n_left_to_next > 0) |
| { |
| vlib_buffer_t *b0; |
| u32 next0; |
| u32 bi0; |
| u8 act0; |
| |
| next0 = 0; |
| bi0 = to_next[0] = from[0]; |
| |
| from += 1; |
| n_left_from -= 1; |
| to_next += 1; |
| n_left_to_next -= 1; |
| |
| b0 = vlib_get_buffer (vm, bi0); |
| |
| if (PREDICT_FALSE (this_thread_index != pol_thread_index)) |
| { |
| next0 = IP_PUNT_POLICER_NEXT_HANDOFF; |
| } |
| else |
| { |
| vnet_get_config_data (&cm->config_main, |
| &b0->current_config_index, &next0, 0); |
| |
| act0 = |
| vnet_policer_police (vm, b0, policer_index, |
| time_in_policer_periods, POLICE_CONFORM); |
| if (PREDICT_FALSE (act0 == SSE2_QOS_ACTION_DROP)) |
| { |
| next0 = IP_PUNT_POLICER_NEXT_DROP; |
| b0->error = node->errors[IP_PUNT_POLICER_ERROR_DROP]; |
| } |
| |
| if (PREDICT_FALSE (b0->flags & VLIB_BUFFER_IS_TRACED)) |
| { |
| ip_punt_policer_trace_t *t = |
| vlib_add_trace (vm, node, b0, sizeof (*t)); |
| t->next = next0; |
| t->policer_index = policer_index; |
| } |
| } |
| vlib_validate_buffer_enqueue_x1 (vm, node, next_index, to_next, |
| n_left_to_next, bi0, next0); |
| } |
| vlib_put_next_frame (vm, node, next_index, n_left_to_next); |
| } |
| |
| return frame->n_vectors; |
| } |
| |
| /** |
| * IP4 punt redirect per-rx interface configuration |
| * redirect punted traffic to another location. |
| */ |
| typedef struct ip_punt_redirect_rx_t_ |
| { |
| /** |
| * Node linkage into the FIB graph |
| */ |
| fib_node_t node; |
| |
| fib_protocol_t fproto; |
| fib_forward_chain_type_t payload_type; |
| fib_node_index_t pl; |
| u32 sibling; |
| |
| /** |
| * redirect forwarding |
| */ |
| dpo_id_t dpo; |
| } ip_punt_redirect_rx_t; |
| |
| /** |
| * IP punt redirect configuration |
| */ |
| typedef struct ip_punt_redirect_t_ |
| { |
| ip_punt_redirect_rx_t *pool; |
| |
| /** |
| * per-RX interface configuration. |
| * sw_if_index = 0 (from which packets are never received) is used to |
| * indicate 'from-any' |
| */ |
| index_t *redirect_by_rx_sw_if_index[FIB_PROTOCOL_IP_MAX]; |
| } ip_punt_redirect_cfg_t; |
| |
| extern ip_punt_redirect_cfg_t ip_punt_redirect_cfg; |
| |
| /** |
| * IP punt redirect next nodes |
| */ |
| typedef enum ip_punt_redirect_next_t_ |
| { |
| IP_PUNT_REDIRECT_NEXT_DROP, |
| IP_PUNT_REDIRECT_NEXT_TX, |
| IP_PUNT_REDIRECT_NEXT_ARP, |
| IP_PUNT_REDIRECT_N_NEXT, |
| } ip_punt_redirect_next_t; |
| |
| /** |
| * IP Punt redirect trace |
| */ |
| typedef struct ip4_punt_redirect_trace_t_ |
| { |
| index_t rrxi; |
| u32 next; |
| } ip_punt_redirect_trace_t; |
| |
| /** |
| * Add a punt redirect entry |
| */ |
| extern void ip_punt_redirect_add (fib_protocol_t fproto, |
| u32 rx_sw_if_index, |
| fib_forward_chain_type_t ct, |
| fib_route_path_t * rpaths); |
| |
| extern void ip_punt_redirect_del (fib_protocol_t fproto, u32 rx_sw_if_index); |
| extern index_t ip_punt_redirect_find (fib_protocol_t fproto, |
| u32 rx_sw_if_index); |
| extern u8 *format_ip_punt_redirect (u8 * s, va_list * args); |
| |
| extern u8 *format_ip_punt_redirect_trace (u8 * s, va_list * args); |
| |
| typedef walk_rc_t (*ip_punt_redirect_walk_cb_t) (u32 rx_sw_if_index, |
| const ip_punt_redirect_rx_t * |
| redirect, void *arg); |
| extern void ip_punt_redirect_walk (fib_protocol_t fproto, |
| ip_punt_redirect_walk_cb_t cb, void *ctx); |
| |
| static_always_inline ip_punt_redirect_rx_t * |
| ip_punt_redirect_get (index_t rrxi) |
| { |
| return (pool_elt_at_index (ip_punt_redirect_cfg.pool, rrxi)); |
| } |
| |
| always_inline uword |
| ip_punt_redirect (vlib_main_t * vm, |
| vlib_node_runtime_t * node, |
| vlib_frame_t * frame, u8 arc_index, fib_protocol_t fproto) |
| { |
| u32 *from, *to_next, n_left_from, n_left_to_next, next_index; |
| vnet_feature_main_t *fm = &feature_main; |
| vnet_feature_config_main_t *cm = &fm->feature_config_mains[arc_index]; |
| index_t *redirects; |
| |
| from = vlib_frame_vector_args (frame); |
| n_left_from = frame->n_vectors; |
| next_index = node->cached_next_index; |
| redirects = ip_punt_redirect_cfg.redirect_by_rx_sw_if_index[fproto]; |
| |
| while (n_left_from > 0) |
| { |
| vlib_get_next_frame (vm, node, next_index, to_next, n_left_to_next); |
| |
| while (n_left_from > 0 && n_left_to_next > 0) |
| { |
| u32 rx_sw_if_index0, rrxi0; |
| ip_punt_redirect_rx_t *rrx0; |
| vlib_buffer_t *b0; |
| u32 next0; |
| u32 bi0; |
| |
| rrxi0 = INDEX_INVALID; |
| next0 = 0; |
| bi0 = to_next[0] = from[0]; |
| |
| from += 1; |
| n_left_from -= 1; |
| to_next += 1; |
| n_left_to_next -= 1; |
| |
| b0 = vlib_get_buffer (vm, bi0); |
| |
| vnet_get_config_data (&cm->config_main, |
| &b0->current_config_index, &next0, 0); |
| |
| rx_sw_if_index0 = vnet_buffer (b0)->sw_if_index[VLIB_RX]; |
| |
| /* |
| * If config exists for this particular RX interface use it, |
| * else use the default (at RX = 0) |
| */ |
| if (vec_len (redirects) > rx_sw_if_index0) |
| { |
| rrxi0 = redirects[rx_sw_if_index0]; |
| if (INDEX_INVALID == rrxi0) |
| rrxi0 = redirects[0]; |
| } |
| else if (vec_len (redirects) >= 1) |
| rrxi0 = redirects[0]; |
| |
| if (PREDICT_TRUE (INDEX_INVALID != rrxi0)) |
| { |
| rrx0 = ip_punt_redirect_get (rrxi0); |
| vnet_buffer (b0)->ip.adj_index[VLIB_TX] = rrx0->dpo.dpoi_index; |
| next0 = rrx0->dpo.dpoi_next_node; |
| } |
| |
| if (PREDICT_FALSE (b0->flags & VLIB_BUFFER_IS_TRACED)) |
| { |
| ip_punt_redirect_trace_t *t = |
| vlib_add_trace (vm, node, b0, sizeof (*t)); |
| t->next = next0; |
| t->rrxi = rrxi0; |
| } |
| |
| vlib_validate_buffer_enqueue_x1 (vm, node, next_index, to_next, |
| n_left_to_next, bi0, next0); |
| } |
| |
| vlib_put_next_frame (vm, node, next_index, n_left_to_next); |
| } |
| |
| return frame->n_vectors; |
| } |
| |
| always_inline uword |
| ip_drop_or_punt (vlib_main_t * vm, |
| vlib_node_runtime_t * node, |
| vlib_frame_t * frame, u8 arc_index) |
| { |
| u32 *from, *to_next, n_left_from, n_left_to_next, next_index; |
| |
| from = vlib_frame_vector_args (frame); |
| n_left_from = frame->n_vectors; |
| next_index = node->cached_next_index; |
| |
| while (n_left_from > 0) |
| { |
| vlib_get_next_frame (vm, node, next_index, to_next, n_left_to_next); |
| |
| while (n_left_from >= 8 && n_left_to_next >= 4) |
| { |
| vlib_buffer_t *b0, *b1, *b2, *b3; |
| u32 next0, next1, next2, next3; |
| u32 bi0, bi1, bi2, bi3; |
| |
| next0 = next1 = next2 = next3 = 0; |
| |
| /* Prefetch next iteration. */ |
| { |
| vlib_buffer_t *p4, *p5, *p6, *p7; |
| |
| p4 = vlib_get_buffer (vm, from[4]); |
| p5 = vlib_get_buffer (vm, from[5]); |
| p6 = vlib_get_buffer (vm, from[6]); |
| p7 = vlib_get_buffer (vm, from[7]); |
| |
| vlib_prefetch_buffer_header (p4, LOAD); |
| vlib_prefetch_buffer_header (p5, LOAD); |
| vlib_prefetch_buffer_header (p6, LOAD); |
| vlib_prefetch_buffer_header (p7, LOAD); |
| } |
| |
| bi0 = to_next[0] = from[0]; |
| bi1 = to_next[1] = from[1]; |
| bi2 = to_next[2] = from[2]; |
| bi3 = to_next[3] = from[3]; |
| |
| from += 4; |
| n_left_from -= 4; |
| to_next += 4; |
| n_left_to_next -= 4; |
| |
| b0 = vlib_get_buffer (vm, bi0); |
| b1 = vlib_get_buffer (vm, bi1); |
| b2 = vlib_get_buffer (vm, bi2); |
| b3 = vlib_get_buffer (vm, bi3); |
| |
| /* punt and drop features are not associated with a given interface |
| * so the special index 0 is used */ |
| vnet_feature_arc_start (arc_index, 0, &next0, b0); |
| vnet_feature_arc_start (arc_index, 0, &next1, b1); |
| vnet_feature_arc_start (arc_index, 0, &next2, b2); |
| vnet_feature_arc_start (arc_index, 0, &next3, b3); |
| |
| vlib_validate_buffer_enqueue_x4 (vm, node, next_index, |
| to_next, n_left_to_next, |
| bi0, bi1, bi2, bi3, |
| next0, next1, next2, next3); |
| } |
| |
| while (n_left_from > 0 && n_left_to_next > 0) |
| { |
| vlib_buffer_t *b0; |
| u32 next0; |
| u32 bi0; |
| |
| next0 = 0; |
| bi0 = to_next[0] = from[0]; |
| |
| from += 1; |
| n_left_from -= 1; |
| to_next += 1; |
| n_left_to_next -= 1; |
| |
| b0 = vlib_get_buffer (vm, bi0); |
| |
| vnet_feature_arc_start (arc_index, 0, &next0, b0); |
| |
| vlib_validate_buffer_enqueue_x1 (vm, node, next_index, to_next, |
| n_left_to_next, bi0, next0); |
| } |
| vlib_put_next_frame (vm, node, next_index, n_left_to_next); |
| } |
| |
| return frame->n_vectors; |
| } |
| |
| #endif |
| |
| /* |
| * fd.io coding-style-patch-verification: ON |
| * |
| * Local Variables: |
| * eval: (c-set-style "gnu") |
| * End: |
| */ |