ehca_mrmw.c revision a1a6ff11006c3a056cda9e8b13e7388fba3e69a1
1fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*
2fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *  IBM eServer eHCA Infiniband device driver for Linux on POWER
3fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *
4fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *  MR/MW functions
5fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *
6fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *  Authors: Dietmar Decker <ddecker@de.ibm.com>
7fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *           Christoph Raisch <raisch@de.ibm.com>
8fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *
9fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *  Copyright (c) 2005 IBM Corporation
10fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *
11fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *  All rights reserved.
12fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *
13fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *  This source code is distributed under a dual license of GPL v2.0 and OpenIB
14fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *  BSD.
15fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *
16fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * OpenIB BSD License
17fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *
18fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * Redistribution and use in source and binary forms, with or without
19fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * modification, are permitted provided that the following conditions are met:
20fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *
21fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * Redistributions of source code must retain the above copyright notice, this
22fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * list of conditions and the following disclaimer.
23fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *
24fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * Redistributions in binary form must reproduce the above copyright notice,
25fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * this list of conditions and the following disclaimer in the documentation
26fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * and/or other materials
27fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * provided with the distribution.
28fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick *
29fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
30fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
31fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
32fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
33fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
34fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
35fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
36fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
37fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
38fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
39fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * POSSIBILITY OF SUCH DAMAGE.
40fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick */
41fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
42f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier#include <rdma/ib_umem.h>
43f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier
44fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick#include <asm/current.h>
45fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
46fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick#include "ehca_iverbs.h"
47fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick#include "ehca_mrmw.h"
48fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick#include "hcp_if.h"
49fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick#include "hipz_hw.h"
50fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
51fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickstatic struct kmem_cache *mr_cache;
52fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickstatic struct kmem_cache *mw_cache;
53fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
54fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickstatic struct ehca_mr *ehca_mr_new(void)
55fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
56fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *me;
57fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
58c376222960ae91d5ffb9197ee36771aaed1d9f90Robert P. J. Day	me = kmem_cache_zalloc(mr_cache, GFP_KERNEL);
59fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (me) {
60fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		spin_lock_init(&me->mrlock);
61fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else
62fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_gen_err("alloc failed");
63fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
64fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return me;
65fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick}
66fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
67fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickstatic void ehca_mr_delete(struct ehca_mr *me)
68fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
69fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	kmem_cache_free(mr_cache, me);
70fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick}
71fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
72fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickstatic struct ehca_mw *ehca_mw_new(void)
73fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
74fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mw *me;
75fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
76c376222960ae91d5ffb9197ee36771aaed1d9f90Robert P. J. Day	me = kmem_cache_zalloc(mw_cache, GFP_KERNEL);
77fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (me) {
78fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		spin_lock_init(&me->mwlock);
79fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else
80fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_gen_err("alloc failed");
81fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
82fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return me;
83fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick}
84fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
85fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickstatic void ehca_mw_delete(struct ehca_mw *me)
86fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
87fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	kmem_cache_free(mw_cache, me);
88fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick}
89fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
90fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
91fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
92fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickstruct ib_mr *ehca_get_dma_mr(struct ib_pd *pd, int mr_access_flags)
93fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
94fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_mr *ib_mr;
95fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret;
96fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *e_maxmr;
97fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_pd *e_pd = container_of(pd, struct ehca_pd, ib_pd);
98fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *shca =
99fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		container_of(pd->device, struct ehca_shca, ib_device);
100fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
101fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (shca->maxmr) {
102fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_maxmr = ehca_mr_new();
103fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (!e_maxmr) {
104fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_err(&shca->ib_device, "out of memory");
105fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ib_mr = ERR_PTR(-ENOMEM);
106fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto get_dma_mr_exit0;
107fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
108fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
109fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = ehca_reg_maxmr(shca, e_maxmr, (u64*)KERNELBASE,
110fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     mr_access_flags, e_pd,
111fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     &e_maxmr->ib.ib_mr.lkey,
112fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     &e_maxmr->ib.ib_mr.rkey);
113fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (ret) {
1143df78f81e070c0e3330ae1bd40385e2f0d6fea2cHoang-Nam Nguyen			ehca_mr_delete(e_maxmr);
115fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ib_mr = ERR_PTR(ret);
116fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto get_dma_mr_exit0;
117fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
118fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_mr = &e_maxmr->ib.ib_mr;
119fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else {
120fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "no internal max-MR exist!");
121fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_mr = ERR_PTR(-EINVAL);
122fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto get_dma_mr_exit0;
123fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
124fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
125fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickget_dma_mr_exit0:
126fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (IS_ERR(ib_mr))
127fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "rc=%lx pd=%p mr_access_flags=%x ",
128fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 PTR_ERR(ib_mr), pd, mr_access_flags);
129fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ib_mr;
130fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_get_dma_mr() */
131fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
132fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
133fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
134fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickstruct ib_mr *ehca_reg_phys_mr(struct ib_pd *pd,
135fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			       struct ib_phys_buf *phys_buf_array,
136fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			       int num_phys_buf,
137fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			       int mr_access_flags,
138fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			       u64 *iova_start)
139fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
140fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_mr *ib_mr;
141fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret;
142fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *e_mr;
143fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *shca =
144fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		container_of(pd->device, struct ehca_shca, ib_device);
145fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_pd *e_pd = container_of(pd, struct ehca_pd, ib_pd);
146fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
147fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 size;
148fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_pginfo pginfo={0,0,0,0,0,0,0,NULL,0,NULL,NULL,0,NULL,0};
149fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 num_pages_mr;
150fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 num_pages_4k; /* 4k portion "pages" */
151fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
152fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if ((num_phys_buf <= 0) || !phys_buf_array) {
153fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "bad input values: num_phys_buf=%x "
154fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "phys_buf_array=%p", num_phys_buf, phys_buf_array);
155fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_mr = ERR_PTR(-EINVAL);
156fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto reg_phys_mr_exit0;
157fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
158fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (((mr_access_flags & IB_ACCESS_REMOTE_WRITE) &&
159fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	     !(mr_access_flags & IB_ACCESS_LOCAL_WRITE)) ||
160fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	    ((mr_access_flags & IB_ACCESS_REMOTE_ATOMIC) &&
161fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	     !(mr_access_flags & IB_ACCESS_LOCAL_WRITE))) {
162fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/*
163fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * Remote Write Access requires Local Write Access
164fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * Remote Atomic Access requires Local Write Access
165fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 */
166fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "bad input values: mr_access_flags=%x",
167fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 mr_access_flags);
168fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_mr = ERR_PTR(-EINVAL);
169fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto reg_phys_mr_exit0;
170fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
171fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
172fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* check physical buffer list and calculate size */
173fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ret = ehca_mr_chk_buf_and_calc_size(phys_buf_array, num_phys_buf,
174fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					    iova_start, &size);
175fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret) {
176fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_mr = ERR_PTR(ret);
177fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto reg_phys_mr_exit0;
178fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
179fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if ((size == 0) ||
180fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	    (((u64)iova_start + size) < (u64)iova_start)) {
181fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "bad input values: size=%lx iova_start=%p",
182fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 size, iova_start);
183fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_mr = ERR_PTR(-EINVAL);
184fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto reg_phys_mr_exit0;
185fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
186fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
187fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mr = ehca_mr_new();
188fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!e_mr) {
189fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "out of memory");
190fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_mr = ERR_PTR(-ENOMEM);
191fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto reg_phys_mr_exit0;
192fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
193fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
194fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* determine number of MR pages */
195fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	num_pages_mr = ((((u64)iova_start % PAGE_SIZE) + size +
196fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 PAGE_SIZE - 1) / PAGE_SIZE);
197fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	num_pages_4k = ((((u64)iova_start % EHCA_PAGESIZE) + size +
198fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 EHCA_PAGESIZE - 1) / EHCA_PAGESIZE);
199fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
200fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* register MR on HCA */
201fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ehca_mr_is_maxmr(size, iova_start)) {
202fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_mr->flags |= EHCA_MR_FLAG_MAXMR;
203fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = ehca_reg_maxmr(shca, e_mr, iova_start, mr_access_flags,
204fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     e_pd, &e_mr->ib.ib_mr.lkey,
205fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     &e_mr->ib.ib_mr.rkey);
206fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (ret) {
207fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ib_mr = ERR_PTR(ret);
208fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto reg_phys_mr_exit1;
209fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
210fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else {
211fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.type           = EHCA_MR_PGI_PHYS;
212fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.num_pages      = num_pages_mr;
213fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.num_4k         = num_pages_4k;
214fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.num_phys_buf   = num_phys_buf;
215fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.phys_buf_array = phys_buf_array;
216fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.next_4k        = (((u64)iova_start & ~PAGE_MASK) /
217fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 EHCA_PAGESIZE);
218fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
219fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = ehca_reg_mr(shca, e_mr, iova_start, size, mr_access_flags,
220fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				  e_pd, &pginfo, &e_mr->ib.ib_mr.lkey,
221fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				  &e_mr->ib.ib_mr.rkey);
222fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (ret) {
223fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ib_mr = ERR_PTR(ret);
224fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto reg_phys_mr_exit1;
225fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
226fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
227fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
228fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* successful registration of all pages */
229fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return &e_mr->ib.ib_mr;
230fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
231fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickreg_phys_mr_exit1:
232fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mr_delete(e_mr);
233fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickreg_phys_mr_exit0:
234fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (IS_ERR(ib_mr))
235fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "rc=%lx pd=%p phys_buf_array=%p "
236fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "num_phys_buf=%x mr_access_flags=%x iova_start=%p",
237fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 PTR_ERR(ib_mr), pd, phys_buf_array,
238fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 num_phys_buf, mr_access_flags, iova_start);
239fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ib_mr;
240fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_reg_phys_mr() */
241fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
242fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
243fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
244f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreierstruct ib_mr *ehca_reg_user_mr(struct ib_pd *pd, u64 start, u64 length, u64 virt,
245f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier			       int mr_access_flags, struct ib_udata *udata)
246fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
247fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_mr *ib_mr;
248fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *e_mr;
249fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *shca =
250fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		container_of(pd->device, struct ehca_shca, ib_device);
251fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_pd *e_pd = container_of(pd, struct ehca_pd, ib_pd);
252fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_pginfo pginfo={0,0,0,0,0,0,0,NULL,0,NULL,NULL,0,NULL,0};
253fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret;
254fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 num_pages_mr;
255fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 num_pages_4k; /* 4k portion "pages" */
256fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
257fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!pd) {
258fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_gen_err("bad pd=%p", pd);
259fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		return ERR_PTR(-EFAULT);
260fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
261f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier
262fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (((mr_access_flags & IB_ACCESS_REMOTE_WRITE) &&
263fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	     !(mr_access_flags & IB_ACCESS_LOCAL_WRITE)) ||
264fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	    ((mr_access_flags & IB_ACCESS_REMOTE_ATOMIC) &&
265fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	     !(mr_access_flags & IB_ACCESS_LOCAL_WRITE))) {
266fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/*
267fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * Remote Write Access requires Local Write Access
268fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * Remote Atomic Access requires Local Write Access
269fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 */
270fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "bad input values: mr_access_flags=%x",
271fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 mr_access_flags);
272fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_mr = ERR_PTR(-EINVAL);
273fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto reg_user_mr_exit0;
274fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
275fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
276f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier	if (length == 0 || virt + length < virt) {
277fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "bad input values: length=%lx "
278f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier			 "virt_base=%lx", length, virt);
279fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_mr = ERR_PTR(-EINVAL);
280fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto reg_user_mr_exit0;
281fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
282fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
283fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mr = ehca_mr_new();
284fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!e_mr) {
285fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "out of memory");
286fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_mr = ERR_PTR(-ENOMEM);
287fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto reg_user_mr_exit0;
288fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
289fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
290f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier	e_mr->umem = ib_umem_get(pd->uobject->context, start, length,
291f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier				 mr_access_flags);
292f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier	if (IS_ERR(e_mr->umem)) {
293f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier		ib_mr = (void *) e_mr->umem;
294f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier		goto reg_user_mr_exit1;
295f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier	}
296f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier
297f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier	if (e_mr->umem->page_size != PAGE_SIZE) {
298f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier		ehca_err(pd->device, "page size not supported, "
299f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier			 "e_mr->umem->page_size=%x", e_mr->umem->page_size);
300f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier		ib_mr = ERR_PTR(-EINVAL);
301f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier		goto reg_user_mr_exit2;
302f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier	}
303f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier
304fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* determine number of MR pages */
305f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier	num_pages_mr = (((virt % PAGE_SIZE) + length + PAGE_SIZE - 1) /
306f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier			PAGE_SIZE);
307f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier	num_pages_4k = (((virt % EHCA_PAGESIZE) + length + EHCA_PAGESIZE - 1) /
308f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier			EHCA_PAGESIZE);
309fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
310fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* register MR on HCA */
311fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.type       = EHCA_MR_PGI_USER;
312fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.num_pages  = num_pages_mr;
313fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.num_4k     = num_pages_4k;
314f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier	pginfo.region     = e_mr->umem;
315f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier	pginfo.next_4k	  = e_mr->umem->offset / EHCA_PAGESIZE;
316fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.next_chunk = list_prepare_entry(pginfo.next_chunk,
317f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier					       (&e_mr->umem->chunk_list),
318fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					       list);
319fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
320f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier	ret = ehca_reg_mr(shca, e_mr, (u64*) virt, length, mr_access_flags, e_pd,
321f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier			  &pginfo, &e_mr->ib.ib_mr.lkey, &e_mr->ib.ib_mr.rkey);
322fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret) {
323fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_mr = ERR_PTR(ret);
324f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier		goto reg_user_mr_exit2;
325fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
326fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
327fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* successful registration of all pages */
328fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return &e_mr->ib.ib_mr;
329fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
330f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreierreg_user_mr_exit2:
331f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier	ib_umem_release(e_mr->umem);
332fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickreg_user_mr_exit1:
333fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mr_delete(e_mr);
334fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickreg_user_mr_exit0:
335fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (IS_ERR(ib_mr))
336f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier		ehca_err(pd->device, "rc=%lx pd=%p mr_access_flags=%x"
337fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 " udata=%p",
338f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier			 PTR_ERR(ib_mr), pd, mr_access_flags, udata);
339fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ib_mr;
340fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_reg_user_mr() */
341fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
342fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
343fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
344fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_rereg_phys_mr(struct ib_mr *mr,
345fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		       int mr_rereg_mask,
346fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		       struct ib_pd *pd,
347fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		       struct ib_phys_buf *phys_buf_array,
348fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		       int num_phys_buf,
349fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		       int mr_access_flags,
350fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		       u64 *iova_start)
351fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
352fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret;
353fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
354fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *shca =
355fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		container_of(mr->device, struct ehca_shca, ib_device);
356fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *e_mr = container_of(mr, struct ehca_mr, ib.ib_mr);
357fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_pd *my_pd = container_of(mr->pd, struct ehca_pd, ib_pd);
358fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 new_size;
359fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 *new_start;
360fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 new_acl;
361fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_pd *new_pd;
362fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 tmp_lkey, tmp_rkey;
363fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	unsigned long sl_flags;
364fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 num_pages_mr = 0;
365fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 num_pages_4k = 0; /* 4k portion "pages" */
366fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_pginfo pginfo={0,0,0,0,0,0,0,NULL,0,NULL,NULL,0,NULL,0};
367fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 cur_pid = current->tgid;
368fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
369fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (my_pd->ib_pd.uobject && my_pd->ib_pd.uobject->context &&
370fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	    (my_pd->ownpid != cur_pid)) {
371fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "Invalid caller pid=%x ownpid=%x",
372fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 cur_pid, my_pd->ownpid);
373fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
374fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto rereg_phys_mr_exit0;
375fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
376fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
377fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!(mr_rereg_mask & IB_MR_REREG_TRANS)) {
378fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* TODO not supported, because PHYP rereg hCall needs pages */
379fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "rereg without IB_MR_REREG_TRANS not "
380fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "supported yet, mr_rereg_mask=%x", mr_rereg_mask);
381fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
382fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto rereg_phys_mr_exit0;
383fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
384fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
385fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (mr_rereg_mask & IB_MR_REREG_PD) {
386fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (!pd) {
387fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_err(mr->device, "rereg with bad pd, pd=%p "
388fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "mr_rereg_mask=%x", pd, mr_rereg_mask);
389fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ret = -EINVAL;
390fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto rereg_phys_mr_exit0;
391fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
392fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
393fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
394fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if ((mr_rereg_mask &
395fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	     ~(IB_MR_REREG_TRANS | IB_MR_REREG_PD | IB_MR_REREG_ACCESS)) ||
396fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	    (mr_rereg_mask == 0)) {
397fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
398fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto rereg_phys_mr_exit0;
399fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
400fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
401fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* check other parameters */
402fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (e_mr == shca->maxmr) {
403fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* should be impossible, however reject to be sure */
404fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "rereg internal max-MR impossible, mr=%p "
405fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "shca->maxmr=%p mr->lkey=%x",
406fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 mr, shca->maxmr, mr->lkey);
407fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
408fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto rereg_phys_mr_exit0;
409fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
410fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (mr_rereg_mask & IB_MR_REREG_TRANS) { /* transl., i.e. addr/size */
411fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (e_mr->flags & EHCA_MR_FLAG_FMR) {
412fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_err(mr->device, "not supported for FMR, mr=%p "
413fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "flags=%x", mr, e_mr->flags);
414fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ret = -EINVAL;
415fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto rereg_phys_mr_exit0;
416fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
417fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (!phys_buf_array || num_phys_buf <= 0) {
418fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_err(mr->device, "bad input values: mr_rereg_mask=%x"
419fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 " phys_buf_array=%p num_phys_buf=%x",
420fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 mr_rereg_mask, phys_buf_array, num_phys_buf);
421fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ret = -EINVAL;
422fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto rereg_phys_mr_exit0;
423fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
424fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
425fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if ((mr_rereg_mask & IB_MR_REREG_ACCESS) &&	/* change ACL */
426fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	    (((mr_access_flags & IB_ACCESS_REMOTE_WRITE) &&
427fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	      !(mr_access_flags & IB_ACCESS_LOCAL_WRITE)) ||
428fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	     ((mr_access_flags & IB_ACCESS_REMOTE_ATOMIC) &&
429fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	      !(mr_access_flags & IB_ACCESS_LOCAL_WRITE)))) {
430fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/*
431fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * Remote Write Access requires Local Write Access
432fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * Remote Atomic Access requires Local Write Access
433fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 */
434fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "bad input values: mr_rereg_mask=%x "
435fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "mr_access_flags=%x", mr_rereg_mask, mr_access_flags);
436fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
437fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto rereg_phys_mr_exit0;
438fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
439fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
440fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* set requested values dependent on rereg request */
441fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	spin_lock_irqsave(&e_mr->mrlock, sl_flags);
442fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	new_start = e_mr->start;  /* new == old address */
443fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	new_size  = e_mr->size;	  /* new == old length */
444fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	new_acl   = e_mr->acl;	  /* new == old access control */
445fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	new_pd    = container_of(mr->pd,struct ehca_pd,ib_pd); /*new == old PD*/
446fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
447fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (mr_rereg_mask & IB_MR_REREG_TRANS) {
448fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		new_start = iova_start;	/* change address */
449fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* check physical buffer list and calculate size */
450fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = ehca_mr_chk_buf_and_calc_size(phys_buf_array,
451fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						    num_phys_buf, iova_start,
452fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						    &new_size);
453fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (ret)
454fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto rereg_phys_mr_exit1;
455fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if ((new_size == 0) ||
456fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		    (((u64)iova_start + new_size) < (u64)iova_start)) {
457fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_err(mr->device, "bad input values: new_size=%lx "
458fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "iova_start=%p", new_size, iova_start);
459fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ret = -EINVAL;
460fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto rereg_phys_mr_exit1;
461fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
462fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		num_pages_mr = ((((u64)new_start % PAGE_SIZE) + new_size +
463fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 PAGE_SIZE - 1) / PAGE_SIZE);
464fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		num_pages_4k = ((((u64)new_start % EHCA_PAGESIZE) + new_size +
465fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 EHCA_PAGESIZE - 1) / EHCA_PAGESIZE);
466fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.type           = EHCA_MR_PGI_PHYS;
467fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.num_pages      = num_pages_mr;
468fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.num_4k         = num_pages_4k;
469fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.num_phys_buf   = num_phys_buf;
470fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.phys_buf_array = phys_buf_array;
471fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.next_4k        = (((u64)iova_start & ~PAGE_MASK) /
472fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 EHCA_PAGESIZE);
473fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
474fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (mr_rereg_mask & IB_MR_REREG_ACCESS)
475fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		new_acl = mr_access_flags;
476fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (mr_rereg_mask & IB_MR_REREG_PD)
477fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		new_pd = container_of(pd, struct ehca_pd, ib_pd);
478fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
479fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ret = ehca_rereg_mr(shca, e_mr, new_start, new_size, new_acl,
480fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			    new_pd, &pginfo, &tmp_lkey, &tmp_rkey);
481fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
482fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto rereg_phys_mr_exit1;
483fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
484fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* successful reregistration */
485fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (mr_rereg_mask & IB_MR_REREG_PD)
486fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		mr->pd = pd;
487fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr->lkey = tmp_lkey;
488fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr->rkey = tmp_rkey;
489fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
490fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickrereg_phys_mr_exit1:
491fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	spin_unlock_irqrestore(&e_mr->mrlock, sl_flags);
492fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickrereg_phys_mr_exit0:
493fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
494fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "ret=%x mr=%p mr_rereg_mask=%x pd=%p "
495fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "phys_buf_array=%p num_phys_buf=%x mr_access_flags=%x "
496fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "iova_start=%p",
497fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 ret, mr, mr_rereg_mask, pd, phys_buf_array,
498fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 num_phys_buf, mr_access_flags, iova_start);
499fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
500fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_rereg_phys_mr() */
501fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
502fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
503fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
504fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_query_mr(struct ib_mr *mr, struct ib_mr_attr *mr_attr)
505fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
506fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret = 0;
507fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 h_ret;
508fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *shca =
509fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		container_of(mr->device, struct ehca_shca, ib_device);
510fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *e_mr = container_of(mr, struct ehca_mr, ib.ib_mr);
511fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_pd *my_pd = container_of(mr->pd, struct ehca_pd, ib_pd);
512fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 cur_pid = current->tgid;
513fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	unsigned long sl_flags;
514fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_hipzout_parms hipzout = {{0},0,0,0,0,0};
515fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
516fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (my_pd->ib_pd.uobject && my_pd->ib_pd.uobject->context &&
517fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	    (my_pd->ownpid != cur_pid)) {
518fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "Invalid caller pid=%x ownpid=%x",
519fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 cur_pid, my_pd->ownpid);
520fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
521fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto query_mr_exit0;
522fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
523fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
524fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if ((e_mr->flags & EHCA_MR_FLAG_FMR)) {
525fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "not supported for FMR, mr=%p e_mr=%p "
526fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "e_mr->flags=%x", mr, e_mr, e_mr->flags);
527fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
528fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto query_mr_exit0;
529fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
530fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
531fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	memset(mr_attr, 0, sizeof(struct ib_mr_attr));
532fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	spin_lock_irqsave(&e_mr->mrlock, sl_flags);
533fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
534fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	h_ret = hipz_h_query_mr(shca->ipz_hca_handle, e_mr, &hipzout);
535fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (h_ret != H_SUCCESS) {
536fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "hipz_mr_query failed, h_ret=%lx mr=%p "
537fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "hca_hndl=%lx mr_hndl=%lx lkey=%x",
538fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 h_ret, mr, shca->ipz_hca_handle.handle,
539fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 e_mr->ipz_mr_handle.handle, mr->lkey);
540a1a6ff11006c3a056cda9e8b13e7388fba3e69a1Hoang-Nam Nguyen		ret = ehca2ib_return_code(h_ret);
541fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto query_mr_exit1;
542fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
543fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr_attr->pd               = mr->pd;
544fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr_attr->device_virt_addr = hipzout.vaddr;
545fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr_attr->size             = hipzout.len;
546fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr_attr->lkey             = hipzout.lkey;
547fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr_attr->rkey             = hipzout.rkey;
548fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mrmw_reverse_map_acl(&hipzout.acl, &mr_attr->mr_access_flags);
549fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
550fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickquery_mr_exit1:
551fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	spin_unlock_irqrestore(&e_mr->mrlock, sl_flags);
552fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickquery_mr_exit0:
553fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
554fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "ret=%x mr=%p mr_attr=%p",
555fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 ret, mr, mr_attr);
556fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
557fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_query_mr() */
558fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
559fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
560fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
561fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_dereg_mr(struct ib_mr *mr)
562fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
563fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret = 0;
564fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 h_ret;
565fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *shca =
566fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		container_of(mr->device, struct ehca_shca, ib_device);
567fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *e_mr = container_of(mr, struct ehca_mr, ib.ib_mr);
568fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_pd *my_pd = container_of(mr->pd, struct ehca_pd, ib_pd);
569fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 cur_pid = current->tgid;
570fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
571fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (my_pd->ib_pd.uobject && my_pd->ib_pd.uobject->context &&
572fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	    (my_pd->ownpid != cur_pid)) {
573fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "Invalid caller pid=%x ownpid=%x",
574fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 cur_pid, my_pd->ownpid);
575fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
576fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto dereg_mr_exit0;
577fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
578fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
579fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if ((e_mr->flags & EHCA_MR_FLAG_FMR)) {
580fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "not supported for FMR, mr=%p e_mr=%p "
581fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "e_mr->flags=%x", mr, e_mr, e_mr->flags);
582fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
583fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto dereg_mr_exit0;
584fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else if (e_mr == shca->maxmr) {
585fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* should be impossible, however reject to be sure */
586fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "dereg internal max-MR impossible, mr=%p "
587fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "shca->maxmr=%p mr->lkey=%x",
588fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 mr, shca->maxmr, mr->lkey);
589fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
590fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto dereg_mr_exit0;
591fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
592fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
593fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* TODO: BUSY: MR still has bound window(s) */
594fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	h_ret = hipz_h_free_resource_mr(shca->ipz_hca_handle, e_mr);
595fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (h_ret != H_SUCCESS) {
596fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "hipz_free_mr failed, h_ret=%lx shca=%p "
597fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "e_mr=%p hca_hndl=%lx mr_hndl=%lx mr->lkey=%x",
598fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 h_ret, shca, e_mr, shca->ipz_hca_handle.handle,
599fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 e_mr->ipz_mr_handle.handle, mr->lkey);
600a1a6ff11006c3a056cda9e8b13e7388fba3e69a1Hoang-Nam Nguyen		ret = ehca2ib_return_code(h_ret);
601fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto dereg_mr_exit0;
602fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
603fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
604f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier	if (e_mr->umem)
605f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier		ib_umem_release(e_mr->umem);
606f7c6a7b5d59980b076abbf2ceeb8735591290285Roland Dreier
607fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* successful deregistration */
608fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mr_delete(e_mr);
609fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
610fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickdereg_mr_exit0:
611fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
612fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mr->device, "ret=%x mr=%p", ret, mr);
613fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
614fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_dereg_mr() */
615fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
616fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
617fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
618fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickstruct ib_mw *ehca_alloc_mw(struct ib_pd *pd)
619fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
620fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_mw *ib_mw;
621fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 h_ret;
622fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mw *e_mw;
623fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_pd *e_pd = container_of(pd, struct ehca_pd, ib_pd);
624fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *shca =
625fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		container_of(pd->device, struct ehca_shca, ib_device);
626fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mw_hipzout_parms hipzout = {{0},0};
627fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
628fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mw = ehca_mw_new();
629fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!e_mw) {
630fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_mw = ERR_PTR(-ENOMEM);
631fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto alloc_mw_exit0;
632fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
633fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
634fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	h_ret = hipz_h_alloc_resource_mw(shca->ipz_hca_handle, e_mw,
635fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 e_pd->fw_pd, &hipzout);
636fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (h_ret != H_SUCCESS) {
637fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "hipz_mw_allocate failed, h_ret=%lx "
638fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "shca=%p hca_hndl=%lx mw=%p",
639fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 h_ret, shca, shca->ipz_hca_handle.handle, e_mw);
640a1a6ff11006c3a056cda9e8b13e7388fba3e69a1Hoang-Nam Nguyen		ib_mw = ERR_PTR(ehca2ib_return_code(h_ret));
641fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto alloc_mw_exit1;
642fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
643fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* successful MW allocation */
644fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mw->ipz_mw_handle = hipzout.handle;
645fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mw->ib_mw.rkey    = hipzout.rkey;
646fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return &e_mw->ib_mw;
647fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
648fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickalloc_mw_exit1:
649fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mw_delete(e_mw);
650fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickalloc_mw_exit0:
651fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (IS_ERR(ib_mw))
652fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "rc=%lx pd=%p", PTR_ERR(ib_mw), pd);
653fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ib_mw;
654fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_alloc_mw() */
655fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
656fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
657fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
658fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_bind_mw(struct ib_qp *qp,
659fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 struct ib_mw *mw,
660fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 struct ib_mw_bind *mw_bind)
661fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
662fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* TODO: not supported up to now */
663fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_gen_err("bind MW currently not supported by HCAD");
664fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
665fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return -EPERM;
666fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_bind_mw() */
667fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
668fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
669fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
670fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_dealloc_mw(struct ib_mw *mw)
671fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
672fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 h_ret;
673fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *shca =
674fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		container_of(mw->device, struct ehca_shca, ib_device);
675fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mw *e_mw = container_of(mw, struct ehca_mw, ib_mw);
676fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
677fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	h_ret = hipz_h_free_resource_mw(shca->ipz_hca_handle, e_mw);
678fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (h_ret != H_SUCCESS) {
679fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(mw->device, "hipz_free_mw failed, h_ret=%lx shca=%p "
680fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "mw=%p rkey=%x hca_hndl=%lx mw_hndl=%lx",
681fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 h_ret, shca, mw, mw->rkey, shca->ipz_hca_handle.handle,
682fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 e_mw->ipz_mw_handle.handle);
683a1a6ff11006c3a056cda9e8b13e7388fba3e69a1Hoang-Nam Nguyen		return ehca2ib_return_code(h_ret);
684fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
685fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* successful deallocation */
686fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mw_delete(e_mw);
687fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return 0;
688fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_dealloc_mw() */
689fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
690fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
691fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
692fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickstruct ib_fmr *ehca_alloc_fmr(struct ib_pd *pd,
693fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			      int mr_access_flags,
694fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			      struct ib_fmr_attr *fmr_attr)
695fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
696fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_fmr *ib_fmr;
697fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *shca =
698fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		container_of(pd->device, struct ehca_shca, ib_device);
699fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_pd *e_pd = container_of(pd, struct ehca_pd, ib_pd);
700fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *e_fmr;
701fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret;
702fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 tmp_lkey, tmp_rkey;
703fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_pginfo pginfo={0,0,0,0,0,0,0,NULL,0,NULL,NULL,0,NULL,0};
704fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
705fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* check other parameters */
706fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (((mr_access_flags & IB_ACCESS_REMOTE_WRITE) &&
707fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	     !(mr_access_flags & IB_ACCESS_LOCAL_WRITE)) ||
708fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	    ((mr_access_flags & IB_ACCESS_REMOTE_ATOMIC) &&
709fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	     !(mr_access_flags & IB_ACCESS_LOCAL_WRITE))) {
710fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/*
711fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * Remote Write Access requires Local Write Access
712fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * Remote Atomic Access requires Local Write Access
713fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 */
714fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "bad input values: mr_access_flags=%x",
715fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 mr_access_flags);
716fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_fmr = ERR_PTR(-EINVAL);
717fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto alloc_fmr_exit0;
718fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
719fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (mr_access_flags & IB_ACCESS_MW_BIND) {
720fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "bad input values: mr_access_flags=%x",
721fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 mr_access_flags);
722fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_fmr = ERR_PTR(-EINVAL);
723fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto alloc_fmr_exit0;
724fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
725fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if ((fmr_attr->max_pages == 0) || (fmr_attr->max_maps == 0)) {
726fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "bad input values: fmr_attr->max_pages=%x "
727fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "fmr_attr->max_maps=%x fmr_attr->page_shift=%x",
728fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 fmr_attr->max_pages, fmr_attr->max_maps,
729fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 fmr_attr->page_shift);
730fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_fmr = ERR_PTR(-EINVAL);
731fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto alloc_fmr_exit0;
732fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
733fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (((1 << fmr_attr->page_shift) != EHCA_PAGESIZE) &&
734fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	    ((1 << fmr_attr->page_shift) != PAGE_SIZE)) {
735fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "unsupported fmr_attr->page_shift=%x",
736fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 fmr_attr->page_shift);
737fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_fmr = ERR_PTR(-EINVAL);
738fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto alloc_fmr_exit0;
739fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
740fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
741fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_fmr = ehca_mr_new();
742fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!e_fmr) {
743fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_fmr = ERR_PTR(-ENOMEM);
744fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto alloc_fmr_exit0;
745fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
746fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_fmr->flags |= EHCA_MR_FLAG_FMR;
747fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
748fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* register MR on HCA */
749fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ret = ehca_reg_mr(shca, e_fmr, NULL,
750fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			  fmr_attr->max_pages * (1 << fmr_attr->page_shift),
751fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			  mr_access_flags, e_pd, &pginfo,
752fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			  &tmp_lkey, &tmp_rkey);
753fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret) {
754fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ib_fmr = ERR_PTR(ret);
755fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto alloc_fmr_exit1;
756fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
757fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
758fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* successful */
759fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_fmr->fmr_page_size = 1 << fmr_attr->page_shift;
760fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_fmr->fmr_max_pages = fmr_attr->max_pages;
761fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_fmr->fmr_max_maps = fmr_attr->max_maps;
762fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_fmr->fmr_map_cnt = 0;
763fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return &e_fmr->ib.ib_fmr;
764fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
765fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickalloc_fmr_exit1:
766fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mr_delete(e_fmr);
767fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickalloc_fmr_exit0:
768fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (IS_ERR(ib_fmr))
769fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(pd->device, "rc=%lx pd=%p mr_access_flags=%x "
770fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "fmr_attr=%p", PTR_ERR(ib_fmr), pd,
771fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 mr_access_flags, fmr_attr);
772fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ib_fmr;
773fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_alloc_fmr() */
774fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
775fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
776fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
777fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_map_phys_fmr(struct ib_fmr *fmr,
778fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		      u64 *page_list,
779fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		      int list_len,
780fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		      u64 iova)
781fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
782fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret;
783fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *shca =
784fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		container_of(fmr->device, struct ehca_shca, ib_device);
785fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *e_fmr = container_of(fmr, struct ehca_mr, ib.ib_fmr);
786fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_pd *e_pd = container_of(fmr->pd, struct ehca_pd, ib_pd);
787fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_pginfo pginfo={0,0,0,0,0,0,0,NULL,0,NULL,NULL,0,NULL,0};
788fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 tmp_lkey, tmp_rkey;
789fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
790fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!(e_fmr->flags & EHCA_MR_FLAG_FMR)) {
791fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(fmr->device, "not a FMR, e_fmr=%p e_fmr->flags=%x",
792fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 e_fmr, e_fmr->flags);
793fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
794fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto map_phys_fmr_exit0;
795fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
796fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ret = ehca_fmr_check_page_list(e_fmr, page_list, list_len);
797fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
798fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto map_phys_fmr_exit0;
799fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (iova % e_fmr->fmr_page_size) {
800fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* only whole-numbered pages */
801fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(fmr->device, "bad iova, iova=%lx fmr_page_size=%x",
802fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 iova, e_fmr->fmr_page_size);
803fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
804fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto map_phys_fmr_exit0;
805fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
806fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (e_fmr->fmr_map_cnt >= e_fmr->fmr_max_maps) {
807fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* HCAD does not limit the maps, however trace this anyway */
808fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_info(fmr->device, "map limit exceeded, fmr=%p "
809fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			  "e_fmr->fmr_map_cnt=%x e_fmr->fmr_max_maps=%x",
810fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			  fmr, e_fmr->fmr_map_cnt, e_fmr->fmr_max_maps);
811fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
812fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
813fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.type      = EHCA_MR_PGI_FMR;
814fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.num_pages = list_len;
815fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.num_4k    = list_len * (e_fmr->fmr_page_size / EHCA_PAGESIZE);
816fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.page_list = page_list;
817fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.next_4k   = ((iova & (e_fmr->fmr_page_size-1)) /
818fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			    EHCA_PAGESIZE);
819fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
820fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ret = ehca_rereg_mr(shca, e_fmr, (u64*)iova,
821fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			    list_len * e_fmr->fmr_page_size,
822fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			    e_fmr->acl, e_pd, &pginfo, &tmp_lkey, &tmp_rkey);
823fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
824fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto map_phys_fmr_exit0;
825fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
826fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* successful reregistration */
827fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_fmr->fmr_map_cnt++;
828fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_fmr->ib.ib_fmr.lkey = tmp_lkey;
829fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_fmr->ib.ib_fmr.rkey = tmp_rkey;
830fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return 0;
831fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
832fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickmap_phys_fmr_exit0:
833fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
834fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(fmr->device, "ret=%x fmr=%p page_list=%p list_len=%x "
835fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "iova=%lx",
836fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 ret, fmr, page_list, list_len, iova);
837fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
838fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_map_phys_fmr() */
839fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
840fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
841fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
842fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_unmap_fmr(struct list_head *fmr_list)
843fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
844fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret = 0;
845fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_fmr *ib_fmr;
846fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *shca = NULL;
847fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *prev_shca;
848fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *e_fmr;
849fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 num_fmr = 0;
850fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 unmap_fmr_cnt = 0;
851fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
852fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* check all FMR belong to same SHCA, and check internal flag */
853fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	list_for_each_entry(ib_fmr, fmr_list, list) {
854fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		prev_shca = shca;
855fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (!ib_fmr) {
856fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_gen_err("bad fmr=%p in list", ib_fmr);
857fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ret = -EINVAL;
858fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto unmap_fmr_exit0;
859fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
860fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		shca = container_of(ib_fmr->device, struct ehca_shca,
861fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				    ib_device);
862fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_fmr = container_of(ib_fmr, struct ehca_mr, ib.ib_fmr);
863fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if ((shca != prev_shca) && prev_shca) {
864fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_err(&shca->ib_device, "SHCA mismatch, shca=%p "
865fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "prev_shca=%p e_fmr=%p",
866fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 shca, prev_shca, e_fmr);
867fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ret = -EINVAL;
868fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto unmap_fmr_exit0;
869fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
870fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (!(e_fmr->flags & EHCA_MR_FLAG_FMR)) {
871fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_err(&shca->ib_device, "not a FMR, e_fmr=%p "
872fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "e_fmr->flags=%x", e_fmr, e_fmr->flags);
873fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ret = -EINVAL;
874fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto unmap_fmr_exit0;
875fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
876fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		num_fmr++;
877fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
878fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
879fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* loop over all FMRs to unmap */
880fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	list_for_each_entry(ib_fmr, fmr_list, list) {
881fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		unmap_fmr_cnt++;
882fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_fmr = container_of(ib_fmr, struct ehca_mr, ib.ib_fmr);
883fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		shca = container_of(ib_fmr->device, struct ehca_shca,
884fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				    ib_device);
885fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = ehca_unmap_one_fmr(shca, e_fmr);
886fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (ret) {
887fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			/* unmap failed, stop unmapping of rest of FMRs */
888fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_err(&shca->ib_device, "unmap of one FMR failed, "
889fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "stop rest, e_fmr=%p num_fmr=%x "
890fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "unmap_fmr_cnt=%x lkey=%x", e_fmr, num_fmr,
891fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 unmap_fmr_cnt, e_fmr->ib.ib_fmr.lkey);
892fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto unmap_fmr_exit0;
893fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
894fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
895fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
896fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickunmap_fmr_exit0:
897fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
898fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_gen_err("ret=%x fmr_list=%p num_fmr=%x unmap_fmr_cnt=%x",
899fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     ret, fmr_list, num_fmr, unmap_fmr_cnt);
900fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
901fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_unmap_fmr() */
902fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
903fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
904fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
905fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_dealloc_fmr(struct ib_fmr *fmr)
906fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
907fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret;
908fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 h_ret;
909fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *shca =
910fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		container_of(fmr->device, struct ehca_shca, ib_device);
911fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *e_fmr = container_of(fmr, struct ehca_mr, ib.ib_fmr);
912fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
913fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!(e_fmr->flags & EHCA_MR_FLAG_FMR)) {
914fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(fmr->device, "not a FMR, e_fmr=%p e_fmr->flags=%x",
915fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 e_fmr, e_fmr->flags);
916fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
917fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto free_fmr_exit0;
918fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
919fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
920fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	h_ret = hipz_h_free_resource_mr(shca->ipz_hca_handle, e_fmr);
921fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (h_ret != H_SUCCESS) {
922fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(fmr->device, "hipz_free_mr failed, h_ret=%lx e_fmr=%p "
923fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "hca_hndl=%lx fmr_hndl=%lx fmr->lkey=%x",
924fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 h_ret, e_fmr, shca->ipz_hca_handle.handle,
925fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 e_fmr->ipz_mr_handle.handle, fmr->lkey);
926a1a6ff11006c3a056cda9e8b13e7388fba3e69a1Hoang-Nam Nguyen		ret = ehca2ib_return_code(h_ret);
927fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto free_fmr_exit0;
928fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
929fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* successful deregistration */
930fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mr_delete(e_fmr);
931fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return 0;
932fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
933fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickfree_fmr_exit0:
934fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
935fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "ret=%x fmr=%p", ret, fmr);
936fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
937fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_dealloc_fmr() */
938fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
939fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
940fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
941fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_reg_mr(struct ehca_shca *shca,
942fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		struct ehca_mr *e_mr,
943fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		u64 *iova_start,
944fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		u64 size,
945fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		int acl,
946fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		struct ehca_pd *e_pd,
947fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		struct ehca_mr_pginfo *pginfo,
948fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		u32 *lkey, /*OUT*/
949fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		u32 *rkey) /*OUT*/
950fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
951fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret;
952fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 h_ret;
953fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 hipz_acl;
954fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_hipzout_parms hipzout = {{0},0,0,0,0,0};
955fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
956fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mrmw_map_acl(acl, &hipz_acl);
957fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mrmw_set_pgsize_hipz_acl(&hipz_acl);
958fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ehca_use_hp_mr == 1)
959fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	        hipz_acl |= 0x00000001;
960fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
961fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	h_ret = hipz_h_alloc_resource_mr(shca->ipz_hca_handle, e_mr,
962fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 (u64)iova_start, size, hipz_acl,
963fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 e_pd->fw_pd, &hipzout);
964fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (h_ret != H_SUCCESS) {
965fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "hipz_alloc_mr failed, h_ret=%lx "
966fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "hca_hndl=%lx", h_ret, shca->ipz_hca_handle.handle);
967a1a6ff11006c3a056cda9e8b13e7388fba3e69a1Hoang-Nam Nguyen		ret = ehca2ib_return_code(h_ret);
968fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto ehca_reg_mr_exit0;
969fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
970fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
971fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mr->ipz_mr_handle = hipzout.handle;
972fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
973fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ret = ehca_reg_mr_rpages(shca, e_mr, pginfo);
974fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
975fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto ehca_reg_mr_exit1;
976fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
977fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* successful registration */
978fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mr->num_pages = pginfo->num_pages;
979fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mr->num_4k    = pginfo->num_4k;
980fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mr->start     = iova_start;
981fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mr->size      = size;
982fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mr->acl       = acl;
983fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	*lkey = hipzout.lkey;
984fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	*rkey = hipzout.rkey;
985fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return 0;
986fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
987fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_reg_mr_exit1:
988fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	h_ret = hipz_h_free_resource_mr(shca->ipz_hca_handle, e_mr);
989fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (h_ret != H_SUCCESS) {
990fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "h_ret=%lx shca=%p e_mr=%p "
991fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "iova_start=%p size=%lx acl=%x e_pd=%p lkey=%x "
992fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "pginfo=%p num_pages=%lx num_4k=%lx ret=%x",
993fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 h_ret, shca, e_mr, iova_start, size, acl, e_pd,
994fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 hipzout.lkey, pginfo, pginfo->num_pages,
995fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 pginfo->num_4k, ret);
996fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "internal error in ehca_reg_mr, "
997fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "not recoverable");
998fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
999fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_reg_mr_exit0:
1000fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
1001fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "ret=%x shca=%p e_mr=%p "
1002fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "iova_start=%p size=%lx acl=%x e_pd=%p pginfo=%p "
1003fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "num_pages=%lx num_4k=%lx",
1004fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 ret, shca, e_mr, iova_start, size, acl, e_pd, pginfo,
1005fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 pginfo->num_pages, pginfo->num_4k);
1006fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
1007fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_reg_mr() */
1008fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1009fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1010fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1011fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_reg_mr_rpages(struct ehca_shca *shca,
1012fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		       struct ehca_mr *e_mr,
1013fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		       struct ehca_mr_pginfo *pginfo)
1014fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1015fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret = 0;
1016fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 h_ret;
1017fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 rnum;
1018fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 rpage;
1019fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 i;
1020fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 *kpage;
1021fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1022f2d9136133de257abbd97fec6f624d3a73d1e1fdHoang-Nam Nguyen	kpage = ehca_alloc_fw_ctrlblock(GFP_KERNEL);
1023fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!kpage) {
1024fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "kpage alloc failed");
1025fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -ENOMEM;
1026fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto ehca_reg_mr_rpages_exit0;
1027fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1028fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1029fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* max 512 pages per shot */
1030fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	for (i = 0; i < ((pginfo->num_4k + 512 - 1) / 512); i++) {
1031fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1032fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (i == ((pginfo->num_4k + 512 - 1) / 512) - 1) {
1033fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			rnum = pginfo->num_4k % 512; /* last shot */
1034fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			if (rnum == 0)
1035fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				rnum = 512;      /* last shot is full */
1036fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		} else
1037fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			rnum = 512;
1038fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1039fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (rnum > 1) {
1040fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ret = ehca_set_pagebuf(e_mr, pginfo, rnum, kpage);
1041fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			if (ret) {
1042fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				ehca_err(&shca->ib_device, "ehca_set_pagebuf "
1043fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 "bad rc, ret=%x rnum=%x kpage=%p",
1044fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 ret, rnum, kpage);
1045fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				ret = -EFAULT;
1046fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				goto ehca_reg_mr_rpages_exit1;
1047fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			}
1048fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			rpage = virt_to_abs(kpage);
1049fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			if (!rpage) {
1050fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				ehca_err(&shca->ib_device, "kpage=%p i=%x",
1051fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 kpage, i);
1052fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				ret = -EFAULT;
1053fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				goto ehca_reg_mr_rpages_exit1;
1054fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			}
1055fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		} else {  /* rnum==1 */
1056fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ret = ehca_set_pagebuf_1(e_mr, pginfo, &rpage);
1057fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			if (ret) {
1058fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				ehca_err(&shca->ib_device, "ehca_set_pagebuf_1 "
1059fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 "bad rc, ret=%x i=%x", ret, i);
1060fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				ret = -EFAULT;
1061fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				goto ehca_reg_mr_rpages_exit1;
1062fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			}
1063fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1064fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1065fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		h_ret = hipz_h_register_rpage_mr(shca->ipz_hca_handle, e_mr,
1066fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						 0, /* pagesize 4k */
1067fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						 0, rpage, rnum);
1068fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1069fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (i == ((pginfo->num_4k + 512 - 1) / 512) - 1) {
1070fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			/*
1071fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 * check for 'registration complete'==H_SUCCESS
1072fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 * and for 'page registered'==H_PAGE_REGISTERED
1073fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 */
1074fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			if (h_ret != H_SUCCESS) {
1075fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				ehca_err(&shca->ib_device, "last "
1076fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 "hipz_reg_rpage_mr failed, h_ret=%lx "
1077fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 "e_mr=%p i=%x hca_hndl=%lx mr_hndl=%lx"
1078fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 " lkey=%x", h_ret, e_mr, i,
1079fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 shca->ipz_hca_handle.handle,
1080fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 e_mr->ipz_mr_handle.handle,
1081fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					 e_mr->ib.ib_mr.lkey);
1082a1a6ff11006c3a056cda9e8b13e7388fba3e69a1Hoang-Nam Nguyen				ret = ehca2ib_return_code(h_ret);
1083fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				break;
1084fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			} else
1085fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				ret = 0;
1086fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		} else if (h_ret != H_PAGE_REGISTERED) {
1087fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_err(&shca->ib_device, "hipz_reg_rpage_mr failed, "
1088fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "h_ret=%lx e_mr=%p i=%x lkey=%x hca_hndl=%lx "
1089fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "mr_hndl=%lx", h_ret, e_mr, i,
1090fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 e_mr->ib.ib_mr.lkey,
1091fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 shca->ipz_hca_handle.handle,
1092fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 e_mr->ipz_mr_handle.handle);
1093a1a6ff11006c3a056cda9e8b13e7388fba3e69a1Hoang-Nam Nguyen			ret = ehca2ib_return_code(h_ret);
1094fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			break;
1095fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		} else
1096fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ret = 0;
1097fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} /* end for(i) */
1098fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1099fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1100fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_reg_mr_rpages_exit1:
11017e28db5d8ff63b1cabc221c5cb84a5f45752f1c2Hoang-Nam Nguyen	ehca_free_fw_ctrlblock(kpage);
1102fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_reg_mr_rpages_exit0:
1103fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
1104fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "ret=%x shca=%p e_mr=%p pginfo=%p "
1105fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "num_pages=%lx num_4k=%lx", ret, shca, e_mr, pginfo,
1106fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 pginfo->num_pages, pginfo->num_4k);
1107fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
1108fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_reg_mr_rpages() */
1109fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1110fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1111fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1112fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickinline int ehca_rereg_mr_rereg1(struct ehca_shca *shca,
1113fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				struct ehca_mr *e_mr,
1114fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				u64 *iova_start,
1115fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				u64 size,
1116fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				u32 acl,
1117fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				struct ehca_pd *e_pd,
1118fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				struct ehca_mr_pginfo *pginfo,
1119fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				u32 *lkey, /*OUT*/
1120fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				u32 *rkey) /*OUT*/
1121fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1122fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret;
1123fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 h_ret;
1124fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 hipz_acl;
1125fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 *kpage;
1126fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 rpage;
1127fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_pginfo pginfo_save;
1128fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_hipzout_parms hipzout = {{0},0,0,0,0,0};
1129fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1130fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mrmw_map_acl(acl, &hipz_acl);
1131fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mrmw_set_pgsize_hipz_acl(&hipz_acl);
1132fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1133f2d9136133de257abbd97fec6f624d3a73d1e1fdHoang-Nam Nguyen	kpage = ehca_alloc_fw_ctrlblock(GFP_KERNEL);
1134fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!kpage) {
1135fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "kpage alloc failed");
1136fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -ENOMEM;
1137fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto ehca_rereg_mr_rereg1_exit0;
1138fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1139fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1140fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo_save = *pginfo;
1141fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ret = ehca_set_pagebuf(e_mr, pginfo, pginfo->num_4k, kpage);
1142fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret) {
1143fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "set pagebuf failed, e_mr=%p "
1144fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "pginfo=%p type=%x num_pages=%lx num_4k=%lx kpage=%p",
1145fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 e_mr, pginfo, pginfo->type, pginfo->num_pages,
1146fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 pginfo->num_4k,kpage);
1147fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto ehca_rereg_mr_rereg1_exit1;
1148fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1149fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	rpage = virt_to_abs(kpage);
1150fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!rpage) {
1151fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "kpage=%p", kpage);
1152fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EFAULT;
1153fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto ehca_rereg_mr_rereg1_exit1;
1154fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1155fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	h_ret = hipz_h_reregister_pmr(shca->ipz_hca_handle, e_mr,
1156fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				      (u64)iova_start, size, hipz_acl,
1157fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				      e_pd->fw_pd, rpage, &hipzout);
1158fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (h_ret != H_SUCCESS) {
1159fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/*
1160fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * reregistration unsuccessful, try it again with the 3 hCalls,
1161fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * e.g. this is required in case H_MR_CONDITION
1162fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * (MW bound or MR is shared)
1163fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 */
1164fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_warn(&shca->ib_device, "hipz_h_reregister_pmr failed "
1165fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			  "(Rereg1), h_ret=%lx e_mr=%p", h_ret, e_mr);
1166fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*pginfo = pginfo_save;
1167fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EAGAIN;
1168fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else if ((u64*)hipzout.vaddr != iova_start) {
1169fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "PHYP changed iova_start in "
1170fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "rereg_pmr, iova_start=%p iova_start_out=%lx e_mr=%p "
1171fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "mr_handle=%lx lkey=%x lkey_out=%x", iova_start,
1172fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 hipzout.vaddr, e_mr, e_mr->ipz_mr_handle.handle,
1173fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 e_mr->ib.ib_mr.lkey, hipzout.lkey);
1174fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EFAULT;
1175fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else {
1176fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/*
1177fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * successful reregistration
1178fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * note: start and start_out are identical for eServer HCAs
1179fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 */
1180fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_mr->num_pages = pginfo->num_pages;
1181fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_mr->num_4k    = pginfo->num_4k;
1182fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_mr->start     = iova_start;
1183fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_mr->size      = size;
1184fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_mr->acl       = acl;
1185fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*lkey = hipzout.lkey;
1186fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*rkey = hipzout.rkey;
1187fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1188fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1189fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_rereg_mr_rereg1_exit1:
11907e28db5d8ff63b1cabc221c5cb84a5f45752f1c2Hoang-Nam Nguyen	ehca_free_fw_ctrlblock(kpage);
1191fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_rereg_mr_rereg1_exit0:
1192fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if ( ret && (ret != -EAGAIN) )
1193fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "ret=%x lkey=%x rkey=%x "
1194fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "pginfo=%p num_pages=%lx num_4k=%lx",
1195fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 ret, *lkey, *rkey, pginfo, pginfo->num_pages,
1196fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 pginfo->num_4k);
1197fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
1198fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_rereg_mr_rereg1() */
1199fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1200fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1201fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1202fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_rereg_mr(struct ehca_shca *shca,
1203fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		  struct ehca_mr *e_mr,
1204fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		  u64 *iova_start,
1205fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		  u64 size,
1206fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		  int acl,
1207fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		  struct ehca_pd *e_pd,
1208fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		  struct ehca_mr_pginfo *pginfo,
1209fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		  u32 *lkey,
1210fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		  u32 *rkey)
1211fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1212fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret = 0;
1213fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 h_ret;
1214fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int rereg_1_hcall = 1; /* 1: use hipz_h_reregister_pmr directly */
1215fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int rereg_3_hcall = 0; /* 1: use 3 hipz calls for reregistration */
1216fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1217fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* first determine reregistration hCall(s) */
1218fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if ((pginfo->num_4k > 512) || (e_mr->num_4k > 512) ||
1219fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	    (pginfo->num_4k > e_mr->num_4k)) {
1220fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_dbg(&shca->ib_device, "Rereg3 case, pginfo->num_4k=%lx "
1221fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "e_mr->num_4k=%x", pginfo->num_4k, e_mr->num_4k);
1222fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		rereg_1_hcall = 0;
1223fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		rereg_3_hcall = 1;
1224fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1225fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1226fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (e_mr->flags & EHCA_MR_FLAG_MAXMR) {	/* check for max-MR */
1227fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		rereg_1_hcall = 0;
1228fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		rereg_3_hcall = 1;
1229fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_mr->flags &= ~EHCA_MR_FLAG_MAXMR;
1230fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "Rereg MR for max-MR! e_mr=%p",
1231fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 e_mr);
1232fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1233fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1234fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (rereg_1_hcall) {
1235fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = ehca_rereg_mr_rereg1(shca, e_mr, iova_start, size,
1236fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					   acl, e_pd, pginfo, lkey, rkey);
1237fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (ret) {
1238fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			if (ret == -EAGAIN)
1239fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				rereg_3_hcall = 1;
1240fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			else
1241fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				goto ehca_rereg_mr_exit0;
1242fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1243fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1244fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1245fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (rereg_3_hcall) {
1246fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		struct ehca_mr save_mr;
1247fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1248fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* first deregister old MR */
1249fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		h_ret = hipz_h_free_resource_mr(shca->ipz_hca_handle, e_mr);
1250fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (h_ret != H_SUCCESS) {
1251fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_err(&shca->ib_device, "hipz_free_mr failed, "
1252fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "h_ret=%lx e_mr=%p hca_hndl=%lx mr_hndl=%lx "
1253fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "mr->lkey=%x",
1254fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 h_ret, e_mr, shca->ipz_hca_handle.handle,
1255fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 e_mr->ipz_mr_handle.handle,
1256fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 e_mr->ib.ib_mr.lkey);
1257a1a6ff11006c3a056cda9e8b13e7388fba3e69a1Hoang-Nam Nguyen			ret = ehca2ib_return_code(h_ret);
1258fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto ehca_rereg_mr_exit0;
1259fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1260fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* clean ehca_mr_t, without changing struct ib_mr and lock */
1261fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		save_mr = *e_mr;
1262fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_mr_deletenew(e_mr);
1263fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1264fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* set some MR values */
1265fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_mr->flags = save_mr.flags;
1266fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_mr->fmr_page_size = save_mr.fmr_page_size;
1267fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_mr->fmr_max_pages = save_mr.fmr_max_pages;
1268fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_mr->fmr_max_maps = save_mr.fmr_max_maps;
1269fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_mr->fmr_map_cnt = save_mr.fmr_map_cnt;
1270fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1271fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = ehca_reg_mr(shca, e_mr, iova_start, size, acl,
1272fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				      e_pd, pginfo, lkey, rkey);
1273fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (ret) {
1274fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			u32 offset = (u64)(&e_mr->flags) - (u64)e_mr;
1275fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			memcpy(&e_mr->flags, &(save_mr.flags),
1276fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			       sizeof(struct ehca_mr) - offset);
1277fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto ehca_rereg_mr_exit0;
1278fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1279fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1280fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1281fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_rereg_mr_exit0:
1282fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
1283fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "ret=%x shca=%p e_mr=%p "
1284fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "iova_start=%p size=%lx acl=%x e_pd=%p pginfo=%p "
1285fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "num_pages=%lx lkey=%x rkey=%x rereg_1_hcall=%x "
1286fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "rereg_3_hcall=%x", ret, shca, e_mr, iova_start, size,
1287fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 acl, e_pd, pginfo, pginfo->num_pages, *lkey, *rkey,
1288fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 rereg_1_hcall, rereg_3_hcall);
1289fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
1290fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_rereg_mr() */
1291fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1292fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1293fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1294fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_unmap_one_fmr(struct ehca_shca *shca,
1295fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		       struct ehca_mr *e_fmr)
1296fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1297fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret = 0;
1298fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 h_ret;
1299fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int rereg_1_hcall = 1; /* 1: use hipz_mr_reregister directly */
1300fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int rereg_3_hcall = 0; /* 1: use 3 hipz calls for unmapping */
1301fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_pd *e_pd =
1302fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		container_of(e_fmr->ib.ib_fmr.pd, struct ehca_pd, ib_pd);
1303fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr save_fmr;
1304fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 tmp_lkey, tmp_rkey;
1305fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_pginfo pginfo={0,0,0,0,0,0,0,NULL,0,NULL,NULL,0,NULL,0};
1306fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_hipzout_parms hipzout = {{0},0,0,0,0,0};
1307fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1308fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* first check if reregistration hCall can be used for unmap */
1309fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (e_fmr->fmr_max_pages > 512) {
1310fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		rereg_1_hcall = 0;
1311fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		rereg_3_hcall = 1;
1312fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1313fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1314fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (rereg_1_hcall) {
1315fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/*
1316fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * note: after using rereg hcall with len=0,
1317fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 * rereg hcall must be used again for registering pages
1318fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 */
1319fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		h_ret = hipz_h_reregister_pmr(shca->ipz_hca_handle, e_fmr, 0,
1320fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					      0, 0, e_pd->fw_pd, 0, &hipzout);
1321fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (h_ret != H_SUCCESS) {
1322fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			/*
1323fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 * should not happen, because length checked above,
1324fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 * FMRs are not shared and no MW bound to FMRs
1325fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 */
1326fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_err(&shca->ib_device, "hipz_reregister_pmr failed "
1327fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "(Rereg1), h_ret=%lx e_fmr=%p hca_hndl=%lx "
1328fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "mr_hndl=%lx lkey=%x lkey_out=%x",
1329fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 h_ret, e_fmr, shca->ipz_hca_handle.handle,
1330fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 e_fmr->ipz_mr_handle.handle,
1331fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 e_fmr->ib.ib_fmr.lkey, hipzout.lkey);
1332fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			rereg_3_hcall = 1;
1333fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		} else {
1334fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			/* successful reregistration */
1335fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			e_fmr->start = NULL;
1336fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			e_fmr->size = 0;
1337fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			tmp_lkey = hipzout.lkey;
1338fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			tmp_rkey = hipzout.rkey;
1339fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1340fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1341fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1342fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (rereg_3_hcall) {
1343fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		struct ehca_mr save_mr;
1344fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1345fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* first free old FMR */
1346fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		h_ret = hipz_h_free_resource_mr(shca->ipz_hca_handle, e_fmr);
1347fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (h_ret != H_SUCCESS) {
1348fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_err(&shca->ib_device, "hipz_free_mr failed, "
1349fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "h_ret=%lx e_fmr=%p hca_hndl=%lx mr_hndl=%lx "
1350fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 "lkey=%x",
1351fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 h_ret, e_fmr, shca->ipz_hca_handle.handle,
1352fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 e_fmr->ipz_mr_handle.handle,
1353fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				 e_fmr->ib.ib_fmr.lkey);
1354a1a6ff11006c3a056cda9e8b13e7388fba3e69a1Hoang-Nam Nguyen			ret = ehca2ib_return_code(h_ret);
1355fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto ehca_unmap_one_fmr_exit0;
1356fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1357fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* clean ehca_mr_t, without changing lock */
1358fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		save_fmr = *e_fmr;
1359fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_mr_deletenew(e_fmr);
1360fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1361fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* set some MR values */
1362fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_fmr->flags = save_fmr.flags;
1363fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_fmr->fmr_page_size = save_fmr.fmr_page_size;
1364fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_fmr->fmr_max_pages = save_fmr.fmr_max_pages;
1365fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_fmr->fmr_max_maps = save_fmr.fmr_max_maps;
1366fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_fmr->fmr_map_cnt = save_fmr.fmr_map_cnt;
1367fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		e_fmr->acl = save_fmr.acl;
1368fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1369fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.type      = EHCA_MR_PGI_FMR;
1370fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.num_pages = 0;
1371fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo.num_4k    = 0;
1372fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = ehca_reg_mr(shca, e_fmr, NULL,
1373fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				  (e_fmr->fmr_max_pages * e_fmr->fmr_page_size),
1374fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				  e_fmr->acl, e_pd, &pginfo, &tmp_lkey,
1375fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				  &tmp_rkey);
1376fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (ret) {
1377fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			u32 offset = (u64)(&e_fmr->flags) - (u64)e_fmr;
1378fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			memcpy(&e_fmr->flags, &(save_mr.flags),
1379fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			       sizeof(struct ehca_mr) - offset);
1380fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto ehca_unmap_one_fmr_exit0;
1381fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1382fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1383fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1384fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_unmap_one_fmr_exit0:
1385fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
1386fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "ret=%x tmp_lkey=%x tmp_rkey=%x "
1387fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "fmr_max_pages=%x rereg_1_hcall=%x rereg_3_hcall=%x",
1388fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 ret, tmp_lkey, tmp_rkey, e_fmr->fmr_max_pages,
1389fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 rereg_1_hcall, rereg_3_hcall);
1390fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
1391fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_unmap_one_fmr() */
1392fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1393fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1394fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1395fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_reg_smr(struct ehca_shca *shca,
1396fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 struct ehca_mr *e_origmr,
1397fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 struct ehca_mr *e_newmr,
1398fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 u64 *iova_start,
1399fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 int acl,
1400fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 struct ehca_pd *e_pd,
1401fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 u32 *lkey, /*OUT*/
1402fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		 u32 *rkey) /*OUT*/
1403fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1404fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret = 0;
1405fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 h_ret;
1406fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 hipz_acl;
1407fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_hipzout_parms hipzout = {{0},0,0,0,0,0};
1408fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1409fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mrmw_map_acl(acl, &hipz_acl);
1410fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mrmw_set_pgsize_hipz_acl(&hipz_acl);
1411fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1412fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	h_ret = hipz_h_register_smr(shca->ipz_hca_handle, e_newmr, e_origmr,
1413fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				    (u64)iova_start, hipz_acl, e_pd->fw_pd,
1414fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				    &hipzout);
1415fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (h_ret != H_SUCCESS) {
1416fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "hipz_reg_smr failed, h_ret=%lx "
1417fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "shca=%p e_origmr=%p e_newmr=%p iova_start=%p acl=%x "
1418fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "e_pd=%p hca_hndl=%lx mr_hndl=%lx lkey=%x",
1419fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 h_ret, shca, e_origmr, e_newmr, iova_start, acl, e_pd,
1420fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 shca->ipz_hca_handle.handle,
1421fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 e_origmr->ipz_mr_handle.handle,
1422fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 e_origmr->ib.ib_mr.lkey);
1423a1a6ff11006c3a056cda9e8b13e7388fba3e69a1Hoang-Nam Nguyen		ret = ehca2ib_return_code(h_ret);
1424fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto ehca_reg_smr_exit0;
1425fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1426fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* successful registration */
1427fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_newmr->num_pages     = e_origmr->num_pages;
1428fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_newmr->num_4k        = e_origmr->num_4k;
1429fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_newmr->start         = iova_start;
1430fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_newmr->size          = e_origmr->size;
1431fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_newmr->acl           = acl;
1432fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_newmr->ipz_mr_handle = hipzout.handle;
1433fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	*lkey = hipzout.lkey;
1434fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	*rkey = hipzout.rkey;
1435fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return 0;
1436fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1437fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_reg_smr_exit0:
1438fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
1439fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "ret=%x shca=%p e_origmr=%p "
1440fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "e_newmr=%p iova_start=%p acl=%x e_pd=%p",
1441fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 ret, shca, e_origmr, e_newmr, iova_start, acl, e_pd);
1442fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
1443fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_reg_smr() */
1444fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1445fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1446fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1447fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/* register internal max-MR to internal SHCA */
1448fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_reg_internal_maxmr(
1449fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_shca *shca,
1450fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_pd *e_pd,
1451fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr **e_maxmr)  /*OUT*/
1452fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1453fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret;
1454fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *e_mr;
1455fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 *iova_start;
1456fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 size_maxmr;
1457fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_pginfo pginfo={0,0,0,0,0,0,0,NULL,0,NULL,NULL,0,NULL,0};
1458fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_phys_buf ib_pbuf;
1459fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 num_pages_mr;
1460fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 num_pages_4k; /* 4k portion "pages" */
1461fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1462fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mr = ehca_mr_new();
1463fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!e_mr) {
1464fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "out of memory");
1465fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -ENOMEM;
1466fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto ehca_reg_internal_maxmr_exit0;
1467fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1468fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mr->flags |= EHCA_MR_FLAG_MAXMR;
1469fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1470fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* register internal max-MR on HCA */
1471fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	size_maxmr = (u64)high_memory - PAGE_OFFSET;
1472fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	iova_start = (u64*)KERNELBASE;
1473fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ib_pbuf.addr = 0;
1474fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ib_pbuf.size = size_maxmr;
1475fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	num_pages_mr = ((((u64)iova_start % PAGE_SIZE) + size_maxmr +
1476fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 PAGE_SIZE - 1) / PAGE_SIZE);
1477fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	num_pages_4k = ((((u64)iova_start % EHCA_PAGESIZE) + size_maxmr +
1478fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 EHCA_PAGESIZE - 1) / EHCA_PAGESIZE);
1479fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1480fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.type           = EHCA_MR_PGI_PHYS;
1481fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.num_pages      = num_pages_mr;
1482fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.num_4k         = num_pages_4k;
1483fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.num_phys_buf   = 1;
1484fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	pginfo.phys_buf_array = &ib_pbuf;
1485fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1486fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ret = ehca_reg_mr(shca, e_mr, iova_start, size_maxmr, 0, e_pd,
1487fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			  &pginfo, &e_mr->ib.ib_mr.lkey,
1488fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			  &e_mr->ib.ib_mr.rkey);
1489fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret) {
1490fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "reg of internal max MR failed, "
1491fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "e_mr=%p iova_start=%p size_maxmr=%lx num_pages_mr=%x "
1492fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "num_pages_4k=%x", e_mr, iova_start, size_maxmr,
1493fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 num_pages_mr, num_pages_4k);
1494fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto ehca_reg_internal_maxmr_exit1;
1495fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1496fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1497fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* successful registration of all pages */
1498fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mr->ib.ib_mr.device = e_pd->ib_pd.device;
1499fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mr->ib.ib_mr.pd = &e_pd->ib_pd;
1500fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_mr->ib.ib_mr.uobject = NULL;
1501fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	atomic_inc(&(e_pd->ib_pd.usecnt));
1502fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	atomic_set(&(e_mr->ib.ib_mr.usecnt), 0);
1503fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	*e_maxmr = e_mr;
1504fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return 0;
1505fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1506fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_reg_internal_maxmr_exit1:
1507fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mr_delete(e_mr);
1508fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_reg_internal_maxmr_exit0:
1509fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
1510fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "ret=%x shca=%p e_pd=%p e_maxmr=%p",
1511fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 ret, shca, e_pd, e_maxmr);
1512fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
1513fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_reg_internal_maxmr() */
1514fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1515fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1516fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1517fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_reg_maxmr(struct ehca_shca *shca,
1518fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		   struct ehca_mr *e_newmr,
1519fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		   u64 *iova_start,
1520fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		   int acl,
1521fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		   struct ehca_pd *e_pd,
1522fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		   u32 *lkey,
1523fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		   u32 *rkey)
1524fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1525fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 h_ret;
1526fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *e_origmr = shca->maxmr;
1527fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 hipz_acl;
1528fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr_hipzout_parms hipzout = {{0},0,0,0,0,0};
1529fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1530fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mrmw_map_acl(acl, &hipz_acl);
1531fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ehca_mrmw_set_pgsize_hipz_acl(&hipz_acl);
1532fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1533fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	h_ret = hipz_h_register_smr(shca->ipz_hca_handle, e_newmr, e_origmr,
1534fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				    (u64)iova_start, hipz_acl, e_pd->fw_pd,
1535fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				    &hipzout);
1536fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (h_ret != H_SUCCESS) {
1537fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "hipz_reg_smr failed, h_ret=%lx "
1538fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "e_origmr=%p hca_hndl=%lx mr_hndl=%lx lkey=%x",
1539fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 h_ret, e_origmr, shca->ipz_hca_handle.handle,
1540fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 e_origmr->ipz_mr_handle.handle,
1541fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 e_origmr->ib.ib_mr.lkey);
1542a1a6ff11006c3a056cda9e8b13e7388fba3e69a1Hoang-Nam Nguyen		return ehca2ib_return_code(h_ret);
1543fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1544fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* successful registration */
1545fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_newmr->num_pages     = e_origmr->num_pages;
1546fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_newmr->num_4k        = e_origmr->num_4k;
1547fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_newmr->start         = iova_start;
1548fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_newmr->size          = e_origmr->size;
1549fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_newmr->acl           = acl;
1550fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_newmr->ipz_mr_handle = hipzout.handle;
1551fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	*lkey = hipzout.lkey;
1552fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	*rkey = hipzout.rkey;
1553fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return 0;
1554fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_reg_maxmr() */
1555fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1556fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1557fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1558fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_dereg_internal_maxmr(struct ehca_shca *shca)
1559fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1560fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret;
1561fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ehca_mr *e_maxmr;
1562fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_pd *ib_pd;
1563fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1564fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!shca->maxmr) {
1565fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "bad call, shca=%p", shca);
1566fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EINVAL;
1567fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto ehca_dereg_internal_maxmr_exit0;
1568fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1569fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1570fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	e_maxmr = shca->maxmr;
1571fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ib_pd = e_maxmr->ib.ib_mr.pd;
1572fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	shca->maxmr = NULL; /* remove internal max-MR indication from SHCA */
1573fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1574fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	ret = ehca_dereg_mr(&e_maxmr->ib.ib_mr);
1575fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret) {
1576fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "dereg internal max-MR failed, "
1577fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 "ret=%x e_maxmr=%p shca=%p lkey=%x",
1578fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 ret, e_maxmr, shca, e_maxmr->ib.ib_mr.lkey);
1579fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		shca->maxmr = e_maxmr;
1580fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto ehca_dereg_internal_maxmr_exit0;
1581fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1582fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1583fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	atomic_dec(&ib_pd->usecnt);
1584fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1585fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_dereg_internal_maxmr_exit0:
1586fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
1587fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_err(&shca->ib_device, "ret=%x shca=%p shca->maxmr=%p",
1588fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			 ret, shca, shca->maxmr);
1589fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
1590fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_dereg_internal_maxmr() */
1591fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1592fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1593fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1594fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*
1595fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * check physical buffer array of MR verbs for validness and
1596fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * calculates MR size
1597fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick */
1598fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_mr_chk_buf_and_calc_size(struct ib_phys_buf *phys_buf_array,
1599fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				  int num_phys_buf,
1600fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				  u64 *iova_start,
1601fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				  u64 *size)
1602fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1603fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_phys_buf *pbuf = phys_buf_array;
1604fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 size_count = 0;
1605fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 i;
1606fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1607fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (num_phys_buf == 0) {
1608fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_gen_err("bad phys buf array len, num_phys_buf=0");
1609fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		return -EINVAL;
1610fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1611fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* check first buffer */
1612fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (((u64)iova_start & ~PAGE_MASK) != (pbuf->addr & ~PAGE_MASK)) {
1613fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_gen_err("iova_start/addr mismatch, iova_start=%p "
1614fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     "pbuf->addr=%lx pbuf->size=%lx",
1615fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     iova_start, pbuf->addr, pbuf->size);
1616fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		return -EINVAL;
1617fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1618fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (((pbuf->addr + pbuf->size) % PAGE_SIZE) &&
1619fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	    (num_phys_buf > 1)) {
1620fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_gen_err("addr/size mismatch in 1st buf, pbuf->addr=%lx "
1621fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     "pbuf->size=%lx", pbuf->addr, pbuf->size);
1622fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		return -EINVAL;
1623fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1624fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1625fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	for (i = 0; i < num_phys_buf; i++) {
1626fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if ((i > 0) && (pbuf->addr % PAGE_SIZE)) {
1627fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_gen_err("bad address, i=%x pbuf->addr=%lx "
1628fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     "pbuf->size=%lx",
1629fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     i, pbuf->addr, pbuf->size);
1630fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			return -EINVAL;
1631fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1632fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (((i > 0) &&	/* not 1st */
1633fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		     (i < (num_phys_buf - 1)) &&	/* not last */
1634fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		     (pbuf->size % PAGE_SIZE)) || (pbuf->size == 0)) {
1635fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_gen_err("bad size, i=%x pbuf->size=%lx",
1636fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     i, pbuf->size);
1637fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			return -EINVAL;
1638fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1639fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		size_count += pbuf->size;
1640fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pbuf++;
1641fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1642fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1643fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	*size = size_count;
1644fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return 0;
1645fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_mr_chk_buf_and_calc_size() */
1646fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1647fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1648fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1649fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/* check page list of map FMR verb for validness */
1650fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_fmr_check_page_list(struct ehca_mr *e_fmr,
1651fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     u64 *page_list,
1652fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     int list_len)
1653fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1654fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 i;
1655fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 *page;
1656fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1657fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if ((list_len == 0) || (list_len > e_fmr->fmr_max_pages)) {
1658fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_gen_err("bad list_len, list_len=%x "
1659fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     "e_fmr->fmr_max_pages=%x fmr=%p",
1660fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     list_len, e_fmr->fmr_max_pages, e_fmr);
1661fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		return -EINVAL;
1662fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1663fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1664fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* each page must be aligned */
1665fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	page = page_list;
1666fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	for (i = 0; i < list_len; i++) {
1667fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (*page % e_fmr->fmr_page_size) {
1668fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_gen_err("bad page, i=%x *page=%lx page=%p fmr=%p "
1669fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     "fmr_page_size=%x", i, *page, page, e_fmr,
1670fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     e_fmr->fmr_page_size);
1671fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			return -EINVAL;
1672fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1673fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		page++;
1674fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1675fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1676fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return 0;
1677fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_fmr_check_page_list() */
1678fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1679fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1680fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1681fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/* setup page buffer from page info */
1682fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_set_pagebuf(struct ehca_mr *e_mr,
1683fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		     struct ehca_mr_pginfo *pginfo,
1684fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		     u32 number,
1685fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		     u64 *kpage)
1686fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1687fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret = 0;
1688fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_umem_chunk *prev_chunk;
1689fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_umem_chunk *chunk;
1690fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_phys_buf *pbuf;
1691fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 *fmrlist;
1692fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 num4k, pgaddr, offs4k;
1693fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 i = 0;
1694fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u32 j = 0;
1695fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1696fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (pginfo->type == EHCA_MR_PGI_PHYS) {
1697fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* loop over desired phys_buf_array entries */
1698fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		while (i < number) {
1699fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			pbuf   = pginfo->phys_buf_array + pginfo->next_buf;
1700fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			num4k  = ((pbuf->addr % EHCA_PAGESIZE) + pbuf->size +
1701fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				  EHCA_PAGESIZE - 1) / EHCA_PAGESIZE;
1702fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			offs4k = (pbuf->addr & ~PAGE_MASK) / EHCA_PAGESIZE;
1703fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			while (pginfo->next_4k < offs4k + num4k) {
1704fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				/* sanity check */
1705fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				if ((pginfo->page_cnt >= pginfo->num_pages) ||
1706fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				    (pginfo->page_4k_cnt >= pginfo->num_4k)) {
1707fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					ehca_gen_err("page_cnt >= num_pages, "
1708fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     "page_cnt=%lx "
1709fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     "num_pages=%lx "
1710fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     "page_4k_cnt=%lx "
1711fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     "num_4k=%lx i=%x",
1712fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     pginfo->page_cnt,
1713fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     pginfo->num_pages,
1714fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     pginfo->page_4k_cnt,
1715fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     pginfo->num_4k, i);
1716fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					ret = -EFAULT;
1717fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					goto ehca_set_pagebuf_exit0;
1718fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				}
1719fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				*kpage = phys_to_abs(
1720fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					(pbuf->addr & EHCA_PAGEMASK)
1721fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					+ (pginfo->next_4k * EHCA_PAGESIZE));
1722fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				if ( !(*kpage) && pbuf->addr ) {
1723fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					ehca_gen_err("pbuf->addr=%lx "
1724fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     "pbuf->size=%lx "
1725fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     "next_4k=%lx", pbuf->addr,
1726fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     pbuf->size,
1727fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     pginfo->next_4k);
1728fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					ret = -EFAULT;
1729fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					goto ehca_set_pagebuf_exit0;
1730fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				}
1731fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				(pginfo->page_4k_cnt)++;
1732fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				(pginfo->next_4k)++;
1733fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				if (pginfo->next_4k %
1734fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				    (PAGE_SIZE / EHCA_PAGESIZE) == 0)
1735fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					(pginfo->page_cnt)++;
1736fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				kpage++;
1737fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				i++;
1738fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				if (i >= number) break;
1739fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			}
1740fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			if (pginfo->next_4k >= offs4k + num4k) {
1741fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				(pginfo->next_buf)++;
1742fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				pginfo->next_4k = 0;
1743fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			}
1744fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1745fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else if (pginfo->type == EHCA_MR_PGI_USER) {
1746fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* loop over desired chunk entries */
1747fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		chunk      = pginfo->next_chunk;
1748fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		prev_chunk = pginfo->next_chunk;
1749fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		list_for_each_entry_continue(chunk,
1750fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     (&(pginfo->region->chunk_list)),
1751fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     list) {
1752fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			for (i = pginfo->next_nmap; i < chunk->nmap; ) {
1753fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				pgaddr = ( page_to_pfn(chunk->page_list[i].page)
1754fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					   << PAGE_SHIFT );
1755fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				*kpage = phys_to_abs(pgaddr +
1756fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     (pginfo->next_4k *
1757fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						      EHCA_PAGESIZE));
1758fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				if ( !(*kpage) ) {
1759fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					ehca_gen_err("pgaddr=%lx "
1760fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     "chunk->page_list[i]=%lx "
1761fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     "i=%x next_4k=%lx mr=%p",
1762fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     pgaddr,
1763fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     (u64)sg_dma_address(
1764fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick							     &chunk->
1765fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick							     page_list[i]),
1766fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     i, pginfo->next_4k, e_mr);
1767fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					ret = -EFAULT;
1768fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					goto ehca_set_pagebuf_exit0;
1769fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				}
1770fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				(pginfo->page_4k_cnt)++;
1771fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				(pginfo->next_4k)++;
1772fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				kpage++;
1773fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				if (pginfo->next_4k %
1774fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				    (PAGE_SIZE / EHCA_PAGESIZE) == 0) {
1775fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					(pginfo->page_cnt)++;
1776fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					(pginfo->next_nmap)++;
1777fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					pginfo->next_4k = 0;
1778fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					i++;
1779fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				}
1780fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				j++;
1781fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				if (j >= number) break;
1782fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			}
1783fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			if ((pginfo->next_nmap >= chunk->nmap) &&
1784fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			    (j >= number)) {
1785fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				pginfo->next_nmap = 0;
1786fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				prev_chunk = chunk;
1787fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				break;
1788fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			} else if (pginfo->next_nmap >= chunk->nmap) {
1789fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				pginfo->next_nmap = 0;
1790fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				prev_chunk = chunk;
1791fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			} else if (j >= number)
1792fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				break;
1793fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			else
1794fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				prev_chunk = chunk;
1795fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1796fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo->next_chunk =
1797fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			list_prepare_entry(prev_chunk,
1798fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					   (&(pginfo->region->chunk_list)),
1799fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					   list);
1800fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else if (pginfo->type == EHCA_MR_PGI_FMR) {
1801fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* loop over desired page_list entries */
1802fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		fmrlist = pginfo->page_list + pginfo->next_listelem;
1803fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		for (i = 0; i < number; i++) {
1804fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			*kpage = phys_to_abs((*fmrlist & EHCA_PAGEMASK) +
1805fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     pginfo->next_4k * EHCA_PAGESIZE);
1806fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			if ( !(*kpage) ) {
1807fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				ehca_gen_err("*fmrlist=%lx fmrlist=%p "
1808fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     "next_listelem=%lx next_4k=%lx",
1809fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     *fmrlist, fmrlist,
1810fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     pginfo->next_listelem,
1811fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     pginfo->next_4k);
1812fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				ret = -EFAULT;
1813fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				goto ehca_set_pagebuf_exit0;
1814fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			}
1815fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			(pginfo->page_4k_cnt)++;
1816fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			(pginfo->next_4k)++;
1817fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			kpage++;
1818fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			if (pginfo->next_4k %
1819fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			    (e_mr->fmr_page_size / EHCA_PAGESIZE) == 0) {
1820fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				(pginfo->page_cnt)++;
1821fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				(pginfo->next_listelem)++;
1822fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				fmrlist++;
1823fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				pginfo->next_4k = 0;
1824fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			}
1825fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1826fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else {
1827fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_gen_err("bad pginfo->type=%x", pginfo->type);
1828fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EFAULT;
1829fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto ehca_set_pagebuf_exit0;
1830fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1831fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1832fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_set_pagebuf_exit0:
1833fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
1834fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_gen_err("ret=%x e_mr=%p pginfo=%p type=%x num_pages=%lx "
1835fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     "num_4k=%lx next_buf=%lx next_4k=%lx number=%x "
1836fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     "kpage=%p page_cnt=%lx page_4k_cnt=%lx i=%x "
1837fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     "next_listelem=%lx region=%p next_chunk=%p "
1838fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     "next_nmap=%lx", ret, e_mr, pginfo, pginfo->type,
1839fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     pginfo->num_pages, pginfo->num_4k,
1840fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     pginfo->next_buf, pginfo->next_4k, number, kpage,
1841fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     pginfo->page_cnt, pginfo->page_4k_cnt, i,
1842fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     pginfo->next_listelem, pginfo->region,
1843fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     pginfo->next_chunk, pginfo->next_nmap);
1844fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
1845fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_set_pagebuf() */
1846fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1847fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1848fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1849fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/* setup 1 page from page info page buffer */
1850fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_set_pagebuf_1(struct ehca_mr *e_mr,
1851fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		       struct ehca_mr_pginfo *pginfo,
1852fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		       u64 *rpage)
1853fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1854fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	int ret = 0;
1855fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_phys_buf *tmp_pbuf;
1856fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 *fmrlist;
1857fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_umem_chunk *chunk;
1858fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	struct ib_umem_chunk *prev_chunk;
1859fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	u64 pgaddr, num4k, offs4k;
1860fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1861fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (pginfo->type == EHCA_MR_PGI_PHYS) {
1862fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		/* sanity check */
1863fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if ((pginfo->page_cnt >= pginfo->num_pages) ||
1864fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		    (pginfo->page_4k_cnt >= pginfo->num_4k)) {
1865fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_gen_err("page_cnt >= num_pages, page_cnt=%lx "
1866fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     "num_pages=%lx page_4k_cnt=%lx num_4k=%lx",
1867fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     pginfo->page_cnt, pginfo->num_pages,
1868fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     pginfo->page_4k_cnt, pginfo->num_4k);
1869fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ret = -EFAULT;
1870fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto ehca_set_pagebuf_1_exit0;
1871fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1872fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		tmp_pbuf = pginfo->phys_buf_array + pginfo->next_buf;
1873fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		num4k  = ((tmp_pbuf->addr % EHCA_PAGESIZE) + tmp_pbuf->size +
1874fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			  EHCA_PAGESIZE - 1) / EHCA_PAGESIZE;
1875fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		offs4k = (tmp_pbuf->addr & ~PAGE_MASK) / EHCA_PAGESIZE;
1876fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*rpage = phys_to_abs((tmp_pbuf->addr & EHCA_PAGEMASK) +
1877fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     (pginfo->next_4k * EHCA_PAGESIZE));
1878fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if ( !(*rpage) && tmp_pbuf->addr ) {
1879fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_gen_err("tmp_pbuf->addr=%lx"
1880fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     " tmp_pbuf->size=%lx next_4k=%lx",
1881fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     tmp_pbuf->addr, tmp_pbuf->size,
1882fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     pginfo->next_4k);
1883fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ret = -EFAULT;
1884fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto ehca_set_pagebuf_1_exit0;
1885fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1886fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		(pginfo->page_4k_cnt)++;
1887fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		(pginfo->next_4k)++;
1888fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (pginfo->next_4k % (PAGE_SIZE / EHCA_PAGESIZE) == 0)
1889fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			(pginfo->page_cnt)++;
1890fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (pginfo->next_4k >= offs4k + num4k) {
1891fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			(pginfo->next_buf)++;
1892fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			pginfo->next_4k = 0;
1893fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1894fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else if (pginfo->type == EHCA_MR_PGI_USER) {
1895fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		chunk      = pginfo->next_chunk;
1896fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		prev_chunk = pginfo->next_chunk;
1897fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		list_for_each_entry_continue(chunk,
1898fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     (&(pginfo->region->chunk_list)),
1899fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     list) {
1900fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			pgaddr = ( page_to_pfn(chunk->page_list[
1901fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						       pginfo->next_nmap].page)
1902fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				   << PAGE_SHIFT);
1903fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			*rpage = phys_to_abs(pgaddr +
1904fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     (pginfo->next_4k * EHCA_PAGESIZE));
1905fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			if ( !(*rpage) ) {
1906fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				ehca_gen_err("pgaddr=%lx chunk->page_list[]=%lx"
1907fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     " next_nmap=%lx next_4k=%lx mr=%p",
1908fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     pgaddr, (u64)sg_dma_address(
1909fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick						     &chunk->page_list[
1910fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick							     pginfo->
1911fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick							     next_nmap]),
1912fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     pginfo->next_nmap, pginfo->next_4k,
1913fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					     e_mr);
1914fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				ret = -EFAULT;
1915fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				goto ehca_set_pagebuf_1_exit0;
1916fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			}
1917fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			(pginfo->page_4k_cnt)++;
1918fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			(pginfo->next_4k)++;
1919fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			if (pginfo->next_4k %
1920fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			    (PAGE_SIZE / EHCA_PAGESIZE) == 0) {
1921fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				(pginfo->page_cnt)++;
1922fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				(pginfo->next_nmap)++;
1923fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				pginfo->next_4k = 0;
1924fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			}
1925fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			if (pginfo->next_nmap >= chunk->nmap) {
1926fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				pginfo->next_nmap = 0;
1927fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				prev_chunk = chunk;
1928fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			}
1929fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			break;
1930fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1931fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		pginfo->next_chunk =
1932fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			list_prepare_entry(prev_chunk,
1933fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					   (&(pginfo->region->chunk_list)),
1934fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick					   list);
1935fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else if (pginfo->type == EHCA_MR_PGI_FMR) {
1936fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		fmrlist = pginfo->page_list + pginfo->next_listelem;
1937fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*rpage = phys_to_abs((*fmrlist & EHCA_PAGEMASK) +
1938fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     pginfo->next_4k * EHCA_PAGESIZE);
1939fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if ( !(*rpage) ) {
1940fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ehca_gen_err("*fmrlist=%lx fmrlist=%p "
1941fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     "next_listelem=%lx next_4k=%lx",
1942fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     *fmrlist, fmrlist, pginfo->next_listelem,
1943fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     pginfo->next_4k);
1944fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			ret = -EFAULT;
1945fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			goto ehca_set_pagebuf_1_exit0;
1946fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1947fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		(pginfo->page_4k_cnt)++;
1948fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		(pginfo->next_4k)++;
1949fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		if (pginfo->next_4k %
1950fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		    (e_mr->fmr_page_size / EHCA_PAGESIZE) == 0) {
1951fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			(pginfo->page_cnt)++;
1952fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			(pginfo->next_listelem)++;
1953fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			pginfo->next_4k = 0;
1954fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		}
1955fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else {
1956fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_gen_err("bad pginfo->type=%x", pginfo->type);
1957fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ret = -EFAULT;
1958fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		goto ehca_set_pagebuf_1_exit0;
1959fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
1960fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1961fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickehca_set_pagebuf_1_exit0:
1962fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ret)
1963fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_gen_err("ret=%x e_mr=%p pginfo=%p type=%x num_pages=%lx "
1964fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     "num_4k=%lx next_buf=%lx next_4k=%lx rpage=%p "
1965fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     "page_cnt=%lx page_4k_cnt=%lx next_listelem=%lx "
1966fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     "region=%p next_chunk=%p next_nmap=%lx", ret, e_mr,
1967fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     pginfo, pginfo->type, pginfo->num_pages,
1968fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     pginfo->num_4k, pginfo->next_buf, pginfo->next_4k,
1969fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     rpage, pginfo->page_cnt, pginfo->page_4k_cnt,
1970fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     pginfo->next_listelem, pginfo->region,
1971fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			     pginfo->next_chunk, pginfo->next_nmap);
1972fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return ret;
1973fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_set_pagebuf_1() */
1974fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1975fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1976fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1977fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*
1978fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * check MR if it is a max-MR, i.e. uses whole memory
1979fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * in case it's a max-MR 1 is returned, else 0
1980fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick */
1981fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_mr_is_maxmr(u64 size,
1982fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		     u64 *iova_start)
1983fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1984fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	/* a MR is treated as max-MR only if it fits following: */
1985fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if ((size == ((u64)high_memory - PAGE_OFFSET)) &&
1986fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	    (iova_start == (void*)KERNELBASE)) {
1987fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		ehca_gen_dbg("this is a max-MR");
1988fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		return 1;
1989fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	} else
1990fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		return 0;
1991fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_mr_is_maxmr() */
1992fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1993fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
1994fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
1995fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/* map access control for MR/MW. This routine is used for MR and MW. */
1996fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickvoid ehca_mrmw_map_acl(int ib_acl,
1997fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		       u32 *hipz_acl)
1998fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
1999fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	*hipz_acl = 0;
2000fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ib_acl & IB_ACCESS_REMOTE_READ)
2001fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*hipz_acl |= HIPZ_ACCESSCTRL_R_READ;
2002fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ib_acl & IB_ACCESS_REMOTE_WRITE)
2003fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*hipz_acl |= HIPZ_ACCESSCTRL_R_WRITE;
2004fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ib_acl & IB_ACCESS_REMOTE_ATOMIC)
2005fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*hipz_acl |= HIPZ_ACCESSCTRL_R_ATOMIC;
2006fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ib_acl & IB_ACCESS_LOCAL_WRITE)
2007fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*hipz_acl |= HIPZ_ACCESSCTRL_L_WRITE;
2008fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (ib_acl & IB_ACCESS_MW_BIND)
2009fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*hipz_acl |= HIPZ_ACCESSCTRL_MW_BIND;
2010fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_mrmw_map_acl() */
2011fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
2012fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
2013fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
2014fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/* sets page size in hipz access control for MR/MW. */
2015fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickvoid ehca_mrmw_set_pgsize_hipz_acl(u32 *hipz_acl) /*INOUT*/
2016fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
2017fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return; /* HCA supports only 4k */
2018fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_mrmw_set_pgsize_hipz_acl() */
2019fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
2020fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
2021fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
2022fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*
2023fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * reverse map access control for MR/MW.
2024fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * This routine is used for MR and MW.
2025fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick */
2026fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickvoid ehca_mrmw_reverse_map_acl(const u32 *hipz_acl,
2027fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick			       int *ib_acl) /*OUT*/
2028fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
2029fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	*ib_acl = 0;
2030fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (*hipz_acl & HIPZ_ACCESSCTRL_R_READ)
2031fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*ib_acl |= IB_ACCESS_REMOTE_READ;
2032fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (*hipz_acl & HIPZ_ACCESSCTRL_R_WRITE)
2033fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*ib_acl |= IB_ACCESS_REMOTE_WRITE;
2034fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (*hipz_acl & HIPZ_ACCESSCTRL_R_ATOMIC)
2035fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*ib_acl |= IB_ACCESS_REMOTE_ATOMIC;
2036fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (*hipz_acl & HIPZ_ACCESSCTRL_L_WRITE)
2037fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*ib_acl |= IB_ACCESS_LOCAL_WRITE;
2038fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (*hipz_acl & HIPZ_ACCESSCTRL_MW_BIND)
2039fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		*ib_acl |= IB_ACCESS_MW_BIND;
2040fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_mrmw_reverse_map_acl() */
2041fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
2042fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
2043fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*----------------------------------------------------------------------*/
2044fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
2045fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick/*
2046fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * MR destructor and constructor
2047fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * used in Reregister MR verb, sets all fields in ehca_mr_t to 0,
2048fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick * except struct ib_mr and spinlock
2049fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick */
2050fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickvoid ehca_mr_deletenew(struct ehca_mr *mr)
2051fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
2052fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr->flags         = 0;
2053fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr->num_pages     = 0;
2054fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr->num_4k        = 0;
2055fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr->acl           = 0;
2056fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr->start         = NULL;
2057fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr->fmr_page_size = 0;
2058fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr->fmr_max_pages = 0;
2059fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr->fmr_max_maps  = 0;
2060fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr->fmr_map_cnt   = 0;
2061fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	memset(&mr->ipz_mr_handle, 0, sizeof(mr->ipz_mr_handle));
2062fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	memset(&mr->galpas, 0, sizeof(mr->galpas));
2063fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr->nr_of_pages   = 0;
2064fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr->pagearray     = NULL;
2065fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick} /* end ehca_mr_deletenew() */
2066fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
2067fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickint ehca_init_mrmw_cache(void)
2068fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
2069fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mr_cache = kmem_cache_create("ehca_cache_mr",
2070fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     sizeof(struct ehca_mr), 0,
2071fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     SLAB_HWCACHE_ALIGN,
2072fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     NULL, NULL);
2073fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!mr_cache)
2074fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		return -ENOMEM;
2075fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	mw_cache = kmem_cache_create("ehca_cache_mw",
2076fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     sizeof(struct ehca_mw), 0,
2077fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     SLAB_HWCACHE_ALIGN,
2078fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick				     NULL, NULL);
2079fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (!mw_cache) {
2080fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		kmem_cache_destroy(mr_cache);
2081fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		mr_cache = NULL;
2082fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		return -ENOMEM;
2083fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	}
2084fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	return 0;
2085fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick}
2086fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick
2087fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schickvoid ehca_cleanup_mrmw_cache(void)
2088fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick{
2089fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (mr_cache)
2090fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		kmem_cache_destroy(mr_cache);
2091fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick	if (mw_cache)
2092fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick		kmem_cache_destroy(mw_cache);
2093fab97220c9e409a98b1956ba677ddd2dd43b0b95Heiko J Schick}
2094