/* Copyright (c) 2008, 2024, Oracle and/or its affiliates. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License, version 2.0, as published by the Free Software Foundation. This program is designed to work with certain software (including but not limited to OpenSSL) that is licensed under separate terms, as designated in a particular file or component or in included license documentation. The authors of MySQL hereby grant you an additional permission to link the program and your derivative works with the separately licensed software that they have either included with the program or referenced in the documentation. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License, version 2.0, for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */ #ifndef COMPONENTS_SERVICES_BITS_PSI_SOCKET_BITS_H #define COMPONENTS_SERVICES_BITS_PSI_SOCKET_BITS_H #ifndef MYSQL_ABI_CHECK #include /* size_t */ #endif #include /* socklen_t */ #include /** @file mysql/components/services/bits/psi_socket_bits.h Performance schema instrumentation interface. @defgroup psi_abi_socket Socket Instrumentation (ABI) @ingroup psi_abi @{ */ /** Instrumented socket key. To instrument a socket, a socket key must be obtained using @c register_socket. Using a zero key always disable the instrumentation. */ typedef unsigned int PSI_socket_key; /** @def PSI_SOCKET_VERSION_1 Performance Schema Socket Interface number for version 1. This version is supported. */ #define PSI_SOCKET_VERSION_1 1 /** @def PSI_CURRENT_SOCKET_VERSION Performance Schema Socket Interface number for the most recent version. The most current version is @c PSI_SOCKET_VERSION_1 */ #define PSI_CURRENT_SOCKET_VERSION 1 /** Interface for an instrumented socket descriptor. This is an opaque structure. */ struct PSI_socket : PSI_instr {}; typedef struct PSI_socket PSI_socket; /** Interface for an instrumented socket operation. This is an opaque structure. */ struct PSI_socket_locker; typedef struct PSI_socket_locker PSI_socket_locker; /** State of an instrumented socket. */ enum PSI_socket_state { /** Idle, waiting for the next command. */ PSI_SOCKET_STATE_IDLE = 1, /** Active, executing a command. */ PSI_SOCKET_STATE_ACTIVE = 2 }; typedef enum PSI_socket_state PSI_socket_state; /** Operation performed on an instrumented socket. */ enum PSI_socket_operation { /** Socket creation, as in @c socket() or @c socketpair(). */ PSI_SOCKET_CREATE = 0, /** Socket connection, as in @c connect(), @c listen() and @c accept(). */ PSI_SOCKET_CONNECT = 1, /** Socket bind, as in @c bind(), @c getsockname() and @c getpeername(). */ PSI_SOCKET_BIND = 2, /** Socket close, as in @c shutdown(). */ PSI_SOCKET_CLOSE = 3, /** Socket send, @c send(). */ PSI_SOCKET_SEND = 4, /** Socket receive, @c recv(). */ PSI_SOCKET_RECV = 5, /** Socket send, @c sendto(). */ PSI_SOCKET_SENDTO = 6, /** Socket receive, @c recvfrom). */ PSI_SOCKET_RECVFROM = 7, /** Socket send, @c sendmsg(). */ PSI_SOCKET_SENDMSG = 8, /** Socket receive, @c recvmsg(). */ PSI_SOCKET_RECVMSG = 9, /** Socket seek, such as @c fseek() or @c seek(). */ PSI_SOCKET_SEEK = 10, /** Socket options, as in @c getsockopt() and @c setsockopt(). */ PSI_SOCKET_OPT = 11, /** Socket status, as in @c sockatmark() and @c isfdtype(). */ PSI_SOCKET_STAT = 12, /** Socket shutdown, as in @c shutdown(). */ PSI_SOCKET_SHUTDOWN = 13, /** Socket select, as in @c select() and @c poll(). */ PSI_SOCKET_SELECT = 14 }; typedef enum PSI_socket_operation PSI_socket_operation; /** Socket instrument information. @since PSI_SOCKET_VERSION_1 This structure is used to register an instrumented socket. */ struct PSI_socket_info_v1 { /** Pointer to the key assigned to the registered socket. */ PSI_socket_key *m_key; /** The name of the socket instrument to register. */ const char *m_name; /** The flags of the socket instrument to register. @sa PSI_FLAG_SINGLETON */ unsigned int m_flags; /** Volatility index. */ int m_volatility; /** Documentation. */ const char *m_documentation; }; typedef struct PSI_socket_info_v1 PSI_socket_info_v1; /** State data storage for @c start_socket_wait_v1_t. This structure provide temporary storage to a socket locker. The content of this structure is considered opaque, the fields are only hints of what an implementation of the psi interface can use. This memory is provided by the instrumented code for performance reasons. @sa start_socket_wait_v1_t */ struct PSI_socket_locker_state_v1 { /** Internal state. */ unsigned int m_flags; /** Current socket. */ struct PSI_socket *m_socket; /** Current thread. */ struct PSI_thread *m_thread; /** Operation number of bytes. */ size_t m_number_of_bytes; /** Timer start. */ unsigned long long m_timer_start{0}; /** Timer function. */ unsigned long long (*m_timer)(void); /** Current operation. */ enum PSI_socket_operation m_operation; /** Source file. */ const char *m_src_file; /** Source line number. */ int m_src_line; /** Internal data. */ void *m_wait; }; typedef struct PSI_socket_locker_state_v1 PSI_socket_locker_state_v1; /** Socket registration API. @param category a category name (typically a plugin name) @param info an array of socket info to register @param count the size of the info array */ typedef void (*register_socket_v1_t)(const char *category, struct PSI_socket_info_v1 *info, int count); /** Socket instrumentation initialisation API. @param key the registered socket key @param fd socket file descriptor @param addr the socket ip address @param addr_len length of socket ip address @return an instrumented socket */ typedef struct PSI_socket *(*init_socket_v1_t)(PSI_socket_key key, const my_socket *fd, const struct sockaddr *addr, socklen_t addr_len); /** socket instrumentation destruction API. @param socket the socket to destroy */ typedef void (*destroy_socket_v1_t)(struct PSI_socket *socket); /** Record a socket instrumentation start event. @param state data storage for the locker @param socket the instrumented socket @param op socket operation to be performed @param count the number of bytes requested, or 0 if not applicable @param src_file the source file name @param src_line the source line number @return a socket locker, or NULL */ typedef struct PSI_socket_locker *(*start_socket_wait_v1_t)( struct PSI_socket_locker_state_v1 *state, struct PSI_socket *socket, enum PSI_socket_operation op, size_t count, const char *src_file, unsigned int src_line); /** Record a socket instrumentation end event. Note that for socket close operations, the instrumented socket handle associated with the socket (which was provided to obtain a locker) is invalid after this call. @param locker a socket locker for the running thread @param count the number of bytes actually used in the operation, or 0 if not applicable, or -1 if the operation failed @sa get_thread_socket_locker */ typedef void (*end_socket_wait_v1_t)(struct PSI_socket_locker *locker, size_t count); /** Set the socket state for an instrumented socket. @param socket the instrumented socket @param state socket state */ typedef void (*set_socket_state_v1_t)(struct PSI_socket *socket, enum PSI_socket_state state); /** Set the socket info for an instrumented socket. @param socket the instrumented socket @param fd the socket descriptor @param addr the socket ip address @param addr_len length of socket ip address */ typedef void (*set_socket_info_v1_t)(struct PSI_socket *socket, const my_socket *fd, const struct sockaddr *addr, socklen_t addr_len); /** Bind a socket to the thread that owns it. @param socket instrumented socket */ typedef void (*set_socket_thread_owner_v1_t)(struct PSI_socket *socket); typedef struct PSI_socket_info_v1 PSI_socket_info; typedef struct PSI_socket_locker_state_v1 PSI_socket_locker_state; /** @} (end of group psi_abi_socket) */ #endif /* COMPONENTS_SERVICES_BITS_PSI_SOCKET_BITS_H */