Commit 86f1742b authored by Ilya Dryomov's avatar Ilya Dryomov Committed by Sage Weil

libceph: fixup error handling in osdmap_apply_incremental()

The existing error handling scheme requires resetting err to -EINVAL
prior to calling any ceph_decode_* macro.  This is ugly and fragile,
and there already are a few places where we would return 0 on error,
due to a missing reset.  Follow osdmap_decode() and fix this by adding
a special e_inval label to be used by all ceph_decode_* macros.
Signed-off-by: default avatarIlya Dryomov <ilya.dryomov@inktank.com>
Reviewed-by: default avatarAlex Elder <elder@linaro.org>
parent 9902e682
...@@ -867,19 +867,19 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end, ...@@ -867,19 +867,19 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end,
__s64 new_pool_max; __s64 new_pool_max;
__s32 new_flags, max; __s32 new_flags, max;
void *start = *p; void *start = *p;
int err = -EINVAL; int err;
u16 version; u16 version;
dout("%s %p to %p len %d\n", __func__, *p, end, (int)(end - *p)); dout("%s %p to %p len %d\n", __func__, *p, end, (int)(end - *p));
ceph_decode_16_safe(p, end, version, bad); ceph_decode_16_safe(p, end, version, e_inval);
if (version != 6) { if (version != 6) {
pr_warning("got unknown v %d != 6 of inc osdmap\n", version); pr_warning("got unknown v %d != 6 of inc osdmap\n", version);
goto bad; goto e_inval;
} }
ceph_decode_need(p, end, sizeof(fsid)+sizeof(modified)+2*sizeof(u32), ceph_decode_need(p, end, sizeof(fsid)+sizeof(modified)+2*sizeof(u32),
bad); e_inval);
ceph_decode_copy(p, &fsid, sizeof(fsid)); ceph_decode_copy(p, &fsid, sizeof(fsid));
epoch = ceph_decode_32(p); epoch = ceph_decode_32(p);
BUG_ON(epoch != map->epoch+1); BUG_ON(epoch != map->epoch+1);
...@@ -888,7 +888,7 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end, ...@@ -888,7 +888,7 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end,
new_flags = ceph_decode_32(p); new_flags = ceph_decode_32(p);
/* full map? */ /* full map? */
ceph_decode_32_safe(p, end, len, bad); ceph_decode_32_safe(p, end, len, e_inval);
if (len > 0) { if (len > 0) {
dout("apply_incremental full map len %d, %p to %p\n", dout("apply_incremental full map len %d, %p to %p\n",
len, *p, end); len, *p, end);
...@@ -896,13 +896,14 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end, ...@@ -896,13 +896,14 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end,
} }
/* new crush? */ /* new crush? */
ceph_decode_32_safe(p, end, len, bad); ceph_decode_32_safe(p, end, len, e_inval);
if (len > 0) { if (len > 0) {
dout("apply_incremental new crush map len %d, %p to %p\n",
len, *p, end);
newcrush = crush_decode(*p, min(*p+len, end)); newcrush = crush_decode(*p, min(*p+len, end));
if (IS_ERR(newcrush)) if (IS_ERR(newcrush)) {
return ERR_CAST(newcrush); err = PTR_ERR(newcrush);
newcrush = NULL;
goto bad;
}
*p += len; *p += len;
} }
...@@ -912,13 +913,13 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end, ...@@ -912,13 +913,13 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end,
if (new_pool_max >= 0) if (new_pool_max >= 0)
map->pool_max = new_pool_max; map->pool_max = new_pool_max;
ceph_decode_need(p, end, 5*sizeof(u32), bad); ceph_decode_need(p, end, 5*sizeof(u32), e_inval);
/* new max? */ /* new max? */
max = ceph_decode_32(p); max = ceph_decode_32(p);
if (max >= 0) { if (max >= 0) {
err = osdmap_set_max_osd(map, max); err = osdmap_set_max_osd(map, max);
if (err < 0) if (err)
goto bad; goto bad;
} }
...@@ -932,11 +933,11 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end, ...@@ -932,11 +933,11 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end,
} }
/* new_pool */ /* new_pool */
ceph_decode_32_safe(p, end, len, bad); ceph_decode_32_safe(p, end, len, e_inval);
while (len--) { while (len--) {
struct ceph_pg_pool_info *pi; struct ceph_pg_pool_info *pi;
ceph_decode_64_safe(p, end, pool, bad); ceph_decode_64_safe(p, end, pool, e_inval);
pi = __lookup_pg_pool(&map->pg_pools, pool); pi = __lookup_pg_pool(&map->pg_pools, pool);
if (!pi) { if (!pi) {
pi = kzalloc(sizeof(*pi), GFP_NOFS); pi = kzalloc(sizeof(*pi), GFP_NOFS);
...@@ -953,29 +954,28 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end, ...@@ -953,29 +954,28 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end,
} }
if (version >= 5) { if (version >= 5) {
err = __decode_pool_names(p, end, map); err = __decode_pool_names(p, end, map);
if (err < 0) if (err)
goto bad; goto bad;
} }
/* old_pool */ /* old_pool */
ceph_decode_32_safe(p, end, len, bad); ceph_decode_32_safe(p, end, len, e_inval);
while (len--) { while (len--) {
struct ceph_pg_pool_info *pi; struct ceph_pg_pool_info *pi;
ceph_decode_64_safe(p, end, pool, bad); ceph_decode_64_safe(p, end, pool, e_inval);
pi = __lookup_pg_pool(&map->pg_pools, pool); pi = __lookup_pg_pool(&map->pg_pools, pool);
if (pi) if (pi)
__remove_pg_pool(&map->pg_pools, pi); __remove_pg_pool(&map->pg_pools, pi);
} }
/* new_up */ /* new_up */
err = -EINVAL; ceph_decode_32_safe(p, end, len, e_inval);
ceph_decode_32_safe(p, end, len, bad);
while (len--) { while (len--) {
u32 osd; u32 osd;
struct ceph_entity_addr addr; struct ceph_entity_addr addr;
ceph_decode_32_safe(p, end, osd, bad); ceph_decode_32_safe(p, end, osd, e_inval);
ceph_decode_copy_safe(p, end, &addr, sizeof(addr), bad); ceph_decode_copy_safe(p, end, &addr, sizeof(addr), e_inval);
ceph_decode_addr(&addr); ceph_decode_addr(&addr);
pr_info("osd%d up\n", osd); pr_info("osd%d up\n", osd);
BUG_ON(osd >= map->max_osd); BUG_ON(osd >= map->max_osd);
...@@ -984,11 +984,11 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end, ...@@ -984,11 +984,11 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end,
} }
/* new_state */ /* new_state */
ceph_decode_32_safe(p, end, len, bad); ceph_decode_32_safe(p, end, len, e_inval);
while (len--) { while (len--) {
u32 osd; u32 osd;
u8 xorstate; u8 xorstate;
ceph_decode_32_safe(p, end, osd, bad); ceph_decode_32_safe(p, end, osd, e_inval);
xorstate = **(u8 **)p; xorstate = **(u8 **)p;
(*p)++; /* clean flag */ (*p)++; /* clean flag */
if (xorstate == 0) if (xorstate == 0)
...@@ -1000,10 +1000,10 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end, ...@@ -1000,10 +1000,10 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end,
} }
/* new_weight */ /* new_weight */
ceph_decode_32_safe(p, end, len, bad); ceph_decode_32_safe(p, end, len, e_inval);
while (len--) { while (len--) {
u32 osd, off; u32 osd, off;
ceph_decode_need(p, end, sizeof(u32)*2, bad); ceph_decode_need(p, end, sizeof(u32)*2, e_inval);
osd = ceph_decode_32(p); osd = ceph_decode_32(p);
off = ceph_decode_32(p); off = ceph_decode_32(p);
pr_info("osd%d weight 0x%x %s\n", osd, off, pr_info("osd%d weight 0x%x %s\n", osd, off,
...@@ -1014,7 +1014,7 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end, ...@@ -1014,7 +1014,7 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end,
} }
/* new_pg_temp */ /* new_pg_temp */
ceph_decode_32_safe(p, end, len, bad); ceph_decode_32_safe(p, end, len, e_inval);
while (len--) { while (len--) {
struct ceph_pg_mapping *pg; struct ceph_pg_mapping *pg;
int j; int j;
...@@ -1024,22 +1024,22 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end, ...@@ -1024,22 +1024,22 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end,
err = ceph_decode_pgid(p, end, &pgid); err = ceph_decode_pgid(p, end, &pgid);
if (err) if (err)
goto bad; goto bad;
ceph_decode_need(p, end, sizeof(u32), bad); ceph_decode_need(p, end, sizeof(u32), e_inval);
pglen = ceph_decode_32(p); pglen = ceph_decode_32(p);
if (pglen) { if (pglen) {
ceph_decode_need(p, end, pglen*sizeof(u32), bad); ceph_decode_need(p, end, pglen*sizeof(u32), e_inval);
/* removing existing (if any) */ /* removing existing (if any) */
(void) __remove_pg_mapping(&map->pg_temp, pgid); (void) __remove_pg_mapping(&map->pg_temp, pgid);
/* insert */ /* insert */
err = -EINVAL;
if (pglen > (UINT_MAX - sizeof(*pg)) / sizeof(u32)) if (pglen > (UINT_MAX - sizeof(*pg)) / sizeof(u32))
goto bad; goto e_inval;
err = -ENOMEM;
pg = kmalloc(sizeof(*pg) + sizeof(u32)*pglen, GFP_NOFS); pg = kmalloc(sizeof(*pg) + sizeof(u32)*pglen, GFP_NOFS);
if (!pg) if (!pg) {
err = -ENOMEM;
goto bad; goto bad;
}
pg->pgid = pgid; pg->pgid = pgid;
pg->len = pglen; pg->len = pglen;
for (j = 0; j < pglen; j++) for (j = 0; j < pglen; j++)
...@@ -1063,6 +1063,8 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end, ...@@ -1063,6 +1063,8 @@ struct ceph_osdmap *osdmap_apply_incremental(void **p, void *end,
dout("inc osdmap epoch %d max_osd %d\n", map->epoch, map->max_osd); dout("inc osdmap epoch %d max_osd %d\n", map->epoch, map->max_osd);
return map; return map;
e_inval:
err = -EINVAL;
bad: bad:
pr_err("corrupt inc osdmap (%d) epoch %d off %d (%p of %p-%p)\n", pr_err("corrupt inc osdmap (%d) epoch %d off %d (%p of %p-%p)\n",
err, epoch, (int)(*p - start), *p, start, end); err, epoch, (int)(*p - start), *p, start, end);
......
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