2  * Copyright (c) 2006 Chelsio, Inc. All rights reserved.
 
   4  * This software is available to you under a choice of one of two
 
   5  * licenses.  You may choose to be licensed under the terms of the GNU
 
   6  * General Public License (GPL) Version 2, available from the file
 
   7  * COPYING in the main directory of this source tree, or the
 
   8  * OpenIB.org BSD license below:
 
  10  *     Redistribution and use in source and binary forms, with or
 
  11  *     without modification, are permitted provided that the following
 
  14  *      - Redistributions of source code must retain the above
 
  15  *        copyright notice, this list of conditions and the following
 
  18  *      - Redistributions in binary form must reproduce the above
 
  19  *        copyright notice, this list of conditions and the following
 
  20  *        disclaimer in the documentation and/or other materials
 
  21  *        provided with the distribution.
 
  23  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 
  24  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 
  25  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 
  26  * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
 
  27  * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
 
  28  * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
 
  29  * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 
  32 #include <asm/byteorder.h>
 
  34 #include <rdma/iw_cm.h>
 
  35 #include <rdma/ib_verbs.h>
 
  39 #include "iwch_provider.h"
 
  41 int iwch_register_mem(struct iwch_dev *rhp, struct iwch_pd *php,
 
  50         if (cxio_register_phys_mem(&rhp->rdev,
 
  51                                    &stag, mhp->attr.pdid,
 
  58                                    &mhp->attr.pbl_size, &mhp->attr.pbl_addr))
 
  61         mhp->attr.stag = stag;
 
  63         mhp->ibmr.rkey = mhp->ibmr.lkey = stag;
 
  64         insert_handle(rhp, &rhp->mmidr, mhp, mmid);
 
  65         PDBG("%s mmid 0x%x mhp %p\n", __FUNCTION__, mmid, mhp);
 
  69 int iwch_reregister_mem(struct iwch_dev *rhp, struct iwch_pd *php,
 
  79         /* We could support this... */
 
  80         if (npages > mhp->attr.pbl_size)
 
  83         stag = mhp->attr.stag;
 
  84         if (cxio_reregister_phys_mem(&rhp->rdev,
 
  85                                    &stag, mhp->attr.pdid,
 
  92                                    &mhp->attr.pbl_size, &mhp->attr.pbl_addr))
 
  95         mhp->attr.stag = stag;
 
  97         mhp->ibmr.rkey = mhp->ibmr.lkey = stag;
 
  98         insert_handle(rhp, &rhp->mmidr, mhp, mmid);
 
  99         PDBG("%s mmid 0x%x mhp %p\n", __FUNCTION__, mmid, mhp);
 
 103 int build_phys_page_list(struct ib_phys_buf *buffer_list,
 
 116         for (i = 0; i < num_phys_buf; ++i) {
 
 117                 if (i != 0 && buffer_list[i].addr & ~PAGE_MASK)
 
 119                 if (i != 0 && i != num_phys_buf - 1 &&
 
 120                     (buffer_list[i].size & ~PAGE_MASK))
 
 122                 *total_size += buffer_list[i].size;
 
 124                         mask |= buffer_list[i].addr;
 
 126                         mask |= buffer_list[i].addr & PAGE_MASK;
 
 127                 if (i != num_phys_buf - 1)
 
 128                         mask |= buffer_list[i].addr + buffer_list[i].size;
 
 130                         mask |= (buffer_list[i].addr + buffer_list[i].size +
 
 131                                 PAGE_SIZE - 1) & PAGE_MASK;
 
 134         if (*total_size > 0xFFFFFFFFULL)
 
 137         /* Find largest page shift we can use to cover buffers */
 
 138         for (*shift = PAGE_SHIFT; *shift < 27; ++(*shift))
 
 139                 if (num_phys_buf > 1) {
 
 140                         if ((1ULL << *shift) & mask)
 
 143                         if (1ULL << *shift >=
 
 144                             buffer_list[0].size +
 
 145                             (buffer_list[0].addr & ((1ULL << *shift) - 1)))
 
 148         buffer_list[0].size += buffer_list[0].addr & ((1ULL << *shift) - 1);
 
 149         buffer_list[0].addr &= ~0ull << *shift;
 
 152         for (i = 0; i < num_phys_buf; ++i)
 
 153                 *npages += (buffer_list[i].size +
 
 154                         (1ULL << *shift) - 1) >> *shift;
 
 159         *page_list = kmalloc(sizeof(u64) * *npages, GFP_KERNEL);
 
 164         for (i = 0; i < num_phys_buf; ++i)
 
 166                      j < (buffer_list[i].size + (1ULL << *shift) - 1) >> *shift;
 
 168                         (*page_list)[n++] = cpu_to_be64(buffer_list[i].addr +
 
 169                             ((u64) j << *shift));
 
 171         PDBG("%s va 0x%llx mask 0x%llx shift %d len %lld pbl_size %d\n",
 
 172              __FUNCTION__, (unsigned long long) *iova_start,
 
 173              (unsigned long long) mask, *shift, (unsigned long long) *total_size,