aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/infiniband/core
diff options
context:
space:
mode:
authorIgor Ivanov <Igor.Ivanov@itseez.com>2013-08-14 06:58:29 -0400
committerRoland Dreier <roland@purestorage.com>2013-08-28 12:52:03 -0400
commit400dbc96583ff3b8ad4c09bd7e9dcd35a6215922 (patch)
treed3ca0674c024dead92e3d50302dde73c09c3005d /drivers/infiniband/core
parent319a441d1361ea703b091caf92418f8121eadfc5 (diff)
IB/core: Infrastructure for extensible uverbs commands
Add infrastructure to support extended uverbs capabilities in a forward/backward manner. Uverbs command opcodes which are based on the verbs extensions approach should be greater or equal to IB_USER_VERBS_CMD_THRESHOLD. They have new header format and processed a bit differently. Whenever a specific IB_USER_VERBS_CMD_XXX is extended, which practically means it needs to have additional arguments, we will be able to add them without creating a completely new IB_USER_VERBS_CMD_YYY command or bumping the uverbs ABI version. This patch for itself doesn't provide the whole scheme which is also dependent on adding a comp_mask field to each extended uverbs command struct. The new header framework allows for future extension of the CMD arguments (ib_uverbs_cmd_hdr.in_words, ib_uverbs_cmd_hdr.out_words) for an existing new command (that is a command that supports the new uverbs command header format suggested in this patch) w/o bumping ABI version and with maintaining backward and formward compatibility to new and old libibverbs versions. In the uverbs command we are passing both uverbs arguments and the provider arguments. We split the ib_uverbs_cmd_hdr.in_words to ib_uverbs_cmd_hdr.in_words which will now carry only uverbs input argument struct size and ib_uverbs_cmd_hdr.provider_in_words that will carry the provider input argument size. Same goes for the response (the uverbs CMD output argument). For example take the create_cq call and the mlx4_ib provider: The uverbs layer gets libibverb's struct ibv_create_cq (named struct ib_uverbs_create_cq in the kernel), mlx4_ib gets libmlx4's struct mlx4_create_cq (which includes struct ibv_create_cq and is named struct mlx4_ib_create_cq in the kernel) and in_words = sizeof(mlx4_create_cq)/4 . Thus ib_uverbs_cmd_hdr.in_words carry both uverbs plus mlx4_ib input argument sizes, where uverbs assumes it knows the size of its input argument - struct ibv_create_cq. Now, if we wish to add a variable to struct ibv_create_cq, we can add a comp_mask field to the struct which is basically bit field indicating which fields exists in the struct (as done for the libibverbs API extension), but we need a way to tell what is the total size of the struct and not assume the struct size is predefined (since we may get different struct sizes from different user libibverbs versions). So we know at which point the provider input argument (struct mlx4_create_cq) begins. Same goes for extending the provider struct mlx4_create_cq. Thus we split the ib_uverbs_cmd_hdr.in_words to ib_uverbs_cmd_hdr.in_words which will now carry only uverbs input argument struct size and ib_uverbs_cmd_hdr.provider_in_words that will carry the provider (mlx4_ib) input argument size. Signed-off-by: Igor Ivanov <Igor.Ivanov@itseez.com> Signed-off-by: Hadar Hen Zion <hadarh@mellanox.com> Signed-off-by: Or Gerlitz <ogerlitz@mellanox.com> Signed-off-by: Roland Dreier <roland@purestorage.com>
Diffstat (limited to 'drivers/infiniband/core')
-rw-r--r--drivers/infiniband/core/uverbs_main.c29
1 files changed, 24 insertions, 5 deletions
diff --git a/drivers/infiniband/core/uverbs_main.c b/drivers/infiniband/core/uverbs_main.c
index 2c6f0f2ecd9d..e4e7b2449d19 100644
--- a/drivers/infiniband/core/uverbs_main.c
+++ b/drivers/infiniband/core/uverbs_main.c
@@ -583,9 +583,6 @@ static ssize_t ib_uverbs_write(struct file *filp, const char __user *buf,
583 if (copy_from_user(&hdr, buf, sizeof hdr)) 583 if (copy_from_user(&hdr, buf, sizeof hdr))
584 return -EFAULT; 584 return -EFAULT;
585 585
586 if (hdr.in_words * 4 != count)
587 return -EINVAL;
588
589 if (hdr.command >= ARRAY_SIZE(uverbs_cmd_table) || 586 if (hdr.command >= ARRAY_SIZE(uverbs_cmd_table) ||
590 !uverbs_cmd_table[hdr.command]) 587 !uverbs_cmd_table[hdr.command])
591 return -EINVAL; 588 return -EINVAL;
@@ -597,8 +594,30 @@ static ssize_t ib_uverbs_write(struct file *filp, const char __user *buf,
597 if (!(file->device->ib_dev->uverbs_cmd_mask & (1ull << hdr.command))) 594 if (!(file->device->ib_dev->uverbs_cmd_mask & (1ull << hdr.command)))
598 return -ENOSYS; 595 return -ENOSYS;
599 596
600 return uverbs_cmd_table[hdr.command](file, buf + sizeof hdr, 597 if (hdr.command >= IB_USER_VERBS_CMD_THRESHOLD) {
601 hdr.in_words * 4, hdr.out_words * 4); 598 struct ib_uverbs_cmd_hdr_ex hdr_ex;
599
600 if (copy_from_user(&hdr_ex, buf, sizeof(hdr_ex)))
601 return -EFAULT;
602
603 if (((hdr_ex.in_words + hdr_ex.provider_in_words) * 4) != count)
604 return -EINVAL;
605
606 return uverbs_cmd_table[hdr.command](file,
607 buf + sizeof(hdr_ex),
608 (hdr_ex.in_words +
609 hdr_ex.provider_in_words) * 4,
610 (hdr_ex.out_words +
611 hdr_ex.provider_out_words) * 4);
612 } else {
613 if (hdr.in_words * 4 != count)
614 return -EINVAL;
615
616 return uverbs_cmd_table[hdr.command](file,
617 buf + sizeof(hdr),
618 hdr.in_words * 4,
619 hdr.out_words * 4);
620 }
602} 621}
603 622
604static int ib_uverbs_mmap(struct file *filp, struct vm_area_struct *vma) 623static int ib_uverbs_mmap(struct file *filp, struct vm_area_struct *vma)