builtinurl: mark api as deprecated
[vpp.git] / src / vnet / ipfix-export / flow_report.h
1 /*
2  * Copyright (c) 2015 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 #ifndef __included_vnet_flow_report_h__
16 #define __included_vnet_flow_report_h__
17
18 #include <vlib/vlib.h>
19 #include <vnet/vnet.h>
20 #include <vnet/ethernet/ethernet.h>
21 #include <vnet/ethernet/packet.h>
22 #include <vnet/ip/ip_packet.h>
23 #include <vnet/ip/ip_types.h>
24 #include <vnet/ip/ip4_packet.h>
25 #include <vnet/ip/ip6_packet.h>
26 #include <vnet/udp/udp_packet.h>
27 #include <vlib/cli.h>
28 #include <vppinfra/error.h>
29 #include <vppinfra/hash.h>
30 #include <vppinfra/cache.h>
31
32 #include <vnet/ipfix-export/ipfix_packet.h>
33
34 /* ipfix field definitions for a particular report */
35 typedef struct
36 {
37   u32 info_element;
38   u32 size;
39 } ipfix_report_element_t;
40
41 /* Used to build the rewrite */
42 typedef struct
43 {
44   ip4_header_t ip4;
45   udp_header_t udp;
46   ipfix_template_packet_t ipfix;
47 } ip4_ipfix_template_packet_t;
48
49 /* Used to build the rewrite */
50 typedef struct
51 {
52   ip6_header_t ip6;
53   udp_header_t udp;
54   ipfix_template_packet_t ipfix;
55 } ip6_ipfix_template_packet_t;
56
57 struct flow_report_main;
58 struct flow_report;
59 struct ipfix_exporter;
60
61 typedef vlib_frame_t *(vnet_flow_data_callback_t) (
62   struct flow_report_main *frm, struct ipfix_exporter *exp,
63   struct flow_report *, vlib_frame_t *, u32 *, u32);
64
65 typedef u8 *(vnet_flow_rewrite_callback_t) (struct ipfix_exporter *exp,
66                                             struct flow_report *,
67                                             u16, ipfix_report_element_t *elts,
68                                             u32 n_elts, u32 *stream_index);
69
70 u8 *vnet_flow_rewrite_generic_callback (struct ipfix_exporter *exp,
71                                         struct flow_report *, u16,
72                                         ipfix_report_element_t *elts,
73                                         u32 n_elts, u32 *stream_index);
74
75 typedef union
76 {
77   void *as_ptr;
78   uword as_uword;
79 } opaque_t;
80
81 /*
82  * A stream represents an IPFIX session to a destination. We can have
83  * multiple streams to the same destination, but each one has its own
84  * domain and source port. A stream has a sequence number for that
85  * session. A stream may contain multiple templates (i.e multiple for
86  * reports) and each stream also has its own template space.
87  *
88  * A stream has per thread state so that data packets can be built
89  * and send on multiple threads at the same time.
90  */
91 typedef struct
92 {
93   u32 domain_id;
94   u32 sequence_number;
95   u16 src_port;
96   u16 n_reports;
97   u16 next_template_no;
98 } flow_report_stream_t;
99
100 /*
101  * For each flow_report we want to be able to build buffers/frames per thread.
102  */
103 typedef struct
104 {
105   vlib_buffer_t *buffer;
106   vlib_frame_t *frame;
107   u16 next_data_offset;
108   /*
109    * We need this per stream as the IPFIX sequence number is the count of
110    * data record sent, not the count of packets with data records sent.
111    * See RFC 7011, Sec 3.1
112    */
113   u8 n_data_records;
114 } flow_report_per_thread_t;
115
116 /*
117  * A flow report represents a group of fields that are to be exported.
118  * Each flow_report has an associated template that is generated when
119  * the flow_report is added. Each flow_report is associated with a
120  * stream, and multiple flow_reports can use the same stream. When
121  * adding a flow_report the keys for the stream are the domain_id
122  * and the source_port.
123  */
124 typedef struct flow_report
125 {
126   /* ipfix rewrite, set by callback */
127   u8 *rewrite;
128   u16 template_id;
129   int data_record_size;
130   flow_report_per_thread_t *per_thread_data;
131   u32 stream_index;
132   f64 last_template_sent;
133   int update_rewrite;
134
135   /* Bitmap of fields to send */
136   uword *fields_to_send;
137
138   /* Opaque data */
139   opaque_t opaque;
140
141   /* build-the-template-packet rewrite callback */
142   vnet_flow_rewrite_callback_t *rewrite_callback;
143   ipfix_report_element_t *report_elements;
144   u32 n_report_elements;
145   u32 *stream_indexp;
146
147   /* Send-flow-data callback */
148   vnet_flow_data_callback_t *flow_data_callback;
149 } flow_report_t;
150
151 /*
152  * The maximum number of ipfix exporters we can have at once
153  */
154 #define IPFIX_EXPORTERS_MAX 5
155
156 /*
157  * We support multiple exporters. Each one has its own configured
158  * destination, and its own set of reports and streams.
159  */
160 typedef struct ipfix_exporter
161 {
162   flow_report_t *reports;
163   flow_report_stream_t *streams;
164
165   /* ipfix collector ip address, port, our ip address, fib index */
166   ip_address_t ipfix_collector;
167   u16 collector_port;
168   ip_address_t src_address;
169   u32 fib_index;
170
171   /* Path MTU */
172   u32 path_mtu;
173
174   /* time interval in seconds after which to resend templates */
175   u32 template_interval;
176
177   /* UDP checksum calculation enable flag */
178   u8 udp_checksum;
179
180   /*
181    * The amount of data needed for all the headers, prior to the first
182    * flowset (template or data or ...) This is mostly dependent on the
183    * L3 and L4 protocols in use.
184    */
185   u32 all_headers_size;
186 } ipfix_exporter_t;
187
188 typedef struct flow_report_main
189 {
190   /*
191    * A pool of the exporters. Entry 0 is always there for backwards
192    * compatability reasons. Entries 1 and above have to be created by
193    * the users.
194    */
195   ipfix_exporter_t *exporters;
196
197   /* time scale transform. Joy. */
198   u32 unix_time_0;
199   f64 vlib_time_0;
200
201   /* convenience variables */
202   vlib_main_t *vlib_main;
203   vnet_main_t *vnet_main;
204
205   u16 msg_id_base;
206 } flow_report_main_t;
207
208 extern flow_report_main_t flow_report_main;
209
210 extern vlib_node_registration_t flow_report_process_node;
211
212 typedef struct
213 {
214   vnet_flow_data_callback_t *flow_data_callback;
215   vnet_flow_rewrite_callback_t *rewrite_callback;
216   ipfix_report_element_t *report_elements;
217   u32 n_report_elements;
218   opaque_t opaque;
219   int is_add;
220   u32 domain_id;
221   u16 src_port;
222   u32 *stream_indexp;
223   /*
224    * When adding a flow report, the index of the flow report is stored
225    * here on success.
226    */
227   u32 flow_report_index;
228 } vnet_flow_report_add_del_args_t;
229
230 int vnet_flow_report_add_del (ipfix_exporter_t *exp,
231                               vnet_flow_report_add_del_args_t *a,
232                               u16 *template_id);
233
234 clib_error_t *flow_report_add_del_error_to_clib_error (int error);
235
236 void vnet_flow_reports_reset (ipfix_exporter_t *exp);
237
238 void vnet_stream_reset (ipfix_exporter_t *exp, u32 stream_index);
239
240 int vnet_stream_change (ipfix_exporter_t *exp, u32 old_domain_id,
241                         u16 old_src_port, u32 new_domain_id, u16 new_src_port);
242
243 /*
244  * Search all the exporters for one that has a matching destination address.
245  */
246 ipfix_exporter_t *
247 vnet_ipfix_exporter_lookup (const ip_address_t *ipfix_collector);
248
249 /*
250  * Get the currently in use buffer for the given stream on the given core.
251  * If there is no current buffer then allocate a new one and return that.
252  * This is the buffer that data records should be written into. The offset
253  * currently in use is stored in the per-thread data for the stream and
254  * should be updated as new records are written in.
255  */
256 vlib_buffer_t *vnet_ipfix_exp_get_buffer (vlib_main_t *vm,
257                                           ipfix_exporter_t *exp,
258                                           flow_report_t *fr, u32 thread_index);
259
260 /*
261  * Send the provided buffer. At this stage the buffer should be populated
262  * with data records, with the offset in use stored in the stream per thread
263  * data. This func will fix up all the headers and then send the buffer.
264  */
265 void vnet_ipfix_exp_send_buffer (vlib_main_t *vm, ipfix_exporter_t *exp,
266                                  flow_report_t *fr,
267                                  flow_report_stream_t *stream,
268                                  u32 thread_index, vlib_buffer_t *b0);
269
270 #endif /* __included_vnet_flow_report_h__ */
271
272 /*
273  * fd.io coding-style-patch-verification: ON
274  *
275  * Local Variables:
276  * eval: (c-set-style "gnu")
277  * End:
278  */