blob: 27f881758d16c541560d616b22efc16649389241 [file] [log] [blame]
Jan-Bernd Themann7a291082006-09-13 17:44:31 +02001/*
Paul Gortmaker3396c782012-01-27 13:36:01 +00002 * linux/drivers/net/ethernet/ibm/ehea/ehea_qmr.c
Jan-Bernd Themann7a291082006-09-13 17:44:31 +02003 *
4 * eHEA ethernet device driver for IBM eServer System p
5 *
6 * (C) Copyright IBM Corp. 2006
7 *
8 * Authors:
9 * Christoph Raisch <raisch@de.ibm.com>
10 * Jan-Bernd Themann <themann@de.ibm.com>
11 * Thomas Klein <tklein@de.ibm.com>
12 *
13 *
14 * This program is free software; you can redistribute it and/or modify
15 * it under the terms of the GNU General Public License as published by
16 * the Free Software Foundation; either version 2, or (at your option)
17 * any later version.
18 *
19 * This program is distributed in the hope that it will be useful,
20 * but WITHOUT ANY WARRANTY; without even the implied warranty of
21 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
22 * GNU General Public License for more details.
23 *
24 * You should have received a copy of the GNU General Public License
25 * along with this program; if not, write to the Free Software
26 * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
27 */
28
Joe Perches8c4877a2010-12-13 10:05:14 -080029#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
30
Al Virod7fe0f22006-12-03 23:15:30 -050031#include <linux/mm.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090032#include <linux/slab.h>
Jan-Bernd Themann7a291082006-09-13 17:44:31 +020033#include "ehea.h"
34#include "ehea_phyp.h"
35#include "ehea_qmr.h"
36
Thadeu Lima de Souza Cascardo1886e5d2012-01-13 08:06:32 +000037static struct ehea_bmap *ehea_bmap;
Thomas Klein44c82152007-07-11 16:32:00 +020038
Jan-Bernd Themann7a291082006-09-13 17:44:31 +020039static void *hw_qpageit_get_inc(struct hw_queue *queue)
40{
41 void *retvalue = hw_qeit_get(queue);
42
43 queue->current_q_offset += queue->pagesize;
44 if (queue->current_q_offset > queue->queue_length) {
45 queue->current_q_offset -= queue->pagesize;
46 retvalue = NULL;
47 } else if (((u64) retvalue) & (EHEA_PAGESIZE-1)) {
Joe Perches8c4877a2010-12-13 10:05:14 -080048 pr_err("not on pageboundary\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +020049 retvalue = NULL;
50 }
51 return retvalue;
52}
53
54static int hw_queue_ctor(struct hw_queue *queue, const u32 nr_of_pages,
55 const u32 pagesize, const u32 qe_size)
56{
57 int pages_per_kpage = PAGE_SIZE / pagesize;
58 int i, k;
59
60 if ((pagesize > PAGE_SIZE) || (!pages_per_kpage)) {
Joe Perches8c4877a2010-12-13 10:05:14 -080061 pr_err("pagesize conflict! kernel pagesize=%d, ehea pagesize=%d\n",
62 (int)PAGE_SIZE, (int)pagesize);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +020063 return -EINVAL;
64 }
65
66 queue->queue_length = nr_of_pages * pagesize;
Doug Maxeyf67c6272008-01-31 20:20:51 -060067 queue->queue_pages = kmalloc(nr_of_pages * sizeof(void *), GFP_KERNEL);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +020068 if (!queue->queue_pages) {
Joe Perches8c4877a2010-12-13 10:05:14 -080069 pr_err("no mem for queue_pages\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +020070 return -ENOMEM;
71 }
72
73 /*
74 * allocate pages for queue:
75 * outer loop allocates whole kernel pages (page aligned) and
76 * inner loop divides a kernel page into smaller hea queue pages
77 */
78 i = 0;
79 while (i < nr_of_pages) {
Doug Maxeyf67c6272008-01-31 20:20:51 -060080 u8 *kpage = (u8 *)get_zeroed_page(GFP_KERNEL);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +020081 if (!kpage)
82 goto out_nomem;
83 for (k = 0; k < pages_per_kpage && i < nr_of_pages; k++) {
Doug Maxeyf67c6272008-01-31 20:20:51 -060084 (queue->queue_pages)[i] = (struct ehea_page *)kpage;
Jan-Bernd Themann7a291082006-09-13 17:44:31 +020085 kpage += pagesize;
86 i++;
87 }
88 }
89
90 queue->current_q_offset = 0;
91 queue->qe_size = qe_size;
92 queue->pagesize = pagesize;
93 queue->toggle_state = 1;
94
95 return 0;
96out_nomem:
97 for (i = 0; i < nr_of_pages; i += pages_per_kpage) {
98 if (!(queue->queue_pages)[i])
99 break;
100 free_page((unsigned long)(queue->queue_pages)[i]);
101 }
102 return -ENOMEM;
103}
104
105static void hw_queue_dtor(struct hw_queue *queue)
106{
107 int pages_per_kpage = PAGE_SIZE / queue->pagesize;
108 int i, nr_pages;
109
110 if (!queue || !queue->queue_pages)
111 return;
112
113 nr_pages = queue->queue_length / queue->pagesize;
114
115 for (i = 0; i < nr_pages; i += pages_per_kpage)
116 free_page((unsigned long)(queue->queue_pages)[i]);
117
118 kfree(queue->queue_pages);
119}
120
121struct ehea_cq *ehea_create_cq(struct ehea_adapter *adapter,
122 int nr_of_cqe, u64 eq_handle, u32 cq_token)
123{
124 struct ehea_cq *cq;
125 struct h_epa epa;
126 u64 *cq_handle_ref, hret, rpage;
127 u32 act_nr_of_entries, act_pages, counter;
128 int ret;
129 void *vpage;
130
131 cq = kzalloc(sizeof(*cq), GFP_KERNEL);
132 if (!cq) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800133 pr_err("no mem for cq\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200134 goto out_nomem;
135 }
136
137 cq->attr.max_nr_of_cqes = nr_of_cqe;
138 cq->attr.cq_token = cq_token;
139 cq->attr.eq_handle = eq_handle;
140
141 cq->adapter = adapter;
142
143 cq_handle_ref = &cq->fw_handle;
144 act_nr_of_entries = 0;
145 act_pages = 0;
146
147 hret = ehea_h_alloc_resource_cq(adapter->handle, &cq->attr,
148 &cq->fw_handle, &cq->epas);
149 if (hret != H_SUCCESS) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800150 pr_err("alloc_resource_cq failed\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200151 goto out_freemem;
152 }
153
154 ret = hw_queue_ctor(&cq->hw_queue, cq->attr.nr_pages,
155 EHEA_PAGESIZE, sizeof(struct ehea_cqe));
156 if (ret)
157 goto out_freeres;
158
159 for (counter = 0; counter < cq->attr.nr_pages; counter++) {
160 vpage = hw_qpageit_get_inc(&cq->hw_queue);
161 if (!vpage) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800162 pr_err("hw_qpageit_get_inc failed\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200163 goto out_kill_hwq;
164 }
165
Michael Ellerman67e1dbc2012-07-25 21:19:55 +0000166 rpage = __pa(vpage);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200167 hret = ehea_h_register_rpage(adapter->handle,
168 0, EHEA_CQ_REGISTER_ORIG,
169 cq->fw_handle, rpage, 1);
170 if (hret < H_SUCCESS) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800171 pr_err("register_rpage_cq failed ehea_cq=%p hret=%llx counter=%i act_pages=%i\n",
172 cq, hret, counter, cq->attr.nr_pages);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200173 goto out_kill_hwq;
174 }
175
176 if (counter == (cq->attr.nr_pages - 1)) {
177 vpage = hw_qpageit_get_inc(&cq->hw_queue);
178
179 if ((hret != H_SUCCESS) || (vpage)) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800180 pr_err("registration of pages not complete hret=%llx\n",
181 hret);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200182 goto out_kill_hwq;
183 }
184 } else {
Julia Lawall662f44a2008-12-25 18:03:09 -0800185 if (hret != H_PAGE_REGISTERED) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800186 pr_err("CQ: registration of page failed hret=%llx\n",
187 hret);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200188 goto out_kill_hwq;
189 }
190 }
191 }
192
193 hw_qeit_reset(&cq->hw_queue);
194 epa = cq->epas.kernel;
195 ehea_reset_cq_ep(cq);
196 ehea_reset_cq_n1(cq);
197
198 return cq;
199
200out_kill_hwq:
201 hw_queue_dtor(&cq->hw_queue);
202
203out_freeres:
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100204 ehea_h_free_resource(adapter->handle, cq->fw_handle, FORCE_FREE);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200205
206out_freemem:
207 kfree(cq);
208
209out_nomem:
210 return NULL;
211}
212
Thadeu Lima de Souza Cascardo1886e5d2012-01-13 08:06:32 +0000213static u64 ehea_destroy_cq_res(struct ehea_cq *cq, u64 force)
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100214{
215 u64 hret;
216 u64 adapter_handle = cq->adapter->handle;
217
Jan-Bernd Themannd1d25aa2007-07-02 13:00:46 +0200218 /* deregister all previous registered pages */
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100219 hret = ehea_h_free_resource(adapter_handle, cq->fw_handle, force);
220 if (hret != H_SUCCESS)
221 return hret;
222
223 hw_queue_dtor(&cq->hw_queue);
224 kfree(cq);
225
226 return hret;
227}
228
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200229int ehea_destroy_cq(struct ehea_cq *cq)
230{
Thomas Kleinea96cea2010-04-20 23:10:55 +0000231 u64 hret, aer, aerr;
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200232 if (!cq)
233 return 0;
234
Jan-Bernd Themann28721c82007-08-22 16:21:28 +0200235 hcp_epas_dtor(&cq->epas);
Doug Maxeyf67c6272008-01-31 20:20:51 -0600236 hret = ehea_destroy_cq_res(cq, NORMAL_FREE);
237 if (hret == H_R_STATE) {
Thomas Kleinea96cea2010-04-20 23:10:55 +0000238 ehea_error_data(cq->adapter, cq->fw_handle, &aer, &aerr);
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100239 hret = ehea_destroy_cq_res(cq, FORCE_FREE);
240 }
Thomas Klein1b5135d2006-11-03 17:47:20 +0100241
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200242 if (hret != H_SUCCESS) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800243 pr_err("destroy CQ failed\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200244 return -EIO;
245 }
246
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200247 return 0;
248}
249
250struct ehea_eq *ehea_create_eq(struct ehea_adapter *adapter,
251 const enum ehea_eq_type type,
252 const u32 max_nr_of_eqes, const u8 eqe_gen)
253{
254 int ret, i;
255 u64 hret, rpage;
256 void *vpage;
257 struct ehea_eq *eq;
258
259 eq = kzalloc(sizeof(*eq), GFP_KERNEL);
260 if (!eq) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800261 pr_err("no mem for eq\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200262 return NULL;
263 }
264
265 eq->adapter = adapter;
266 eq->attr.type = type;
267 eq->attr.max_nr_of_eqes = max_nr_of_eqes;
268 eq->attr.eqe_gen = eqe_gen;
269 spin_lock_init(&eq->spinlock);
270
271 hret = ehea_h_alloc_resource_eq(adapter->handle,
272 &eq->attr, &eq->fw_handle);
273 if (hret != H_SUCCESS) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800274 pr_err("alloc_resource_eq failed\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200275 goto out_freemem;
276 }
277
278 ret = hw_queue_ctor(&eq->hw_queue, eq->attr.nr_pages,
279 EHEA_PAGESIZE, sizeof(struct ehea_eqe));
280 if (ret) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800281 pr_err("can't allocate eq pages\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200282 goto out_freeres;
283 }
284
285 for (i = 0; i < eq->attr.nr_pages; i++) {
286 vpage = hw_qpageit_get_inc(&eq->hw_queue);
287 if (!vpage) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800288 pr_err("hw_qpageit_get_inc failed\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200289 hret = H_RESOURCE;
290 goto out_kill_hwq;
291 }
292
Michael Ellerman67e1dbc2012-07-25 21:19:55 +0000293 rpage = __pa(vpage);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200294
295 hret = ehea_h_register_rpage(adapter->handle, 0,
296 EHEA_EQ_REGISTER_ORIG,
297 eq->fw_handle, rpage, 1);
298
299 if (i == (eq->attr.nr_pages - 1)) {
300 /* last page */
301 vpage = hw_qpageit_get_inc(&eq->hw_queue);
Doug Maxeyf67c6272008-01-31 20:20:51 -0600302 if ((hret != H_SUCCESS) || (vpage))
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200303 goto out_kill_hwq;
Doug Maxeyf67c6272008-01-31 20:20:51 -0600304
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200305 } else {
Julia Lawall662f44a2008-12-25 18:03:09 -0800306 if (hret != H_PAGE_REGISTERED)
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200307 goto out_kill_hwq;
Doug Maxeyf67c6272008-01-31 20:20:51 -0600308
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200309 }
310 }
311
312 hw_qeit_reset(&eq->hw_queue);
313 return eq;
314
315out_kill_hwq:
316 hw_queue_dtor(&eq->hw_queue);
317
318out_freeres:
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100319 ehea_h_free_resource(adapter->handle, eq->fw_handle, FORCE_FREE);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200320
321out_freemem:
322 kfree(eq);
323 return NULL;
324}
325
326struct ehea_eqe *ehea_poll_eq(struct ehea_eq *eq)
327{
328 struct ehea_eqe *eqe;
329 unsigned long flags;
330
331 spin_lock_irqsave(&eq->spinlock, flags);
Joe Perches43d620c2011-06-16 19:08:06 +0000332 eqe = hw_eqit_eq_get_inc_valid(&eq->hw_queue);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200333 spin_unlock_irqrestore(&eq->spinlock, flags);
334
335 return eqe;
336}
337
Thadeu Lima de Souza Cascardo1886e5d2012-01-13 08:06:32 +0000338static u64 ehea_destroy_eq_res(struct ehea_eq *eq, u64 force)
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200339{
340 u64 hret;
341 unsigned long flags;
342
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200343 spin_lock_irqsave(&eq->spinlock, flags);
344
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100345 hret = ehea_h_free_resource(eq->adapter->handle, eq->fw_handle, force);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200346 spin_unlock_irqrestore(&eq->spinlock, flags);
347
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100348 if (hret != H_SUCCESS)
349 return hret;
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200350
351 hw_queue_dtor(&eq->hw_queue);
352 kfree(eq);
353
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100354 return hret;
355}
356
357int ehea_destroy_eq(struct ehea_eq *eq)
358{
Thomas Kleinea96cea2010-04-20 23:10:55 +0000359 u64 hret, aer, aerr;
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100360 if (!eq)
361 return 0;
362
Jan-Bernd Themann28721c82007-08-22 16:21:28 +0200363 hcp_epas_dtor(&eq->epas);
364
Doug Maxeyf67c6272008-01-31 20:20:51 -0600365 hret = ehea_destroy_eq_res(eq, NORMAL_FREE);
366 if (hret == H_R_STATE) {
Thomas Kleinea96cea2010-04-20 23:10:55 +0000367 ehea_error_data(eq->adapter, eq->fw_handle, &aer, &aerr);
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100368 hret = ehea_destroy_eq_res(eq, FORCE_FREE);
369 }
370
371 if (hret != H_SUCCESS) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800372 pr_err("destroy EQ failed\n");
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100373 return -EIO;
Jan-Bernd Themannd1d25aa2007-07-02 13:00:46 +0200374 }
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100375
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200376 return 0;
377}
378
Ben Hutchings1aa8b472012-07-10 10:56:59 +0000379/* allocates memory for a queue and registers pages in phyp */
Thadeu Lima de Souza Cascardo1886e5d2012-01-13 08:06:32 +0000380static int ehea_qp_alloc_register(struct ehea_qp *qp, struct hw_queue *hw_queue,
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200381 int nr_pages, int wqe_size, int act_nr_sges,
382 struct ehea_adapter *adapter, int h_call_q_selector)
383{
384 u64 hret, rpage;
385 int ret, cnt;
386 void *vpage;
387
388 ret = hw_queue_ctor(hw_queue, nr_pages, EHEA_PAGESIZE, wqe_size);
389 if (ret)
390 return ret;
391
392 for (cnt = 0; cnt < nr_pages; cnt++) {
393 vpage = hw_qpageit_get_inc(hw_queue);
394 if (!vpage) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800395 pr_err("hw_qpageit_get_inc failed\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200396 goto out_kill_hwq;
397 }
Michael Ellerman67e1dbc2012-07-25 21:19:55 +0000398 rpage = __pa(vpage);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200399 hret = ehea_h_register_rpage(adapter->handle,
400 0, h_call_q_selector,
401 qp->fw_handle, rpage, 1);
402 if (hret < H_SUCCESS) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800403 pr_err("register_rpage_qp failed\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200404 goto out_kill_hwq;
405 }
406 }
407 hw_qeit_reset(hw_queue);
408 return 0;
409
410out_kill_hwq:
411 hw_queue_dtor(hw_queue);
412 return -EIO;
413}
414
415static inline u32 map_wqe_size(u8 wqe_enc_size)
416{
417 return 128 << wqe_enc_size;
418}
419
420struct ehea_qp *ehea_create_qp(struct ehea_adapter *adapter,
421 u32 pd, struct ehea_qp_init_attr *init_attr)
422{
423 int ret;
424 u64 hret;
425 struct ehea_qp *qp;
426 u32 wqe_size_in_bytes_sq, wqe_size_in_bytes_rq1;
427 u32 wqe_size_in_bytes_rq2, wqe_size_in_bytes_rq3;
428
429
430 qp = kzalloc(sizeof(*qp), GFP_KERNEL);
431 if (!qp) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800432 pr_err("no mem for qp\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200433 return NULL;
434 }
435
436 qp->adapter = adapter;
437
438 hret = ehea_h_alloc_resource_qp(adapter->handle, init_attr, pd,
439 &qp->fw_handle, &qp->epas);
440 if (hret != H_SUCCESS) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800441 pr_err("ehea_h_alloc_resource_qp failed\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200442 goto out_freemem;
443 }
444
445 wqe_size_in_bytes_sq = map_wqe_size(init_attr->act_wqe_size_enc_sq);
446 wqe_size_in_bytes_rq1 = map_wqe_size(init_attr->act_wqe_size_enc_rq1);
447 wqe_size_in_bytes_rq2 = map_wqe_size(init_attr->act_wqe_size_enc_rq2);
448 wqe_size_in_bytes_rq3 = map_wqe_size(init_attr->act_wqe_size_enc_rq3);
449
450 ret = ehea_qp_alloc_register(qp, &qp->hw_squeue, init_attr->nr_sq_pages,
451 wqe_size_in_bytes_sq,
452 init_attr->act_wqe_size_enc_sq, adapter,
453 0);
454 if (ret) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800455 pr_err("can't register for sq ret=%x\n", ret);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200456 goto out_freeres;
457 }
458
459 ret = ehea_qp_alloc_register(qp, &qp->hw_rqueue1,
460 init_attr->nr_rq1_pages,
461 wqe_size_in_bytes_rq1,
462 init_attr->act_wqe_size_enc_rq1,
463 adapter, 1);
464 if (ret) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800465 pr_err("can't register for rq1 ret=%x\n", ret);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200466 goto out_kill_hwsq;
467 }
468
469 if (init_attr->rq_count > 1) {
470 ret = ehea_qp_alloc_register(qp, &qp->hw_rqueue2,
471 init_attr->nr_rq2_pages,
472 wqe_size_in_bytes_rq2,
473 init_attr->act_wqe_size_enc_rq2,
474 adapter, 2);
475 if (ret) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800476 pr_err("can't register for rq2 ret=%x\n", ret);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200477 goto out_kill_hwr1q;
478 }
479 }
480
481 if (init_attr->rq_count > 2) {
482 ret = ehea_qp_alloc_register(qp, &qp->hw_rqueue3,
483 init_attr->nr_rq3_pages,
484 wqe_size_in_bytes_rq3,
485 init_attr->act_wqe_size_enc_rq3,
486 adapter, 3);
487 if (ret) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800488 pr_err("can't register for rq3 ret=%x\n", ret);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200489 goto out_kill_hwr2q;
490 }
491 }
492
493 qp->init_attr = *init_attr;
494
495 return qp;
496
497out_kill_hwr2q:
498 hw_queue_dtor(&qp->hw_rqueue2);
499
500out_kill_hwr1q:
501 hw_queue_dtor(&qp->hw_rqueue1);
502
503out_kill_hwsq:
504 hw_queue_dtor(&qp->hw_squeue);
505
506out_freeres:
507 ehea_h_disable_and_get_hea(adapter->handle, qp->fw_handle);
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100508 ehea_h_free_resource(adapter->handle, qp->fw_handle, FORCE_FREE);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200509
510out_freemem:
511 kfree(qp);
512 return NULL;
513}
514
Thadeu Lima de Souza Cascardo1886e5d2012-01-13 08:06:32 +0000515static u64 ehea_destroy_qp_res(struct ehea_qp *qp, u64 force)
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200516{
Jan-Bernd Themannd1d25aa2007-07-02 13:00:46 +0200517 u64 hret;
518 struct ehea_qp_init_attr *qp_attr = &qp->init_attr;
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200519
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200520
Jan-Bernd Themannd1d25aa2007-07-02 13:00:46 +0200521 ehea_h_disable_and_get_hea(qp->adapter->handle, qp->fw_handle);
522 hret = ehea_h_free_resource(qp->adapter->handle, qp->fw_handle, force);
523 if (hret != H_SUCCESS)
524 return hret;
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200525
Jan-Bernd Themannd1d25aa2007-07-02 13:00:46 +0200526 hw_queue_dtor(&qp->hw_squeue);
527 hw_queue_dtor(&qp->hw_rqueue1);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200528
Jan-Bernd Themannd1d25aa2007-07-02 13:00:46 +0200529 if (qp_attr->rq_count > 1)
530 hw_queue_dtor(&qp->hw_rqueue2);
531 if (qp_attr->rq_count > 2)
532 hw_queue_dtor(&qp->hw_rqueue3);
533 kfree(qp);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200534
Jan-Bernd Themannd1d25aa2007-07-02 13:00:46 +0200535 return hret;
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200536}
537
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100538int ehea_destroy_qp(struct ehea_qp *qp)
539{
Thomas Kleinea96cea2010-04-20 23:10:55 +0000540 u64 hret, aer, aerr;
Jan-Bernd Themannd1d25aa2007-07-02 13:00:46 +0200541 if (!qp)
542 return 0;
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100543
Jan-Bernd Themann28721c82007-08-22 16:21:28 +0200544 hcp_epas_dtor(&qp->epas);
545
Doug Maxeyf67c6272008-01-31 20:20:51 -0600546 hret = ehea_destroy_qp_res(qp, NORMAL_FREE);
547 if (hret == H_R_STATE) {
Thomas Kleinea96cea2010-04-20 23:10:55 +0000548 ehea_error_data(qp->adapter, qp->fw_handle, &aer, &aerr);
Jan-Bernd Themannd1d25aa2007-07-02 13:00:46 +0200549 hret = ehea_destroy_qp_res(qp, FORCE_FREE);
550 }
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100551
Jan-Bernd Themannd1d25aa2007-07-02 13:00:46 +0200552 if (hret != H_SUCCESS) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800553 pr_err("destroy QP failed\n");
Jan-Bernd Themannd1d25aa2007-07-02 13:00:46 +0200554 return -EIO;
555 }
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100556
Jan-Bernd Themannd1d25aa2007-07-02 13:00:46 +0200557 return 0;
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100558}
559
Hannes Hering48cfb142008-05-07 14:43:36 +0200560static inline int ehea_calc_index(unsigned long i, unsigned long s)
Thomas Klein44c82152007-07-11 16:32:00 +0200561{
Hannes Hering48cfb142008-05-07 14:43:36 +0200562 return (i >> s) & EHEA_INDEX_MASK;
563}
Thomas Klein44c82152007-07-11 16:32:00 +0200564
Hannes Hering48cfb142008-05-07 14:43:36 +0200565static inline int ehea_init_top_bmap(struct ehea_top_bmap *ehea_top_bmap,
566 int dir)
567{
Hannes Heringd4f12da2008-10-16 11:36:42 +0200568 if (!ehea_top_bmap->dir[dir]) {
Hannes Hering48cfb142008-05-07 14:43:36 +0200569 ehea_top_bmap->dir[dir] =
570 kzalloc(sizeof(struct ehea_dir_bmap), GFP_KERNEL);
571 if (!ehea_top_bmap->dir[dir])
572 return -ENOMEM;
573 }
574 return 0;
575}
Thomas Klein44c82152007-07-11 16:32:00 +0200576
Hannes Hering48cfb142008-05-07 14:43:36 +0200577static inline int ehea_init_bmap(struct ehea_bmap *ehea_bmap, int top, int dir)
578{
Hannes Heringd4f12da2008-10-16 11:36:42 +0200579 if (!ehea_bmap->top[top]) {
Hannes Hering48cfb142008-05-07 14:43:36 +0200580 ehea_bmap->top[top] =
581 kzalloc(sizeof(struct ehea_top_bmap), GFP_KERNEL);
582 if (!ehea_bmap->top[top])
583 return -ENOMEM;
584 }
585 return ehea_init_top_bmap(ehea_bmap->top[top], dir);
586}
Thomas Klein44c82152007-07-11 16:32:00 +0200587
Hannes Heringd4f12da2008-10-16 11:36:42 +0200588static DEFINE_MUTEX(ehea_busmap_mutex);
589static unsigned long ehea_mr_len;
590
591#define EHEA_BUSMAP_ADD_SECT 1
592#define EHEA_BUSMAP_REM_SECT 0
593
594static void ehea_rebuild_busmap(void)
Hannes Hering48cfb142008-05-07 14:43:36 +0200595{
Hannes Heringd4f12da2008-10-16 11:36:42 +0200596 u64 vaddr = EHEA_BUSMAP_START;
597 int top, dir, idx;
Thomas Klein44c82152007-07-11 16:32:00 +0200598
Hannes Heringd4f12da2008-10-16 11:36:42 +0200599 for (top = 0; top < EHEA_MAP_ENTRIES; top++) {
600 struct ehea_top_bmap *ehea_top;
601 int valid_dir_entries = 0;
602
603 if (!ehea_bmap->top[top])
604 continue;
605 ehea_top = ehea_bmap->top[top];
606 for (dir = 0; dir < EHEA_MAP_ENTRIES; dir++) {
607 struct ehea_dir_bmap *ehea_dir;
608 int valid_entries = 0;
609
610 if (!ehea_top->dir[dir])
611 continue;
612 valid_dir_entries++;
613 ehea_dir = ehea_top->dir[dir];
614 for (idx = 0; idx < EHEA_MAP_ENTRIES; idx++) {
615 if (!ehea_dir->ent[idx])
616 continue;
617 valid_entries++;
618 ehea_dir->ent[idx] = vaddr;
619 vaddr += EHEA_SECTSIZE;
620 }
621 if (!valid_entries) {
622 ehea_top->dir[dir] = NULL;
623 kfree(ehea_dir);
624 }
625 }
626 if (!valid_dir_entries) {
627 ehea_bmap->top[top] = NULL;
628 kfree(ehea_top);
629 }
630 }
631}
632
Thomas Klein3fd09c42008-10-27 10:38:46 +0100633static int ehea_update_busmap(unsigned long pfn, unsigned long nr_pages, int add)
Hannes Heringd4f12da2008-10-16 11:36:42 +0200634{
635 unsigned long i, start_section, end_section;
636
Thomas Klein3fd09c42008-10-27 10:38:46 +0100637 if (!nr_pages)
638 return 0;
639
Hannes Heringd4f12da2008-10-16 11:36:42 +0200640 if (!ehea_bmap) {
Hannes Hering70666c72008-09-05 16:36:26 +0200641 ehea_bmap = kzalloc(sizeof(struct ehea_bmap), GFP_KERNEL);
Hannes Heringd4f12da2008-10-16 11:36:42 +0200642 if (!ehea_bmap)
643 return -ENOMEM;
Thomas Klein44c82152007-07-11 16:32:00 +0200644 }
645
Hannes Heringd4f12da2008-10-16 11:36:42 +0200646 start_section = (pfn * PAGE_SIZE) / EHEA_SECTSIZE;
Thomas Klein3fd09c42008-10-27 10:38:46 +0100647 end_section = start_section + ((nr_pages * PAGE_SIZE) / EHEA_SECTSIZE);
Hannes Heringd4f12da2008-10-16 11:36:42 +0200648 /* Mark entries as valid or invalid only; address is assigned later */
649 for (i = start_section; i < end_section; i++) {
650 u64 flag;
651 int top = ehea_calc_index(i, EHEA_TOP_INDEX_SHIFT);
652 int dir = ehea_calc_index(i, EHEA_DIR_INDEX_SHIFT);
653 int idx = i & EHEA_INDEX_MASK;
Hannes Heringc5916cf2008-11-05 13:18:21 +0100654
Hannes Heringd4f12da2008-10-16 11:36:42 +0200655 if (add) {
656 int ret = ehea_init_bmap(ehea_bmap, top, dir);
657 if (ret)
658 return ret;
659 flag = 1; /* valid */
660 ehea_mr_len += EHEA_SECTSIZE;
661 } else {
662 if (!ehea_bmap->top[top])
663 continue;
664 if (!ehea_bmap->top[top]->dir[dir])
665 continue;
666 flag = 0; /* invalid */
667 ehea_mr_len -= EHEA_SECTSIZE;
668 }
Hannes Hering48cfb142008-05-07 14:43:36 +0200669
Hannes Heringd4f12da2008-10-16 11:36:42 +0200670 ehea_bmap->top[top]->dir[dir]->ent[idx] = flag;
671 }
672 ehea_rebuild_busmap(); /* Assign contiguous addresses for mr */
Thomas Klein44c82152007-07-11 16:32:00 +0200673 return 0;
674}
675
Hannes Heringd4f12da2008-10-16 11:36:42 +0200676int ehea_add_sect_bmap(unsigned long pfn, unsigned long nr_pages)
677{
678 int ret;
Hannes Hering48cfb142008-05-07 14:43:36 +0200679
Hannes Heringd4f12da2008-10-16 11:36:42 +0200680 mutex_lock(&ehea_busmap_mutex);
681 ret = ehea_update_busmap(pfn, nr_pages, EHEA_BUSMAP_ADD_SECT);
682 mutex_unlock(&ehea_busmap_mutex);
683 return ret;
684}
685
686int ehea_rem_sect_bmap(unsigned long pfn, unsigned long nr_pages)
687{
688 int ret;
689
690 mutex_lock(&ehea_busmap_mutex);
691 ret = ehea_update_busmap(pfn, nr_pages, EHEA_BUSMAP_REM_SECT);
692 mutex_unlock(&ehea_busmap_mutex);
693 return ret;
694}
695
Thomas Klein3fd09c42008-10-27 10:38:46 +0100696static int ehea_is_hugepage(unsigned long pfn)
Hannes Heringd4f12da2008-10-16 11:36:42 +0200697{
Thomas Klein3fd09c42008-10-27 10:38:46 +0100698 int page_order;
699
700 if (pfn & EHEA_HUGEPAGE_PFN_MASK)
701 return 0;
702
703 page_order = compound_order(pfn_to_page(pfn));
704 if (page_order + PAGE_SHIFT != EHEA_HUGEPAGESHIFT)
705 return 0;
706
707 return 1;
708}
709
710static int ehea_create_busmap_callback(unsigned long initial_pfn,
711 unsigned long total_nr_pages, void *arg)
712{
713 int ret;
714 unsigned long pfn, start_pfn, end_pfn, nr_pages;
715
716 if ((total_nr_pages * PAGE_SIZE) < EHEA_HUGEPAGE_SIZE)
717 return ehea_update_busmap(initial_pfn, total_nr_pages,
718 EHEA_BUSMAP_ADD_SECT);
719
720 /* Given chunk is >= 16GB -> check for hugepages */
721 start_pfn = initial_pfn;
722 end_pfn = initial_pfn + total_nr_pages;
723 pfn = start_pfn;
724
725 while (pfn < end_pfn) {
726 if (ehea_is_hugepage(pfn)) {
727 /* Add mem found in front of the hugepage */
728 nr_pages = pfn - start_pfn;
729 ret = ehea_update_busmap(start_pfn, nr_pages,
730 EHEA_BUSMAP_ADD_SECT);
731 if (ret)
732 return ret;
733
734 /* Skip the hugepage */
735 pfn += (EHEA_HUGEPAGE_SIZE / PAGE_SIZE);
736 start_pfn = pfn;
737 } else
738 pfn += (EHEA_SECTSIZE / PAGE_SIZE);
739 }
740
741 /* Add mem found behind the hugepage(s) */
742 nr_pages = pfn - start_pfn;
743 return ehea_update_busmap(start_pfn, nr_pages, EHEA_BUSMAP_ADD_SECT);
Hannes Heringd4f12da2008-10-16 11:36:42 +0200744}
Hannes Hering48cfb142008-05-07 14:43:36 +0200745
746int ehea_create_busmap(void)
747{
748 int ret;
Hannes Heringd4f12da2008-10-16 11:36:42 +0200749
Hannes Hering48cfb142008-05-07 14:43:36 +0200750 mutex_lock(&ehea_busmap_mutex);
751 ehea_mr_len = 0;
KAMEZAWA Hiroyuki908eedc2009-09-22 16:45:46 -0700752 ret = walk_system_ram_range(0, 1ULL << MAX_PHYSMEM_BITS, NULL,
Hannes Hering48cfb142008-05-07 14:43:36 +0200753 ehea_create_busmap_callback);
754 mutex_unlock(&ehea_busmap_mutex);
755 return ret;
756}
757
Doug Maxeyf67c6272008-01-31 20:20:51 -0600758void ehea_destroy_busmap(void)
Thomas Klein44c82152007-07-11 16:32:00 +0200759{
Hannes Hering48cfb142008-05-07 14:43:36 +0200760 int top, dir;
761 mutex_lock(&ehea_busmap_mutex);
762 if (!ehea_bmap)
763 goto out_destroy;
764
765 for (top = 0; top < EHEA_MAP_ENTRIES; top++) {
766 if (!ehea_bmap->top[top])
767 continue;
768
769 for (dir = 0; dir < EHEA_MAP_ENTRIES; dir++) {
770 if (!ehea_bmap->top[top]->dir[dir])
771 continue;
772
773 kfree(ehea_bmap->top[top]->dir[dir]);
774 }
775
776 kfree(ehea_bmap->top[top]);
777 }
778
779 kfree(ehea_bmap);
780 ehea_bmap = NULL;
Hannes Heringc5916cf2008-11-05 13:18:21 +0100781out_destroy:
Hannes Hering48cfb142008-05-07 14:43:36 +0200782 mutex_unlock(&ehea_busmap_mutex);
Thomas Klein44c82152007-07-11 16:32:00 +0200783}
784
785u64 ehea_map_vaddr(void *caddr)
786{
Hannes Hering48cfb142008-05-07 14:43:36 +0200787 int top, dir, idx;
788 unsigned long index, offset;
Thomas Klein44c82152007-07-11 16:32:00 +0200789
Hannes Hering48cfb142008-05-07 14:43:36 +0200790 if (!ehea_bmap)
791 return EHEA_INVAL_ADDR;
Thomas Klein44c82152007-07-11 16:32:00 +0200792
Michael Ellerman67e1dbc2012-07-25 21:19:55 +0000793 index = __pa(caddr) >> SECTION_SIZE_BITS;
Hannes Hering48cfb142008-05-07 14:43:36 +0200794 top = (index >> EHEA_TOP_INDEX_SHIFT) & EHEA_INDEX_MASK;
795 if (!ehea_bmap->top[top])
796 return EHEA_INVAL_ADDR;
Thomas Klein44c82152007-07-11 16:32:00 +0200797
Hannes Hering48cfb142008-05-07 14:43:36 +0200798 dir = (index >> EHEA_DIR_INDEX_SHIFT) & EHEA_INDEX_MASK;
799 if (!ehea_bmap->top[top]->dir[dir])
800 return EHEA_INVAL_ADDR;
801
802 idx = index & EHEA_INDEX_MASK;
803 if (!ehea_bmap->top[top]->dir[dir]->ent[idx])
804 return EHEA_INVAL_ADDR;
805
806 offset = (unsigned long)caddr & (EHEA_SECTSIZE - 1);
807 return ehea_bmap->top[top]->dir[dir]->ent[idx] | offset;
808}
809
810static inline void *ehea_calc_sectbase(int top, int dir, int idx)
811{
812 unsigned long ret = idx;
813 ret |= dir << EHEA_DIR_INDEX_SHIFT;
814 ret |= top << EHEA_TOP_INDEX_SHIFT;
Michael Ellerman67e1dbc2012-07-25 21:19:55 +0000815 return __va(ret << SECTION_SIZE_BITS);
Hannes Hering48cfb142008-05-07 14:43:36 +0200816}
817
818static u64 ehea_reg_mr_section(int top, int dir, int idx, u64 *pt,
819 struct ehea_adapter *adapter,
820 struct ehea_mr *mr)
821{
822 void *pg;
823 u64 j, m, hret;
824 unsigned long k = 0;
Michael Ellerman67e1dbc2012-07-25 21:19:55 +0000825 u64 pt_abs = __pa(pt);
Hannes Hering48cfb142008-05-07 14:43:36 +0200826
827 void *sectbase = ehea_calc_sectbase(top, dir, idx);
828
829 for (j = 0; j < (EHEA_PAGES_PER_SECTION / EHEA_MAX_RPAGE); j++) {
830
831 for (m = 0; m < EHEA_MAX_RPAGE; m++) {
832 pg = sectbase + ((k++) * EHEA_PAGESIZE);
Michael Ellerman67e1dbc2012-07-25 21:19:55 +0000833 pt[m] = __pa(pg);
Hannes Hering48cfb142008-05-07 14:43:36 +0200834 }
835 hret = ehea_h_register_rpage_mr(adapter->handle, mr->handle, 0,
836 0, pt_abs, EHEA_MAX_RPAGE);
837
Joe Perches8e95a202009-12-03 07:58:21 +0000838 if ((hret != H_SUCCESS) &&
839 (hret != H_PAGE_REGISTERED)) {
Hannes Hering48cfb142008-05-07 14:43:36 +0200840 ehea_h_free_resource(adapter->handle, mr->handle,
841 FORCE_FREE);
Joe Perches8c4877a2010-12-13 10:05:14 -0800842 pr_err("register_rpage_mr failed\n");
Hannes Hering48cfb142008-05-07 14:43:36 +0200843 return hret;
844 }
845 }
846 return hret;
847}
848
849static u64 ehea_reg_mr_sections(int top, int dir, u64 *pt,
850 struct ehea_adapter *adapter,
851 struct ehea_mr *mr)
852{
853 u64 hret = H_SUCCESS;
854 int idx;
855
856 for (idx = 0; idx < EHEA_MAP_ENTRIES; idx++) {
857 if (!ehea_bmap->top[top]->dir[dir]->ent[idx])
858 continue;
Hannes Heringc5916cf2008-11-05 13:18:21 +0100859
Hannes Hering48cfb142008-05-07 14:43:36 +0200860 hret = ehea_reg_mr_section(top, dir, idx, pt, adapter, mr);
861 if ((hret != H_SUCCESS) && (hret != H_PAGE_REGISTERED))
Hannes Heringc5916cf2008-11-05 13:18:21 +0100862 return hret;
Hannes Hering48cfb142008-05-07 14:43:36 +0200863 }
864 return hret;
865}
866
867static u64 ehea_reg_mr_dir_sections(int top, u64 *pt,
868 struct ehea_adapter *adapter,
869 struct ehea_mr *mr)
870{
871 u64 hret = H_SUCCESS;
872 int dir;
873
874 for (dir = 0; dir < EHEA_MAP_ENTRIES; dir++) {
875 if (!ehea_bmap->top[top]->dir[dir])
876 continue;
877
878 hret = ehea_reg_mr_sections(top, dir, pt, adapter, mr);
879 if ((hret != H_SUCCESS) && (hret != H_PAGE_REGISTERED))
Hannes Heringc5916cf2008-11-05 13:18:21 +0100880 return hret;
Hannes Hering48cfb142008-05-07 14:43:36 +0200881 }
882 return hret;
Thomas Klein44c82152007-07-11 16:32:00 +0200883}
884
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100885int ehea_reg_kernel_mr(struct ehea_adapter *adapter, struct ehea_mr *mr)
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200886{
Thomas Klein44c82152007-07-11 16:32:00 +0200887 int ret;
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200888 u64 *pt;
Hannes Hering48cfb142008-05-07 14:43:36 +0200889 u64 hret;
Thomas Klein44c82152007-07-11 16:32:00 +0200890 u32 acc_ctrl = EHEA_MR_ACC_CTRL;
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200891
Hannes Hering48cfb142008-05-07 14:43:36 +0200892 unsigned long top;
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200893
Nathan Lynchd76e56b2008-12-22 08:42:11 +0000894 pt = (void *)get_zeroed_page(GFP_KERNEL);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200895 if (!pt) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800896 pr_err("no mem\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200897 ret = -ENOMEM;
898 goto out;
899 }
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200900
Hannes Hering48cfb142008-05-07 14:43:36 +0200901 hret = ehea_h_alloc_resource_mr(adapter->handle, EHEA_BUSMAP_START,
902 ehea_mr_len, acc_ctrl, adapter->pd,
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100903 &mr->handle, &mr->lkey);
Hannes Hering48cfb142008-05-07 14:43:36 +0200904
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200905 if (hret != H_SUCCESS) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800906 pr_err("alloc_resource_mr failed\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200907 ret = -EIO;
908 goto out;
909 }
910
Hannes Hering48cfb142008-05-07 14:43:36 +0200911 if (!ehea_bmap) {
912 ehea_h_free_resource(adapter->handle, mr->handle, FORCE_FREE);
Joe Perches8c4877a2010-12-13 10:05:14 -0800913 pr_err("no busmap available\n");
Hannes Hering48cfb142008-05-07 14:43:36 +0200914 ret = -EIO;
915 goto out;
916 }
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200917
Hannes Hering48cfb142008-05-07 14:43:36 +0200918 for (top = 0; top < EHEA_MAP_ENTRIES; top++) {
919 if (!ehea_bmap->top[top])
920 continue;
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200921
Hannes Hering48cfb142008-05-07 14:43:36 +0200922 hret = ehea_reg_mr_dir_sections(top, pt, adapter, mr);
923 if((hret != H_PAGE_REGISTERED) && (hret != H_SUCCESS))
924 break;
925 }
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200926
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200927 if (hret != H_SUCCESS) {
Thomas Klein44c82152007-07-11 16:32:00 +0200928 ehea_h_free_resource(adapter->handle, mr->handle, FORCE_FREE);
Joe Perches8c4877a2010-12-13 10:05:14 -0800929 pr_err("registering mr failed\n");
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200930 ret = -EIO;
931 goto out;
932 }
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100933
Thomas Klein44c82152007-07-11 16:32:00 +0200934 mr->vaddr = EHEA_BUSMAP_START;
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100935 mr->adapter = adapter;
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200936 ret = 0;
937out:
Nathan Lynchd76e56b2008-12-22 08:42:11 +0000938 free_page((unsigned long)pt);
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200939 return ret;
940}
941
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100942int ehea_rem_mr(struct ehea_mr *mr)
943{
944 u64 hret;
945
946 if (!mr || !mr->adapter)
947 return -EINVAL;
948
949 hret = ehea_h_free_resource(mr->adapter->handle, mr->handle,
950 FORCE_FREE);
951 if (hret != H_SUCCESS) {
Joe Perches8c4877a2010-12-13 10:05:14 -0800952 pr_err("destroy MR failed\n");
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100953 return -EIO;
954 }
955
956 return 0;
957}
958
959int ehea_gen_smr(struct ehea_adapter *adapter, struct ehea_mr *old_mr,
960 struct ehea_mr *shared_mr)
961{
962 u64 hret;
963
964 hret = ehea_h_register_smr(adapter->handle, old_mr->handle,
965 old_mr->vaddr, EHEA_MR_ACC_CTRL,
966 adapter->pd, shared_mr);
967 if (hret != H_SUCCESS)
968 return -EIO;
969
970 shared_mr->adapter = adapter;
971
972 return 0;
973}
974
Thadeu Lima de Souza Cascardo1886e5d2012-01-13 08:06:32 +0000975static void print_error_data(u64 *data)
Jan-Bernd Themannd2db9ee2007-02-09 09:10:51 +0100976{
977 int length;
978 u64 type = EHEA_BMASK_GET(ERROR_DATA_TYPE, data[2]);
979 u64 resource = data[1];
Jan-Bernd Themann7a291082006-09-13 17:44:31 +0200980
Jan-Bernd Themannd2db9ee2007-02-09 09:10:51 +0100981 length = EHEA_BMASK_GET(ERROR_DATA_LENGTH, data[0]);
982
983 if (length > EHEA_PAGESIZE)
984 length = EHEA_PAGESIZE;
985
Thomas Kleinea96cea2010-04-20 23:10:55 +0000986 if (type == EHEA_AER_RESTYPE_QP)
Joe Perches8c4877a2010-12-13 10:05:14 -0800987 pr_err("QP (resource=%llX) state: AER=0x%llX, AERR=0x%llX, port=%llX\n",
988 resource, data[6], data[12], data[22]);
Thomas Kleinea96cea2010-04-20 23:10:55 +0000989 else if (type == EHEA_AER_RESTYPE_CQ)
Joe Perches8c4877a2010-12-13 10:05:14 -0800990 pr_err("CQ (resource=%llX) state: AER=0x%llX\n",
991 resource, data[6]);
Thomas Kleinea96cea2010-04-20 23:10:55 +0000992 else if (type == EHEA_AER_RESTYPE_EQ)
Joe Perches8c4877a2010-12-13 10:05:14 -0800993 pr_err("EQ (resource=%llX) state: AER=0x%llX\n",
994 resource, data[6]);
Jan-Bernd Themanne542aa62007-03-22 17:50:24 +0100995
Jan-Bernd Themannd2db9ee2007-02-09 09:10:51 +0100996 ehea_dump(data, length, "error data");
997}
998
Thomas Kleinea96cea2010-04-20 23:10:55 +0000999u64 ehea_error_data(struct ehea_adapter *adapter, u64 res_handle,
1000 u64 *aer, u64 *aerr)
Jan-Bernd Themannd2db9ee2007-02-09 09:10:51 +01001001{
1002 unsigned long ret;
1003 u64 *rblock;
Thomas Kleinea96cea2010-04-20 23:10:55 +00001004 u64 type = 0;
Jan-Bernd Themannd2db9ee2007-02-09 09:10:51 +01001005
Thomas Klein3faf2692009-01-21 14:45:33 -08001006 rblock = (void *)get_zeroed_page(GFP_KERNEL);
Jan-Bernd Themannd2db9ee2007-02-09 09:10:51 +01001007 if (!rblock) {
Joe Perches8c4877a2010-12-13 10:05:14 -08001008 pr_err("Cannot allocate rblock memory\n");
Thomas Kleinea96cea2010-04-20 23:10:55 +00001009 goto out;
Jan-Bernd Themannd2db9ee2007-02-09 09:10:51 +01001010 }
1011
Thomas Kleinea96cea2010-04-20 23:10:55 +00001012 ret = ehea_h_error_data(adapter->handle, res_handle, rblock);
Jan-Bernd Themannd2db9ee2007-02-09 09:10:51 +01001013
Thomas Kleinea96cea2010-04-20 23:10:55 +00001014 if (ret == H_SUCCESS) {
1015 type = EHEA_BMASK_GET(ERROR_DATA_TYPE, rblock[2]);
1016 *aer = rblock[6];
1017 *aerr = rblock[12];
Jan-Bernd Themannd2db9ee2007-02-09 09:10:51 +01001018 print_error_data(rblock);
Thomas Kleinea96cea2010-04-20 23:10:55 +00001019 } else if (ret == H_R_STATE) {
Joe Perches8c4877a2010-12-13 10:05:14 -08001020 pr_err("No error data available: %llX\n", res_handle);
Thomas Kleinea96cea2010-04-20 23:10:55 +00001021 } else
Joe Perches8c4877a2010-12-13 10:05:14 -08001022 pr_err("Error data could not be fetched: %llX\n", res_handle);
Jan-Bernd Themannd2db9ee2007-02-09 09:10:51 +01001023
Thomas Klein3faf2692009-01-21 14:45:33 -08001024 free_page((unsigned long)rblock);
Thomas Kleinea96cea2010-04-20 23:10:55 +00001025out:
1026 return type;
Jan-Bernd Themannd2db9ee2007-02-09 09:10:51 +01001027}