提交 593ceb0c 编写于 作者: D David Fries 提交者: Greg Kroah-Hartman

w1: fix netlink refcnt leak on error path

If the message type is W1_MASTER_CMD or W1_SLAVE_CMD, then a reference
is taken when searching for the slave or master device.  If there
isn't any following data m->len (mlen is a copy) is 0 and packing up
the message for later execution is skipped leaving nothing to
decrement the reference counts.

Way back when, m->len was checked before the search that increments the
reference count, but W1_LIST_MASTERS has no additional data, the check
was moved in 9be62e0b causing this bug.

This change reorders to put the check before the reference count is
incremented avoiding the problem.
Signed-off-by: NDavid Fries <David@Fries.net>
Acked-by: NEvgeniy Polyakov <zbr@ioremap.net>
Signed-off-by: NGreg Kroah-Hartman <gregkh@linuxfoundation.org>
上级 a82cb8b9
...@@ -300,12 +300,6 @@ static int w1_process_command_root(struct cn_msg *msg, ...@@ -300,12 +300,6 @@ static int w1_process_command_root(struct cn_msg *msg,
struct w1_netlink_msg *w; struct w1_netlink_msg *w;
u32 *id; u32 *id;
if (mcmd->type != W1_LIST_MASTERS) {
printk(KERN_NOTICE "%s: msg: %x.%x, wrong type: %u, len: %u.\n",
__func__, msg->id.idx, msg->id.val, mcmd->type, mcmd->len);
return -EPROTO;
}
cn = kmalloc(PAGE_SIZE, GFP_KERNEL); cn = kmalloc(PAGE_SIZE, GFP_KERNEL);
if (!cn) if (!cn)
return -ENOMEM; return -ENOMEM;
...@@ -441,6 +435,9 @@ static void w1_process_cb(struct w1_master *dev, struct w1_async_cmd *async_cmd) ...@@ -441,6 +435,9 @@ static void w1_process_cb(struct w1_master *dev, struct w1_async_cmd *async_cmd)
w1_netlink_send_error(&node->block->msg, node->m, cmd, w1_netlink_send_error(&node->block->msg, node->m, cmd,
node->block->portid, err); node->block->portid, err);
/* ref taken in w1_search_slave or w1_search_master_id when building
* the block
*/
if (sl) if (sl)
w1_unref_slave(sl); w1_unref_slave(sl);
else else
...@@ -503,30 +500,42 @@ static void w1_cn_callback(struct cn_msg *msg, struct netlink_skb_parms *nsp) ...@@ -503,30 +500,42 @@ static void w1_cn_callback(struct cn_msg *msg, struct netlink_skb_parms *nsp)
msg_len = msg->len; msg_len = msg->len;
while (msg_len && !err) { while (msg_len && !err) {
struct w1_reg_num id;
u16 mlen = m->len;
dev = NULL; dev = NULL;
sl = NULL; sl = NULL;
memcpy(&id, m->id.id, sizeof(id));
#if 0
printk("%s: %02x.%012llx.%02x: type=%02x, len=%u.\n",
__func__, id.family, (unsigned long long)id.id, id.crc, m->type, m->len);
#endif
if (m->len + sizeof(struct w1_netlink_msg) > msg_len) { if (m->len + sizeof(struct w1_netlink_msg) > msg_len) {
err = -E2BIG; err = -E2BIG;
break; break;
} }
/* execute on this thread, no need to process later */
if (m->type == W1_LIST_MASTERS) {
err = w1_process_command_root(msg, m, nsp->portid);
goto out_cont;
}
/* All following message types require additional data,
* check here before references are taken.
*/
if (!m->len) {
err = -EPROTO;
goto out_cont;
}
/* both search calls take reference counts */
if (m->type == W1_MASTER_CMD) { if (m->type == W1_MASTER_CMD) {
dev = w1_search_master_id(m->id.mst.id); dev = w1_search_master_id(m->id.mst.id);
} else if (m->type == W1_SLAVE_CMD) { } else if (m->type == W1_SLAVE_CMD) {
sl = w1_search_slave(&id); sl = w1_search_slave((struct w1_reg_num *)m->id.id);
if (sl) if (sl)
dev = sl->master; dev = sl->master;
} else { } else {
err = w1_process_command_root(msg, m, nsp->portid); printk(KERN_NOTICE
"%s: msg: %x.%x, wrong type: %u, len: %u.\n",
__func__, msg->id.idx, msg->id.val,
m->type, m->len);
err = -EPROTO;
goto out_cont; goto out_cont;
} }
...@@ -536,8 +545,6 @@ static void w1_cn_callback(struct cn_msg *msg, struct netlink_skb_parms *nsp) ...@@ -536,8 +545,6 @@ static void w1_cn_callback(struct cn_msg *msg, struct netlink_skb_parms *nsp)
} }
err = 0; err = 0;
if (!mlen)
goto out_cont;
atomic_inc(&block->refcnt); atomic_inc(&block->refcnt);
node->async.cb = w1_process_cb; node->async.cb = w1_process_cb;
...@@ -557,7 +564,8 @@ static void w1_cn_callback(struct cn_msg *msg, struct netlink_skb_parms *nsp) ...@@ -557,7 +564,8 @@ static void w1_cn_callback(struct cn_msg *msg, struct netlink_skb_parms *nsp)
if (err) if (err)
w1_netlink_send_error(msg, m, NULL, nsp->portid, err); w1_netlink_send_error(msg, m, NULL, nsp->portid, err);
msg_len -= sizeof(struct w1_netlink_msg) + m->len; msg_len -= sizeof(struct w1_netlink_msg) + m->len;
m = (struct w1_netlink_msg *)(((u8 *)m) + sizeof(struct w1_netlink_msg) + m->len); m = (struct w1_netlink_msg *)(((u8 *)m) +
sizeof(struct w1_netlink_msg) + m->len);
/* /*
* Let's allow requests for nonexisting devices. * Let's allow requests for nonexisting devices.
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册