ethtool: fix genlmsg_put() failure handling in ethnl_default_dumpit()
If the genlmsg_put() call in ethnl_default_dumpit() fails, we bail out
without checking if we already have some messages in current skb like we do
with ethnl_default_dump_one() failure later. Therefore if existing messages
almost fill up the buffer so that there is not enough space even for
netlink and genetlink header, we lose all prepared messages and return and
error.
Rather than duplicating the skb->len check, move the genlmsg_put(),
genlmsg_cancel() and genlmsg_end() calls into ethnl_default_dump_one().
This is also more logical as all message composition will be in
ethnl_default_dump_one() and only iteration logic will be left in
ethnl_default_dumpit().
Fixes: 728480f124
("ethtool: default handlers for GET requests")
Reported-by: Jakub Kicinski <kuba@kernel.org>
Signed-off-by: Michal Kubecek <mkubecek@suse.cz>
Reviewed-by: Jakub Kicinski <kuba@kernel.org>
Signed-off-by: David S. Miller <davem@davemloft.net>
Signed-off-by: hongrongxuan <hongrongxuan@huawei.com>
This commit is contained in:
parent
43d13a08f0
commit
e2fd4f9fe3
|
@ -370,10 +370,17 @@ err_dev:
|
||||||
}
|
}
|
||||||
|
|
||||||
static int ethnl_default_dump_one(struct sk_buff *skb, struct net_device *dev,
|
static int ethnl_default_dump_one(struct sk_buff *skb, struct net_device *dev,
|
||||||
const struct ethnl_dump_ctx *ctx)
|
const struct ethnl_dump_ctx *ctx,
|
||||||
|
struct netlink_callback *cb)
|
||||||
{
|
{
|
||||||
|
void *ehdr;
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
|
ehdr = genlmsg_put(skb, NETLINK_CB(cb->skb).portid, cb->nlh->nlmsg_seq,
|
||||||
|
ðtool_genl_family, 0, ctx->ops->reply_cmd);
|
||||||
|
if (!ehdr)
|
||||||
|
return -EMSGSIZE;
|
||||||
|
|
||||||
ethnl_init_reply_data(ctx->reply_data, ctx->ops, dev);
|
ethnl_init_reply_data(ctx->reply_data, ctx->ops, dev);
|
||||||
rtnl_lock();
|
rtnl_lock();
|
||||||
ret = ctx->ops->prepare_data(ctx->req_info, ctx->reply_data, NULL);
|
ret = ctx->ops->prepare_data(ctx->req_info, ctx->reply_data, NULL);
|
||||||
|
@ -389,6 +396,10 @@ out:
|
||||||
if (ctx->ops->cleanup_data)
|
if (ctx->ops->cleanup_data)
|
||||||
ctx->ops->cleanup_data(ctx->reply_data);
|
ctx->ops->cleanup_data(ctx->reply_data);
|
||||||
ctx->reply_data->dev = NULL;
|
ctx->reply_data->dev = NULL;
|
||||||
|
if (ret < 0)
|
||||||
|
genlmsg_cancel(skb, ehdr);
|
||||||
|
else
|
||||||
|
genlmsg_end(skb, ehdr);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -405,7 +416,6 @@ static int ethnl_default_dumpit(struct sk_buff *skb,
|
||||||
int s_idx = ctx->pos_idx;
|
int s_idx = ctx->pos_idx;
|
||||||
int h, idx = 0;
|
int h, idx = 0;
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
void *ehdr;
|
|
||||||
|
|
||||||
rtnl_lock();
|
rtnl_lock();
|
||||||
for (h = ctx->pos_hash; h < NETDEV_HASHENTRIES; h++, s_idx = 0) {
|
for (h = ctx->pos_hash; h < NETDEV_HASHENTRIES; h++, s_idx = 0) {
|
||||||
|
@ -425,26 +435,15 @@ restart_chain:
|
||||||
dev_hold(dev);
|
dev_hold(dev);
|
||||||
rtnl_unlock();
|
rtnl_unlock();
|
||||||
|
|
||||||
ehdr = genlmsg_put(skb, NETLINK_CB(cb->skb).portid,
|
ret = ethnl_default_dump_one(skb, dev, ctx, cb);
|
||||||
cb->nlh->nlmsg_seq,
|
|
||||||
ðtool_genl_family, 0,
|
|
||||||
ctx->ops->reply_cmd);
|
|
||||||
if (!ehdr) {
|
|
||||||
dev_put(dev);
|
|
||||||
ret = -EMSGSIZE;
|
|
||||||
goto out;
|
|
||||||
}
|
|
||||||
ret = ethnl_default_dump_one(skb, dev, ctx);
|
|
||||||
dev_put(dev);
|
dev_put(dev);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
genlmsg_cancel(skb, ehdr);
|
|
||||||
if (ret == -EOPNOTSUPP)
|
if (ret == -EOPNOTSUPP)
|
||||||
goto lock_and_cont;
|
goto lock_and_cont;
|
||||||
if (likely(skb->len))
|
if (likely(skb->len))
|
||||||
ret = skb->len;
|
ret = skb->len;
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
genlmsg_end(skb, ehdr);
|
|
||||||
lock_and_cont:
|
lock_and_cont:
|
||||||
rtnl_lock();
|
rtnl_lock();
|
||||||
if (net->dev_base_seq != seq) {
|
if (net->dev_base_seq != seq) {
|
||||||
|
|
Loading…
Reference in New Issue