summaryrefslogtreecommitdiffstats
path: root/src/vnet/l2/l2_bd.h
blob: 35c0e0bf0a16a2df86f779cfdad1fe90517e6cdd (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
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
/*
 * l2_bd.h : layer 2 bridge domain
 *
 * Copyright (c) 2013 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 included_l2bd_h
#define included_l2bd_h

#include <vlib/vlib.h>
#include <vnet/vnet.h>
#include <vnet/ip/ip46_address.h>
#include <vnet/ethernet/mac_address.h>

typedef enum l2_bd_port_type_t_
{
  L2_BD_PORT_TYPE_NORMAL = 0,
  L2_BD_PORT_TYPE_BVI = 1,
  L2_BD_PORT_TYPE_UU_FWD = 2,
} l2_bd_port_type_t;

typedef struct
{
  /* hash bd_id -> bd_index */
  uword *bd_index_by_bd_id;

  /* Busy bd_index bitmap */
  uword *bd_index_bitmap;

  /* convenience */
  vlib_main_t *vlib_main;
  vnet_main_t *vnet_main;
} bd_main_t;

extern bd_main_t bd_main;

/* Bridge domain member  */

#define L2_FLOOD_MEMBER_NORMAL 0
#define L2_FLOOD_MEMBER_BVI    1

typedef struct
{
  u32 sw_if_index;		/* the output L2 interface */
  u8 flags;			/* 0=normal, 1=bvi */
  u8 shg;			/* split horizon group number  */
  u16 spare;
} l2_flood_member_t;

/* Per-bridge domain configuration */

typedef struct
{
  /*
   * Contains bit enables for flooding, learning, and forwarding.
   * All other feature bits should always be set.
   */
  u32 feature_bitmap;
  /*
   * identity of the bridge-domain's BVI interface
   * set to ~0 if there is no BVI
   */
  u32 bvi_sw_if_index;

  /*
   * identity of the bridge-domain's UU flood interface
   * set to ~0 if there is no such configuration
   */
  u32 uu_fwd_sw_if_index;

  /* bridge domain id, not to be confused with bd_index */
  u32 bd_id;

  /* Vector of member ports */
  l2_flood_member_t *members;

  /* First flood_count member ports are flooded */
  u32 flood_count;

  /* Tunnel Master (Multicast vxlan) are always flooded */
  u32 tun_master_count;

  /* Tunnels (Unicast vxlan) are flooded if there are no masters */
  u32 tun_normal_count;

  /* Interface on which packets are not flooded */
  u32 no_flood_count;

  /* hash ip4/ip6 -> mac for arp/nd termination */
  uword *mac_by_ip4;
  uword *mac_by_ip6;

  /* mac aging */
  u8 mac_age;

  /* sequence number for bridge domain based flush of MACs */
  u8 seq_num;

  /* Bridge domain tag (C string NULL terminated) */
  u8 *bd_tag;

} l2_bridge_domain_t;

/* Limit Bridge Domain ID to 24 bits to match 24-bit VNI range */
#define L2_BD_ID_MAX ((1<<24)-1)

typedef struct
{
  u32 bd_id;
  u8 flood;
  u8 uu_flood;
  u8 forward;
  u8 learn;
  u8 arp_term;
  u8 arp_ufwd;
  u8 mac_age;
  u8 *bd_tag;
  u8 is_add;
} l2_bridge_domain_add_del_args_t;

/* Return 1 if bridge domain has been initialized */
always_inline u32
bd_is_valid (l2_bridge_domain_t * bd_config)
{
  return (bd_config->feature_bitmap != 0);
}

/* Init bridge domain if not done already */
void bd_validate (l2_bridge_domain_t * bd_config);


void
bd_add_member (l2_bridge_domain_t * bd_config, l2_flood_member_t * member);

u32 bd_remove_member (l2_bridge_domain_t * bd_config, u32 sw_if_index);

typedef enum bd_flags_t_
{
  L2_NONE = 0,
  L2_LEARN = (1 << 0),
  L2_FWD = (1 << 1),
  L2_FLOOD = (1 << 2),
  L2_UU_FLOOD = (1 << 3),
  L2_ARP_TERM = (1 << 4),
  L2_ARP_UFWD = (1 << 5),
} bd_flags_t;

