blob: 8f092eed1c252abfe1a32380c3353250982b7d54 [file] [log] [blame]
Klement Sekera8f2a4ea2017-05-04 06:15:18 +02001/*
2 *------------------------------------------------------------------
3 * Copyright (c) 2017 Cisco and/or its affiliates.
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at:
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 *------------------------------------------------------------------
16 */
17
18#ifndef vpp_api_h_included
19#define vpp_api_h_included
20
21#include <string.h>
22#include <stdbool.h>
23#include <vppinfra/types.h>
Ole Troan003d5da2018-12-18 12:23:13 +010024#include <vlibapi/api_types.h>
Klement Sekeradc15be22017-06-12 06:49:33 +020025#include <vapi/vapi_common.h>
Mohsin Kazmi3fca5672018-01-04 18:57:26 +010026#include <svm/queue.h>
Klement Sekeradc15be22017-06-12 06:49:33 +020027
28#ifdef __cplusplus
29extern "C"
30{
31#endif
Klement Sekera8f2a4ea2017-05-04 06:15:18 +020032
33/**
34 * @file vapi.h
35 *
36 * common vpp api C declarations
37 *
38 * This file declares the common C API functions. These include connect,
39 * disconnect and utility functions as well as the low-level vapi_send and
40 * vapi_recv API. This is only the transport layer.
41 *
42 * Message formats and higher-level APIs are generated by running the
43 * vapi_c_gen.py script (which is run for in-tree APIs as part of the build
44 * process). It's not recommended to mix the higher and lower level APIs. Due
45 * to version issues, the higher-level APIs are not part of the shared library.
46 */
Ole Troan2ca88ff2022-01-27 16:25:43 +010047typedef struct vapi_ctx_s *vapi_ctx_t;
Klement Sekera8f2a4ea2017-05-04 06:15:18 +020048
49/**
50 * @brief allocate vapi message of given size
51 *
52 * @note message must be freed by vapi_msg_free if not consumed by vapi_send
53 * call
54 *
55 * @param ctx opaque vapi context
56 *
57 * @return pointer to message or NULL if out of memory
58 */
Ole Troan2ca88ff2022-01-27 16:25:43 +010059void *vapi_msg_alloc (vapi_ctx_t ctx, size_t size);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +020060
61/**
62 * @brief free a vapi message
63 *
64 * @note messages received by vapi_recv must be freed when no longer needed
65 *
66 * @param ctx opaque vapi context
67 * @param msg message to be freed
68 */
Ole Troan2ca88ff2022-01-27 16:25:43 +010069void vapi_msg_free (vapi_ctx_t ctx, void *msg);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +020070
71/**
72 * @brief allocate vapi context
73 *
74 * @param[out] pointer to result variable
75 *
76 * @return VAPI_OK on success, other error code on error
77 */
Ole Troan2ca88ff2022-01-27 16:25:43 +010078vapi_error_e vapi_ctx_alloc (vapi_ctx_t *result);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +020079
80/**
81 * @brief free vapi context
82 */
Ole Troan2ca88ff2022-01-27 16:25:43 +010083void vapi_ctx_free (vapi_ctx_t ctx);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +020084
85/**
86 * @brief check if message identified by it's message id is known by the vpp to
87 * which the connection is open
88 */
Ole Troan2ca88ff2022-01-27 16:25:43 +010089bool vapi_is_msg_available (vapi_ctx_t ctx, vapi_msg_id_t type);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +020090
91/**
92 * @brief connect to vpp
93 *
94 * @param ctx opaque vapi context, must be allocated using vapi_ctx_alloc first
95 * @param name application name
96 * @param chroot_prefix shared memory prefix
97 * @param max_outstanding_requests max number of outstanding requests queued
98 * @param response_queue_size size of the response queue
99 * @param mode mode of operation - blocking or nonblocking
Klement Sekeradab732a2018-07-04 13:43:46 +0200100 * @param handle_keepalives - if true, automatically handle memclnt_keepalive
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200101 *
102 * @return VAPI_OK on success, other error code on error
103 */
Ole Troan2ca88ff2022-01-27 16:25:43 +0100104vapi_error_e vapi_connect (vapi_ctx_t ctx, const char *name,
105 const char *chroot_prefix,
106 int max_outstanding_requests,
107 int response_queue_size, vapi_mode_e mode,
108 bool handle_keepalives);
109
110/**
Stanislav Zaikindc4d21e2024-03-06 19:48:30 +0100111 * @brief connect to vpp
112 *
113 * @param ctx opaque vapi context, must be allocated using vapi_ctx_alloc first
114 * @param name application name
115 * @param path shared memory prefix or path to unix socket
116 * @param max_outstanding_requests max number of outstanding requests queued
117 * @param response_queue_size size of the response queue
118 * @param mode mode of operation - blocking or nonblocking
119 * @param handle_keepalives - if true, automatically handle memclnt_keepalive
120 * @param use_uds - if true, use unix domain socket transport
121 *
122 * @return VAPI_OK on success, other error code on error
123 */
124vapi_error_e vapi_connect_ex (vapi_ctx_t ctx, const char *name,
125 const char *path, int max_outstanding_requests,
126 int response_queue_size, vapi_mode_e mode,
127 bool handle_keepalives, bool use_uds);
128
129/**
Ole Troan2ca88ff2022-01-27 16:25:43 +0100130 * @brief connect to vpp from a client in same process
131 * @remark This MUST be called from a separate thread. If called
132 * from the main thread, it will deadlock.
133 *
134 * @param ctx opaque vapi context, must be allocated using vapi_ctx_alloc first
135 * @param name application name
136 * @param max_outstanding_requests max number of outstanding requests queued
137 * @param response_queue_size size of the response queue
138 * @param mode mode of operation - blocking or nonblocking
139 * @param handle_keepalives - if true, automatically handle memclnt_keepalive
140 *
141 * @return VAPI_OK on success, other error code on error
142 */
143vapi_error_e vapi_connect_from_vpp (vapi_ctx_t ctx, const char *name,
144 int max_outstanding_requests,
145 int response_queue_size, vapi_mode_e mode,
146 bool handle_keepalives);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200147
148/**
149 * @brief disconnect from vpp
150 *
151 * @param ctx opaque vapi context
152 *
153 * @return VAPI_OK on success, other error code on error
154 */
Ole Troan2ca88ff2022-01-27 16:25:43 +0100155vapi_error_e vapi_disconnect (vapi_ctx_t ctx);
156vapi_error_e vapi_disconnect_from_vpp (vapi_ctx_t ctx);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200157
158/**
159 * @brief get event file descriptor
160 *
161 * @note this file descriptor becomes readable when messages (from vpp)
162 * are waiting in queue
163 *
164 * @param ctx opaque vapi context
165 * @param[out] fd pointer to result variable
166 *
167 * @return VAPI_OK on success, other error code on error
168 */
Ole Troan2ca88ff2022-01-27 16:25:43 +0100169vapi_error_e vapi_get_fd (vapi_ctx_t ctx, int *fd);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200170
171/**
172 * @brief low-level api for sending messages to vpp
173 *
174 * @note it is not recommended to use this api directly, use generated api
175 * instead
176 *
177 * @param ctx opaque vapi context
178 * @param msg message to send
179 *
180 * @return VAPI_OK on success, other error code on error
181 */
Ole Troan2ca88ff2022-01-27 16:25:43 +0100182vapi_error_e vapi_send (vapi_ctx_t ctx, void *msg);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200183
184/**
185 * @brief low-level api for atomically sending two messages to vpp - either
186 * both messages are sent or neither one is
187 *
188 * @note it is not recommended to use this api directly, use generated api
189 * instead
190 *
191 * @param ctx opaque vapi context
192 * @param msg1 first message to send
193 * @param msg2 second message to send
194 *
195 * @return VAPI_OK on success, other error code on error
196 */
Matthew Smith4b9935c2022-12-02 20:46:16 +0000197vapi_error_e vapi_send2 (vapi_ctx_t ctx, void *msg1, void *msg2);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200198
199/**
200 * @brief low-level api for reading messages from vpp
201 *
202 * @note it is not recommended to use this api directly, use generated api
203 * instead
204 *
205 * @param ctx opaque vapi context
206 * @param[out] msg pointer to result variable containing message
207 * @param[out] msg_size pointer to result variable containing message size
Mohsin Kazmi3fca5672018-01-04 18:57:26 +0100208 * @param cond enum type for blocking, non-blocking or timed wait call
209 * @param time in sec for timed wait
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200210 *
211 * @return VAPI_OK on success, other error code on error
212 */
Matthew Smith4b9935c2022-12-02 20:46:16 +0000213vapi_error_e vapi_recv (vapi_ctx_t ctx, void **msg, size_t *msg_size,
214 svm_q_conditional_wait_t cond, u32 time);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200215
216/**
Matthew Smith4b9935c2022-12-02 20:46:16 +0000217 * @brief wait for connection to become readable
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200218 *
219 * @param ctx opaque vapi context
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200220 *
221 * @return VAPI_OK on success, other error code on error
222 */
Matthew Smith4b9935c2022-12-02 20:46:16 +0000223vapi_error_e vapi_wait (vapi_ctx_t ctx);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200224
225/**
226 * @brief pick next message sent by vpp and call the appropriate callback
227 *
Dau Do10dbb372024-06-10 19:55:19 -0700228 * @note if using block mode, it will be blocked indefinitely until the next
229 * msg available. If using non-blocking mode, it will block for time_wait
230 * seconds until the next msg available if time_wait > 0, or does not block if
231 * time_wait == 0.
232 *
233 * @return VAPI_OK on success, other error code on error
234 */
235vapi_error_e vapi_dispatch_one_timedwait (vapi_ctx_t ctx, u32 wait_time);
236
237/**
238 * @brief pick next message sent by vpp and call the appropriate callback
239 *
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200240 * @return VAPI_OK on success, other error code on error
241 */
Matthew Smith57f177d2022-12-15 22:18:08 +0000242vapi_error_e vapi_dispatch_one (vapi_ctx_t ctx);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200243
244/**
245 * @brief loop vapi_dispatch_one until responses to all currently outstanding
246 * requests have been received and their callbacks called
247 *
248 * @note the dispatch loop is interrupted if any error is encountered or
249 * returned from the callback, in which case this error is returned as the
Dau Do10dbb372024-06-10 19:55:19 -0700250 * result of vapi_dispatch. In this case it might be necessary to call
251 * dispatch again to process the remaining messages. Returning VAPI_EUSER
252 * from a callback allows the user to break the dispatch loop (and
253 * distinguish this case in the calling code from other failures). VAPI never
254 * returns VAPI_EUSER on its own.
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200255 *
256 * @return VAPI_OK on success, other error code on error
257 */
Matthew Smith57f177d2022-12-15 22:18:08 +0000258vapi_error_e vapi_dispatch (vapi_ctx_t ctx);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200259
260/** generic vapi event callback */
Matthew Smith57f177d2022-12-15 22:18:08 +0000261typedef vapi_error_e (*vapi_event_cb) (vapi_ctx_t ctx, void *callback_ctx,
262 void *payload);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200263
264/**
265 * @brief set event callback to call when message with given id is dispatched
266 *
267 * @param ctx opaque vapi context
268 * @param id message id
269 * @param callback callback
270 * @param callback_ctx context pointer stored and passed to callback
271 */
Matthew Smith57f177d2022-12-15 22:18:08 +0000272void vapi_set_event_cb (vapi_ctx_t ctx, vapi_msg_id_t id,
273 vapi_event_cb callback, void *callback_ctx);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200274
275/**
276 * @brief clear event callback for given message id
277 *
278 * @param ctx opaque vapi context
279 * @param id message id
280 */
Matthew Smith57f177d2022-12-15 22:18:08 +0000281void vapi_clear_event_cb (vapi_ctx_t ctx, vapi_msg_id_t id);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200282
283/** generic vapi event callback */
Matthew Smith57f177d2022-12-15 22:18:08 +0000284typedef vapi_error_e (*vapi_generic_event_cb) (vapi_ctx_t ctx,
285 void *callback_ctx,
286 vapi_msg_id_t id, void *msg);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200287/**
288 * @brief set generic event callback
289 *
290 * @note this callback is called by dispatch if no message-type specific
291 * callback is set (so it's a fallback callback)
292 *
293 * @param ctx opaque vapi context
294 * @param callback callback
295 * @param callback_ctx context pointer stored and passed to callback
296 */
Matthew Smith57f177d2022-12-15 22:18:08 +0000297void vapi_set_generic_event_cb (vapi_ctx_t ctx, vapi_generic_event_cb callback,
298 void *callback_ctx);
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200299
300/**
301 * @brief clear generic event callback
302 *
303 * @param ctx opaque vapi context
304 */
Matthew Smith57f177d2022-12-15 22:18:08 +0000305void vapi_clear_generic_event_cb (vapi_ctx_t ctx);
306
307/**
308 * @brief signal RX thread to exit
309 *
310 * @note This adds a message to the client input queue that indicates that
311 * an RX thread should stop processing incoming messages and exit. If an
312 * application has an RX thread which sleeps while waiting for incoming
313 * messages using vapi_wait(), this call will allow the application to
314 * wake up from the vapi_wait() call and figure out that it should stop
315 * running.
316 *
317 * @param ctx opaque vapi context
318 */
319void vapi_stop_rx_thread (vapi_ctx_t ctx);
Klement Sekeradc15be22017-06-12 06:49:33 +0200320
321#ifdef __cplusplus
322}
323#endif
Klement Sekera8f2a4ea2017-05-04 06:15:18 +0200324
325#endif
326
327/*
328 * fd.io coding-style-patch-verification: ON
329 *
330 * Local Variables:
331 * eval: (c-set-style "gnu")
332 * End:
333 */