1 /* SPDX-License-Identifier: Apache-2.0
2 * Copyright (c) 2023 Cisco Systems, Inc.
8 #include <vppinfra/clib.h>
9 #include <vppinfra/error_bootstrap.h>
10 #include <vppinfra/format.h>
11 #include <vnet/vnet.h>
12 #include <vnet/dev/types.h>
13 #include <vnet/dev/args.h>
15 #define VNET_DEV_DEVICE_ID_PREFIX_DELIMITER "/"
17 #define foreach_vnet_dev_port_type \
23 #define _(b, n) VNET_DEV_PORT_TYPE_##n = (1U << (b)),
24 foreach_vnet_dev_port_type
26 } vnet_dev_port_type_t;
28 #define foreach_vnet_dev_port_caps \
31 _ (change_max_rx_frame_size) \
34 #define foreach_vnet_dev_port_rx_offloads _ (ip4_cksum)
36 #define foreach_vnet_dev_port_tx_offloads \
45 #define _(n) u8 n : 1;
46 foreach_vnet_dev_port_caps
50 } vnet_dev_port_caps_t;
56 #define _(n) u8 n : 1;
57 foreach_vnet_dev_port_rx_offloads
61 } vnet_dev_port_rx_offloads_t;
67 #define _(n) u8 n : 1;
68 foreach_vnet_dev_port_tx_offloads
72 } vnet_dev_port_tx_offloads_t;
80 typedef struct vnet_dev_bus_registration vnet_dev_bus_registration_t;
81 typedef struct vnet_dev_driver_registration vnet_dev_driver_registration_t;
83 typedef struct vnet_dev vnet_dev_t;
84 typedef struct vnet_dev_port vnet_dev_port_t;
85 typedef struct vnet_dev_rx_queue vnet_dev_rx_queue_t;
86 typedef struct vnet_dev_tx_queue vnet_dev_tx_queue_t;
87 typedef struct vnet_dev_bus_registration vnet_dev_bus_registration_t;
88 typedef struct vnet_dev_driver_registration vnet_dev_driver_registration_t;
89 typedef struct vnet_dev_counter vnet_dev_counter_t;
90 typedef struct vnet_dev_counter_main vnet_dev_counter_main_t;
91 typedef struct vnet_dev_port_cfg_change_req vnet_dev_port_cfg_change_req_t;
93 typedef vnet_dev_rv_t (vnet_dev_op_t) (vlib_main_t *, vnet_dev_t *);
94 typedef vnet_dev_rv_t (vnet_dev_port_op_t) (vlib_main_t *, vnet_dev_port_t *);
95 typedef vnet_dev_rv_t (vnet_dev_port_cfg_change_op_t) (
96 vlib_main_t *, vnet_dev_port_t *, vnet_dev_port_cfg_change_req_t *);
97 typedef vnet_dev_rv_t (vnet_dev_rx_queue_op_t) (vlib_main_t *,
98 vnet_dev_rx_queue_t *);
99 typedef vnet_dev_rv_t (vnet_dev_tx_queue_op_t) (vlib_main_t *,
100 vnet_dev_tx_queue_t *);
101 typedef void (vnet_dev_op_no_rv_t) (vlib_main_t *, vnet_dev_t *);
102 typedef void (vnet_dev_port_op_no_rv_t) (vlib_main_t *, vnet_dev_port_t *);
103 typedef void (vnet_dev_rx_queue_op_no_rv_t) (vlib_main_t *,
104 vnet_dev_rx_queue_t *);
105 typedef void (vnet_dev_tx_queue_op_no_rv_t) (vlib_main_t *,
106 vnet_dev_tx_queue_t *);
108 typedef u16 vnet_dev_queue_id_t;
109 typedef u16 vnet_dev_bus_index_t;
110 typedef u16 vnet_dev_driver_index_t;
114 vnet_dev_rx_queue_op_t *alloc;
115 vnet_dev_rx_queue_op_t *start;
116 vnet_dev_rx_queue_op_no_rv_t *stop;
117 vnet_dev_rx_queue_op_no_rv_t *free;
118 } vnet_dev_rx_queue_ops_t;
122 vnet_dev_tx_queue_op_t *alloc;
123 vnet_dev_tx_queue_op_t *start;
124 vnet_dev_tx_queue_op_no_rv_t *stop;
125 vnet_dev_tx_queue_op_no_rv_t *free;
126 } vnet_dev_tx_queue_ops_t;
135 u8 size_is_power_of_two : 1;
136 } vnet_dev_queue_config_t;
138 #define foreach_vnet_dev_port_cfg_type \
140 _ (MAX_RX_FRAME_SIZE) \
141 _ (CHANGE_PRIMARY_HW_ADDR) \
142 _ (ADD_SECONDARY_HW_ADDR) \
143 _ (REMOVE_SECONDARY_HW_ADDR) \
144 _ (RXQ_INTR_MODE_ENABLE) \
145 _ (RXQ_INTR_MODE_DISABLE)
149 VNET_DEV_PORT_CFG_UNKNOWN,
150 #define _(n) VNET_DEV_PORT_CFG_##n,
151 foreach_vnet_dev_port_cfg_type
153 } __clib_packed vnet_dev_port_cfg_type_t;
155 typedef struct vnet_dev_port_cfg_change_req
157 vnet_dev_port_cfg_type_t type;
164 vnet_dev_hw_addr_t addr;
165 u16 max_rx_frame_size;
166 vnet_dev_queue_id_t queue_id;
169 } vnet_dev_port_cfg_change_req_t;
173 vnet_dev_hw_addr_t hw_addr;
176 u16 max_supported_rx_frame_size;
177 vnet_dev_port_type_t type;
178 vnet_dev_port_caps_t caps;
179 vnet_dev_port_rx_offloads_t rx_offloads;
180 vnet_dev_port_tx_offloads_t tx_offloads;
181 } vnet_dev_port_attr_t;
185 VNET_DEV_PERIODIC_OP_TYPE_DEV = 1,
186 VNET_DEV_PERIODIC_OP_TYPE_PORT = 2,
187 } __clib_packed vnet_dev_periodic_op_type_t;
193 vnet_dev_periodic_op_type_t type;
197 vnet_dev_port_t *port;
202 vnet_dev_op_no_rv_t *dev_op;
203 vnet_dev_port_op_no_rv_t *port_op;
206 } vnet_dev_periodic_op_t;
210 struct _vlib_node_fn_registration *registrations;
211 format_function_t *format_trace;
212 vlib_error_desc_t *error_counters;
213 u16 n_error_counters;
218 vnet_dev_op_t *alloc;
220 vnet_dev_op_no_rv_t *deinit;
221 vnet_dev_op_t *reset;
222 vnet_dev_op_no_rv_t *free;
223 u8 *(*probe) (vlib_main_t *, vnet_dev_bus_index_t, void *);
224 format_function_t *format_info;
229 vnet_dev_port_op_t *alloc;
230 vnet_dev_port_op_t *init;
231 vnet_dev_port_cfg_change_op_t *config_change;
232 vnet_dev_port_cfg_change_op_t *config_change_validate;
233 vnet_dev_port_op_t *start;
234 vnet_dev_port_op_no_rv_t *stop;
235 vnet_dev_port_op_no_rv_t *deinit;
236 vnet_dev_port_op_no_rv_t *free;
237 format_function_t *format_status;
238 } vnet_dev_port_ops_t;
244 u8 update_next_index : 1;
245 u8 update_feature_arc : 1;
250 } vnet_dev_rx_queue_rt_req_t;
252 typedef struct vnet_dev_rx_queue
254 CLIB_CACHE_LINE_ALIGN_MARK (cacheline0);
255 vnet_dev_port_t *port;
258 vnet_dev_counter_main_t *counter_main;
259 CLIB_CACHE_LINE_ALIGN_MARK (runtime0);
260 vnet_dev_rx_queue_t *next_on_thread;
261 u8 interrupt_mode : 1;
265 vnet_dev_queue_id_t queue_id;
268 vnet_dev_rx_queue_rt_req_t runtime_request;
269 CLIB_CACHE_LINE_ALIGN_MARK (runtime1);
270 vlib_buffer_template_t buffer_template;
271 CLIB_CACHE_LINE_ALIGN_MARK (driver_data);
273 } vnet_dev_rx_queue_t;
275 STATIC_ASSERT_SIZEOF (vnet_dev_rx_queue_t, 3 * CLIB_CACHE_LINE_BYTES);
277 typedef struct vnet_dev_tx_queue
279 CLIB_CACHE_LINE_ALIGN_MARK (cacheline0);
280 vnet_dev_port_t *port;
281 clib_bitmap_t *assigned_threads;
283 vnet_dev_counter_main_t *counter_main;
284 CLIB_CACHE_LINE_ALIGN_MARK (runtime0);
285 vnet_dev_queue_id_t queue_id;
291 CLIB_ALIGN_MARK (private_data, 16);
293 } vnet_dev_tx_queue_t;
295 STATIC_ASSERT_SIZEOF (vnet_dev_tx_queue_t, 2 * CLIB_CACHE_LINE_BYTES);
297 typedef struct vnet_dev_port
299 CLIB_CACHE_LINE_ALIGN_MARK (cacheline0);
301 vnet_dev_port_id_t port_id;
302 vnet_dev_driver_index_t driver_index;
307 u8 interface_created : 1;
308 u8 rx_node_assigned : 1;
309 vnet_dev_counter_main_t *counter_main;
310 vnet_dev_queue_config_t rx_queue_config;
311 vnet_dev_queue_config_t tx_queue_config;
312 vnet_dev_port_attr_t attr;
313 u32 max_rx_frame_size;
314 vnet_dev_hw_addr_t primary_hw_addr;
315 vnet_dev_hw_addr_t *secondary_hw_addr;
318 vnet_dev_rx_queue_t **rx_queues;
319 vnet_dev_tx_queue_t **tx_queues;
320 vnet_dev_port_ops_t port_ops;
321 vnet_dev_arg_t *args;
322 vnet_dev_rx_queue_ops_t rx_queue_ops;
323 vnet_dev_tx_queue_ops_t tx_queue_ops;
324 vnet_dev_node_t rx_node;
325 vnet_dev_node_t tx_node;
329 vnet_dev_if_name_t name;
332 u32 current_config_index;
334 u16 redirect_to_node_next_index;
335 u8 feature_arc_index;
337 u8 redirect_to_node : 1;
338 u8 default_is_intr_mode : 1;
348 CLIB_CACHE_LINE_ALIGN_MARK (data0);
352 typedef struct vnet_dev
354 vnet_dev_device_id_t device_id;
356 u16 not_first_init : 1;
358 u16 process_node_quit : 1;
359 u16 process_node_periodic : 1;
365 vnet_dev_driver_index_t driver_index;
367 u32 process_node_index;
368 u8 bus_data[32] __clib_aligned (16);
370 vnet_dev_port_t **ports;
371 vnet_dev_periodic_op_t *periodic_ops;
373 vnet_dev_arg_t *args;
374 u8 __clib_aligned (16)
380 u16 vendor_id, device_id;
384 #define VNET_DEV_MATCH(...) \
385 (vnet_dev_match_t[]) \
392 vnet_dev_op_t *device_open;
393 vnet_dev_op_no_rv_t *device_close;
394 vnet_dev_rv_t (*dma_mem_alloc_fn) (vlib_main_t *, vnet_dev_t *, u32, u32,
396 void (*dma_mem_free_fn) (vlib_main_t *, vnet_dev_t *, void *);
397 void *(*get_device_info) (vlib_main_t *, char *);
398 void (*free_device_info) (vlib_main_t *, void *);
399 format_function_t *format_device_info;
400 format_function_t *format_device_addr;
401 } vnet_dev_bus_ops_t;
403 struct vnet_dev_bus_registration
405 vnet_dev_bus_registration_t *next_registration;
406 vnet_dev_driver_name_t name;
407 u16 device_data_size;
408 vnet_dev_bus_ops_t ops;
411 struct vnet_dev_driver_registration
413 vnet_dev_driver_registration_t *next_registration;
414 u8 bus_master_enable : 1;
415 vnet_dev_driver_name_t name;
416 vnet_dev_bus_name_t bus;
418 u16 runtime_temp_space_sz;
419 vnet_dev_match_t *match;
422 vnet_dev_arg_t *args;
428 vnet_dev_bus_registration_t *registration;
429 vnet_dev_bus_ops_t ops;
436 vnet_dev_driver_registration_t *registration;
438 vnet_dev_bus_index_t bus_index;
444 vnet_dev_bus_t *buses;
445 vnet_dev_driver_t *drivers;
446 vnet_dev_t **devices;
447 vnet_dev_port_t **ports_by_dev_instance;
448 vnet_dev_bus_registration_t *bus_registrations;
449 vnet_dev_driver_registration_t *driver_registrations;
450 void *runtime_temp_spaces;
451 u32 log2_runtime_temp_space_sz;
452 u32 *free_process_node_indices;
453 u32 *free_rx_node_indices;
454 uword *device_index_by_id;
457 u16 next_rx_queue_thread;
458 u8 eth_port_rx_feature_arc_index;
461 extern vnet_dev_main_t vnet_dev_main;
467 vnet_dev_port_attr_t attr;
468 vnet_dev_port_ops_t ops;
469 vnet_dev_arg_t *args;
474 vnet_dev_node_t *rx_node;
475 vnet_dev_node_t *tx_node;
479 vnet_dev_queue_config_t config;
480 vnet_dev_rx_queue_ops_t ops;
485 vnet_dev_queue_config_t config;
486 vnet_dev_tx_queue_ops_t ops;
488 } vnet_dev_port_add_args_t;
505 } vnet_dev_port_state_changes_t;
508 vnet_dev_rv_t vnet_dev_arg_parse (vlib_main_t *, vnet_dev_t *,
509 vnet_dev_arg_t *, u8 *);
510 void vnet_dev_arg_free (vnet_dev_arg_t **);
511 void vnet_dev_arg_clear_value (vnet_dev_arg_t *);
512 format_function_t format_vnet_dev_arg_type;
513 format_function_t format_vnet_dev_arg_value;
514 format_function_t format_vnet_dev_args;
517 vnet_dev_t *vnet_dev_alloc (vlib_main_t *, vnet_dev_device_id_t,
518 vnet_dev_driver_t *);
519 void vnet_dev_free (vlib_main_t *, vnet_dev_t *);
520 vnet_dev_rv_t vnet_dev_init (vlib_main_t *, vnet_dev_t *);
521 void vnet_dev_deinit (vlib_main_t *, vnet_dev_t *);
522 vnet_dev_rv_t vnet_dev_reset (vlib_main_t *, vnet_dev_t *);
523 void vnet_dev_detach (vlib_main_t *, vnet_dev_t *);
524 vnet_dev_rv_t vnet_dev_port_add (vlib_main_t *, vnet_dev_t *,
526 vnet_dev_port_add_args_t *);
527 vnet_dev_rv_t vnet_dev_dma_mem_alloc (vlib_main_t *, vnet_dev_t *, u32, u32,
529 void vnet_dev_dma_mem_free (vlib_main_t *, vnet_dev_t *, void *);
530 vnet_dev_bus_t *vnet_dev_find_device_bus (vlib_main_t *, vnet_dev_device_id_t);
531 void *vnet_dev_get_device_info (vlib_main_t *, vnet_dev_device_id_t);
534 clib_error_t *vnet_dev_port_err (vlib_main_t *, vnet_dev_port_t *,
535 vnet_dev_rv_t, char *, ...);
538 clib_error_t *vnet_dev_port_set_max_frame_size (vnet_main_t *,
539 vnet_hw_interface_t *, u32);
540 u32 vnet_dev_port_eth_flag_change (vnet_main_t *, vnet_hw_interface_t *, u32);
541 clib_error_t *vnet_dev_port_mac_change (vnet_hw_interface_t *, const u8 *,
543 clib_error_t *vnet_dev_add_del_mac_address (vnet_hw_interface_t *, const u8 *,
545 int vnet_dev_flow_ops_fn (vnet_main_t *, vnet_flow_dev_op_t, u32, u32,
547 clib_error_t *vnet_dev_interface_set_rss_queues (vnet_main_t *,
548 vnet_hw_interface_t *,
550 void vnet_dev_clear_hw_interface_counters (u32);
551 void vnet_dev_set_interface_next_node (vnet_main_t *, u32, u32);
554 vnet_dev_rv_t vnet_dev_port_start (vlib_main_t *, vnet_dev_port_t *);
555 vnet_dev_rv_t vnet_dev_port_start_all_rx_queues (vlib_main_t *,
557 vnet_dev_rv_t vnet_dev_port_start_all_tx_queues (vlib_main_t *,
559 void vnet_dev_port_stop (vlib_main_t *, vnet_dev_port_t *);
560 void vnet_dev_port_deinit (vlib_main_t *, vnet_dev_port_t *);
561 void vnet_dev_port_free (vlib_main_t *, vnet_dev_port_t *);
562 void vnet_dev_port_add_counters (vlib_main_t *, vnet_dev_port_t *,
563 vnet_dev_counter_t *, u16);
564 void vnet_dev_port_free_counters (vlib_main_t *, vnet_dev_port_t *);
565 void vnet_dev_port_update_tx_node_runtime (vlib_main_t *, vnet_dev_port_t *);
566 void vnet_dev_port_state_change (vlib_main_t *, vnet_dev_port_t *,
567 vnet_dev_port_state_changes_t);
568 void vnet_dev_port_clear_counters (vlib_main_t *, vnet_dev_port_t *);
570 vnet_dev_port_cfg_change_req_validate (vlib_main_t *, vnet_dev_port_t *,
571 vnet_dev_port_cfg_change_req_t *);
572 vnet_dev_rv_t vnet_dev_port_cfg_change (vlib_main_t *, vnet_dev_port_t *,
573 vnet_dev_port_cfg_change_req_t *);
574 vnet_dev_rv_t vnet_dev_port_if_create (vlib_main_t *, vnet_dev_port_t *);
575 vnet_dev_rv_t vnet_dev_port_if_remove (vlib_main_t *, vnet_dev_port_t *);
578 vnet_dev_rv_t vnet_dev_rx_queue_alloc (vlib_main_t *, vnet_dev_port_t *, u16);
579 vnet_dev_rv_t vnet_dev_tx_queue_alloc (vlib_main_t *, vnet_dev_port_t *, u16);
580 void vnet_dev_rx_queue_free (vlib_main_t *, vnet_dev_rx_queue_t *);
581 void vnet_dev_tx_queue_free (vlib_main_t *, vnet_dev_tx_queue_t *);
582 void vnet_dev_rx_queue_add_counters (vlib_main_t *, vnet_dev_rx_queue_t *,
583 vnet_dev_counter_t *, u16);
584 void vnet_dev_rx_queue_free_counters (vlib_main_t *, vnet_dev_rx_queue_t *);
585 void vnet_dev_tx_queue_add_counters (vlib_main_t *, vnet_dev_tx_queue_t *,
586 vnet_dev_counter_t *, u16);
587 void vnet_dev_tx_queue_free_counters (vlib_main_t *, vnet_dev_tx_queue_t *);
588 vnet_dev_rv_t vnet_dev_rx_queue_start (vlib_main_t *, vnet_dev_rx_queue_t *);
589 vnet_dev_rv_t vnet_dev_tx_queue_start (vlib_main_t *, vnet_dev_tx_queue_t *);
590 void vnet_dev_rx_queue_stop (vlib_main_t *, vnet_dev_rx_queue_t *);
591 void vnet_dev_tx_queue_stop (vlib_main_t *, vnet_dev_tx_queue_t *);
594 vnet_dev_rv_t vnet_dev_process_create (vlib_main_t *, vnet_dev_t *);
595 vnet_dev_rv_t vnet_dev_process_call_op (vlib_main_t *, vnet_dev_t *,
597 vnet_dev_rv_t vnet_dev_process_call_op_no_rv (vlib_main_t *, vnet_dev_t *,
598 vnet_dev_op_no_rv_t *);
599 void vnet_dev_process_call_op_no_wait (vlib_main_t *, vnet_dev_t *,
600 vnet_dev_op_no_rv_t *);
601 vnet_dev_rv_t vnet_dev_process_call_port_op (vlib_main_t *, vnet_dev_port_t *,
602 vnet_dev_port_op_t *);
603 vnet_dev_rv_t vnet_dev_process_call_port_op_no_rv (vlib_main_t *vm,
605 vnet_dev_port_op_no_rv_t *);
606 void vnet_dev_process_call_port_op_no_wait (vlib_main_t *, vnet_dev_port_t *,
607 vnet_dev_port_op_no_rv_t *);
609 vnet_dev_process_port_cfg_change_req (vlib_main_t *, vnet_dev_port_t *,
610 vnet_dev_port_cfg_change_req_t *);
611 void vnet_dev_process_quit (vlib_main_t *, vnet_dev_t *);
612 void vnet_dev_poll_dev_add (vlib_main_t *, vnet_dev_t *, f64,
613 vnet_dev_op_no_rv_t *);
614 void vnet_dev_poll_dev_remove (vlib_main_t *, vnet_dev_t *,
615 vnet_dev_op_no_rv_t *);
616 void vnet_dev_poll_port_add (vlib_main_t *, vnet_dev_port_t *, f64,
617 vnet_dev_port_op_no_rv_t *);
618 void vnet_dev_poll_port_remove (vlib_main_t *, vnet_dev_port_t *,
619 vnet_dev_port_op_no_rv_t *);
626 vnet_dev_port_t *port;
629 vnet_dev_rv_t vnet_dev_rt_exec_ops (vlib_main_t *, vnet_dev_t *,
630 vnet_dev_rt_op_t *, u32);
636 u8 show_zero_counters : 1;
638 } vnet_dev_format_args_t;
640 format_function_t format_vnet_dev_addr;
641 format_function_t format_vnet_dev_flags;
642 format_function_t format_vnet_dev_hw_addr;
643 format_function_t format_vnet_dev_info;
644 format_function_t format_vnet_dev_interface_info;
645 format_function_t format_vnet_dev_interface_name;
646 format_function_t format_vnet_dev_log;
647 format_function_t format_vnet_dev_port_caps;
648 format_function_t format_vnet_dev_port_flags;
649 format_function_t format_vnet_dev_port_info;
650 format_function_t format_vnet_dev_port_rx_offloads;
651 format_function_t format_vnet_dev_port_tx_offloads;
652 format_function_t format_vnet_dev_rv;
653 format_function_t format_vnet_dev_rx_queue_info;
654 format_function_t format_vnet_dev_tx_queue_info;
655 unformat_function_t unformat_vnet_dev_flags;
656 unformat_function_t unformat_vnet_dev_port_flags;
660 vnet_dev_rx_queue_t *first_rx_queue;
661 } vnet_dev_rx_node_runtime_t;
663 STATIC_ASSERT (sizeof (vnet_dev_rx_node_runtime_t) <=
664 VLIB_NODE_RUNTIME_DATA_SIZE,
665 "must fit into runtime data");
667 #define foreach_vnet_dev_port_rx_next \
668 _ (ETH_INPUT, "ethernet-input") \
669 _ (DROP, "error-drop")
673 #define _(n, s) VNET_DEV_ETH_RX_PORT_NEXT_##n,
674 foreach_vnet_dev_port_rx_next
676 VNET_DEV_ETH_RX_PORT_N_NEXTS
677 } vnet_dev_eth_port_rx_next_t;
679 extern u16 vnet_dev_default_next_index_by_port_type[];
680 extern vlib_node_registration_t port_rx_eth_node;
682 typedef vnet_interface_output_runtime_t vnet_dev_tx_node_runtime_t;
684 STATIC_ASSERT (sizeof (vnet_dev_tx_node_runtime_t) <=
685 VLIB_NODE_RUNTIME_DATA_SIZE,
686 "must fit into runtime data");
688 #define VNET_DEV_REGISTER_BUS(x, ...) \
689 __VA_ARGS__ vnet_dev_bus_registration_t __vnet_dev_bus_registration_##x; \
690 static void __clib_constructor __vnet_dev_bus_registration_fn_##x (void) \
692 vnet_dev_main_t *dm = &vnet_dev_main; \
693 __vnet_dev_bus_registration_##x.next_registration = \
694 dm->bus_registrations; \
695 dm->bus_registrations = &__vnet_dev_bus_registration_##x; \
697 __VA_ARGS__ vnet_dev_bus_registration_t __vnet_dev_bus_registration_##x
699 #define VNET_DEV_REGISTER_DRIVER(x, ...) \
700 __VA_ARGS__ vnet_dev_driver_registration_t \
701 __vnet_dev_driver_registration_##x; \
702 static void __clib_constructor __vnet_dev_driver_registration_fn_##x (void) \
704 vnet_dev_main_t *dm = &vnet_dev_main; \
705 __vnet_dev_driver_registration_##x.next_registration = \
706 dm->driver_registrations; \
707 dm->driver_registrations = &__vnet_dev_driver_registration_##x; \
709 __VA_ARGS__ vnet_dev_driver_registration_t __vnet_dev_driver_registration_##x
711 #define VNET_DEV_NODE_FN(node) \
712 uword CLIB_MARCH_SFX (node##_fn) (vlib_main_t *, vlib_node_runtime_t *, \
714 static vlib_node_fn_registration_t CLIB_MARCH_SFX ( \
715 node##_fn_registration) = { \
716 .function = &CLIB_MARCH_SFX (node##_fn), \
719 static void __clib_constructor CLIB_MARCH_SFX ( \
720 node##_fn_multiarch_register) (void) \
722 extern vnet_dev_node_t node; \
723 vlib_node_fn_registration_t *r; \
724 r = &CLIB_MARCH_SFX (node##_fn_registration); \
725 r->march_variant = CLIB_MARCH_SFX (CLIB_MARCH_VARIANT_TYPE); \
726 r->next_registration = (node).registrations; \
727 (node).registrations = r; \
729 uword CLIB_MARCH_SFX (node##_fn)
731 #define foreach_vnet_dev_port(p, d) pool_foreach_pointer (p, d->ports)
732 #define foreach_vnet_dev_port_rx_queue(q, p) \
733 pool_foreach_pointer (q, p->rx_queues)
734 #define foreach_vnet_dev_port_tx_queue(q, p) \
735 pool_foreach_pointer (q, p->tx_queues)
737 #include <vnet/dev/dev_funcs.h>
739 #endif /* _VNET_DEV_H_ */