u32 bd_set_flags (vlib_main_t * vm, u32 bd_index, bd_flags_t flags,
		  u32 enable);
void bd_set_mac_age (vlib_main_t * vm, u32 bd_index, u8 age);
int bd_add_del (l2_bridge_domain_add_del_args_t * args);

/**
 * \brief Get a bridge domain.
 *
 * Get a bridge domain with the given bridge domain ID.
 *
 * \param bdm bd_main pointer.
 * \param bd_id The bridge domain ID
 * \return The bridge domain index in \c l2input_main->l2_bridge_domain_t vector.
 */
u32 bd_find_index (bd_main_t * bdm, u32 bd_id);

/**
 * \brief Create a bridge domain.
 *
 * Create a bridge domain with the given bridge domain ID
 *
 * \param bdm bd_main pointer.
 * \return The bridge domain index in \c l2input_main->l2_bridge_domain_t vector.
 */
u32 bd_add_bd_index (bd_main_t * bdm, u32 bd_id);

/**
 * \brief Get or create a bridge domain.
 *
 * Get a bridge domain with the given bridge domain ID, if one exists, otherwise
 * create one with the given ID, or the first unused ID if the given ID is ~0..
 *
 * \param bdm bd_main pointer.
 * \param bd_id The bridge domain ID
 * \return The bridge domain index in \c l2input_main->l2_bridge_domain_t vector.
 */
static inline u32
bd_find_or_add_bd_index (bd_main_t * bdm, u32 bd_id)
{
  u32 bd_index = bd_find_index (bdm, bd_id);
  if (bd_index == ~0)
    return bd_add_bd_index (bdm, bd_id);
  return bd_index;
}

u32 bd_add_del_ip_mac (u32 bd_index,
		       ip46_type_t type,
		       const ip46_address_t * ip_addr,
		       const mac_address_t * mac, u8 is_add);

void bd_flush_ip_mac (u32 bd_index);

#endif

/*
 * fd.io coding-style-patch-verification: ON
 *
 * Local Variables:
 * eval: (c-set-style "gnu")
 * End:
 */
