Merge branches 'cxgb4', 'flowsteer', 'ipoib', 'iser', 'mlx4', 'ocrdma' and 'qib'...
[firefly-linux-kernel-4.4.55.git] / drivers / infiniband / core / uverbs_cmd.c
index b3c07b0c9f2655bb13b9fb81c6ef38fc2efd8757..f2b81b9ee0d6849c67693e50c351721c64df27fe 100644 (file)
@@ -54,6 +54,7 @@ static struct uverbs_lock_class qp_lock_class = { .name = "QP-uobj" };
 static struct uverbs_lock_class ah_lock_class  = { .name = "AH-uobj" };
 static struct uverbs_lock_class srq_lock_class = { .name = "SRQ-uobj" };
 static struct uverbs_lock_class xrcd_lock_class = { .name = "XRCD-uobj" };
+static struct uverbs_lock_class rule_lock_class = { .name = "RULE-uobj" };
 
 #define INIT_UDATA(udata, ibuf, obuf, ilen, olen)                      \
        do {                                                            \
@@ -330,6 +331,7 @@ ssize_t ib_uverbs_get_context(struct ib_uverbs_file *file,
        INIT_LIST_HEAD(&ucontext->srq_list);
        INIT_LIST_HEAD(&ucontext->ah_list);
        INIT_LIST_HEAD(&ucontext->xrcd_list);
+       INIT_LIST_HEAD(&ucontext->rule_list);
        ucontext->closing = 0;
 
        resp.num_comp_vectors = file->device->num_comp_vectors;
@@ -1526,7 +1528,7 @@ ssize_t ib_uverbs_create_qp(struct ib_uverbs_file *file,
                   (unsigned long) cmd.response + sizeof resp,
                   in_len - sizeof cmd, out_len - sizeof resp);
 
-       obj = kmalloc(sizeof *obj, GFP_KERNEL);
+       obj = kzalloc(sizeof *obj, GFP_KERNEL);
        if (!obj)
                return -ENOMEM;
 
@@ -1642,8 +1644,13 @@ ssize_t ib_uverbs_create_qp(struct ib_uverbs_file *file,
                goto err_copy;
        }
 
-       if (xrcd)
+       if (xrcd) {
+               obj->uxrcd = container_of(xrcd_uobj, struct ib_uxrcd_object,
+                                         uobject);
+               atomic_inc(&obj->uxrcd->refcnt);
                put_xrcd_read(xrcd_uobj);
+       }
+
        if (pd)
                put_pd_read(pd);
        if (scq)
@@ -1753,6 +1760,8 @@ ssize_t ib_uverbs_open_qp(struct ib_uverbs_file *file,
                goto err_remove;
        }
 
+       obj->uxrcd = container_of(xrcd_uobj, struct ib_uxrcd_object, uobject);
+       atomic_inc(&obj->uxrcd->refcnt);
        put_xrcd_read(xrcd_uobj);
 
        mutex_lock(&file->mutex);
@@ -2019,6 +2028,9 @@ ssize_t ib_uverbs_destroy_qp(struct ib_uverbs_file *file,
        if (ret)
                return ret;
 
+       if (obj->uxrcd)
+               atomic_dec(&obj->uxrcd->refcnt);
+
        idr_remove_uobj(&ib_uverbs_qp_idr, uobj);
 
        mutex_lock(&file->mutex);
@@ -2587,6 +2599,232 @@ out_put:
        return ret ? ret : in_len;
 }
 
+static int kern_spec_to_ib_spec(struct ib_kern_spec *kern_spec,
+                               union ib_flow_spec *ib_spec)
+{
+       ib_spec->type = kern_spec->type;
+
+       switch (ib_spec->type) {
+       case IB_FLOW_SPEC_ETH:
+               ib_spec->eth.size = sizeof(struct ib_flow_spec_eth);
+               if (ib_spec->eth.size != kern_spec->eth.size)
+                       return -EINVAL;
+               memcpy(&ib_spec->eth.val, &kern_spec->eth.val,
+                      sizeof(struct ib_flow_eth_filter));
+               memcpy(&ib_spec->eth.mask, &kern_spec->eth.mask,
+                      sizeof(struct ib_flow_eth_filter));
+               break;
+       case IB_FLOW_SPEC_IPV4:
+               ib_spec->ipv4.size = sizeof(struct ib_flow_spec_ipv4);
+               if (ib_spec->ipv4.size != kern_spec->ipv4.size)
+                       return -EINVAL;
+               memcpy(&ib_spec->ipv4.val, &kern_spec->ipv4.val,
+                      sizeof(struct ib_flow_ipv4_filter));
+               memcpy(&ib_spec->ipv4.mask, &kern_spec->ipv4.mask,
+                      sizeof(struct ib_flow_ipv4_filter));
+               break;
+       case IB_FLOW_SPEC_TCP:
+       case IB_FLOW_SPEC_UDP:
+               ib_spec->tcp_udp.size = sizeof(struct ib_flow_spec_tcp_udp);
+               if (ib_spec->tcp_udp.size != kern_spec->tcp_udp.size)
+                       return -EINVAL;
+               memcpy(&ib_spec->tcp_udp.val, &kern_spec->tcp_udp.val,
+                      sizeof(struct ib_flow_tcp_udp_filter));
+               memcpy(&ib_spec->tcp_udp.mask, &kern_spec->tcp_udp.mask,
+                      sizeof(struct ib_flow_tcp_udp_filter));
+               break;
+       default:
+               return -EINVAL;
+       }
+       return 0;
+}
+
+ssize_t ib_uverbs_create_flow(struct ib_uverbs_file *file,
+                             const char __user *buf, int in_len,
+                             int out_len)
+{
+       struct ib_uverbs_create_flow      cmd;
+       struct ib_uverbs_create_flow_resp resp;
+       struct ib_uobject                 *uobj;
+       struct ib_flow                    *flow_id;
+       struct ib_kern_flow_attr          *kern_flow_attr;
+       struct ib_flow_attr               *flow_attr;
+       struct ib_qp                      *qp;
+       int err = 0;
+       void *kern_spec;
+       void *ib_spec;
+       int i;
+       int kern_attr_size;
+
+       if (out_len < sizeof(resp))
+               return -ENOSPC;
+
+       if (copy_from_user(&cmd, buf, sizeof(cmd)))
+               return -EFAULT;
+
+       if (cmd.comp_mask)
+               return -EINVAL;
+
+       if ((cmd.flow_attr.type == IB_FLOW_ATTR_SNIFFER &&
+            !capable(CAP_NET_ADMIN)) || !capable(CAP_NET_RAW))
+               return -EPERM;
+
+       if (cmd.flow_attr.num_of_specs < 0 ||
+           cmd.flow_attr.num_of_specs > IB_FLOW_SPEC_SUPPORT_LAYERS)
+               return -EINVAL;
+
+       kern_attr_size = cmd.flow_attr.size - sizeof(cmd) -
+                        sizeof(struct ib_uverbs_cmd_hdr_ex);
+
+       if (cmd.flow_attr.size < 0 || cmd.flow_attr.size > in_len ||
+           kern_attr_size < 0 || kern_attr_size >
+           (cmd.flow_attr.num_of_specs * sizeof(struct ib_kern_spec)))
+               return -EINVAL;
+
+       if (cmd.flow_attr.num_of_specs) {
+               kern_flow_attr = kmalloc(cmd.flow_attr.size, GFP_KERNEL);
+               if (!kern_flow_attr)
+                       return -ENOMEM;
+
+               memcpy(kern_flow_attr, &cmd.flow_attr, sizeof(*kern_flow_attr));
+               if (copy_from_user(kern_flow_attr + 1, buf + sizeof(cmd),
+                                  kern_attr_size)) {
+                       err = -EFAULT;
+                       goto err_free_attr;
+               }
+       } else {
+               kern_flow_attr = &cmd.flow_attr;
+               kern_attr_size = sizeof(cmd.flow_attr);
+       }
+
+       uobj = kmalloc(sizeof(*uobj), GFP_KERNEL);
+       if (!uobj) {
+               err = -ENOMEM;
+               goto err_free_attr;
+       }
+       init_uobj(uobj, 0, file->ucontext, &rule_lock_class);
+       down_write(&uobj->mutex);
+
+       qp = idr_read_qp(cmd.qp_handle, file->ucontext);
+       if (!qp) {
+               err = -EINVAL;
+               goto err_uobj;
+       }
+
+       flow_attr = kmalloc(cmd.flow_attr.size, GFP_KERNEL);
+       if (!flow_attr) {
+               err = -ENOMEM;
+               goto err_put;
+       }
+
+       flow_attr->type = kern_flow_attr->type;
+       flow_attr->priority = kern_flow_attr->priority;
+       flow_attr->num_of_specs = kern_flow_attr->num_of_specs;
+       flow_attr->port = kern_flow_attr->port;
+       flow_attr->flags = kern_flow_attr->flags;
+       flow_attr->size = sizeof(*flow_attr);
+
+       kern_spec = kern_flow_attr + 1;
+       ib_spec = flow_attr + 1;
+       for (i = 0; i < flow_attr->num_of_specs && kern_attr_size > 0; i++) {
+               err = kern_spec_to_ib_spec(kern_spec, ib_spec);
+               if (err)
+                       goto err_free;
+               flow_attr->size +=
+                       ((union ib_flow_spec *) ib_spec)->size;
+               kern_attr_size -= ((struct ib_kern_spec *) kern_spec)->size;
+               kern_spec += ((struct ib_kern_spec *) kern_spec)->size;
+               ib_spec += ((union ib_flow_spec *) ib_spec)->size;
+       }
+       if (kern_attr_size) {
+               pr_warn("create flow failed, %d bytes left from uverb cmd\n",
+                       kern_attr_size);
+               goto err_free;
+       }
+       flow_id = ib_create_flow(qp, flow_attr, IB_FLOW_DOMAIN_USER);
+       if (IS_ERR(flow_id)) {
+               err = PTR_ERR(flow_id);
+               goto err_free;
+       }
+       flow_id->qp = qp;
+       flow_id->uobject = uobj;
+       uobj->object = flow_id;
+
+       err = idr_add_uobj(&ib_uverbs_rule_idr, uobj);
+       if (err)
+               goto destroy_flow;
+
+       memset(&resp, 0, sizeof(resp));
+       resp.flow_handle = uobj->id;
+
+       if (copy_to_user((void __user *)(unsigned long) cmd.response,
+                        &resp, sizeof(resp))) {
+               err = -EFAULT;
+               goto err_copy;
+       }
+
+       put_qp_read(qp);
+       mutex_lock(&file->mutex);
+       list_add_tail(&uobj->list, &file->ucontext->rule_list);
+       mutex_unlock(&file->mutex);
+
+       uobj->live = 1;
+
+       up_write(&uobj->mutex);
+       kfree(flow_attr);
+       if (cmd.flow_attr.num_of_specs)
+               kfree(kern_flow_attr);
+       return in_len;
+err_copy:
+       idr_remove_uobj(&ib_uverbs_rule_idr, uobj);
+destroy_flow:
+       ib_destroy_flow(flow_id);
+err_free:
+       kfree(flow_attr);
+err_put:
+       put_qp_read(qp);
+err_uobj:
+       put_uobj_write(uobj);
+err_free_attr:
+       if (cmd.flow_attr.num_of_specs)
+               kfree(kern_flow_attr);
+       return err;
+}
+
+ssize_t ib_uverbs_destroy_flow(struct ib_uverbs_file *file,
+                              const char __user *buf, int in_len,
+                              int out_len) {
+       struct ib_uverbs_destroy_flow   cmd;
+       struct ib_flow                  *flow_id;
+       struct ib_uobject               *uobj;
+       int                             ret;
+
+       if (copy_from_user(&cmd, buf, sizeof(cmd)))
+               return -EFAULT;
+
+       uobj = idr_write_uobj(&ib_uverbs_rule_idr, cmd.flow_handle,
+                             file->ucontext);
+       if (!uobj)
+               return -EINVAL;
+       flow_id = uobj->object;
+
+       ret = ib_destroy_flow(flow_id);
+       if (!ret)
+               uobj->live = 0;
+
+       put_uobj_write(uobj);
+
+       idr_remove_uobj(&ib_uverbs_rule_idr, uobj);
+
+       mutex_lock(&file->mutex);
+       list_del(&uobj->list);
+       mutex_unlock(&file->mutex);
+
+       put_uobj(uobj);
+
+       return ret ? ret : in_len;
+}
+
 static int __uverbs_create_xsrq(struct ib_uverbs_file *file,
                                struct ib_uverbs_create_xsrq *cmd,
                                struct ib_udata *udata)
@@ -2860,6 +3098,8 @@ ssize_t ib_uverbs_destroy_srq(struct ib_uverbs_file *file,
        struct ib_srq                    *srq;
        struct ib_uevent_object          *obj;
        int                               ret = -EINVAL;
+       struct ib_usrq_object            *us;
+       enum ib_srq_type                  srq_type;
 
        if (copy_from_user(&cmd, buf, sizeof cmd))
                return -EFAULT;
@@ -2869,6 +3109,7 @@ ssize_t ib_uverbs_destroy_srq(struct ib_uverbs_file *file,
                return -EINVAL;
        srq = uobj->object;
        obj = container_of(uobj, struct ib_uevent_object, uobject);
+       srq_type = srq->srq_type;
 
        ret = ib_destroy_srq(srq);
        if (!ret)
@@ -2879,6 +3120,11 @@ ssize_t ib_uverbs_destroy_srq(struct ib_uverbs_file *file,
        if (ret)
                return ret;
 
+       if (srq_type == IB_SRQT_XRC) {
+               us = container_of(obj, struct ib_usrq_object, uevent);
+               atomic_dec(&us->uxrcd->refcnt);
+       }
+
        idr_remove_uobj(&ib_uverbs_srq_idr, uobj);
 
        mutex_lock(&file->mutex);