Commit 1ff22883 authored by David Howells's avatar David Howells Committed by Greg Kroah-Hartman

fscache: Fix incomplete initialisation of inline key space

The inline key in struct rxrpc_cookie is insufficiently initialized,
zeroing only 3 of the 4 slots, therefore an index_key_len between 13 and 15
bytes will end up hashing uninitialized memory because the memcpy only
partially fills the last buf[] element.

Fix this by clearing fscache_cookie objects on allocation rather than using
the slab constructor to initialise them.  We're going to pretty much fill
in the entire struct anyway, so bringing it into our dcache writably
shouldn't incur much overhead.

This removes the need to do clearance in fscache_set_key() (where we aren't
doing it correctly anyway).

Also, we don't need to set cookie->key_len in fscache_set_key() as we
already did it in the only caller, so remove that.

Fixes: ec0328e4 ("fscache: Maintain a catalogue of allocated cookies")
Reported-by: syzbot+a95b989b2dde8e806af8@syzkaller.appspotmail.com
Reported-by: default avatarEric Sandeen <sandeen@redhat.com>
Cc: stable <stable@vger.kernel.org>
Signed-off-by: default avatarDavid Howells <dhowells@redhat.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 169b8033
...@@ -69,19 +69,6 @@ void fscache_free_cookie(struct fscache_cookie *cookie) ...@@ -69,19 +69,6 @@ void fscache_free_cookie(struct fscache_cookie *cookie)
} }
} }
/*
* initialise an cookie jar slab element prior to any use
*/
void fscache_cookie_init_once(void *_cookie)
{
struct fscache_cookie *cookie = _cookie;
memset(cookie, 0, sizeof(*cookie));
spin_lock_init(&cookie->lock);
spin_lock_init(&cookie->stores_lock);
INIT_HLIST_HEAD(&cookie->backing_objects);
}
/* /*
* Set the index key in a cookie. The cookie struct has space for a 12-byte * Set the index key in a cookie. The cookie struct has space for a 12-byte
* key plus length and hash, but if that's not big enough, it's instead a * key plus length and hash, but if that's not big enough, it's instead a
...@@ -95,8 +82,6 @@ static int fscache_set_key(struct fscache_cookie *cookie, ...@@ -95,8 +82,6 @@ static int fscache_set_key(struct fscache_cookie *cookie,
u32 *buf; u32 *buf;
int i; int i;
cookie->key_len = index_key_len;
if (index_key_len > sizeof(cookie->inline_key)) { if (index_key_len > sizeof(cookie->inline_key)) {
buf = kzalloc(index_key_len, GFP_KERNEL); buf = kzalloc(index_key_len, GFP_KERNEL);
if (!buf) if (!buf)
...@@ -104,9 +89,6 @@ static int fscache_set_key(struct fscache_cookie *cookie, ...@@ -104,9 +89,6 @@ static int fscache_set_key(struct fscache_cookie *cookie,
cookie->key = buf; cookie->key = buf;
} else { } else {
buf = (u32 *)cookie->inline_key; buf = (u32 *)cookie->inline_key;
buf[0] = 0;
buf[1] = 0;
buf[2] = 0;
} }
memcpy(buf, index_key, index_key_len); memcpy(buf, index_key, index_key_len);
...@@ -161,7 +143,7 @@ struct fscache_cookie *fscache_alloc_cookie( ...@@ -161,7 +143,7 @@ struct fscache_cookie *fscache_alloc_cookie(
struct fscache_cookie *cookie; struct fscache_cookie *cookie;
/* allocate and initialise a cookie */ /* allocate and initialise a cookie */
cookie = kmem_cache_alloc(fscache_cookie_jar, GFP_KERNEL); cookie = kmem_cache_zalloc(fscache_cookie_jar, GFP_KERNEL);
if (!cookie) if (!cookie)
return NULL; return NULL;
...@@ -192,6 +174,9 @@ struct fscache_cookie *fscache_alloc_cookie( ...@@ -192,6 +174,9 @@ struct fscache_cookie *fscache_alloc_cookie(
cookie->netfs_data = netfs_data; cookie->netfs_data = netfs_data;
cookie->flags = (1 << FSCACHE_COOKIE_NO_DATA_YET); cookie->flags = (1 << FSCACHE_COOKIE_NO_DATA_YET);
cookie->type = def->type; cookie->type = def->type;
spin_lock_init(&cookie->lock);
spin_lock_init(&cookie->stores_lock);
INIT_HLIST_HEAD(&cookie->backing_objects);
/* radix tree insertion won't use the preallocation pool unless it's /* radix tree insertion won't use the preallocation pool unless it's
* told it may not wait */ * told it may not wait */
......
...@@ -51,7 +51,6 @@ extern struct fscache_cache *fscache_select_cache_for_object( ...@@ -51,7 +51,6 @@ extern struct fscache_cache *fscache_select_cache_for_object(
extern struct kmem_cache *fscache_cookie_jar; extern struct kmem_cache *fscache_cookie_jar;
extern void fscache_free_cookie(struct fscache_cookie *); extern void fscache_free_cookie(struct fscache_cookie *);
extern void fscache_cookie_init_once(void *);
extern struct fscache_cookie *fscache_alloc_cookie(struct fscache_cookie *, extern struct fscache_cookie *fscache_alloc_cookie(struct fscache_cookie *,
const struct fscache_cookie_def *, const struct fscache_cookie_def *,
const void *, size_t, const void *, size_t,
......
...@@ -143,9 +143,7 @@ static int __init fscache_init(void) ...@@ -143,9 +143,7 @@ static int __init fscache_init(void)
fscache_cookie_jar = kmem_cache_create("fscache_cookie_jar", fscache_cookie_jar = kmem_cache_create("fscache_cookie_jar",
sizeof(struct fscache_cookie), sizeof(struct fscache_cookie),
0, 0, 0, NULL);
0,
fscache_cookie_init_once);
if (!fscache_cookie_jar) { if (!fscache_cookie_jar) {
pr_notice("Failed to allocate a cookie jar\n"); pr_notice("Failed to allocate a cookie jar\n");
ret = -ENOMEM; ret = -ENOMEM;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment