On Tue, Jan 17, 2006 at 03:24:37PM -0800, Sean Hefty wrote:
> +static void cm_mask_compare_data(u8 *dst, u8 *src, u8 *mask)
> +{
> +     int i;
> +
> +     for (i = 0; i < IB_CM_PRIVATE_DATA_COMPARE_SIZE; i++)
> +             dst[i] = src[i] & mask[i];
> +}

Is this code going to get invoked very often?

If so, can the mask operation use a "native" size since
IB_CM_PRIVATE_DATA_COMPARE_SIZE is hard coded to 64 byte?

e.g something like:
        for (i = 0; i < IB_CM_PRIVATE_DATA_COMPARE_SIZE/sizeof(unsigned long);
                                                                        i++)
                ((unsigned long *)dst)[i] = ((unsigned long *)src)[i] 
                                                & ((unsigned long *)mask)[i];

thanks,
grant

> +
> +static int cm_compare_data(struct ib_cm_private_data_compare *src_data,
> +                        struct ib_cm_private_data_compare *dst_data)
> +{
> +     u8 src[IB_CM_PRIVATE_DATA_COMPARE_SIZE];
> +     u8 dst[IB_CM_PRIVATE_DATA_COMPARE_SIZE];
> +
> +     if (!src_data || !dst_data)
> +             return 0;
> +     
> +     cm_mask_compare_data(src, src_data->data, dst_data->mask);
> +     cm_mask_compare_data(dst, dst_data->data, src_data->mask);
> +     return memcmp(src, dst, IB_CM_PRIVATE_DATA_COMPARE_SIZE);
> +}
> +
> +static int cm_compare_private_data(u8 *private_data,
> +                                struct ib_cm_private_data_compare *dst_data)
> +{
> +     u8 src[IB_CM_PRIVATE_DATA_COMPARE_SIZE];
> +
> +     if (!dst_data)
> +             return 0;
> +     
> +     cm_mask_compare_data(src, private_data, dst_data->mask);
> +     return memcmp(src, dst_data->data, IB_CM_PRIVATE_DATA_COMPARE_SIZE);
> +}
> +
>  static struct cm_id_private * cm_insert_listen(struct cm_id_private 
> *cm_id_priv)
>  {
>       struct rb_node **link = &cm.listen_service_table.rb_node;
> @@ -362,14 +397,18 @@ static struct cm_id_private * cm_insert_
>       struct cm_id_private *cur_cm_id_priv;
>       __be64 service_id = cm_id_priv->id.service_id;
>       __be64 service_mask = cm_id_priv->id.service_mask;
> +     int data_cmp;
>  
>       while (*link) {
>               parent = *link;
>               cur_cm_id_priv = rb_entry(parent, struct cm_id_private,
>                                         service_node);
> +             data_cmp = cm_compare_data(cm_id_priv->compare_data,
> +                                        cur_cm_id_priv->compare_data);
>               if ((cur_cm_id_priv->id.service_mask & service_id) ==
>                   (service_mask & cur_cm_id_priv->id.service_id) &&
> -                 (cm_id_priv->id.device == cur_cm_id_priv->id.device))
> +                 (cm_id_priv->id.device == cur_cm_id_priv->id.device) &&
> +                 !data_cmp)
>                       return cur_cm_id_priv;
>  
>               if (cm_id_priv->id.device < cur_cm_id_priv->id.device)
> @@ -378,6 +417,10 @@ static struct cm_id_private * cm_insert_
>                       link = &(*link)->rb_right;
>               else if (service_id < cur_cm_id_priv->id.service_id)
>                       link = &(*link)->rb_left;
> +             else if (service_id > cur_cm_id_priv->id.service_id)
> +                     link = &(*link)->rb_right;
> +             else if (data_cmp < 0)
> +                     link = &(*link)->rb_left;
>               else
>                       link = &(*link)->rb_right;
>       }
> @@ -387,16 +430,20 @@ static struct cm_id_private * cm_insert_
>  }
>  
>  static struct cm_id_private * cm_find_listen(struct ib_device *device,
> -                                          __be64 service_id)
> +                                          __be64 service_id,
> +                                          u8 *private_data)
>  {
>       struct rb_node *node = cm.listen_service_table.rb_node;
>       struct cm_id_private *cm_id_priv;
> +     int data_cmp;
>  
>       while (node) {
>               cm_id_priv = rb_entry(node, struct cm_id_private, service_node);
> +             data_cmp = cm_compare_private_data(private_data,
> +                                                cm_id_priv->compare_data);
>               if ((cm_id_priv->id.service_mask & service_id) ==
>                    cm_id_priv->id.service_id &&
> -                 (cm_id_priv->id.device == device))
> +                 (cm_id_priv->id.device == device) && !data_cmp)
>                       return cm_id_priv;
>  
>               if (device < cm_id_priv->id.device)
> @@ -405,6 +452,10 @@ static struct cm_id_private * cm_find_li
>                       node = node->rb_right;
>               else if (service_id < cm_id_priv->id.service_id)
>                       node = node->rb_left;
> +             else if (service_id > cm_id_priv->id.service_id)
> +                     node = node->rb_right;
> +             else if (data_cmp < 0)
> +                     node = node->rb_left;
>               else
>                       node = node->rb_right;
>       }
> @@ -728,15 +779,14 @@ retest:
>       wait_event(cm_id_priv->wait, !atomic_read(&cm_id_priv->refcount));
>       while ((work = cm_dequeue_work(cm_id_priv)) != NULL)
>               cm_free_work(work);
> -     if (cm_id_priv->private_data && cm_id_priv->private_data_len)
> -             kfree(cm_id_priv->private_data);
> +     kfree(cm_id_priv->compare_data);
> +     kfree(cm_id_priv->private_data);
>       kfree(cm_id_priv);
>  }
>  EXPORT_SYMBOL(ib_destroy_cm_id);
>  
> -int ib_cm_listen(struct ib_cm_id *cm_id,
> -              __be64 service_id,
> -              __be64 service_mask)
> +int ib_cm_listen(struct ib_cm_id *cm_id, __be64 service_id, __be64 
> service_mask,
> +              struct ib_cm_private_data_compare *compare_data)
>  {
>       struct cm_id_private *cm_id_priv, *cur_cm_id_priv;
>       unsigned long flags;
> @@ -750,7 +800,19 @@ int ib_cm_listen(struct ib_cm_id *cm_id,
>               return -EINVAL;
>  
>       cm_id_priv = container_of(cm_id, struct cm_id_private, id);
> -     BUG_ON(cm_id->state != IB_CM_IDLE);
> +     if (cm_id->state != IB_CM_IDLE)
> +             return -EINVAL;
> +
> +     if (compare_data) {
> +             cm_id_priv->compare_data = kzalloc(sizeof *compare_data,
> +                                                GFP_KERNEL);
> +             if (!cm_id_priv->compare_data)
> +                     return -ENOMEM;
> +             cm_mask_compare_data(cm_id_priv->compare_data->data,
> +                                  compare_data->data, compare_data->mask);
> +             memcpy(cm_id_priv->compare_data->mask, compare_data->mask,
> +                    IB_CM_PRIVATE_DATA_COMPARE_SIZE);
> +     }
>  
>       cm_id->state = IB_CM_LISTEN;
>  
> @@ -767,6 +829,8 @@ int ib_cm_listen(struct ib_cm_id *cm_id,
>  
>       if (cur_cm_id_priv) {
>               cm_id->state = IB_CM_IDLE;
> +             kfree(cm_id_priv->compare_data);
> +             cm_id_priv->compare_data = NULL;
>               ret = -EBUSY;
>       }
>       return ret;
> @@ -1239,7 +1303,8 @@ static struct cm_id_private * cm_match_r
>  
>       /* Find matching listen request. */
>       listen_cm_id_priv = cm_find_listen(cm_id_priv->id.device,
> -                                        req_msg->service_id);
> +                                        req_msg->service_id,
> +                                        req_msg->private_data);
>       if (!listen_cm_id_priv) {
>               spin_unlock_irqrestore(&cm.lock, flags);
>               cm_issue_rej(work->port, work->mad_recv_wc,
> @@ -2646,7 +2711,8 @@ static int cm_sidr_req_handler(struct cm
>               goto out; /* Duplicate message. */
>       }
>       cur_cm_id_priv = cm_find_listen(cm_id->device,
> -                                     sidr_req_msg->service_id);
> +                                     sidr_req_msg->service_id,
> +                                     sidr_req_msg->private_data);
>       if (!cur_cm_id_priv) {
>               rb_erase(&cm_id_priv->sidr_id_node, &cm.remote_sidr_table);
>               spin_unlock_irqrestore(&cm.lock, flags);
> diff -uprN -X linux-2.6.git/Documentation/dontdiff 
> linux-2.6.git/drivers/infiniband/core/ucm.c 
> linux-2.6.ib/drivers/infiniband/core/ucm.c
> --- linux-2.6.git/drivers/infiniband/core/ucm.c       2006-01-16 
> 16:03:08.000000000 -0800
> +++ linux-2.6.ib/drivers/infiniband/core/ucm.c        2006-01-16 
> 16:03:35.000000000 -0800
> @@ -646,6 +646,17 @@ out:
>       return result;
>  }
>  
> +static int ucm_validate_listen(__be64 service_id, __be64 service_mask)
> +{
> +     service_id &= service_mask;
> +
> +     if (((service_id & IB_CMA_SERVICE_ID_MASK) == IB_CMA_SERVICE_ID) ||
> +         ((service_id & IB_SDP_SERVICE_ID_MASK) == IB_SDP_SERVICE_ID))
> +             return -EINVAL;
> +
> +     return 0;
> +}
> +
>  static ssize_t ib_ucm_listen(struct ib_ucm_file *file,
>                            const char __user *inbuf,
>                            int in_len, int out_len)
> @@ -661,7 +672,13 @@ static ssize_t ib_ucm_listen(struct ib_u
>       if (IS_ERR(ctx))
>               return PTR_ERR(ctx);
>  
> -     result = ib_cm_listen(ctx->cm_id, cmd.service_id, cmd.service_mask);
> +     result = ucm_validate_listen(cmd.service_id, cmd.service_mask);
> +     if (result)
> +             goto out;
> +
> +     result = ib_cm_listen(ctx->cm_id, cmd.service_id, cmd.service_mask,
> +                           NULL);
> +out:
>       ib_ucm_ctx_put(ctx);
>       return result;
>  }
> diff -uprN -X linux-2.6.git/Documentation/dontdiff 
> linux-2.6.git/include/rdma/ib_cm.h 
> linux-2.6.ib/include/rdma/ib_cm.h
> --- linux-2.6.git/include/rdma/ib_cm.h        2006-01-16 10:26:47.000000000 
> -0800
> +++ linux-2.6.ib/include/rdma/ib_cm.h 2006-01-16 16:03:35.000000000 -0800
> @@ -32,7 +32,7 @@
>   * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
>   * SOFTWARE.
>   *
> - * $Id: ib_cm.h 2730 2005-06-28 16:43:03Z sean.hefty $
> + * $Id: ib_cm.h 4311 2005-12-05 18:42:01Z sean.hefty $
>   */
>  #if !defined(IB_CM_H)
>  #define IB_CM_H
> @@ -102,7 +102,8 @@ enum ib_cm_data_size {
>       IB_CM_APR_INFO_LENGTH            = 72,
>       IB_CM_SIDR_REQ_PRIVATE_DATA_SIZE = 216,
>       IB_CM_SIDR_REP_PRIVATE_DATA_SIZE = 136,
> -     IB_CM_SIDR_REP_INFO_LENGTH       = 72
> +     IB_CM_SIDR_REP_INFO_LENGTH       = 72,
> +     IB_CM_PRIVATE_DATA_COMPARE_SIZE  = 64
>  };
>  
>  struct ib_cm_id;
> @@ -238,7 +239,6 @@ struct ib_cm_sidr_rep_event_param {
>       u32                     qpn;
>       void                    *info;
>       u8                      info_len;
> -
>  };
>  
>  struct ib_cm_event {
> @@ -317,6 +317,15 @@ void ib_destroy_cm_id(struct ib_cm_id *c
>  
>  #define IB_SERVICE_ID_AGN_MASK       
> __constant_cpu_to_be64(0xFF00000000000000ULL)
>  #define IB_CM_ASSIGN_SERVICE_ID __constant_cpu_to_be64(0x0200000000000000ULL)
> +#define IB_CMA_SERVICE_ID    __constant_cpu_to_be64(0x0000000001000000ULL)
> +#define IB_CMA_SERVICE_ID_MASK       
> __constant_cpu_to_be64(0xFFFFFFFFFF000000ULL)
> +#define IB_SDP_SERVICE_ID    __constant_cpu_to_be64(0x0000000000010000ULL)
> +#define IB_SDP_SERVICE_ID_MASK       
> __constant_cpu_to_be64(0xFFFFFFFFFFFF0000ULL)
> +
> +struct ib_cm_private_data_compare {
> +     u8  data[IB_CM_PRIVATE_DATA_COMPARE_SIZE];
> +     u8  mask[IB_CM_PRIVATE_DATA_COMPARE_SIZE];
> +};
>  
>  /**
>   * ib_cm_listen - Initiates listening on the specified service ID for
> @@ -330,10 +339,12 @@ void ib_destroy_cm_id(struct ib_cm_id *c
>   *   range of service IDs.  If set to 0, the service ID is matched
>   *   exactly.  This parameter is ignored if %service_id is set to
>   *   IB_CM_ASSIGN_SERVICE_ID.
> + * @compare_data: This parameter is optional.  It specifies data that must
> + *   appear in the private data of a connection request for the specified
> + *   listen request.
>   */
> -int ib_cm_listen(struct ib_cm_id *cm_id,
> -              __be64 service_id,
> -              __be64 service_mask);
> +int ib_cm_listen(struct ib_cm_id *cm_id, __be64 service_id, __be64 
> service_mask,
> +              struct ib_cm_private_data_compare *compare_data);
>  
>  struct ib_cm_req_param {
>       struct ib_sa_path_rec   *primary_path;
> 
> 
> 
> _______________________________________________
> openib-general mailing list
> openib-general@openib.org
> http://openib.org/mailman/listinfo/openib-general
> 
> To unsubscribe, please visit http://openib.org/mailman/listinfo/openib-general
-
To unsubscribe from this list: send the line "unsubscribe netdev" in
the body of a message to [EMAIL PROTECTED]
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to