\ _ (avx, 1, ecx, 28) \ _ (rdrand, 1, ecx, 30) \ _ (avx2, 7, ebx, 5) \ _ (bmi2, 7, ebx, 8) \ _ (rtm, 7, ebx, 11) \ _ (pqm, 7, ebx, 12) \ _ (pqe, 7, ebx, 15) \ _ (avx512f, 7, ebx, 16) \ _ (rdseed, 7, ebx, 18) \ _ (x86_aes, 1, ecx, 25) \ _ (sha, 7, ebx, 29) \ _ (vaes, 7, ecx, 9) \ _ (vpclmulqdq, 7, ecx, 10) \ _ (avx512_vnni, 7, ecx, 11) \ _ (avx512_bitalg, 7, ecx, 12) \ _ (avx512_vpopcntdq, 7, ecx, 14) \ _ (movdiri, 7, ecx, 27) \ _ (movdir64b, 7, ecx, 28) \ _ (enqcmd, 7, ecx, 29) \ _ (avx512_fp16, 7, edx, 23) \ _ (invariant_tsc, 0x80000007, edx, 8) #define foreach_aarch64_flags \ _ (fp, 0) \ _ (asimd, 1) \ _ (evtstrm, 2) \ _ (aarch64_aes, 3) \ _ (pmull, 4) \ _ (sha1, 5) \ _ (sha2, 6) \ _ (crc32, 7) \ _ (atomics, 8) \ _ (fphp, 9) \ _ (asimdhp, 10) \ _ (cpuid, 11) \ _ (asimdrdm, 12) \ _ (jscvt, 13) \ _ (fcma, 14) \ _ (lrcpc, 15) \ _ (dcpop, 16) \ _ (sha3, 17) \ _ (sm3, 18) \ _ (sm4, 19) \ _ (asimddp, 20) \ _ (sha512, 21) \ _ (sve, 22) u32 clib_get_current_cpu_id (void); u32 clib_get_current_numa_node (void); typedef int (*clib_cpu_supports_func_t) (void); #if defined(__x86_64__) #include "cpuid.h" static inline int clib_get_cpuid (const u32 lev, u32 * eax, u32 * ebx, u32 * ecx, u32 * edx) { if ((u32) __get_cpuid_max (0x80000000 & lev, 0) < lev) return 0; if (lev == 7) __cpuid_count (lev, 0, *eax, *ebx, *ecx, *edx); else __cpuid (lev, *eax, *ebx, *ecx, *edx); return 1; } #define _(flag, func, reg, bit) \ static inline int \ clib_cpu_supports_ ## flag() \ { \ u32 __attribute__((unused)) eax, ebx = 0, ecx = 0, edx = 0; \ clib_get_cpuid (func, &eax, &ebx, &ecx, &edx); \ \ return ((reg & (1 << bit)) != 0); \ } foreach_x86_64_flags #undef _ #else /* __x86_64__ */ #define _(flag, func, reg, bit) \ static inline int clib_cpu_supports_ ## flag() { return 0; } foreach_x86_64_flags #undef _ #endif /* __x86_64__ */ #if defined(__aarch64__) #include <sys/auxv.h> #define _(flag, bit) \ static inline int \ clib_cpu_supports_ ## flag() \ { \ unsigned long hwcap = getauxval(AT_HWCAP); \ return (hwcap & (1 << bit)); \ } foreach_aarch64_flags #undef _ #else /* ! __x86_64__ && !__aarch64__ */ #define _(flag, bit) \ static inline int clib_cpu_supports_ ## flag() { return 0; } foreach_aarch64_flags #undef _ #endif /* __x86_64__, __aarch64__ */ /* * aes is the only feature with the same name in both flag lists * handle this by prefixing it with the arch name, and handling it * with the custom function below */ static inline int clib_cpu_supports_aes () { #if defined(__x86_64__) return clib_cpu_supports_x86_aes (); #elif defined (__aarch64__) return clib_cpu_supports_aarch64_aes (); #else return 0; #endif } static inline int clib_cpu_march_priority_scalar () { return 1; } static inline int clib_cpu_march_priority_spr () { if (clib_cpu_supports_enqcmd ()) return 300; return -1; } static inline int clib_cpu_march_priority_icl () { if (clib_cpu_supports_avx512_bitalg ()) return 200; return -1; } static inline int clib_cpu_march_priority_adl () { if (clib_cpu_supports_movdiri () && clib_cpu_supports_avx2 ()) return 150; return -1; } static inline int clib_cpu_march_priority_skx () { if (clib_cpu_supports_avx512f ()) return 100; return -1; } static inline int clib_cpu_march_priority_trm () { if (clib_cpu_supports_movdiri ()) return 40; return -1; } static inline int clib_cpu_march_priority_hsw () { if (clib_cpu_supports_avx2 ()) return 50; return -1; } #define X86_CPU_ARCH_PERF_FUNC 0xA static inline int clib_get_pmu_counter_count (u8 *fixed, u8 *general) { #if defined(__x86_64__) u32 __clib_unused eax = 0, ebx = 0, ecx = 0, edx = 0; clib_get_cpuid (X86_CPU_ARCH_PERF_FUNC, &eax, &ebx, &ecx, &edx); *general = (eax & 0xFF00) >> 8; *fixed = (edx & 0xF); return 1; #else return 0; #endif } static inline u32 clib_cpu_implementer () { char buf[128]; static u32 implementer = -1; if (-1 != implementer) return implementer; FILE *fp = fopen ("/proc/cpuinfo", "r"); if (!fp) return implementer; while (!feof (fp)) { if (!fgets (buf, sizeof (buf), fp)) break; buf[127] = '\0'; if (strstr (buf, "CPU implementer")) implementer = (u32) strtol (memchr (buf, ':', 128) + 2, NULL, 0); if (-1 != implementer) break; } fclose (fp); return implementer; } static inline u32 clib_cpu_part () { char buf[128]; static u32 part = -1; if (-1 != part) return part; FILE *fp = fopen ("/proc/cpuinfo", "r"); if (!fp) return part; while (!feof (fp)) { if (!fgets (buf, sizeof (buf), fp)) break; buf[127] = '\0'; if (strstr (buf, "CPU part")) part = (u32) strtol (memchr (buf, ':', 128) + 2, NULL, 0); if (-1 != part) break; } fclose (fp); return part; } #define AARCH64_CPU_IMPLEMENTER_CAVIUM 0x43 #define AARCH64_CPU_PART_THUNDERX2 0x0af #define AARCH64_CPU_PART_OCTEONTX2T96 0x0b2 #define AARCH64_CPU_PART_OCTEONTX2T98 0x0b1 #define AARCH64_CPU_IMPLEMENTER_QDF24XX 0x51 #define AARCH64_CPU_PART_QDF24XX 0xc00 #define AARCH64_CPU_IMPLEMENTER_CORTEXA72 0x41 #define AARCH64_CPU_PART_CORTEXA72 0xd08 #define AARCH64_CPU_IMPLEMENTER_NEOVERSEN1 0x41 #define AARCH64_CPU_PART_NEOVERSEN1 0xd0c static inline int clib_cpu_march_priority_octeontx2 () { if ((AARCH64_CPU_IMPLEMENTER_CAVIUM == clib_cpu_implementer ()) && ((AARCH64_CPU_PART_OCTEONTX2T96 == clib_cpu_part ()) || AARCH64_CPU_PART_OCTEONTX2T98 == clib_cpu_part ())) return 20; return -1; } static inline int clib_cpu_march_priority_thunderx2t99 () { if ((AARCH64_CPU_IMPLEMENTER_CAVIUM == clib_cpu_implementer ()) && (AARCH64_CPU_PART_THUNDERX2 == clib_cpu_part ())) return 20; return -1; } static inline int clib_cpu_march_priority_qdf24xx () { if ((AARCH64_CPU_IMPLEMENTER_QDF24XX == clib_cpu_implementer ()) && (AARCH64_CPU_PART_QDF24XX == clib_cpu_part ())) return 20; return -1; } static inline int clib_cpu_march_priority_cortexa72 () { if ((AARCH64_CPU_IMPLEMENTER_CORTEXA72 == clib_cpu_implementer ()) && (AARCH64_CPU_PART_CORTEXA72 == clib_cpu_part ())) return 10; return -1; } static inline int clib_cpu_march_priority_neoversen1 () { if ((AARCH64_CPU_IMPLEMENTER_NEOVERSEN1 == clib_cpu_implementer ()) && (AARCH64_CPU_PART_NEOVERSEN1 == clib_cpu_part ())) return 10; return -1; } #ifdef CLIB_MARCH_VARIANT #define CLIB_MARCH_FN_PRIORITY() CLIB_MARCH_SFX(clib_cpu_march_priority)() #else #define CLIB_MARCH_FN_PRIORITY() 0 #endif #endif /* included_clib_cpu_h */ #define CLIB_MARCH_FN_CONSTRUCTOR(fn) \ static void __clib_constructor \ CLIB_MARCH_SFX(fn ## _march_constructor) (void) \ { \ if (CLIB_MARCH_FN_PRIORITY() > fn ## _selected_priority) \ { \ fn ## _selected = & CLIB_MARCH_SFX (fn ## _ma); \ fn ## _selected_priority = CLIB_MARCH_FN_PRIORITY(); \ } \ } \ #ifndef CLIB_MARCH_VARIANT #define CLIB_MARCH_FN(fn, rtype, _args...) \ static rtype CLIB_MARCH_SFX (fn##_ma) (_args); \ rtype (*fn##_selected) (_args) = &CLIB_MARCH_SFX (fn##_ma); \ int fn##_selected_priority = 0; \ static inline rtype CLIB_MARCH_SFX (fn##_ma) (_args) #else #define CLIB_MARCH_FN(fn, rtype, _args...) \ static rtype CLIB_MARCH_SFX (fn##_ma) (_args); \ extern rtype (*fn##_selected) (_args); \ extern int fn##_selected_priority; \ CLIB_MARCH_FN_CONSTRUCTOR (fn) \ static rtype CLIB_MARCH_SFX (fn##_ma) (_args) #endif #define CLIB_MARCH_FN_SELECT(fn) (* fn ## _selected) format_function_t format_cpu_uarch; format_function_t format_cpu_model_name; format_function_t format_cpu_flags; format_function_t format_march_variant; /* * fd.io coding-style-patch-verification: ON * * Local Variables: * eval: (c-set-style "gnu") * End: */