162306a36Sopenharmony_ci/* SPDX-License-Identifier: GPL-2.0-only */ 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * Kernel Connection Multiplexor 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (c) 2016 Tom Herbert <tom@herbertland.com> 662306a36Sopenharmony_ci */ 762306a36Sopenharmony_ci 862306a36Sopenharmony_ci#ifndef __NET_KCM_H_ 962306a36Sopenharmony_ci#define __NET_KCM_H_ 1062306a36Sopenharmony_ci 1162306a36Sopenharmony_ci#include <linux/skbuff.h> 1262306a36Sopenharmony_ci#include <net/sock.h> 1362306a36Sopenharmony_ci#include <net/strparser.h> 1462306a36Sopenharmony_ci#include <uapi/linux/kcm.h> 1562306a36Sopenharmony_ci 1662306a36Sopenharmony_ciextern unsigned int kcm_net_id; 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_ci#define KCM_STATS_ADD(stat, count) ((stat) += (count)) 1962306a36Sopenharmony_ci#define KCM_STATS_INCR(stat) ((stat)++) 2062306a36Sopenharmony_ci 2162306a36Sopenharmony_cistruct kcm_psock_stats { 2262306a36Sopenharmony_ci unsigned long long tx_msgs; 2362306a36Sopenharmony_ci unsigned long long tx_bytes; 2462306a36Sopenharmony_ci unsigned long long reserved; 2562306a36Sopenharmony_ci unsigned long long unreserved; 2662306a36Sopenharmony_ci unsigned int tx_aborts; 2762306a36Sopenharmony_ci}; 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_cistruct kcm_mux_stats { 3062306a36Sopenharmony_ci unsigned long long rx_msgs; 3162306a36Sopenharmony_ci unsigned long long rx_bytes; 3262306a36Sopenharmony_ci unsigned long long tx_msgs; 3362306a36Sopenharmony_ci unsigned long long tx_bytes; 3462306a36Sopenharmony_ci unsigned int rx_ready_drops; 3562306a36Sopenharmony_ci unsigned int tx_retries; 3662306a36Sopenharmony_ci unsigned int psock_attach; 3762306a36Sopenharmony_ci unsigned int psock_unattach_rsvd; 3862306a36Sopenharmony_ci unsigned int psock_unattach; 3962306a36Sopenharmony_ci}; 4062306a36Sopenharmony_ci 4162306a36Sopenharmony_cistruct kcm_stats { 4262306a36Sopenharmony_ci unsigned long long rx_msgs; 4362306a36Sopenharmony_ci unsigned long long rx_bytes; 4462306a36Sopenharmony_ci unsigned long long tx_msgs; 4562306a36Sopenharmony_ci unsigned long long tx_bytes; 4662306a36Sopenharmony_ci}; 4762306a36Sopenharmony_ci 4862306a36Sopenharmony_cistruct kcm_tx_msg { 4962306a36Sopenharmony_ci unsigned int sent; 5062306a36Sopenharmony_ci unsigned int frag_offset; 5162306a36Sopenharmony_ci unsigned int msg_flags; 5262306a36Sopenharmony_ci bool started_tx; 5362306a36Sopenharmony_ci struct sk_buff *frag_skb; 5462306a36Sopenharmony_ci struct sk_buff *last_skb; 5562306a36Sopenharmony_ci}; 5662306a36Sopenharmony_ci 5762306a36Sopenharmony_ci/* Socket structure for KCM client sockets */ 5862306a36Sopenharmony_cistruct kcm_sock { 5962306a36Sopenharmony_ci struct sock sk; 6062306a36Sopenharmony_ci struct kcm_mux *mux; 6162306a36Sopenharmony_ci struct list_head kcm_sock_list; 6262306a36Sopenharmony_ci int index; 6362306a36Sopenharmony_ci u32 done : 1; 6462306a36Sopenharmony_ci struct work_struct done_work; 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_ci struct kcm_stats stats; 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_ci /* Transmit */ 6962306a36Sopenharmony_ci struct kcm_psock *tx_psock; 7062306a36Sopenharmony_ci struct work_struct tx_work; 7162306a36Sopenharmony_ci struct list_head wait_psock_list; 7262306a36Sopenharmony_ci struct sk_buff *seq_skb; 7362306a36Sopenharmony_ci u32 tx_stopped : 1; 7462306a36Sopenharmony_ci 7562306a36Sopenharmony_ci /* Don't use bit fields here, these are set under different locks */ 7662306a36Sopenharmony_ci bool tx_wait; 7762306a36Sopenharmony_ci bool tx_wait_more; 7862306a36Sopenharmony_ci 7962306a36Sopenharmony_ci /* Receive */ 8062306a36Sopenharmony_ci struct kcm_psock *rx_psock; 8162306a36Sopenharmony_ci struct list_head wait_rx_list; /* KCMs waiting for receiving */ 8262306a36Sopenharmony_ci bool rx_wait; 8362306a36Sopenharmony_ci u32 rx_disabled : 1; 8462306a36Sopenharmony_ci}; 8562306a36Sopenharmony_ci 8662306a36Sopenharmony_cistruct bpf_prog; 8762306a36Sopenharmony_ci 8862306a36Sopenharmony_ci/* Structure for an attached lower socket */ 8962306a36Sopenharmony_cistruct kcm_psock { 9062306a36Sopenharmony_ci struct sock *sk; 9162306a36Sopenharmony_ci struct strparser strp; 9262306a36Sopenharmony_ci struct kcm_mux *mux; 9362306a36Sopenharmony_ci int index; 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_ci u32 tx_stopped : 1; 9662306a36Sopenharmony_ci u32 done : 1; 9762306a36Sopenharmony_ci u32 unattaching : 1; 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_ci void (*save_state_change)(struct sock *sk); 10062306a36Sopenharmony_ci void (*save_data_ready)(struct sock *sk); 10162306a36Sopenharmony_ci void (*save_write_space)(struct sock *sk); 10262306a36Sopenharmony_ci 10362306a36Sopenharmony_ci struct list_head psock_list; 10462306a36Sopenharmony_ci 10562306a36Sopenharmony_ci struct kcm_psock_stats stats; 10662306a36Sopenharmony_ci 10762306a36Sopenharmony_ci /* Receive */ 10862306a36Sopenharmony_ci struct list_head psock_ready_list; 10962306a36Sopenharmony_ci struct bpf_prog *bpf_prog; 11062306a36Sopenharmony_ci struct kcm_sock *rx_kcm; 11162306a36Sopenharmony_ci unsigned long long saved_rx_bytes; 11262306a36Sopenharmony_ci unsigned long long saved_rx_msgs; 11362306a36Sopenharmony_ci struct sk_buff *ready_rx_msg; 11462306a36Sopenharmony_ci 11562306a36Sopenharmony_ci /* Transmit */ 11662306a36Sopenharmony_ci struct kcm_sock *tx_kcm; 11762306a36Sopenharmony_ci struct list_head psock_avail_list; 11862306a36Sopenharmony_ci unsigned long long saved_tx_bytes; 11962306a36Sopenharmony_ci unsigned long long saved_tx_msgs; 12062306a36Sopenharmony_ci}; 12162306a36Sopenharmony_ci 12262306a36Sopenharmony_ci/* Per net MUX list */ 12362306a36Sopenharmony_cistruct kcm_net { 12462306a36Sopenharmony_ci struct mutex mutex; 12562306a36Sopenharmony_ci struct kcm_psock_stats aggregate_psock_stats; 12662306a36Sopenharmony_ci struct kcm_mux_stats aggregate_mux_stats; 12762306a36Sopenharmony_ci struct strp_aggr_stats aggregate_strp_stats; 12862306a36Sopenharmony_ci struct list_head mux_list; 12962306a36Sopenharmony_ci int count; 13062306a36Sopenharmony_ci}; 13162306a36Sopenharmony_ci 13262306a36Sopenharmony_ci/* Structure for a MUX */ 13362306a36Sopenharmony_cistruct kcm_mux { 13462306a36Sopenharmony_ci struct list_head kcm_mux_list; 13562306a36Sopenharmony_ci struct rcu_head rcu; 13662306a36Sopenharmony_ci struct kcm_net *knet; 13762306a36Sopenharmony_ci 13862306a36Sopenharmony_ci struct list_head kcm_socks; /* All KCM sockets on MUX */ 13962306a36Sopenharmony_ci int kcm_socks_cnt; /* Total KCM socket count for MUX */ 14062306a36Sopenharmony_ci struct list_head psocks; /* List of all psocks on MUX */ 14162306a36Sopenharmony_ci int psocks_cnt; /* Total attached sockets */ 14262306a36Sopenharmony_ci 14362306a36Sopenharmony_ci struct kcm_mux_stats stats; 14462306a36Sopenharmony_ci struct kcm_psock_stats aggregate_psock_stats; 14562306a36Sopenharmony_ci struct strp_aggr_stats aggregate_strp_stats; 14662306a36Sopenharmony_ci 14762306a36Sopenharmony_ci /* Receive */ 14862306a36Sopenharmony_ci spinlock_t rx_lock ____cacheline_aligned_in_smp; 14962306a36Sopenharmony_ci struct list_head kcm_rx_waiters; /* KCMs waiting for receiving */ 15062306a36Sopenharmony_ci struct list_head psocks_ready; /* List of psocks with a msg ready */ 15162306a36Sopenharmony_ci struct sk_buff_head rx_hold_queue; 15262306a36Sopenharmony_ci 15362306a36Sopenharmony_ci /* Transmit */ 15462306a36Sopenharmony_ci spinlock_t lock ____cacheline_aligned_in_smp; /* TX and mux locking */ 15562306a36Sopenharmony_ci struct list_head psocks_avail; /* List of available psocks */ 15662306a36Sopenharmony_ci struct list_head kcm_tx_waiters; /* KCMs waiting for a TX psock */ 15762306a36Sopenharmony_ci}; 15862306a36Sopenharmony_ci 15962306a36Sopenharmony_ci#ifdef CONFIG_PROC_FS 16062306a36Sopenharmony_ciint kcm_proc_init(void); 16162306a36Sopenharmony_civoid kcm_proc_exit(void); 16262306a36Sopenharmony_ci#else 16362306a36Sopenharmony_cistatic inline int kcm_proc_init(void) { return 0; } 16462306a36Sopenharmony_cistatic inline void kcm_proc_exit(void) { } 16562306a36Sopenharmony_ci#endif 16662306a36Sopenharmony_ci 16762306a36Sopenharmony_cistatic inline void aggregate_psock_stats(struct kcm_psock_stats *stats, 16862306a36Sopenharmony_ci struct kcm_psock_stats *agg_stats) 16962306a36Sopenharmony_ci{ 17062306a36Sopenharmony_ci /* Save psock statistics in the mux when psock is being unattached. */ 17162306a36Sopenharmony_ci 17262306a36Sopenharmony_ci#define SAVE_PSOCK_STATS(_stat) (agg_stats->_stat += stats->_stat) 17362306a36Sopenharmony_ci SAVE_PSOCK_STATS(tx_msgs); 17462306a36Sopenharmony_ci SAVE_PSOCK_STATS(tx_bytes); 17562306a36Sopenharmony_ci SAVE_PSOCK_STATS(reserved); 17662306a36Sopenharmony_ci SAVE_PSOCK_STATS(unreserved); 17762306a36Sopenharmony_ci SAVE_PSOCK_STATS(tx_aborts); 17862306a36Sopenharmony_ci#undef SAVE_PSOCK_STATS 17962306a36Sopenharmony_ci} 18062306a36Sopenharmony_ci 18162306a36Sopenharmony_cistatic inline void aggregate_mux_stats(struct kcm_mux_stats *stats, 18262306a36Sopenharmony_ci struct kcm_mux_stats *agg_stats) 18362306a36Sopenharmony_ci{ 18462306a36Sopenharmony_ci /* Save psock statistics in the mux when psock is being unattached. */ 18562306a36Sopenharmony_ci 18662306a36Sopenharmony_ci#define SAVE_MUX_STATS(_stat) (agg_stats->_stat += stats->_stat) 18762306a36Sopenharmony_ci SAVE_MUX_STATS(rx_msgs); 18862306a36Sopenharmony_ci SAVE_MUX_STATS(rx_bytes); 18962306a36Sopenharmony_ci SAVE_MUX_STATS(tx_msgs); 19062306a36Sopenharmony_ci SAVE_MUX_STATS(tx_bytes); 19162306a36Sopenharmony_ci SAVE_MUX_STATS(rx_ready_drops); 19262306a36Sopenharmony_ci SAVE_MUX_STATS(psock_attach); 19362306a36Sopenharmony_ci SAVE_MUX_STATS(psock_unattach_rsvd); 19462306a36Sopenharmony_ci SAVE_MUX_STATS(psock_unattach); 19562306a36Sopenharmony_ci#undef SAVE_MUX_STATS 19662306a36Sopenharmony_ci} 19762306a36Sopenharmony_ci 19862306a36Sopenharmony_ci#endif /* __NET_KCM_H_ */ 199