From: Wei Wang <weiwan@google.com>
To: "David S . Miller" <davem@davemloft.net>, netdev@vger.kernel.org
Cc: Jakub Kicinski <kuba@kernel.org>,
Eric Dumazet <edumazet@google.com>,
Paolo Abeni <pabeni@redhat.com>,
Hannes Frederic Sowa <hannes@stressinduktion.org>,
Felix Fietkau <nbd@nbd.name>, Wei Wang <weiwan@google.com>
Subject: [RFC PATCH net-next 6/6] net: improve napi threaded config
Date: Mon, 14 Sep 2020 10:24:53 -0700 [thread overview]
Message-ID: <20200914172453.1833883-7-weiwan@google.com> (raw)
In-Reply-To: <20200914172453.1833883-1-weiwan@google.com>
This commit mainly addresses the threaded config to make the switch
between softirq based and kthread based NAPI processing not require
a device down/up.
It also moves the kthread_create() call to the sysfs handler when user
tries to enable "threaded" on napi, and properly handles the
kthread_create() failure. This is because certain drivers do not have
the napi created and linked to the dev when dev_open() is called. So
the previous implementation does not work properly there.
Signed-off-by: Wei Wang <weiwan@google.com>
---
net/core/dev.c | 49 +++++++++++++++++++++++++-------------------
net/core/net-sysfs.c | 9 +++-----
2 files changed, 31 insertions(+), 27 deletions(-)
diff --git a/net/core/dev.c b/net/core/dev.c
index ab8af727058b..9f7df61f7c9a 100644
--- a/net/core/dev.c
+++ b/net/core/dev.c
@@ -1489,17 +1489,24 @@ EXPORT_SYMBOL(netdev_notify_peers);
static int napi_threaded_poll(void *data);
-static void napi_thread_start(struct napi_struct *n)
+static int napi_kthread_create(struct napi_struct *n)
{
- if (test_bit(NAPI_STATE_THREADED, &n->state) && !n->thread)
- n->thread = kthread_create(napi_threaded_poll, n, "%s-%d",
- n->dev->name, n->napi_id);
+ int err = 0;
+
+ n->thread = kthread_create(napi_threaded_poll, n, "%s-%d",
+ n->dev->name, n->napi_id);
+ if (IS_ERR(n->thread)) {
+ err = PTR_ERR(n->thread);
+ pr_err("kthread_create failed with err %d\n", err);
+ n->thread = NULL;
+ }
+
+ return err;
}
static int __dev_open(struct net_device *dev, struct netlink_ext_ack *extack)
{
const struct net_device_ops *ops = dev->netdev_ops;
- struct napi_struct *n;
int ret;
ASSERT_RTNL();
@@ -1531,9 +1538,6 @@ static int __dev_open(struct net_device *dev, struct netlink_ext_ack *extack)
if (!ret && ops->ndo_open)
ret = ops->ndo_open(dev);
- list_for_each_entry(n, &dev->napi_list, dev_list)
- napi_thread_start(n);
-
netpoll_poll_enable(dev);
if (ret)
@@ -1584,6 +1588,7 @@ static void napi_thread_stop(struct napi_struct *n)
if (!n->thread)
return;
kthread_stop(n->thread);
+ clear_bit(NAPI_STATE_THREADED, &n->state);
n->thread = NULL;
}
@@ -4266,7 +4271,7 @@ int gro_normal_batch __read_mostly = 8;
static inline void ____napi_schedule(struct softnet_data *sd,
struct napi_struct *napi)
{
- if (napi->thread) {
+ if (test_bit(NAPI_STATE_THREADED, &napi->state)) {
wake_up_process(napi->thread);
return;
}
@@ -6623,25 +6628,25 @@ static void init_gro_hash(struct napi_struct *napi)
int napi_set_threaded(struct napi_struct *n, bool threaded)
{
- ASSERT_RTNL();
+ int err = 0;
- if (n->dev->flags & IFF_UP)
- return -EBUSY;
+ ASSERT_RTNL();
if (threaded == !!test_bit(NAPI_STATE_THREADED, &n->state))
return 0;
- if (threaded)
+ if (threaded) {
+ if (!n->thread) {
+ err = napi_kthread_create(n);
+ if (err)
+ goto out;
+ }
set_bit(NAPI_STATE_THREADED, &n->state);
- else
+ } else {
clear_bit(NAPI_STATE_THREADED, &n->state);
+ }
- /* if the device is initializing, nothing todo */
- if (test_bit(__LINK_STATE_START, &n->dev->state))
- return 0;
-
- napi_thread_stop(n);
- napi_thread_start(n);
- return 0;
+out:
+ return err;
}
EXPORT_SYMBOL(napi_set_threaded);
@@ -6686,6 +6691,7 @@ void napi_disable(struct napi_struct *n)
msleep(1);
hrtimer_cancel(&n->timer);
+ napi_thread_stop(n);
clear_bit(NAPI_STATE_DISABLE, &n->state);
}
@@ -6806,6 +6812,7 @@ static int napi_thread_wait(struct napi_struct *napi)
while (!kthread_should_stop() && !napi_disable_pending(napi)) {
if (test_bit(NAPI_STATE_SCHED, &napi->state)) {
+ WARN_ON(!list_empty(&napi->poll_list));
__set_current_state(TASK_RUNNING);
return 0;
}
diff --git a/net/core/net-sysfs.c b/net/core/net-sysfs.c
index 0172457a1bfe..48b7582a0372 100644
--- a/net/core/net-sysfs.c
+++ b/net/core/net-sysfs.c
@@ -608,11 +608,6 @@ static ssize_t threaded_store(struct device *dev,
goto unlock;
}
- if (netdev->flags & IFF_UP) {
- ret = -EBUSY;
- goto unlock;
- }
-
bmap = __alloc_thread_bitmap(netdev, &bits);
if (!bmap) {
ret = -ENOMEM;
@@ -625,7 +620,9 @@ static ssize_t threaded_store(struct device *dev,
i = 0;
list_for_each_entry(n, &netdev->napi_list, dev_list) {
- napi_set_threaded(n, test_bit(i, bmap));
+ ret = napi_set_threaded(n, test_bit(i, bmap));
+ if (ret)
+ goto free_unlock;
i++;
}
ret = len;
--
2.28.0.618.gf4bc123cb7-goog
next prev parent reply other threads:[~2020-09-14 17:28 UTC|newest]
Thread overview: 32+ messages / expand[flat|nested] mbox.gz Atom feed top
2020-09-14 17:24 [RFC PATCH net-next 0/6] implement kthread based napi poll Wei Wang
2020-09-14 17:24 ` [RFC PATCH net-next 1/6] net: implement threaded-able napi poll loop support Wei Wang
2020-09-25 19:45 ` Hannes Frederic Sowa
2020-09-25 23:50 ` Wei Wang
2020-09-26 14:22 ` Hannes Frederic Sowa
2020-09-28 8:45 ` Paolo Abeni
2020-09-28 18:13 ` Wei Wang
2020-09-14 17:24 ` [RFC PATCH net-next 2/6] net: add sysfs attribute to control napi threaded mode Wei Wang
2020-09-15 2:50 ` kernel test robot
2020-09-15 3:47 ` kernel test robot
2020-09-14 17:24 ` [RFC PATCH net-next 3/6] net: extract napi poll functionality to __napi_poll() Wei Wang
2020-09-14 17:24 ` [RFC PATCH net-next 4/6] net: modify kthread handler to use __napi_poll() Wei Wang
2020-09-14 17:24 ` [RFC PATCH net-next 5/6] net: process RPS/RFS work in kthread context Wei Wang
2020-09-18 22:44 ` Wei Wang
2020-09-21 8:11 ` Eric Dumazet
2020-09-14 17:24 ` Wei Wang [this message]
2020-09-25 13:48 ` [RFC PATCH net-next 0/6] implement kthread based napi poll Magnus Karlsson
2020-09-25 17:15 ` Wei Wang
2020-09-25 17:30 ` Eric Dumazet
2020-09-25 18:16 ` Stephen Hemminger
2020-09-25 18:23 ` Eric Dumazet
2020-09-25 19:00 ` Stephen Hemminger
2020-09-25 19:06 ` Jakub Kicinski
2020-09-28 14:07 ` Magnus Karlsson
2020-09-28 17:43 ` Eric Dumazet
2020-09-28 18:15 ` Wei Wang
2020-09-29 19:19 ` Jakub Kicinski
2020-09-29 20:16 ` Wei Wang
2020-09-29 21:48 ` Jakub Kicinski
2020-09-30 8:23 ` David Laight
2020-09-30 8:58 ` Paolo Abeni
2020-09-30 15:58 ` Jakub Kicinski
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20200914172453.1833883-7-weiwan@google.com \
--to=weiwan@google.com \
--cc=davem@davemloft.net \
--cc=edumazet@google.com \
--cc=hannes@stressinduktion.org \
--cc=kuba@kernel.org \
--cc=nbd@nbd.name \
--cc=netdev@vger.kernel.org \
--cc=pabeni@redhat.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.