diff options
Diffstat (limited to 'drivers/infiniband')
-rw-r--r-- | drivers/infiniband/hw/mthca/mthca_provider.c | 82 |
1 files changed, 82 insertions, 0 deletions
diff --git a/drivers/infiniband/hw/mthca/mthca_provider.c b/drivers/infiniband/hw/mthca/mthca_provider.c index 318356c19abe..bbdfcbe6bade 100644 --- a/drivers/infiniband/hw/mthca/mthca_provider.c +++ b/drivers/infiniband/hw/mthca/mthca_provider.c | |||
@@ -654,6 +654,87 @@ static struct ib_mr *mthca_reg_phys_mr(struct ib_pd *pd, | |||
654 | return &mr->ibmr; | 654 | return &mr->ibmr; |
655 | } | 655 | } |
656 | 656 | ||
657 | static struct ib_mr *mthca_reg_user_mr(struct ib_pd *pd, struct ib_umem *region, | ||
658 | int acc, struct ib_udata *udata) | ||
659 | { | ||
660 | struct mthca_dev *dev = to_mdev(pd->device); | ||
661 | struct ib_umem_chunk *chunk; | ||
662 | struct mthca_mr *mr; | ||
663 | u64 *pages; | ||
664 | int shift, n, len; | ||
665 | int i, j, k; | ||
666 | int err = 0; | ||
667 | |||
668 | shift = ffs(region->page_size) - 1; | ||
669 | |||
670 | mr = kmalloc(sizeof *mr, GFP_KERNEL); | ||
671 | if (!mr) | ||
672 | return ERR_PTR(-ENOMEM); | ||
673 | |||
674 | n = 0; | ||
675 | list_for_each_entry(chunk, ®ion->chunk_list, list) | ||
676 | n += chunk->nents; | ||
677 | |||
678 | mr->mtt = mthca_alloc_mtt(dev, n); | ||
679 | if (IS_ERR(mr->mtt)) { | ||
680 | err = PTR_ERR(mr->mtt); | ||
681 | goto err; | ||
682 | } | ||
683 | |||
684 | pages = (u64 *) __get_free_page(GFP_KERNEL); | ||
685 | if (!pages) { | ||
686 | err = -ENOMEM; | ||
687 | goto err_mtt; | ||
688 | } | ||
689 | |||
690 | i = n = 0; | ||
691 | |||
692 | list_for_each_entry(chunk, ®ion->chunk_list, list) | ||
693 | for (j = 0; j < chunk->nmap; ++j) { | ||
694 | len = sg_dma_len(&chunk->page_list[j]) >> shift; | ||
695 | for (k = 0; k < len; ++k) { | ||
696 | pages[i++] = sg_dma_address(&chunk->page_list[j]) + | ||
697 | region->page_size * k; | ||
698 | /* | ||
699 | * Be friendly to WRITE_MTT command | ||
700 | * and leave two empty slots for the | ||
701 | * index and reserved fields of the | ||
702 | * mailbox. | ||
703 | */ | ||
704 | if (i == PAGE_SIZE / sizeof (u64) - 2) { | ||
705 | err = mthca_write_mtt(dev, mr->mtt, | ||
706 | n, pages, i); | ||
707 | if (err) | ||
708 | goto mtt_done; | ||
709 | n += i; | ||
710 | i = 0; | ||
711 | } | ||
712 | } | ||
713 | } | ||
714 | |||
715 | if (i) | ||
716 | err = mthca_write_mtt(dev, mr->mtt, n, pages, i); | ||
717 | mtt_done: | ||
718 | free_page((unsigned long) pages); | ||
719 | if (err) | ||
720 | goto err_mtt; | ||
721 | |||
722 | err = mthca_mr_alloc(dev, to_mpd(pd)->pd_num, shift, region->virt_base, | ||
723 | region->length, convert_access(acc), mr); | ||
724 | |||
725 | if (err) | ||
726 | goto err_mtt; | ||
727 | |||
728 | return &mr->ibmr; | ||
729 | |||
730 | err_mtt: | ||
731 | mthca_free_mtt(dev, mr->mtt); | ||
732 | |||
733 | err: | ||
734 | kfree(mr); | ||
735 | return ERR_PTR(err); | ||
736 | } | ||
737 | |||
657 | static int mthca_dereg_mr(struct ib_mr *mr) | 738 | static int mthca_dereg_mr(struct ib_mr *mr) |
658 | { | 739 | { |
659 | struct mthca_mr *mmr = to_mmr(mr); | 740 | struct mthca_mr *mmr = to_mmr(mr); |
@@ -804,6 +885,7 @@ int mthca_register_device(struct mthca_dev *dev) | |||
804 | dev->ib_dev.poll_cq = mthca_poll_cq; | 885 | dev->ib_dev.poll_cq = mthca_poll_cq; |
805 | dev->ib_dev.get_dma_mr = mthca_get_dma_mr; | 886 | dev->ib_dev.get_dma_mr = mthca_get_dma_mr; |
806 | dev->ib_dev.reg_phys_mr = mthca_reg_phys_mr; | 887 | dev->ib_dev.reg_phys_mr = mthca_reg_phys_mr; |
888 | dev->ib_dev.reg_user_mr = mthca_reg_user_mr; | ||
807 | dev->ib_dev.dereg_mr = mthca_dereg_mr; | 889 | dev->ib_dev.dereg_mr = mthca_dereg_mr; |
808 | 890 | ||
809 | if (dev->mthca_flags & MTHCA_FLAG_FMR) { | 891 | if (dev->mthca_flags & MTHCA_FLAG_FMR) { |