ll_poll.h 3.4 KB
Newer Older
E
Eliezer Tamir 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32
/*
 * Low Latency Sockets
 * Copyright(c) 2013 Intel Corporation.
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms and conditions of the GNU General Public License,
 * version 2, as published by the Free Software Foundation.
 *
 * This program is distributed in the hope 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 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.
 *
 * Author: Eliezer Tamir
 *
 * Contact Information:
 * e1000-devel Mailing List <e1000-devel@lists.sourceforge.net>
 */

#ifndef _LINUX_NET_LL_POLL_H
#define _LINUX_NET_LL_POLL_H

#include <linux/netdevice.h>
#include <net/ip.h>

#ifdef CONFIG_NET_LL_RX_POLL

struct napi_struct;
33
extern unsigned int sysctl_net_ll_poll __read_mostly;
E
Eliezer Tamir 已提交
34 35 36 37 38

/* return values from ndo_ll_poll */
#define LL_FLUSH_FAILED		-1
#define LL_FLUSH_BUSY		-2

39 40 41
/* we can use sched_clock() because we don't care much about precision
 * we only care that the average is bounded
 */
42
static inline u64 ll_end_time(struct sock *sk)
E
Eliezer Tamir 已提交
43
{
44
	u64 end_time = ACCESS_ONCE(sk->sk_ll_usec);
45 46

	/* we don't mind a ~2.5% imprecision
47
	 * sk->sk_ll_usec is a u_int so this can't overflow
48 49 50 51
	 */
	end_time = (end_time << 10) + sched_clock();

	return end_time;
E
Eliezer Tamir 已提交
52 53 54 55
}

static inline bool sk_valid_ll(struct sock *sk)
{
56
	return sk->sk_ll_usec && sk->sk_napi_id &&
E
Eliezer Tamir 已提交
57 58 59
	       !need_resched() && !signal_pending(current);
}

60
static inline bool can_poll_ll(u64 end_time)
E
Eliezer Tamir 已提交
61
{
62
	return !time_after64(sched_clock(), end_time);
E
Eliezer Tamir 已提交
63 64 65 66 67
}

static inline bool sk_poll_ll(struct sock *sk, int nonblock)
{
	const struct net_device_ops *ops;
68
	u64 end_time = ll_end_time(sk);
E
Eliezer Tamir 已提交
69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120
	struct napi_struct *napi;
	int rc = false;

	/*
	 * rcu read lock for napi hash
	 * bh so we don't race with net_rx_action
	 */
	rcu_read_lock_bh();

	napi = napi_by_id(sk->sk_napi_id);
	if (!napi)
		goto out;

	ops = napi->dev->netdev_ops;
	if (!ops->ndo_ll_poll)
		goto out;

	do {

		rc = ops->ndo_ll_poll(napi);

		if (rc == LL_FLUSH_FAILED)
			break; /* permanent failure */

		if (rc > 0)
			/* local bh are disabled so it is ok to use _BH */
			NET_ADD_STATS_BH(sock_net(sk),
					 LINUX_MIB_LOWLATENCYRXPACKETS, rc);

	} while (skb_queue_empty(&sk->sk_receive_queue)
			&& can_poll_ll(end_time) && !nonblock);

	rc = !skb_queue_empty(&sk->sk_receive_queue);
out:
	rcu_read_unlock_bh();
	return rc;
}

/* used in the NIC receive handler to mark the skb */
static inline void skb_mark_ll(struct sk_buff *skb, struct napi_struct *napi)
{
	skb->napi_id = napi->napi_id;
}

/* used in the protocol hanlder to propagate the napi_id to the socket */
static inline void sk_mark_ll(struct sock *sk, struct sk_buff *skb)
{
	sk->sk_napi_id = skb->napi_id;
}

#else /* CONFIG_NET_LL_RX_POLL */

121
static inline u64 ll_end_time(struct sock *sk)
E
Eliezer Tamir 已提交
122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143
{
	return 0;
}

static inline bool sk_valid_ll(struct sock *sk)
{
	return false;
}

static inline bool sk_poll_ll(struct sock *sk, int nonblock)
{
	return false;
}

static inline void skb_mark_ll(struct sk_buff *skb, struct napi_struct *napi)
{
}

static inline void sk_mark_ll(struct sock *sk, struct sk_buff *skb)
{
}

144
static inline bool can_poll_ll(u64 end_time)
E
Eliezer Tamir 已提交
145 146 147 148 149 150
{
	return false;
}

#endif /* CONFIG_NET_LL_RX_POLL */
#endif /* _LINUX_NET_LL_POLL_H */