--- zzzz-none-000/linux-4.4.271/drivers/net/ppp/pptp.c 2021-06-03 06:22:09.000000000 +0000 +++ hawkeye-5590-750/linux-4.4.271/drivers/net/ppp/pptp.c 2023-04-19 10:22:29.000000000 +0000 @@ -53,6 +53,8 @@ static const struct ppp_channel_ops pptp_chan_ops; static const struct proto_ops pptp_ops; +static pptp_gre_seq_offload_callback_t __rcu pptp_gre_offload_xmit_cb; + #define PPP_LCP_ECHOREQ 0x09 #define PPP_LCP_ECHOREP 0x0A #define SC_RCV_BITS (SC_RCV_B7_1|SC_RCV_B7_0|SC_RCV_ODDP|SC_RCV_EVNP) @@ -129,6 +131,79 @@ return i < MAX_CALLID; } +/* Search a pptp session based on local call id, local and remote ip address */ +static int lookup_session_src(struct pptp_opt *opt, u16 call_id, __be32 daddr, __be32 saddr) +{ + struct pppox_sock *sock; + int i = 1; + + rcu_read_lock(); + for_each_set_bit_from(i, callid_bitmap, MAX_CALLID) { + sock = rcu_dereference(callid_sock[i]); + if (!sock) + continue; + + if (sock->proto.pptp.src_addr.call_id == call_id && + sock->proto.pptp.dst_addr.sin_addr.s_addr == daddr && + sock->proto.pptp.src_addr.sin_addr.s_addr == saddr) { + sock_hold(sk_pppox(sock)); + memcpy(opt, &sock->proto.pptp, sizeof(struct pptp_opt)); + sock_put(sk_pppox(sock)); + rcu_read_unlock(); + return 0; + } + } + rcu_read_unlock(); + return -EINVAL; +} + +/* Search a pptp session based on peer call id and peer ip address */ +static int lookup_session_dst(struct pptp_opt *opt, u16 call_id, __be32 d_addr) +{ + struct pppox_sock *sock; + int i = 1; + + rcu_read_lock(); + for_each_set_bit_from(i, callid_bitmap, MAX_CALLID) { + sock = rcu_dereference(callid_sock[i]); + if (!sock) + continue; + + if (sock->proto.pptp.dst_addr.call_id == call_id && + sock->proto.pptp.dst_addr.sin_addr.s_addr == d_addr) { + sock_hold(sk_pppox(sock)); + memcpy(opt, &sock->proto.pptp, sizeof(struct pptp_opt)); + sock_put(sk_pppox(sock)); + rcu_read_unlock(); + return 0; + } + } + rcu_read_unlock(); + return -EINVAL; +} + +/* If offload mode set then this function sends all packets to + * offload module instead of network stack + */ +static int pptp_client_skb_xmit(struct sk_buff *skb, + struct net_device *pptp_dev) +{ + pptp_gre_seq_offload_callback_t pptp_gre_offload_cb_f; + int ret; + + rcu_read_lock(); + pptp_gre_offload_cb_f = rcu_dereference(pptp_gre_offload_xmit_cb); + + if (!pptp_gre_offload_cb_f) { + rcu_read_unlock(); + return -1; + } + + ret = pptp_gre_offload_cb_f(skb, pptp_dev); + rcu_read_unlock(); + return ret; +} + static int add_chan(struct pppox_sock *sock, struct pptp_addr *sa) { @@ -182,11 +257,13 @@ unsigned char *data; __u32 seq_recv; - struct rtable *rt; struct net_device *tdev; + struct net_device *pptp_dev; struct iphdr *iph; int max_headroom; + int pptp_ifindex; + int ret; if (sk_pppox(po)->sk_state & PPPOX_DEAD) goto tx_error; @@ -286,9 +363,33 @@ ip_select_ident(net, skb, NULL); ip_send_check(iph); - ip_local_out(net, skb->sk, skb); - return 1; + pptp_ifindex = ppp_dev_index(chan); + + /* set incoming interface as the ppp interface */ + if (skb->skb_iif) + skb->skb_iif = pptp_ifindex; + + /* If the PPTP GRE seq number offload module is not enabled yet + * then sends all PPTP GRE packets through linux network stack + */ + if (!opt->pptp_offload_mode) { + ip_local_out(net, skb->sk, skb); + return 1; + } + + pptp_dev = dev_get_by_index(&init_net, pptp_ifindex); + if (!pptp_dev) + goto tx_error; + /* If PPTP offload module is enabled then forward all PPTP GRE + * packets to PPTP GRE offload module + */ + ret = pptp_client_skb_xmit(skb, pptp_dev); + dev_put(pptp_dev); + if (ret < 0) + goto tx_error; + + return 1; tx_error: kfree_skb(skb); return 1; @@ -344,6 +445,13 @@ goto drop; payload = skb->data + headersize; + + /* If offload is enabled, we expect the offload module + * to handle PPTP GRE sequence number checks + */ + if (opt->pptp_offload_mode) + goto allow_packet; + /* check for expected sequence number */ if (seq < opt->seq_recv + 1 || WRAPPED(opt->seq_recv, seq)) { if ((payload[0] == PPP_ALLSTATIONS) && (payload[1] == PPP_UI) && @@ -406,6 +514,7 @@ if (po) { skb_dst_drop(skb); nf_reset(skb); + skb->skb_iif = ppp_dev_index(&po->chan); return sk_receive_skb(sk_pppox(po), skb, 0); } drop: @@ -512,7 +621,7 @@ opt->dst_addr = sp->sa_addr.pptp; sk->sk_state |= PPPOX_CONNECTED; - + opt->pptp_offload_mode = false; end: release_sock(sk); return error; @@ -645,9 +754,169 @@ return err; } +/* pptp_channel_addressing_get() + * Return PPTP channel specific addressing information. + */ +void pptp_channel_addressing_get(struct pptp_opt *opt, struct ppp_channel *chan) +{ + struct sock *sk; + struct pppox_sock *po; + + if (!opt) + return; + + sk = (struct sock *)chan->private; + if (!sk) + return; + + sock_hold(sk); + + /* This is very unlikely, but check the socket is connected state */ + if (unlikely(sock_flag(sk, SOCK_DEAD) || + !(sk->sk_state & PPPOX_CONNECTED))) { + sock_put(sk); + return; + } + + po = pppox_sk(sk); + memcpy(opt, &po->proto.pptp, sizeof(struct pptp_opt)); + sock_put(sk); +} +EXPORT_SYMBOL(pptp_channel_addressing_get); + +/* pptp_session_find() + * Search and return a PPTP session info based on peer callid and IP + * address. The function accepts the parameters in network byte order. + */ +int pptp_session_find(struct pptp_opt *opt, __be16 peer_call_id, + __be32 peer_ip_addr) +{ + if (!opt) + return -EINVAL; + + return lookup_session_dst(opt, ntohs(peer_call_id), peer_ip_addr); +} +EXPORT_SYMBOL(pptp_session_find); + +/* pptp_session_find_by_src_callid() + * Search and return a PPTP session info based on src callid and IP + * address. The function accepts the parameters in network byte order. + */ +int pptp_session_find_by_src_callid(struct pptp_opt *opt, __be16 src_call_id, + __be32 daddr, __be32 saddr) +{ + if (!opt) + return -EINVAL; + + return lookup_session_src(opt, ntohs(src_call_id), daddr, saddr); +} +EXPORT_SYMBOL(pptp_session_find_by_src_callid); + + /* Function to change the offload mode true/false for a PPTP session */ +static int pptp_set_offload_mode(bool accel_mode, + __be16 peer_call_id, __be32 peer_ip_addr) +{ + struct pppox_sock *sock; + int i = 1; + + rcu_read_lock(); + for_each_set_bit_from(i, callid_bitmap, MAX_CALLID) { + sock = rcu_dereference(callid_sock[i]); + if (!sock) + continue; + + if (sock->proto.pptp.dst_addr.call_id == peer_call_id && + sock->proto.pptp.dst_addr.sin_addr.s_addr == peer_ip_addr) { + sock_hold(sk_pppox(sock)); + sock->proto.pptp.pptp_offload_mode = accel_mode; + sock_put(sk_pppox(sock)); + rcu_read_unlock(); + return 0; + } + } + rcu_read_unlock(); + return -EINVAL; +} + +/* Enable the PPTP session offload flag */ +int pptp_session_enable_offload_mode(__be16 peer_call_id, __be32 peer_ip_addr) +{ + return pptp_set_offload_mode(true, peer_call_id, peer_ip_addr); +} +EXPORT_SYMBOL(pptp_session_enable_offload_mode); + +/* Disable the PPTP session offload flag */ +int pptp_session_disable_offload_mode(__be16 peer_call_id, __be32 peer_ip_addr) +{ + return pptp_set_offload_mode(false, peer_call_id, peer_ip_addr); +} +EXPORT_SYMBOL(pptp_session_disable_offload_mode); + +/* Register the offload callback function on behalf of the module which + * will own the sequence and acknowledgment number updates for all + * PPTP GRE packets. All PPTP GRE packets are then transmitted to this + * module after encapsulation in order to ensure the correct seq/ack + * fields are set in the packets before transmission. This is required + * when PPTP flows are offloaded to acceleration engines, in-order to + * ensure consistency in sequence and ack numbers between PPTP control + * (PPP LCP) and data packets + */ +int pptp_register_gre_seq_offload_callback(pptp_gre_seq_offload_callback_t + pptp_gre_offload_cb) +{ + pptp_gre_seq_offload_callback_t pptp_gre_offload_cb_f; + + rcu_read_lock(); + pptp_gre_offload_cb_f = rcu_dereference(pptp_gre_offload_xmit_cb); + + if (pptp_gre_offload_cb_f) { + rcu_read_unlock(); + return -1; + } + + rcu_assign_pointer(pptp_gre_offload_xmit_cb, pptp_gre_offload_cb); + rcu_read_unlock(); + return 0; +} +EXPORT_SYMBOL(pptp_register_gre_seq_offload_callback); + +/* Unregister the PPTP GRE packets sequence number offload callback */ +void pptp_unregister_gre_seq_offload_callback(void) +{ + rcu_assign_pointer(pptp_gre_offload_xmit_cb, NULL); +} +EXPORT_SYMBOL(pptp_unregister_gre_seq_offload_callback); + +/* pptp_hold_chan() */ +static void pptp_hold_chan(struct ppp_channel *chan) +{ + struct sock *sk = (struct sock *)chan->private; + + sock_hold(sk); +} + +/* pptp_release_chan() */ +static void pptp_release_chan(struct ppp_channel *chan) +{ + struct sock *sk = (struct sock *)chan->private; + + sock_put(sk); +} + +/* pptp_get_channel_protocol() + * Return the protocol type of the PPTP over PPP protocol + */ +static int pptp_get_channel_protocol(struct ppp_channel *chan) +{ + return PX_PROTO_PPTP; +} + static const struct ppp_channel_ops pptp_chan_ops = { .start_xmit = pptp_xmit, .ioctl = pptp_ppp_ioctl, + .get_channel_protocol = pptp_get_channel_protocol, + .hold = pptp_hold_chan, + .release = pptp_release_chan, }; static struct proto pptp_sk_proto __read_mostly = {