/* * 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_INVALID_INDEX ((u32)~0) #define SESSION_INVALID_HANDLE ((u64)~0) #define SESSION_CTRL_MSG_MAX_SIZE 86 #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; u32 ns_index; u8 original_tp; u8 *hostname; u64 parent_handle; u32 ckpair_index; u8 crypto_engine; u8 flags; } 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, \ .parent_handle = SESSION_INVALID_HANDLE, \ .ckpair_index = 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; typedef enum { SESSION_CLEANUP_TRANSPORT, SESSION_CLEANUP_SESSION, } session_cleanup_ntf_t; typedef enum session_ft_action_ { SESSION_FT_ACTION_ENQUEUED, SESSION_FT_ACTION_DEQUEUED, SESSION_FT_ACTION_N_ACTIONS } session_ft_action_t; /* * Session states */ #define foreach_session_state \ _(CREATED, "created") \ _(LISTENING, "listening") \ _(CONNECTING, "connecting") \ _(ACCEPTING, "accepting") \ _(READY, "ready") \ _(OPENED, "opened") \ _(TRANSPORT_CLOSING, "transport-closing") \ _(CLOSING, "closing") \ _(APP_CLOSED, "app-closed") \ _(TRANSPORT_CLOSED, "transport-closed") \ _(CLOSED, "closed") \ _(TRANSPORT_DELETED, "transport-deleted") \ typedef enum { #define _(sym, str) SESSION_STATE_ ## sym, foreach_session_state #undef _ SESSION_N_STATES, } session_state_t; #define foreach_session_flag \ _(RX_EVT, "rx-event") \ _(PROXY, "proxy") \ _(CUSTOM_TX, "custom-tx") \ _(IS_MIGRATING, "migrating") \ _(UNIDIRECTIONAL, "unidirectional") \ _(CUSTOM_FIFO_TUNING, "custom-fifo-tuning") \ typedef enum session_flags_bits_ { #define _(sym, str) SESSION_F_BIT_ ## sym, foreach_session_flag #undef _ SESSION_N_FLAGS } session_flag_bits_t; typedef enum session_flags_ { #define _(sym, str) SESSION_F_ ## sym = 1 << SESSION_F_BIT_ ## sym, foreach_session_flag #undef _ } session_flags_t; typedef struct session_ { /** Pointers to rx/tx buffers. Once allocated, these do not move */ svm_fifo_t *rx_fifo; svm_fifo_t *tx_fifo; /** Type built from transport and network protocol types */ session_type_t session_type; /** State in session layer state machine. See @ref session_state_t */ volatile u8 session_state; /** Index in thread pool where session was allocated */ u32 session_index; /** Index of the app worker that owns the session */ u32 app_wrk_index; /** Index of the thread that allocated the session */ u8 thread_index; /** Session flags. See @ref session_flags_t */ u32 flags; /** Index of the transport connection associated to the session */ u32 connection_index; /** Index of application that owns the listener. Set only if a listener */ u32 app_index; union { /** Parent listener session index if the result of an accept */ session_handle_t listener_handle; /** App listener index in app's listener pool if a listener */ 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); } static inline session_handle_t session_make_handle (u32 session_index, u32 data) { return (((u64) data << 32) | (u64) session_index); } always_inline u32 session_handle_index (session_handle_t ho_handle) { return (ho_handle & 0xffffffff); } always_inline u32 session_handle_data (session_handle_t ho_handle) { return (ho_handle >> 32); } typedef enum { SESSION_IO_EVT_RX, SESSION_IO_EVT_TX, SESSION_IO_EVT_TX_FLUSH, SESSION_IO_EVT_BUILTIN_RX, SESSION_IO_EVT_BUILTIN_TX, SESSION_CTRL_EVT_RPC, SESSION_CTRL_EVT_CLOSE, SESSION_CTRL_EVT_RESET, SESSION_CTRL_EVT_BOUND, SESSION_CTRL_EVT_UNLISTEN_REPLY, SESSION_CTRL_EVT_ACCEPTED, SESSION_CTRL_EVT_ACCEPTED_REPLY, SESSION_CTRL_EVT_CONNECTED, SESSION_CTRL_EVT_DISCONNECTED, SESSION_CTRL_EVT_DISCONNECTED_REPLY, SESSION_CTRL_EVT_RESET_REPLY, SESSION_CTRL_EVT_REQ_WORKER_UPDATE, SESSION_CTRL_EVT_WORKER_UPDATE, SESSION_CTRL_EVT_WORKER_UPDATE_REPLY, SESSION_CTRL_EVT_DISCONNECT, SESSION_CTRL_EVT_CONNECT, SESSION_CTRL_EVT_CONNECT_URI, SESSION_CTRL_EVT_LISTEN, SESSION_CTRL_EVT_LISTEN_URI, SESSION_CTRL_EVT_UNLISTEN, SESSION_CTRL_EVT_APP_DETACH, SESSION_CTRL_EVT_APP_ADD_SEGMENT, SESSION_CTRL_EVT_APP_DEL_SEGMENT, SESSION_CTRL_EVT_MIGRATED, SESSION_CTRL_EVT_CLEANUP, } session_evt_type_t; #define foreach_session_ctrl_evt \ _(LISTEN, listen) \ _(LISTEN_URI, listen_uri) \ _(BOUND, bound) \ _(UNLISTEN, unlisten) \ _(UNLISTEN_REPLY, unlisten_reply) \ _(ACCEPTED, accepted) \ _(ACCEPTED_REPLY, accepted_reply) \ _(CONNECT, connect) \ _(CONNECT_URI, connect_uri) \ _(CONNECTED, connected) \ _(DISCONNECT, disconnect) \ _(DISCONNECTED, disconnected) \ _(DISCONNECTED_REPLY, disconnected_reply) \ _(RESET_REPLY, reset_reply) \ _(REQ_WORKER_UPDATE, req_worker_update) \ _(WORKER_UPDATE, worker_update) \ _(WORKER_UPDATE_REPLY, worker_update_reply) \ _(APP_DETACH, app_detach) \ _(APP_ADD_SEGMENT, app_add_segment) \ _(APP_DEL_SEGMENT, app_del_segment) \ _(MIGRATED, migrated) \ _(CLEANUP, cleanup) \ /* Deprecated and will be removed. Use types above */ #define FIFO_EVENT_APP_RX SESSION_IO_EVT_RX #define FIFO_EVENT_APP_TX SESSION_IO_EVT_TX #define FIFO_EVENT_DISCONNECT SESSION_CTRL_EVT_CLOSE #define FIFO_EVENT_BUILTIN_RX SESSION_IO_EVT_BUILTIN_RX #define FIFO_EVENT_BUILTIN_TX SESSION_IO_EVT_BUILTIN_TX 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; typedef struct { u8 event_type; u8 postponed; union { u32 session_index; session_handle_t session_handle; session_rpc_args_t rpc_args; u32 ctrl_data_index; struct { u8 data[0]; }; }; } __clib_packed session_event_t; #define SESSION_MSG_NULL { } typedef struct session_dgram_pre_hdr_ { u32 data_length; u32 data_offset; } session_dgram_pre_hdr_t; typedef 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; } __clib_packed session_dgram_hdr_t; #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"); #define foreach_session_error \ _(NONE, "no error") \ _(UNKNOWN, "generic/unknown error") \ _(REFUSED, "refused") \ _(TIMEDOUT, "timedout") \ _(ALLOC, "obj/memory allocation error") \ _(OWNER, "object not owned by application") \ _(NOROUTE, "no route") \ _(NOINTF, "no resolving interface") \ _(NOIP, "no ip for lcl interface") \ _(NOPORT, "no lcl port") \ _(NOSUPPORT, "not supported") \ _(NOLISTEN, "not listening") \ _(NOSESSION, "session does not exist") \ _(NOAPP, "app not attached") \ _(PORTINUSE, "lcl port in use") \ _(IPINUSE, "ip in use") \ _(ALREADY_LISTENING, "ip port pair already listened on") \ _(INVALID_RMT_IP, "invalid remote ip") \ _(INVALID_APPWRK, "invalid app worker") \ _(INVALID_NS, "invalid namespace") \ _(SEG_NO_SPACE, "Couldn't allocate a fifo pair") \ _(SEG_NO_SPACE2, "Created segment, couldn't allocate a fifo pair") \ _(SEG_CREATE, "Couldn't create a new segment") \ _(FILTERED, "session filtered") \ _(SCOPE, "scope not supported") \ _(BAPI_NO_FD, "bapi doesn't have a socket fd") \ _(BAPI_SEND_FD, "couldn't send fd over bapi socket fd") \ _(BAPI_NO_REG, "app bapi registration not found") \ _(MQ_MSG_ALLOC, "failed to alloc mq msg") \ _(TLS_HANDSHAKE, "failed tls handshake") \ typedef enum session_error_p_ { #define _(sym, str) SESSION_EP_##sym, foreach_session_error #undef _ SESSION_N_ERRORS } session_error_p_t; typedef enum session_error_ { #define _(sym, str) SESSION_E_##sym = -SESSION_EP_##sym, foreach_session_error #undef _ } session_error_t; /* Maintained for compatibility. Will be deprecated */ #define SESSION_ERROR_SEG_CREATE SESSION_E_SEG_CREATE #define SESSION_ERROR_NO_SPACE SESSION_E_SEG_NO_SPACE #define SESSION_ERROR_NEW_SEG_NO_SPACE SESSION_E_SEG_NO_SPACE2 #endif /* SRC_VNET_SESSION_SESSION_TYPES_H_ */ /* * fd.io coding-style-patch-verification: ON * * Local Variables: * eval: (c-set-style "gnu") * End: */