diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/vcl/vppcom.c | 8 | ||||
-rw-r--r-- | src/vnet/session/application_local.c | 14 | ||||
-rw-r--r-- | src/vnet/session/session_node.c | 7 | ||||
-rw-r--r-- | src/vnet/session/transport.h | 11 |
4 files changed, 40 insertions, 0 deletions
diff --git a/src/vcl/vppcom.c b/src/vcl/vppcom.c index b5953872f81..e70aa001772 100644 --- a/src/vcl/vppcom.c +++ b/src/vcl/vppcom.c @@ -1530,6 +1530,14 @@ vppcom_session_read_internal (uint32_t session_handle, void *buf, int n, if (svm_fifo_is_empty_cons (rx_fifo)) svm_fifo_unset_event (s->rx_fifo); + /* Cut-through sessions might request tx notifications on rx fifos */ + if (PREDICT_FALSE (rx_fifo->want_tx_ntf)) + { + app_send_io_evt_to_vpp (s->vpp_evt_q, s->rx_fifo->master_session_index, + SESSION_IO_EVT_RX, SVM_Q_WAIT); + svm_fifo_reset_tx_ntf (s->rx_fifo); + } + VDBG (2, "session %u[0x%llx]: read %d bytes from (%p)", s->session_index, s->vpp_handle, n_read, rx_fifo); diff --git a/src/vnet/session/application_local.c b/src/vnet/session/application_local.c index 856f7489f38..d2edd54512a 100644 --- a/src/vnet/session/application_local.c +++ b/src/vnet/session/application_local.c @@ -475,6 +475,19 @@ ct_custom_tx (void *session) return ct_session_tx (s); } +static int +ct_app_rx_evt (transport_connection_t * tc) +{ + ct_connection_t *ct = (ct_connection_t *) tc, *peer_ct; + session_t *ps; + + peer_ct = ct_connection_get (ct->peer_index); + if (!peer_ct) + return -1; + ps = session_get (peer_ct->c_s_index, peer_ct->c_thread_index); + return session_dequeue_notify (ps); +} + static u8 * format_ct_listener (u8 * s, va_list * args) { @@ -535,6 +548,7 @@ const static transport_proto_vft_t cut_thru_proto = { .close = ct_session_close, .get_connection = ct_session_get, .custom_tx = ct_custom_tx, + .app_rx_evt = ct_app_rx_evt, .tx_type = TRANSPORT_TX_INTERNAL, .service_type = TRANSPORT_SERVICE_APP, .format_listener = format_ct_listener, diff --git a/src/vnet/session/session_node.c b/src/vnet/session/session_node.c index c894c437cde..605fd584e2d 100644 --- a/src/vnet/session/session_node.c +++ b/src/vnet/session/session_node.c @@ -947,6 +947,13 @@ session_queue_node_fn (vlib_main_t * vm, vlib_node_runtime_t * node, continue; } break; + case SESSION_IO_EVT_RX: + s = session_event_get_session (e, thread_index); + if (!s) + break; + transport_app_rx_evt (session_get_transport_proto (s), + s->connection_index, s->thread_index); + break; case SESSION_CTRL_EVT_CLOSE: s = session_get_from_handle_if_valid (e->session_handle); if (PREDICT_FALSE (!s)) diff --git a/src/vnet/session/transport.h b/src/vnet/session/transport.h index e446a3934d1..993b8bd7354 100644 --- a/src/vnet/session/transport.h +++ b/src/vnet/session/transport.h @@ -46,6 +46,7 @@ typedef struct _transport_proto_vft void (*update_time) (f64 time_now, u8 thread_index); void (*flush_data) (transport_connection_t *tconn); int (*custom_tx) (void *session); + int (*app_rx_evt) (transport_connection_t *tconn); /* * Connection retrieval @@ -111,6 +112,16 @@ transport_custom_tx (transport_proto_t tp, void *s) return tp_vfts[tp].custom_tx (s); } +static inline int +transport_app_rx_evt (transport_proto_t tp, u32 conn_index, u32 thread_index) +{ + transport_connection_t *tc; + if (!tp_vfts[tp].app_rx_evt) + return 0; + tc = transport_get_connection (tp, conn_index, thread_index); + return tp_vfts[tp].app_rx_evt (tc); +} + void transport_register_protocol (transport_proto_t transport_proto, const transport_proto_vft_t * vft, fib_protocol_t fib_proto, u32 output_node); |