aboutsummaryrefslogtreecommitdiffstats
path: root/src/vnet/ip-neighbor/ip4_neighbor.h
blob: 7941ebdbced52d52aaadeb685fc4cf4b041587b0 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
/*
 * Copyright (c) 2019 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 __IP4_NEIGHBOR_H__
#define __IP4_NEIGHBOR_H__

#include <vnet/ip/ip.h>
#include <vnet/ethernet/arp_packet.h>
#include <vnet/ip-neighbor/ip_neighbor_types.h>

extern void ip4_neighbor_probe_dst (u32 sw_if_index, u32 thread_index,
				    const ip4_address_t *dst);
extern void ip4_neighbor_advertise (vlib_main_t *vm, vnet_main_t *vnm,
				    u32 sw_if_index, u32 thread_index,
				    const ip4_address_t *addr);

always_inline vlib_buffer_t *
ip4_neighbor_probe (vlib_main_t *vm, vnet_main_t *vnm,
		    const ip_adjacency_t *adj0, const ip4_address_t *src,
		    const ip4_address_t *dst)
{
  vnet_hw_interface_t *hw_if0;
  ethernet_arp_header_t *h0;
  vlib_buffer_t *b0;
  u32 bi0;

  hw_if0 = vnet_get_sup_hw_interface (vnm, adj0->rewrite_header.sw_if_index);

  /* if (NULL == hw_if0->hw_address) */
  /*   return (NULL); */

  /* Send ARP request. */
  h0 = vlib_packet_template_get_packet (vm,
					&ip4_main.ip4_arp_request_packet_template,
					&bi0);
  /* Seems we're out of buffers */
  if (PREDICT_FALSE (!h0))
    return (NULL);

  b0 = vlib_get_buffer (vm, bi0);

  /* Add rewrite/encap string for ARP packet. */
  vnet_rewrite_one_header (adj0[0], h0, sizeof (ethernet_header_t));

  /* Src ethernet address in ARP header. */
  mac_address_from_bytes (&h0->ip4_over_ethernet[0].mac, hw_if0->hw_address);

  h0->ip4_over_ethernet[0].ip4 = *src;
  h0->ip4_over_ethernet[1].ip4 = *dst;

  vnet_buffer (b0)->sw_if_index[VLIB_TX] = adj0->rewrite_header.sw_if_index;
  b0->flags |= VNET_BUFFER_F_LOCALLY_ORIGINATED;

  vlib_buffer_advance (b0, -adj0->rewrite_header.data_bytes);

  {
    vlib_frame_t *f = vlib_get_frame_to_node (vm, hw_if0->output_node_index);
    u32 *to_next = vlib_frame_vector_args (f);
    to_next[0] = bi0;
    f->n_vectors = 1;
    vlib_put_frame_to_node (vm, hw_if0->output_node_index, f);
  }

  vlib_increment_simple_counter (
    &ip_neighbor_counters[AF_IP4].ipnc[VLIB_TX][IP_NEIGHBOR_CTR_REQUEST],
    vm->thread_index, adj0->rewrite_header.sw_if_index, 1);

  return b0;
}

#endif

/*
 * fd.io coding-style-patch-verification: ON
 *
 * Local Variables:
 * eval: (c-set-style "gnu")
 * End:
 */
s="n">foreach_vec64u foreach_vec64f foreach_vec128i foreach_vec128u foreach_vec128f foreach_vec256i foreach_vec256u foreach_vec256f foreach_vec512i foreach_vec512u foreach_vec512f #undef _ /* Vector word sized types. */ #ifndef CLIB_VECTOR_WORD_BITS #ifdef CLIB_HAVE_VEC128 #define CLIB_VECTOR_WORD_BITS 128 #else #define CLIB_VECTOR_WORD_BITS 64 #endif #endif /* CLIB_VECTOR_WORD_BITS */ /* Vector word sized types. */ #if CLIB_VECTOR_WORD_BITS == 128 typedef i8 i8x _vector_size (16); typedef i16 i16x _vector_size (16); typedef i32 i32x _vector_size (16); typedef i64 i64x _vector_size (16); typedef u8 u8x _vector_size (16); typedef u16 u16x _vector_size (16); typedef u32 u32x _vector_size (16); typedef u64 u64x _vector_size (16); #endif #if CLIB_VECTOR_WORD_BITS == 64 typedef i8 i8x _vector_size (8); typedef i16 i16x _vector_size (8); typedef i32 i32x _vector_size (8); typedef i64 i64x _vector_size (8); typedef u8 u8x _vector_size (8); typedef u16 u16x _vector_size (8); typedef u32 u32x _vector_size (8); typedef u64 u64x _vector_size (8); #endif #undef _vector_size #define VECTOR_WORD_TYPE(t) t##x #define VECTOR_WORD_TYPE_LEN(t) (sizeof (VECTOR_WORD_TYPE(t)) / sizeof (t)) /* this series of macros generate _is_equal, _is_greater, _is_zero, _add and _sub inline funcitons for each vector type */ #define _(t, s, c) \ static_always_inline t##s##x##c \ t##s##x##c##_is_equal (t##s##x##c v1, t##s##x##c v2) \ { return (v1 == v2); } \ \ static_always_inline t##s##x##c \ t##s##x##c##_is_greater (t##s##x##c v1, t##s##x##c v2) \ { return (v1 > v2); } \ \ static_always_inline t##s##x##c \ t##s##x##c##_is_zero (t##s##x##c v1) \ { t##s##x##c z = {0}; return (v1 == z); } \ \ static_always_inline t##s##x##c \ t##s##x##c##_add (t##s##x##c v1, t##s##x##c v2) \ { return (v1 + v2); } \ \ static_always_inline t##s##x##c \ t##s##x##c##_sub (t##s##x##c v1, t##s##x##c v2) \ { return (v1 - v2); } foreach_vec #undef _ /* this macro generate _splat inline functions for each scalar vector type */ #define _(t, s, c) \ static_always_inline t##s##x##c \ t##s##x##c##_splat (t##s x) \ { \ t##s##x##c r; \ int i; \ \ for (i = 0; i < c; i++) \ r[i] = x; \ \ return r; \ } foreach_vec128i foreach_vec128u #undef _ #if defined (__SSE4_2__) && __GNUC__ >= 4 #include <vppinfra/vector_sse42.h> #endif #if defined (__AVX2__) #include <vppinfra/vector_avx2.h> #endif #if defined (__AVX512F__) #include <vppinfra/vector_avx512.h> #endif #if defined (__ALTIVEC__) #include <vppinfra/vector_altivec.h> #endif #if defined (__aarch64__) #include <vppinfra/vector_neon.h> #endif #if (defined(CLIB_HAVE_VEC128) || defined(CLIB_HAVE_VEC64)) #include <vppinfra/vector_funcs.h> #endif /* *INDENT-ON* */ #endif /* included_clib_vector_h */ /* * fd.io coding-style-patch-verification: ON * * Local Variables: * eval: (c-set-style "gnu") * End: */