/* * Copyright (c) 2017-2019 Cisco and/or its affiliates. * 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 SRC_VNET_SESSION_SESSION_TYPES_H_ #define SRC_VNET_SESSION_SESSION_TYPES_H_ #include #include #define SESSION_LOCAL_HANDLE_PREFIX 0x7FFFFFFF #define SESSION_LISTENER_PREFIX 0x5FFFFFFF #define foreach_session_endpoint_fields \ foreach_transport_endpoint_cfg_fields \ _(u8, transport_proto) \ typedef struct _session_endpoint { #define _(type, name) type name; foreach_session_endpoint_fields #undef _ } session_endpoint_t; typedef struct _session_endpoint_cfg { #define _(type, name) type name; foreach_session_endpoint_fields #undef _ u32 app_wrk_index; u32 opaque; u8 *hostname; } session_endpoint_cfg_t; #define SESSION_IP46_ZERO \ { \ .ip6 = { \ { 0, 0, }, \ }, \ } #define TRANSPORT_ENDPOINT_NULL \ { \ .sw_if_index = ENDPOINT_INVALID_INDEX, \ .ip = SESSION_IP46_ZERO, \ .fib_index = ENDPOINT_INVALID_INDEX, \ .is_ip4 = 0, \ .port = 0, \ } #define SESSION_ENDPOINT_NULL \ { \ .sw_if_index = ENDPOINT_INVALID_INDEX, \ .ip = SESSION_IP46_ZERO, \ .fib_index = ENDPOINT_INVALID_INDEX, \ .is_ip4 = 0, \ .port = 0, \ .peer = TRANSPORT_ENDPOINT_NULL, \ .transport_proto = 0, \ } #define SESSION_ENDPOINT_CFG_NULL \ { \ .sw_if_index = ENDPOINT_INVALID_INDEX, \ .ip = SESSION_IP46_ZERO, \ .fib_index = ENDPOINT_INVALID_INDEX, \ .is_ip4 = 0, \ .port = 0, \ .peer = TRANSPORT_ENDPOINT_NULL, \ .transport_proto = 0, \ .app_wrk_index = ENDPOINT_INVALID_INDEX, \ .opaque = ENDPOINT_INVALID_INDEX, \ .hostname = 0, \ } #define session_endpoint_to_transport(_sep) ((transport_endpoint_t *)_sep) #define session_endpoint_to_transport_cfg(_sep) \ ((transport_endpoint_cfg_t *)_sep) always_inline u8 session_endpoint_fib_proto (session_endpoint_t * sep) { return sep->is_ip4 ? FIB_PROTOCOL_IP4 : FIB_PROTOCOL_IP6; } static inline u8 session_endpoint_is_local (session_endpoint_t * sep) { return (ip_is_zero (&sep->ip, sep->is_ip4) || ip_is_local_host (&sep->ip, sep->is_ip4)); } static inline u8 session_endpoint_is_zero (session_endpoint_t * sep) { return ip_is_zero (&sep->ip, sep->is_ip4); } typedef u8 session_type_t; typedef u64 session_handle_t; /* * Application session state */ typedef enum { SESSION_STATE_LISTENING, SESSION_STATE_CONNECTING, SESSION_STATE_ACCEPTING, SESSION_STATE_READY, SESSION_STATE_OPENED, SESSION_STATE_TRANSPORT_CLOSING, SESSION_STATE_CLOSING, SESSION_STATE_CLOSED_WAITING, SESSION_STATE_TRANSPORT_CLOSED, SESSION_STATE_CLOSED, SESSION_STATE_N_STATES, } session_state_t; typedef struct generic_session_ { svm_fifo_t *rx_fifo; /**< rx fifo */ svm_fifo_t *tx_fifo; /**< tx fifo */ session_type_t session_type; /**< session type */ volatile u8 session_state; /**< session state */ u32 session_index; /**< index in owning pool */ } generic_session_t; typedef struct session_ { /** fifo pointers. Once allocated, these do not move */ svm_fifo_t *rx_fifo; svm_fifo_t *tx_fifo; /** Type */ session_type_t session_type; /** State */ volatile u8 session_state; /** Session index in per_thread pool */ u32 session_index; /** App worker pool index */ u32 app_wrk_index; u8 thread_index; /** To avoid n**2 "one event per frame" check */ u64 enqueue_epoch; /** svm segment index where fifos were allocated */ u32 svm_segment_index; /** Transport specific */ u32 connection_index; union { /** Parent listener session if the result of an accept */ u32 listener_index; /** Application index if a listener */ u32 app_index; }; union { /** Transport app index for apps acting as transports */ u32 t_app_index; /** App listener index */ u32 al_index; /** Opaque, for general use */ u32 opaque; }; CLIB_CACHE_LINE_ALIGN_MARK (pad); } session_t; always_inline session_type_t session_type_from_proto_and_ip (transport_proto_t proto, u8 is_ip4) { return (proto << 1 | is_ip4); } always_inline transport_proto_t session_type_transport_proto (session_type_t st) { return (st >> 1); } always_inline u8 session_type_is_ip4 (session_type_t st) { return (st & 1); } always_inline transport_proto_t session_get_transport_proto (session_t * s) { return (s->session_type >> 1); } always_inline fib_protocol_t session_get_fib_proto (session_t * s) { u8 is_ip4 = s->session_type & 1; return (is_ip4 ? FIB_PROTOCOL_IP4 : FIB_PROTOCOL_IP6); } always_inline u8 session_has_transport (session_t * s) { return (session_get_transport_proto (s) != TRANSPORT_PROTO_NONE); } static inline transport_service_type_t session_transport_service_type (session_t * s) { transport_proto_t tp; tp = session_get_transport_proto (s); return transport_protocol_service_type (tp); } static inline transport_tx_fn_type_t session_transport_tx_fn_type (session_t * s) { transport_proto_t tp; tp = session_get_transport_proto (s); return transport_protocol_tx_fn_type (tp); } static inline u8 session_tx_is_dgram (session_t * s) { return (session_transport_tx_fn_type (s) == TRANSPORT_TX_DGRAM); } always_inline session_handle_t session_handle (session_t * s) { return ((u64) s->thread_index << 32) | (u64) s->session_index; } always_inline u32 session_index_from_handle (session_handle_t handle) { return handle & 0xFFFFFFFF; } always_inline u32 session_thread_from_handle (session_handle_t handle) { return handle >> 32; } always_inline void session_parse_handle (session_handle_t handle, u32 * index, u32 * thread_index) { *index = session_index_from_handle (handle); *thread_index = session_thread_from_handle (handle); } always_inline u8 session_handle_is_local (session_handle_t handle) { if ((handle >> 32) == SESSION_LOCAL_HANDLE_PREFIX) return 1; return 0; } typedef struct local_session_ { /** fifo pointers. Once allocated, these do not move */ svm_fifo_t *rx_fifo; svm_fifo_t *tx_fifo; /** Type */ session_type_t session_type; /** State */ volatile u8 session_state; /** Session index */ u32 session_index; /** Server index */ u32 app_wrk_index; /** Port for connection. Overlaps thread_index/enqueue_epoch */ u16 port; /** Partly overlaps enqueue_epoch */ u8 pad_epoch[7]; /** Segment index where fifos were allocated */ u32 svm_segment_index; /** Transport listener index. Overlaps connection index */ u32 transport_listener_index; union { u32 listener_index; u32 app_index; }; u32 al_index; /** Has transport embedded when listener not purely local */ session_type_t listener_session_type; /** * Client data */ u32 client_wrk_index; u32 client_opaque; u64 server_evt_q; u64 client_evt_q; CLIB_CACHE_LINE_ALIGN_MARK (pad); } local_session_t; always_inline u32 local_session_id (local_session_t * ls) { ASSERT (ls->session_index < (2 << 16)); u32 app_or_wrk_index; if (ls->session_state == SESSION_STATE_LISTENING) { ASSERT (ls->app_index < (2 << 16)); app_or_wrk_index = ls->app_index; } else { ASSERT (ls->app_wrk_index < (2 << 16)); app_or_wrk_index = ls->app_wrk_index; } return ((u32) app_or_wrk_index << 16 | (u32) ls->session_index); } always_inline void local_session_parse_id (u32 ls_id, u32 * app_or_wrk, u32 * session_index) { *app_or_wrk = ls_id >> 16; *session_index = ls_id & 0xFF; } always_inline void local_session_parse_handle (session_handle_t handle, u32 * app_or_wrk_index, u32 * session_index) { u32 bottom; ASSERT ((handle >> 32) == SESSION_LOCAL_HANDLE_PREFIX); bottom = (handle & 0xFFFFFFFF); local_session_parse_id (bottom, app_or_wrk_index, session_index); } always_inline session_handle_t application_local_session_handle (local_session_t * ls) { return ((u64) SESSION_LOCAL_HANDLE_PREFIX << 32) | (u64) local_session_id (ls); } typedef enum { FIFO_EVENT_APP_RX, SESSION_IO_EVT_CT_RX, FIFO_EVENT_APP_TX, SESSION_IO_EVT_CT_TX, SESSION_IO_EVT_TX_FLUSH, FIFO_EVENT_DISCONNECT, FIFO_EVENT_BUILTIN_RX, FIFO_EVENT_BUILTIN_TX, FIFO_EVENT_RPC, SESSION_CTRL_EVT_BOUND, SESSION_CTRL_EVT_ACCEPTED, SESSION_CTRL_EVT_ACCEPTED_REPLY, SESSION_CTRL_EVT_CONNECTED, SESSION_CTRL_EVT_CONNECTED_REPLY, SESSION_CTRL_EVT_DISCONNECTED, SESSION_CTRL_EVT_DISCONNECTED_REPLY, SESSION_CTRL_EVT_RESET, SESSION_CTRL_EVT_RESET_REPLY, SESSION_CTRL_EVT_REQ_WORKER_UPDATE, SESSION_CTRL_EVT_WORKER_UPDATE, SESSION_CTRL_EVT_WORKER_UPDATE_REPLY, } session_evt_type_t; static inline const char * fifo_event_type_str (session_evt_type_t et) { switch (et) { case FIFO_EVENT_APP_RX: return "FIFO_EVENT_APP_RX"; case FIFO_EVENT_APP_TX: return "FIFO_EVENT_APP_TX"; case FIFO_EVENT_DISCONNECT: return "FIFO_EVENT_DISCONNECT"; case FIFO_EVENT_BUILTIN_RX: return "FIFO_EVENT_BUILTIN_RX"; case FIFO_EVENT_RPC: return "FIFO_EVENT_RPC"; default: return "UNKNOWN FIFO EVENT"; } } typedef enum { SESSION_MQ_IO_EVT_RING, SESSION_MQ_CTRL_EVT_RING, SESSION_MQ_N_RINGS } session_mq_rings_e; typedef struct { void *fp; void *arg; } session_rpc_args_t; /* *INDENT-OFF* */ typedef struct { u8 event_type; u8 postponed; union { svm_fifo_t *fifo; session_handle_t session_handle; session_rpc_args_t rpc_args; struct { u8 data[0]; }; }; } __clib_packed session_event_t; /* *INDENT-ON* */ #define SESSION_MSG_NULL { } typedef struct session_dgram_pre_hdr_ { u32 data_length; u32 data_offset; } session_dgram_pre_hdr_t; /* *INDENT-OFF* */ typedef CLIB_PACKED (struct session_dgram_header_ { u32 data_length; u32 data_offset; ip46_address_t rmt_ip; ip46_address_t lcl_ip; u16 rmt_port; u16 lcl_port; u8 is_ip4; }) session_dgram_hdr_t; /* *INDENT-ON* */ #define SESSION_CONN_ID_LEN 37 #define SESSION_CONN_HDR_LEN 45 STATIC_ASSERT (sizeof (session_dgram_hdr_t) == (SESSION_CONN_ID_LEN + 8), "session conn id wrong length"); #endif /* SRC_VNET_SESSION_SESSION_TYPES_H_ */ /* * fd.io coding-style-patch-verification: ON * * Local Variables: * eval: (c-set-style "gnu") * End: */