libceph: decode CRUSH device/bucket types and names
authorIlya Dryomov <idryomov@gmail.com>
Tue, 19 May 2020 15:09:52 +0000 (17:09 +0200)
committerIlya Dryomov <idryomov@gmail.com>
Mon, 1 Jun 2020 11:22:53 +0000 (13:22 +0200)
These would be matched with the provided client location to calculate
the locality value.

Signed-off-by: Ilya Dryomov <idryomov@gmail.com>
Reviewed-by: Jeff Layton <jlayton@kernel.org>
include/linux/crush/crush.h
net/ceph/crush/crush.c
net/ceph/osdmap.c

index 38b0e4d..33c16f2 100644 (file)
@@ -301,6 +301,12 @@ struct crush_map {
 
        __u32 *choose_tries;
 #else
+       /* device/bucket type id -> type name (CrushWrapper::type_map) */
+       struct rb_root type_names;
+
+       /* device/bucket id -> name (CrushWrapper::name_map) */
+       struct rb_root names;
+
        /* CrushWrapper::choose_args */
        struct rb_root choose_args;
 #endif
@@ -342,4 +348,10 @@ struct crush_work {
        struct crush_work_bucket **work; /* Per-bucket working store */
 };
 
+#ifdef __KERNEL__
+/* osdmap.c */
+void clear_crush_names(struct rb_root *root);
+void clear_choose_args(struct crush_map *c);
+#endif
+
 #endif
index 3d70244..254ded0 100644 (file)
@@ -2,7 +2,6 @@
 #ifdef __KERNEL__
 # include <linux/slab.h>
 # include <linux/crush/crush.h>
-void clear_choose_args(struct crush_map *c);
 #else
 # include "crush_compat.h"
 # include "crush.h"
@@ -130,6 +129,8 @@ void crush_destroy(struct crush_map *map)
 #ifndef __KERNEL__
        kfree(map->choose_tries);
 #else
+       clear_crush_names(&map->type_names);
+       clear_crush_names(&map->names);
        clear_choose_args(map);
 #endif
        kfree(map);
index 5d00ce2..e741308 100644 (file)
@@ -138,6 +138,79 @@ bad:
        return -EINVAL;
 }
 
+struct crush_name_node {
+       struct rb_node cn_node;
+       int cn_id;
+       char cn_name[];
+};
+
+static struct crush_name_node *alloc_crush_name(size_t name_len)
+{
+       struct crush_name_node *cn;
+
+       cn = kmalloc(sizeof(*cn) + name_len + 1, GFP_NOIO);
+       if (!cn)
+               return NULL;
+
+       RB_CLEAR_NODE(&cn->cn_node);
+       return cn;
+}
+
+static void free_crush_name(struct crush_name_node *cn)
+{
+       WARN_ON(!RB_EMPTY_NODE(&cn->cn_node));
+
+       kfree(cn);
+}
+
+DEFINE_RB_FUNCS(crush_name, struct crush_name_node, cn_id, cn_node)
+
+static int decode_crush_names(void **p, void *end, struct rb_root *root)
+{
+       u32 n;
+
+       ceph_decode_32_safe(p, end, n, e_inval);
+       while (n--) {
+               struct crush_name_node *cn;
+               int id;
+               u32 name_len;
+
+               ceph_decode_32_safe(p, end, id, e_inval);
+               ceph_decode_32_safe(p, end, name_len, e_inval);
+               ceph_decode_need(p, end, name_len, e_inval);
+
+               cn = alloc_crush_name(name_len);
+               if (!cn)
+                       return -ENOMEM;
+
+               cn->cn_id = id;
+               memcpy(cn->cn_name, *p, name_len);
+               cn->cn_name[name_len] = '\0';
+               *p += name_len;
+
+               if (!__insert_crush_name(root, cn)) {
+                       free_crush_name(cn);
+                       return -EEXIST;
+               }
+       }
+
+       return 0;
+
+e_inval:
+       return -EINVAL;
+}
+
+void clear_crush_names(struct rb_root *root)
+{
+       while (!RB_EMPTY_ROOT(root)) {
+               struct crush_name_node *cn =
+                   rb_entry(rb_first(root), struct crush_name_node, cn_node);
+
+               erase_crush_name(root, cn);
+               free_crush_name(cn);
+       }
+}
+
 static struct crush_choose_arg_map *alloc_choose_arg_map(void)
 {
        struct crush_choose_arg_map *arg_map;
@@ -354,6 +427,8 @@ static struct crush_map *crush_decode(void *pbyval, void *end)
        if (c == NULL)
                return ERR_PTR(-ENOMEM);
 
+       c->type_names = RB_ROOT;
+       c->names = RB_ROOT;
        c->choose_args = RB_ROOT;
 
         /* set tunables to default values */
@@ -510,8 +585,14 @@ static struct crush_map *crush_decode(void *pbyval, void *end)
                }
        }
 
-       ceph_decode_skip_map(p, end, 32, string, bad); /* type_map */
-       ceph_decode_skip_map(p, end, 32, string, bad); /* name_map */
+       err = decode_crush_names(p, end, &c->type_names);
+       if (err)
+               goto fail;
+
+       err = decode_crush_names(p, end, &c->names);
+       if (err)
+               goto fail;
+
        ceph_decode_skip_map(p, end, 32, string, bad); /* rule_name_map */
 
         /* tunables */