FD.io VPP  v21.06-1-gbb7418cf9
Vector Packet Processing
nat44_ei_inlines.h
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2020 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 #ifndef __included_nat44_ei_inlines_h__
17 #define __included_nat44_ei_inlines_h__
18 
19 #include <vppinfra/clib.h>
20 
21 #include <nat/nat44-ei/nat44_ei.h>
23 
26 {
27  ASSERT (fib_index <= (1 << 14) - 1);
28  ASSERT (proto <= (1 << 3) - 1);
29  return (u64) addr.as_u32 << 32 | (u64) port << 16 | fib_index << 3 |
30  (proto & 0x7);
31 }
32 
33 always_inline void
36 {
37  if (addr)
38  {
39  addr->as_u32 = key >> 32;
40  }
41  if (port)
42  {
43  *port = (key >> 16) & (u16) ~0;
44  }
45  if (fib_index)
46  {
47  *fib_index = key >> 3 & ((1 << 13) - 1);
48  }
49  if (proto)
50  {
51  *proto = key & 0x7;
52  }
53 }
54 
55 always_inline void
57  u32 fib_index, nat_protocol_t proto)
58 {
59  kv->key = calc_nat_key (addr, port, fib_index, proto);
60  kv->value = ~0ULL;
61 }
62 
63 always_inline void
66  u32 session_index)
67 {
68  init_nat_k (kv, addr, port, fib_index, proto);
69  kv->value = (u64) thread_index << 32 | session_index;
70 }
71 
72 always_inline void
73 init_nat_i2o_k (clib_bihash_kv_8_8_t *kv, nat44_ei_session_t *s)
74 {
75  return init_nat_k (kv, s->in2out.addr, s->in2out.port, s->in2out.fib_index,
76  s->nat_proto);
77 }
78 
79 always_inline void
80 init_nat_i2o_kv (clib_bihash_kv_8_8_t *kv, nat44_ei_session_t *s,
81  u32 thread_index, u32 session_index)
82 {
83  init_nat_k (kv, s->in2out.addr, s->in2out.port, s->in2out.fib_index,
84  s->nat_proto);
85  kv->value = (u64) thread_index << 32 | session_index;
86 }
87 
88 always_inline void
89 init_nat_o2i_k (clib_bihash_kv_8_8_t *kv, nat44_ei_session_t *s)
90 {
91  return init_nat_k (kv, s->out2in.addr, s->out2in.port, s->out2in.fib_index,
92  s->nat_proto);
93 }
94 
95 always_inline void
96 init_nat_o2i_kv (clib_bihash_kv_8_8_t *kv, nat44_ei_session_t *s,
97  u32 thread_index, u32 session_index)
98 {
99  init_nat_k (kv, s->out2in.addr, s->out2in.port, s->out2in.fib_index,
100  s->nat_proto);
101  kv->value = (u64) thread_index << 32 | session_index;
102 }
103 
106 {
107  return value->value >> 32;
108 }
109 
112 {
113  return value->value & ~(u32) 0;
114 }
115 
118  u32 sw_if_index0, u32 ip4_addr)
119 {
121  ip4_address_t *first_int_addr;
122 
123  if (PREDICT_FALSE (rt->cached_sw_if_index != sw_if_index0))
124  {
125  first_int_addr = ip4_interface_first_address (
126  im, sw_if_index0, 0 /* just want the address */);
127  rt->cached_sw_if_index = sw_if_index0;
128  if (first_int_addr)
129  rt->cached_ip4_address = first_int_addr->as_u32;
130  else
131  rt->cached_ip4_address = 0;
132  }
133 
134  if (PREDICT_FALSE (ip4_addr == rt->cached_ip4_address))
135  return 1;
136  else
137  return 0;
138 }
139 
140 /** \brief Per-user LRU list maintenance */
141 always_inline void
144 {
145  /* don't update too often - timeout is in magnitude of seconds anyway */
146  if (s->last_heard > s->last_lru_update + 1)
147  {
148  clib_dlist_remove (nm->per_thread_data[thread_index].list_pool,
149  s->per_user_index);
150  clib_dlist_addtail (nm->per_thread_data[thread_index].list_pool,
151  s->per_user_list_head_index, s->per_user_index);
152  s->last_lru_update = s->last_heard;
153  }
154 }
155 
156 always_inline void
158  u8 is_static)
159 {
160  if (u->nsessions + u->nstaticsessions < nm->max_translations_per_user)
161  {
162  if (is_static)
163  u->nstaticsessions++;
164  else
165  u->nsessions++;
166  }
167 }
168 
169 always_inline void
172 {
174  nat44_ei_user_key_t u_key;
176  vec_elt_at_index (nm->per_thread_data, thread_index);
177 
178  if (u->nstaticsessions == 0 && u->nsessions == 0)
179  {
180  u_key.addr.as_u32 = u->addr.as_u32;
181  u_key.fib_index = u->fib_index;
182  kv.key = u_key.as_u64;
183  pool_put_index (tnm->list_pool, u->sessions_per_user_list_head_index);
184  pool_put (tnm->users, u);
185  clib_bihash_add_del_8_8 (&tnm->user_hash, &kv, 0);
186  vlib_set_simple_counter (&nm->total_users, thread_index, 0,
187  pool_elts (tnm->users));
188  }
189 }
190 
193 {
194  if (pool_elts (nm->per_thread_data[thread_index].sessions) >=
196  return 1;
197  return 0;
198 }
199 
200 always_inline void
201 nat44_ei_session_update_counters (nat44_ei_session_t *s, f64 now, uword bytes,
203 {
204  s->last_heard = now;
205  s->total_pkts++;
206  s->total_bytes += bytes;
207  nat_ha_sref (&s->out2in.addr, s->out2in.port, &s->ext_host_addr,
208  s->ext_host_port, s->nat_proto, s->out2in.fib_index,
209  s->total_pkts, s->total_bytes, thread_index,
210  &s->ha_last_refreshed, now);
211 }
212 
213 #endif /* __included_nat44_ei_inlines_h__ */
214 
215 /*
216  * fd.io coding-style-patch-verification: ON
217  *
218  * Local Variables:
219  * eval: (c-set-style "gnu")
220  * End:
221  */
static void nat44_ei_delete_user_with_no_session(nat44_ei_main_t *nm, nat44_ei_user_t *u, u32 thread_index)
vnet_interface_output_runtime_t * rt
u8 runtime_data[0]
Function dependent node-runtime data.
Definition: node.h:506
nat44_ei_user_t * users
Definition: nat44_ei.h:280
u32 thread_index
unsigned long u64
Definition: types.h:89
ip4_address_t * ip4_interface_first_address(ip4_main_t *im, u32 sw_if_index, ip_interface_address_t **result_ia)
Definition: ip4_forward.c:281
NAT44 endpoint independent plugin declarations.
nat44_ei_session_t * sessions
Definition: nat44_ei.h:283
static void init_nat_i2o_kv(clib_bihash_kv_8_8_t *kv, nat44_ei_session_t *s, u32 thread_index, u32 session_index)
static u32 nat_value_get_thread_index(clib_bihash_kv_8_8_t *value)
static void init_nat_i2o_k(clib_bihash_kv_8_8_t *kv, nat44_ei_session_t *s)
nat_protocol_t
Definition: lib.h:63
vhost_vring_addr_t addr
Definition: vhost_user.h:130
static void split_nat_key(u64 key, ip4_address_t *addr, u16 *port, u32 *fib_index, nat_protocol_t *proto)
unsigned char u8
Definition: types.h:56
double f64
Definition: types.h:142
static u64 calc_nat_key(ip4_address_t addr, u16 port, u32 fib_index, u8 proto)
unsigned int u32
Definition: types.h:88
static void init_nat_o2i_kv(clib_bihash_kv_8_8_t *kv, nat44_ei_session_t *s, u32 thread_index, u32 session_index)
if(node->flags &VLIB_NODE_FLAG_TRACE) vnet_interface_output_trace(vm
#define static_always_inline
Definition: clib.h:112
static void nat44_ei_user_session_increment(nat44_ei_main_t *nm, nat44_ei_user_t *u, u8 is_static)
static_always_inline u8 nat44_ei_maximum_sessions_exceeded(nat44_ei_main_t *nm, u32 thread_index)
nat44_ei_main_per_thread_data_t * tnm
#define vec_elt_at_index(v, i)
Get vector value at index i checking that i is in bounds.
static void init_nat_kv(clib_bihash_kv_8_8_t *kv, ip4_address_t addr, u16 port, u32 fib_index, nat_protocol_t proto, u32 thread_index, u32 session_index)
static u32 nat_value_get_session_index(clib_bihash_kv_8_8_t *value)
void nat_ha_sref(ip4_address_t *out_addr, u16 out_port, ip4_address_t *eh_addr, u16 eh_port, u8 proto, u32 fib_index, u32 total_pkts, u64 total_bytes, u32 thread_index, f64 *last_refreshed, f64 now)
Create session refresh HA event.
Definition: nat44_ei_ha.c:879
vl_api_ip_proto_t proto
Definition: acl_types.api:51
u64 key
the key
Definition: bihash_8_8.h:43
static void clib_dlist_addtail(dlist_elt_t *pool, u32 head_index, u32 new_index)
Definition: dlist.h:43
unsigned short u16
Definition: types.h:57
#define pool_put(P, E)
Free an object E in pool P.
Definition: pool.h:305
#define PREDICT_FALSE(x)
Definition: clib.h:124
static void vlib_set_simple_counter(vlib_simple_counter_main_t *cm, u32 thread_index, u32 index, u64 value)
Set a simple counter.
Definition: counter.h:109
u32 max_translations_per_user
Definition: nat44_ei.h:347
u64 value
the value
Definition: bihash_8_8.h:44
nat44_ei_main_per_thread_data_t * per_thread_data
Definition: nat44_ei.h:366
vnet_interface_main_t * im
ip4_address_t addr
Definition: nat44_ei.h:104
8 octet key, 8 octet key value pair
Definition: bihash_8_8.h:41
static u8 nat44_ei_is_interface_addr(ip4_main_t *im, vlib_node_runtime_t *node, u32 sw_if_index0, u32 ip4_addr)
NAT active-passive HA.
u8 value
Definition: qos.api:54
#define pool_put_index(p, i)
Free pool element with given index.
Definition: pool.h:337
#define ASSERT(truth)
#define always_inline
Definition: rdma_mlx5dv.h:23
IPv4 main type.
Definition: ip4.h:107
static void clib_dlist_remove(dlist_elt_t *pool, u32 index)
Definition: dlist.h:99
typedef key
Definition: ipsec_types.api:88
static void nat44_ei_session_update_counters(nat44_ei_session_t *s, f64 now, uword bytes, u32 thread_index)
vlib_main_t vlib_node_runtime_t * node
Definition: nat44_ei.c:3047
static void nat44_ei_session_update_lru(nat44_ei_main_t *nm, nat44_ei_session_t *s, u32 thread_index)
Per-user LRU list maintenance.
u64 uword
Definition: types.h:112
static void init_nat_o2i_k(clib_bihash_kv_8_8_t *kv, nat44_ei_session_t *s)
u16 port
Definition: lb_types.api:73
nat44_ei_main_t * nm
f64 now
u32 max_translations_per_thread
Definition: nat44_ei.h:346
static void init_nat_k(clib_bihash_kv_8_8_t *kv, ip4_address_t addr, u16 port, u32 fib_index, nat_protocol_t proto)
clib_bihash_8_8_t user_hash
Definition: nat44_ei.h:277
vlib_simple_counter_main_t total_users
Definition: nat44_ei.h:408
static uword pool_elts(void *v)
Number of active elements in a pool.
Definition: pool.h:127