FD.io VPP  v17.10-9-gd594711
Vector Packet Processing
application.c
Go to the documentation of this file.
1 /*
2  * Copyright (c) 2017 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 
18 #include <vnet/session/session.h>
19 
20 /**
21  * Pool from which we allocate all applications
22  */
24 
25 /**
26  * Hash table of apps by api client index
27  */
29 
30 /**
31  * Default application event queue size
32  */
34 
35 int
37 {
39 
40  /* builtin servers are always OK */
41  if (app->api_client_index == ~0)
42  return 0;
43 
44  q = vl_api_client_index_to_input_queue (app->api_client_index);
45  if (!q)
46  return 1;
47 
48  if (q->cursize == q->maxsize)
49  return 1;
50  return 0;
51 }
52 
53 static void
55 {
56  hash_set (app_by_api_client_index, app->api_client_index, app->index);
57 }
58 
59 static void
61 {
62  hash_unset (app_by_api_client_index, app->api_client_index);
63 }
64 
66 application_lookup (u32 api_client_index)
67 {
68  uword *p;
69  p = hash_get (app_by_api_client_index, api_client_index);
70  if (p)
71  return application_get (p[0]);
72 
73  return 0;
74 }
75 
78 {
79  application_t *app;
80  pool_get (app_pool, app);
81  memset (app, 0, sizeof (*app));
82  app->index = application_get_index (app);
83  app->connects_seg_manager = APP_INVALID_SEGMENT_MANAGER_INDEX;
84  app->first_segment_manager = APP_INVALID_SEGMENT_MANAGER_INDEX;
85  if (CLIB_DEBUG > 1)
86  clib_warning ("[%d] New app (%d)", getpid (), app->index);
87  return app;
88 }
89 
90 void
92 {
94  u64 handle;
95  u32 index, *handles = 0;
96  int i;
97  vnet_unbind_args_t _a, *a = &_a;
98 
99  /*
100  * The app event queue allocated in first segment is cleared with
101  * the segment manager. No need to explicitly free it.
102  */
103  if (CLIB_DEBUG > 1)
104  clib_warning ("[%d] Delete app (%d)", getpid (), app->index);
105 
106  /*
107  * Listener cleanup
108  */
109 
110  /* *INDENT-OFF* */
111  hash_foreach (handle, index, app->listeners_table,
112  ({
113  vec_add1 (handles, handle);
114  sm = segment_manager_get (index);
115  sm->app_index = SEGMENT_MANAGER_INVALID_APP_INDEX;
116  }));
117  /* *INDENT-ON* */
118 
119  for (i = 0; i < vec_len (handles); i++)
120  {
121  a->app_index = app->index;
122  a->handle = handles[i];
123  /* seg manager is removed when unbind completes */
124  vnet_unbind (a);
125  }
126 
127  /*
128  * Connects segment manager cleanup
129  */
130 
131  if (app->connects_seg_manager != APP_INVALID_SEGMENT_MANAGER_INDEX)
132  {
133  sm = segment_manager_get (app->connects_seg_manager);
134  sm->app_index = SEGMENT_MANAGER_INVALID_APP_INDEX;
136  }
137 
138 
139  /* If first segment manager is used by a listener */
140  if (app->first_segment_manager != APP_INVALID_SEGMENT_MANAGER_INDEX
141  && app->first_segment_manager != app->connects_seg_manager)
142  {
143  sm = segment_manager_get (app->first_segment_manager);
144  /* .. and has no fifos, e.g. it might be used for redirected sessions,
145  * remove it */
146  if (!segment_manager_has_fifos (sm))
147  {
148  sm->app_index = SEGMENT_MANAGER_INVALID_APP_INDEX;
149  segment_manager_del (sm);
150  }
151  }
152 
153  application_table_del (app);
154  pool_put (app_pool, app);
155 }
156 
157 static void
159 {
160  if (cb_fns->session_accept_callback == 0)
161  clib_warning ("No accept callback function provided");
162  if (cb_fns->session_connected_callback == 0)
163  clib_warning ("No session connected callback function provided");
164  if (cb_fns->session_disconnect_callback == 0)
165  clib_warning ("No session disconnect callback function provided");
166  if (cb_fns->session_reset_callback == 0)
167  clib_warning ("No session reset callback function provided");
168 }
169 
170 int
171 application_init (application_t * app, u32 api_client_index, u64 * options,
172  session_cb_vft_t * cb_fns)
173 {
174  segment_manager_t *sm;
176  u32 app_evt_queue_size, first_seg_size;
177  u32 default_rx_fifo_size = 16 << 10, default_tx_fifo_size = 16 << 10;
178  int rv;
179 
180  app_evt_queue_size = options[APP_EVT_QUEUE_SIZE] > 0 ?
182 
183  /* Setup segment manager */
184  sm = segment_manager_new ();
185  sm->app_index = app->index;
186  props = &app->sm_properties;
187  props->add_segment_size = options[SESSION_OPTIONS_ADD_SEGMENT_SIZE];
188  props->rx_fifo_size = options[SESSION_OPTIONS_RX_FIFO_SIZE];
189  props->rx_fifo_size =
190  props->rx_fifo_size ? props->rx_fifo_size : default_rx_fifo_size;
191  props->tx_fifo_size = options[SESSION_OPTIONS_TX_FIFO_SIZE];
192  props->tx_fifo_size =
193  props->tx_fifo_size ? props->tx_fifo_size : default_tx_fifo_size;
194  props->add_segment = props->add_segment_size != 0;
195  props->preallocated_fifo_pairs = options[APP_OPTIONS_PREALLOC_FIFO_PAIRS];
196  props->use_private_segment = options[APP_OPTIONS_FLAGS]
197  & APP_OPTIONS_FLAGS_BUILTIN_APP;
198  props->private_segment_count = options[APP_OPTIONS_PRIVATE_SEGMENT_COUNT];
199  props->private_segment_size = options[APP_OPTIONS_PRIVATE_SEGMENT_SIZE];
200 
201  first_seg_size = options[SESSION_OPTIONS_SEGMENT_SIZE];
202  if ((rv = segment_manager_init (sm, props, first_seg_size)))
203  return rv;
204  sm->first_is_protected = 1;
205 
206  app->first_segment_manager = segment_manager_index (sm);
207  app->api_client_index = api_client_index;
208  app->flags = options[APP_OPTIONS_FLAGS];
209  app->cb_fns = *cb_fns;
210 
211  /* Allocate app event queue in the first shared-memory segment */
212  app->event_queue = segment_manager_alloc_queue (sm, app_evt_queue_size);
213 
214  /* Check that the obvious things are properly set up */
215  application_verify_cb_fns (cb_fns);
216 
217  /* Add app to lookup by api_client_index table */
218  application_table_add (app);
219 
220  return 0;
221 }
222 
225 {
226  return pool_elt_at_index (app_pool, index);
227 }
228 
231 {
232  if (pool_is_free_index (app_pool, index))
233  return 0;
234 
235  return pool_elt_at_index (app_pool, index);
236 }
237 
238 u32
240 {
241  return app - app_pool;
242 }
243 
244 static segment_manager_t *
246 {
247  segment_manager_t *sm = 0;
248 
249  /* If the first segment manager is not in use, don't allocate a new one */
250  if (app->first_segment_manager != APP_INVALID_SEGMENT_MANAGER_INDEX
251  && app->first_segment_manager_in_use == 0)
252  {
253  sm = segment_manager_get (app->first_segment_manager);
254  app->first_segment_manager_in_use = 1;
255  return sm;
256  }
257 
258  sm = segment_manager_new ();
259  sm->properties = &app->sm_properties;
260 
261  return sm;
262 }
263 
264 /**
265  * Start listening local transport endpoint for requested transport.
266  *
267  * Creates a 'dummy' stream session with state LISTENING to be used in session
268  * lookups, prior to establishing connection. Requests transport to build
269  * it's own specific listening connection.
270  */
271 int
273  transport_endpoint_t * tep, u64 * res)
274 {
275  segment_manager_t *sm;
276  stream_session_t *s;
277  u64 handle;
278 
279  s = listen_session_new (session_type);
280  s->app_index = srv->index;
281 
282  if (stream_session_listen (s, tep))
283  goto err;
284 
285  /* Allocate segment manager. All sessions derived out of a listen session
286  * have fifos allocated by the same segment manager. */
288  if (sm == 0)
289  goto err;
290 
291  /* Add to app's listener table. Useful to find all child listeners
292  * when app goes down, although, just for unbinding this is not needed */
293  handle = listen_session_get_handle (s);
294  hash_set (srv->listeners_table, handle, segment_manager_index (sm));
295 
296  *res = handle;
297  return 0;
298 
299 err:
300  listen_session_del (s);
301  return -1;
302 }
303 
304 /**
305  * Stop listening on session associated to handle
306  */
307 int
309 {
310  stream_session_t *listener;
311  uword *indexp;
312  segment_manager_t *sm;
313 
314  if (srv && hash_get (srv->listeners_table, handle) == 0)
315  {
316  clib_warning ("app doesn't own handle %llu!", handle);
317  return -1;
318  }
319 
320  listener = listen_session_get_from_handle (handle);
321  stream_session_stop_listen (listener);
322 
323  indexp = hash_get (srv->listeners_table, handle);
324  ASSERT (indexp);
325 
326  sm = segment_manager_get (*indexp);
327  if (srv->first_segment_manager == *indexp)
328  {
329  /* Delete sessions but don't remove segment manager */
330  srv->first_segment_manager_in_use = 0;
332  }
333  else
334  {
336  }
337  hash_unset (srv->listeners_table, handle);
338  listen_session_del (listener);
339 
340  return 0;
341 }
342 
343 int
345  transport_endpoint_t * tep, u32 api_context)
346 {
347  segment_manager_t *sm;
348  transport_connection_t *tc = 0;
349  int rv;
350 
351  /* Make sure we have a segment manager for connects */
352  if (app->connects_seg_manager == (u32) ~ 0)
353  {
355  if (sm == 0)
356  return -1;
357  app->connects_seg_manager = segment_manager_index (sm);
358  }
359 
360  if ((rv = stream_session_open (app->index, sst, tep, &tc)))
361  return rv;
362 
363  /* Store api_context for when the reply comes. Not the nicest thing
364  * but better than allocating a separate half-open pool. */
365  tc->s_index = api_context;
366 
367  return 0;
368 }
369 
372 {
373  ASSERT (app->connects_seg_manager != (u32) ~ 0);
374  return segment_manager_get (app->connects_seg_manager);
375 }
376 
379  stream_session_t * s)
380 {
381  uword *smp;
382  smp = hash_get (app->listeners_table, listen_session_get_handle (s));
383  ASSERT (smp != 0);
384  return segment_manager_get (*smp);
385 }
386 
387 static u8 *
389 {
390  u8 *app_name;
391 
392  vl_api_registration_t *regp;
393  regp = vl_api_client_index_to_registration (app->api_client_index);
394  if (!regp)
395  app_name = format (0, "builtin-%d%c", app->index, 0);
396  else
397  app_name = format (0, "%s%c", regp->name, 0);
398 
399  return app_name;
400 }
401 
402 int
404 {
405  return !(app->flags & APP_OPTIONS_FLAGS_IS_PROXY);
406 }
407 
408 int
409 application_add_segment_notify (u32 app_index, u32 fifo_segment_index)
410 {
411  application_t *app = application_get (app_index);
412  u32 seg_size = 0;
413  u8 *seg_name;
414 
415  /* Send an API message to the external app, to map new segment */
416  ASSERT (app->cb_fns.add_segment_callback);
417 
418  segment_manager_get_segment_info (fifo_segment_index, &seg_name, &seg_size);
419  return app->cb_fns.add_segment_callback (app->api_client_index, seg_name,
420  seg_size);
421 }
422 
423 u8 *
424 format_application_listener (u8 * s, va_list * args)
425 {
426  application_t *app = va_arg (*args, application_t *);
427  u64 handle = va_arg (*args, u64);
428  u32 index = va_arg (*args, u32);
429  int verbose = va_arg (*args, int);
430  stream_session_t *listener;
431  u8 *app_name, *str;
432 
433  if (app == 0)
434  {
435  if (verbose)
436  s = format (s, "%-40s%-20s%-15s%-15s%-10s", "Connection", "App",
437  "API Client", "ListenerID", "SegManager");
438  else
439  s = format (s, "%-40s%-20s", "Connection", "App");
440 
441  return s;
442  }
443 
444  app_name = app_get_name_from_reg_index (app);
445  listener = listen_session_get_from_handle (handle);
446  str = format (0, "%U", format_stream_session, listener, verbose);
447 
448  if (verbose)
449  {
450  s = format (s, "%-40s%-20s%-15u%-15u%-10u", str, app_name,
451  app->api_client_index, handle, index);
452  }
453  else
454  s = format (s, "%-40s%-20s", str, app_name);
455 
456  vec_free (app_name);
457  return s;
458 }
459 
460 void
462 {
464  segment_manager_t *sm;
465  u8 *app_name, *s = 0;
466  int j;
467 
468  /* Header */
469  if (app == 0)
470  {
471  if (verbose)
472  vlib_cli_output (vm, "%-40s%-20s%-15s%-10s", "Connection", "App",
473  "API Client", "SegManager");
474  else
475  vlib_cli_output (vm, "%-40s%-20s", "Connection", "App");
476  return;
477  }
478 
479  /* make sure */
480  if (app->connects_seg_manager == (u32) ~ 0)
481  return;
482 
483  app_name = app_get_name_from_reg_index (app);
484 
485  /* Across all fifo segments */
486  sm = segment_manager_get (app->connects_seg_manager);
487  for (j = 0; j < vec_len (sm->segment_indices); j++)
488  {
489  svm_fifo_segment_private_t *fifo_segment;
490  svm_fifo_t *fifo;
491  u8 *str;
492 
493  fifo_segment = svm_fifo_segment_get_segment (sm->segment_indices[j]);
494  fifo = svm_fifo_segment_get_fifo_list (fifo_segment);
495  while (fifo)
496  {
497  u32 session_index, thread_index;
498  stream_session_t *session;
499 
500  session_index = fifo->master_session_index;
501  thread_index = fifo->master_thread_index;
502 
503  session = stream_session_get (session_index, thread_index);
504  str = format (0, "%U", format_stream_session, session, verbose);
505 
506  if (verbose)
507  s = format (s, "%-40s%-20s%-15u%-10u", str, app_name,
508  app->api_client_index, app->connects_seg_manager);
509  else
510  s = format (s, "%-40s%-20s", str, app_name);
511 
512  vlib_cli_output (vm, "%v", s);
513  vec_reset_length (s);
514  vec_free (str);
515 
516  fifo = fifo->next;
517  }
518  vec_free (s);
519  }
520 
521  vec_free (app_name);
522 }
523 
524 u8 *
525 format_application (u8 * s, va_list * args)
526 {
527  application_t *app = va_arg (*args, application_t *);
528  CLIB_UNUSED (int verbose) = va_arg (*args, int);
529  u8 *app_name;
530 
531  if (app == 0)
532  {
533  if (verbose)
534  s = format (s, "%-10s%-20s%-15s%-15s%-15s%-15s", "Index", "Name",
535  "API Client", "Add seg size", "Rx fifo size",
536  "Tx fifo size");
537  else
538  s = format (s, "%-10s%-20s%-20s", "Index", "Name", "API Client");
539  return s;
540  }
541 
542  app_name = app_get_name_from_reg_index (app);
543  if (verbose)
544  s = format (s, "%-10d%-20s%-15d%-15d%-15d%-15d", app->index, app_name,
545  app->api_client_index, app->sm_properties.add_segment_size,
546  app->sm_properties.rx_fifo_size,
547  app->sm_properties.tx_fifo_size);
548  else
549  s = format (s, "%-10d%-20s%-20d", app->index, app_name,
550  app->api_client_index);
551  return s;
552 }
553 
554 static clib_error_t *
556  vlib_cli_command_t * cmd)
557 {
558  application_t *app;
559  int do_server = 0;
560  int do_client = 0;
561  int verbose = 0;
562 
564  {
565  clib_error_return (0, "session layer is not enabled");
566  }
567 
569  {
570  if (unformat (input, "server"))
571  do_server = 1;
572  else if (unformat (input, "client"))
573  do_client = 1;
574  else if (unformat (input, "verbose"))
575  verbose = 1;
576  else
577  break;
578  }
579 
580  if (do_server)
581  {
582  u64 handle;
583  u32 index;
584  if (pool_elts (app_pool))
585  {
587  0 /* header */ , 0, 0,
588  verbose);
589  /* *INDENT-OFF* */
590  pool_foreach (app, app_pool,
591  ({
592  /* App's listener sessions */
593  if (hash_elts (app->listeners_table) == 0)
594  continue;
595  hash_foreach (handle, index, app->listeners_table,
596  ({
597  vlib_cli_output (vm, "%U", format_application_listener, app,
598  handle, index, verbose);
599  }));
600  }));
601  /* *INDENT-ON* */
602  }
603  else
604  vlib_cli_output (vm, "No active server bindings");
605  }
606 
607  if (do_client)
608  {
609  if (pool_elts (app_pool))
610  {
611  application_format_connects (0, verbose);
612 
613  /* *INDENT-OFF* */
614  pool_foreach (app, app_pool,
615  ({
616  if (app->connects_seg_manager == (u32)~0)
617  continue;
618  application_format_connects (app, verbose);
619  }));
620  /* *INDENT-ON* */
621  }
622  else
623  vlib_cli_output (vm, "No active client bindings");
624  }
625 
626  /* Print app related info */
627  if (!do_server && !do_client)
628  {
629  vlib_cli_output (vm, "%U", format_application, 0, verbose);
630  pool_foreach (app, app_pool, (
631  {
632  vlib_cli_output (vm, "%U",
633  format_application, app,
634  verbose);
635  }
636  ));
637  }
638 
639  return 0;
640 }
641 
642 /* *INDENT-OFF* */
643 VLIB_CLI_COMMAND (show_app_command, static) =
644 {
645  .path = "show app",
646  .short_help = "show app [server|client] [verbose]",
647  .function = show_app_command_fn,
648 };
649 /* *INDENT-ON* */
650 
651 /*
652  * fd.io coding-style-patch-verification: ON
653  *
654  * Local Variables:
655  * eval: (c-set-style "gnu")
656  * End:
657  */
static void application_table_add(application_t *app)
Definition: application.c:54
int application_stop_listen(application_t *srv, u64 handle)
Stop listening on session associated to handle.
Definition: application.c:308
u8 * name
Client name.
Definition: api_common.h:51
#define hash_set(h, key, value)
Definition: hash.h:254
static u32 default_app_evt_queue_size
Default application event queue size.
Definition: application.c:33
sll srl srl sll sra u16x4 i
Definition: vector_sse2.h:337
static stream_session_t * listen_session_get_from_handle(u64 handle)
Definition: session.h:360
int application_add_segment_notify(u32 app_index, u32 fifo_segment_index)
Definition: application.c:409
#define CLIB_UNUSED(x)
Definition: clib.h:79
#define hash_unset(h, key)
Definition: hash.h:260
a
Definition: bitmap.h:516
struct _transport_connection transport_connection_t
struct _segment_manager_properties segment_manager_properties_t
static clib_error_t * show_app_command_fn(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
Definition: application.c:555
application_t * application_new()
Definition: application.c:77
application_t * application_lookup(u32 api_client_index)
Definition: application.c:66
static stream_session_t * listen_session_new(session_type_t type)
Definition: session.h:377
u8 * format(u8 *s, const char *fmt,...)
Definition: format.c:419
static u8 session_manager_is_enabled()
Definition: session.h:423
#define pool_get(P, E)
Allocate an object E from a pool P (unspecified alignment).
Definition: pool.h:225
#define vec_reset_length(v)
Reset vector length to zero NULL-pointer tolerant.
struct _svm_fifo svm_fifo_t
void segment_manager_get_segment_info(u32 index, u8 **name, u32 *size)
segment_manager_t * application_get_listen_segment_manager(application_t *app, stream_session_t *s)
Definition: application.c:378
int application_is_proxy(application_t *app)
Definition: application.c:403
#define pool_foreach(VAR, POOL, BODY)
Iterate through pool.
Definition: pool.h:437
static void application_table_del(application_t *app)
Definition: application.c:60
struct _stream_session_cb_vft session_cb_vft_t
struct _vnet_unbind_args_t vnet_unbind_args_t
#define hash_foreach(key_var, value_var, h, body)
Definition: hash.h:418
#define clib_error_return(e, args...)
Definition: error.h:99
unix_shared_memory_queue_t * segment_manager_alloc_queue(segment_manager_t *sm, u32 queue_size)
Allocates shm queue in the first segment.
unsigned long u64
Definition: types.h:89
static u8 * app_get_name_from_reg_index(application_t *app)
Definition: application.c:388
int stream_session_open(u32 app_index, session_type_t st, transport_endpoint_t *rmt, transport_connection_t **res)
Ask transport to open connection to remote transport endpoint.
Definition: session.c:637
struct _stream_session_t stream_session_t
segment_manager_t * segment_manager_new()
static void application_verify_cb_fns(session_cb_vft_t *cb_fns)
Definition: application.c:158
static void listen_session_del(stream_session_t *s)
Definition: session.h:399
#define hash_get(h, key)
Definition: hash.h:248
static svm_fifo_t * svm_fifo_segment_get_fifo_list(svm_fifo_segment_private_t *fifo_segment)
#define pool_elt_at_index(p, i)
Returns pointer to element at given index.
Definition: pool.h:458
struct _unformat_input_t unformat_input_t
static application_t * app_pool
Pool from which we allocate all applications.
Definition: application.c:23
#define pool_put(P, E)
Free an object E in pool P.
Definition: pool.h:270
u8 segment_manager_has_fifos(segment_manager_t *sm)
session_type_t
#define SEGMENT_MANAGER_INVALID_APP_INDEX
An API client registration, only in vpp/vlib.
Definition: api_common.h:44
#define UNFORMAT_END_OF_INPUT
Definition: format.h:143
vlib_main_t * vm
Definition: buffer.c:283
u8 * format_stream_session(u8 *s, va_list *args)
Format stream session as per the following format.
Definition: session_cli.c:52
#define vec_free(V)
Free vector&#39;s memory (no header).
Definition: vec.h:336
segment_manager_t * application_get_connect_segment_manager(application_t *app)
Definition: application.c:371
#define clib_warning(format, args...)
Definition: error.h:59
unix_shared_memory_queue_t * vl_api_client_index_to_input_queue(u32 index)
#define pool_is_free_index(P, I)
Use free bitmap to query whether given index is free.
Definition: pool.h:267
struct _application application_t
static uword * app_by_api_client_index
Hash table of apps by api client index.
Definition: application.c:28
static svm_fifo_segment_private_t * svm_fifo_segment_get_segment(u32 segment_index)
static u32 segment_manager_index(segment_manager_t *sm)
static segment_manager_t * application_alloc_segment_manager(application_t *app)
Definition: application.c:245
#define VLIB_CLI_COMMAND(x,...)
Definition: cli.h:154
void segment_manager_init_del(segment_manager_t *sm)
static uword hash_elts(void *v)
Definition: hash.h:117
#define ASSERT(truth)
unsigned int u32
Definition: types.h:88
int application_init(application_t *app, u32 api_client_index, u64 *options, session_cb_vft_t *cb_fns)
Definition: application.c:171
void application_del(application_t *app)
Definition: application.c:91
u32 application_get_index(application_t *app)
Definition: application.c:239
int application_start_listen(application_t *srv, session_type_t session_type, transport_endpoint_t *tep, u64 *res)
Start listening local transport endpoint for requested transport.
Definition: application.c:272
int stream_session_stop_listen(stream_session_t *s)
Ask transport to stop listening on local transport endpoint.
Definition: session.c:707
static vlib_main_t * vlib_get_main(void)
Definition: global_funcs.h:23
u64 uword
Definition: types.h:112
void segment_manager_del(segment_manager_t *sm)
Removes segment manager.
int segment_manager_init(segment_manager_t *sm, segment_manager_properties_t *properties, u32 first_seg_size)
Initializes segment manager based on options provided.
u8 * format_application_listener(u8 *s, va_list *args)
Definition: application.c:424
#define vec_len(v)
Number of elements in vector (rvalue-only, NULL tolerant)
unsigned char u8
Definition: types.h:56
u8 * format_application(u8 *s, va_list *args)
Definition: application.c:525
application_t * application_get(u32 index)
Definition: application.c:224
struct _transport_endpoint transport_endpoint_t
void segment_manager_del_sessions(segment_manager_t *sm)
Initiate disconnects for all sessions &#39;owned&#39; by a segment manager.
int vnet_unbind(vnet_unbind_args_t *a)
struct _segment_manager segment_manager_t
int stream_session_listen(stream_session_t *s, transport_endpoint_t *tep)
Ask transport to listen on local transport endpoint.
Definition: session.c:676
vl_api_registration_t * vl_api_client_index_to_registration(u32 index)
int application_api_queue_is_full(application_t *app)
Definition: application.c:36
#define APP_INVALID_SEGMENT_MANAGER_INDEX
Definition: application.h:98
void vlib_cli_output(vlib_main_t *vm, char *fmt,...)
Definition: cli.c:680
static segment_manager_t * segment_manager_get(u32 index)
static u64 listen_session_get_handle(stream_session_t *s)
Definition: session.h:353
int application_open_session(application_t *app, session_type_t sst, transport_endpoint_t *tep, u32 api_context)
Definition: application.c:344
application_t * application_get_if_valid(u32 index)
Definition: application.c:230
void application_format_connects(application_t *app, int verbose)
Definition: application.c:461
uword unformat(unformat_input_t *i, const char *fmt,...)
Definition: unformat.c:972
static uword unformat_check_input(unformat_input_t *i)
Definition: format.h:169
struct _unix_shared_memory_queue unix_shared_memory_queue_t
static stream_session_t * stream_session_get(u32 si, u32 thread_index)
Definition: session.h:217
static uword pool_elts(void *v)
Number of active elements in a pool.
Definition: pool.h:128