summaryrefslogtreecommitdiffstats
path: root/src/plugins/nsim/nsim.h
blob: d7d32b9d234dc8535dd779fae8e1a5933f0f5229 (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
/*
 * nsim.h - skeleton vpp engine plug-in header file
 *
 * Copyright (c) <current-year> <your-organization>
 * 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_nsim_h__
#define __included_nsim_h__

#include <vnet/vnet.h>
#include <vnet/ip/ip.h>
#include <vnet/ethernet/ethernet.h>

#include <vppinfra/hash.h>
#include <vppinfra/error.h>

#define NSIM_MAX_TX_BURST 32	/**< max packets in a tx burst */

typedef struct
{
  f64 tx_time;
  u32 rx_sw_if_index;
  u32 tx_sw_if_index;
  u32 output_next_index;
  u32 buffer_index;
  u32 pad;			/* pad to 32-bytes */
} nsim_wheel_entry_t;

typedef struct
{
  u32 wheel_size;
  u32 cursize;
  u32 head;
  u32 tail;
  nsim_wheel_entry_t *entries;
    CLIB_CACHE_LINE_ALIGN_MARK (pad);
} nsim_wheel_t;

typedef struct nsim_node_ctx
{
  vnet_feature_config_main_t *fcm;
  f64 expires;
  u32 *drop;
  u32 *reord;
  u16 *reord_nexts;
  u8 *action;
  u64 n_buffered;
  u64 n_loss;
} nsim_node_ctx_t;

#define foreach_nsm_action			\
  _(DROP, "Packet loss")			\
  _(REORDER, "Packet reorder")

enum nsm_action_bit
{
#define _(sym, str) NSIM_ACTION_##sym##_BIT,
  foreach_nsm_action
#undef _
};

typedef enum nsm_action
{
#define _(sym, str) NSIM_ACTION_##sym = 1 << NSIM_ACTION_##sym##_BIT,
  foreach_nsm_action
#undef _
} nsm_action_e;

typedef struct
{
  /* API message ID base */
  u16 msg_id_base;

  /* output feature arc index */
  u16 arc_index;

  /* Two interfaces, cross-connected with delay */
  u32 sw_if_index0, sw_if_index1;
  u32 output_next_index0, output_next_index1;

  /* N interfaces, using the output feature */
  u32 *output_next_index_by_sw_if_index;

  /* Random seed for loss-rate simulation */
  u32 seed;

  /* Per-thread scheduler wheels */
  nsim_wheel_t **wheel_by_thread;

  /* Config parameters */
  f64 delay;
  f64 bandwidth;
  f64 packet_size;
  f64 drop_fraction;
  f64 reorder_fraction;
  u32 poll_main_thread;

  u64 mmap_size;

  /* Wheels are configured */
  int is_configured;

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

extern nsim_main_t nsim_main;

extern vlib_node_registration_t nsim_node;
extern vlib_node_registration_t nsim_input_node;

#endif /* __included_nsim_h__ */

/*
 * fd.io coding-style-patch-verification: ON
 *
 * Local Variables:
 * eval: (c-set-style "gnu")
 * End:
 */
int msglen, int fds[], int num_fds); uword private_data; } clib_socket_t; /* socket config format is host:port. Unspecified port causes a free one to be chosen starting from IPPORT_USERRESERVED (5000). */ clib_error_t *clib_socket_init (clib_socket_t * socket); clib_error_t *clib_socket_accept (clib_socket_t * server, clib_socket_t * client); always_inline uword clib_socket_is_server (clib_socket_t * sock) { return (sock->flags & CLIB_SOCKET_F_IS_SERVER) != 0; } always_inline uword clib_socket_is_client (clib_socket_t * s) { return !clib_socket_is_server (s); } always_inline uword clib_socket_is_connected (clib_socket_t * sock) { return sock->fd > 0; } always_inline int clib_socket_rx_end_of_file (clib_socket_t * s) { return s->flags & CLIB_SOCKET_F_RX_END_OF_FILE; } always_inline void * clib_socket_tx_add (clib_socket_t * s, int n_bytes) { u8 *result; vec_add2 (s->tx_buffer, result, n_bytes); return result; } always_inline void clib_socket_tx_add_va_formatted (clib_socket_t * s, char *fmt, va_list * va) { s->tx_buffer = va_format (s->tx_buffer, fmt, va); } always_inline clib_error_t * clib_socket_tx (clib_socket_t * s) { return s->write_func (s); } always_inline clib_error_t * clib_socket_rx (clib_socket_t * s, int n_bytes) { return s->read_func (s, n_bytes); } always_inline clib_error_t * clib_socket_sendmsg (clib_socket_t * s, void *msg, int msglen, int fds[], int num_fds) { return s->sendmsg_func (s, msg, msglen, fds, num_fds); } always_inline clib_error_t * clib_socket_recvmsg (clib_socket_t * s, void *msg, int msglen, int fds[], int num_fds) { return s->recvmsg_func (s, msg, msglen, fds, num_fds); } always_inline void clib_socket_free (clib_socket_t * s) { vec_free (s->tx_buffer); vec_free (s->rx_buffer); if (clib_mem_is_heap_object (s->config)) vec_free (s->config); memset (s, 0, sizeof (s[0])); } always_inline clib_error_t * clib_socket_close (clib_socket_t * sock) { clib_error_t *err; err = (*sock->close_func) (sock); return err; } void clib_socket_tx_add_formatted (clib_socket_t * s, char *fmt, ...); #endif /* _clib_included_socket_h */ /* * fd.io coding-style-patch-verification: ON * * Local Variables: * eval: (c-set-style "gnu") * End: */