xref: /onnv-gate/usr/src/uts/common/io/ib/clients/rdsv3/ib_rdma.c (revision 12863:5783b6fe7ed5)
112198SEiji.Ota@Sun.COM /*
212198SEiji.Ota@Sun.COM  * Copyright (c) 2010, Oracle and/or its affiliates. All rights reserved.
312198SEiji.Ota@Sun.COM  */
412198SEiji.Ota@Sun.COM 
512198SEiji.Ota@Sun.COM /*
612763SGiri.Adari@Sun.COM  * This file contains code imported from the OFED rds source file ib_rdma.c
712763SGiri.Adari@Sun.COM  * Oracle elects to have and use the contents of ib_rdma.c under and governed
812763SGiri.Adari@Sun.COM  * by the OpenIB.org BSD license (see below for full license text). However,
912763SGiri.Adari@Sun.COM  * the following notice accompanied the original version of this file:
1012763SGiri.Adari@Sun.COM  */
1112763SGiri.Adari@Sun.COM 
1212763SGiri.Adari@Sun.COM /*
1312198SEiji.Ota@Sun.COM  * Copyright (c) 2006 Oracle.  All rights reserved.
1412198SEiji.Ota@Sun.COM  *
1512198SEiji.Ota@Sun.COM  * This software is available to you under a choice of one of two
1612198SEiji.Ota@Sun.COM  * licenses.  You may choose to be licensed under the terms of the GNU
1712198SEiji.Ota@Sun.COM  * General Public License (GPL) Version 2, available from the file
1812198SEiji.Ota@Sun.COM  * COPYING in the main directory of this source tree, or the
1912198SEiji.Ota@Sun.COM  * OpenIB.org BSD license below:
2012198SEiji.Ota@Sun.COM  *
2112198SEiji.Ota@Sun.COM  *     Redistribution and use in source and binary forms, with or
2212198SEiji.Ota@Sun.COM  *     without modification, are permitted provided that the following
2312198SEiji.Ota@Sun.COM  *     conditions are met:
2412198SEiji.Ota@Sun.COM  *
2512198SEiji.Ota@Sun.COM  *      - Redistributions of source code must retain the above
2612198SEiji.Ota@Sun.COM  *        copyright notice, this list of conditions and the following
2712198SEiji.Ota@Sun.COM  *        disclaimer.
2812198SEiji.Ota@Sun.COM  *
2912198SEiji.Ota@Sun.COM  *      - Redistributions in binary form must reproduce the above
3012198SEiji.Ota@Sun.COM  *        copyright notice, this list of conditions and the following
3112198SEiji.Ota@Sun.COM  *        disclaimer in the documentation and/or other materials
3212198SEiji.Ota@Sun.COM  *        provided with the distribution.
3312198SEiji.Ota@Sun.COM  *
3412198SEiji.Ota@Sun.COM  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
3512198SEiji.Ota@Sun.COM  * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
3612198SEiji.Ota@Sun.COM  * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
3712198SEiji.Ota@Sun.COM  * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
3812198SEiji.Ota@Sun.COM  * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
3912198SEiji.Ota@Sun.COM  * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
4012198SEiji.Ota@Sun.COM  * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
4112198SEiji.Ota@Sun.COM  * SOFTWARE.
4212198SEiji.Ota@Sun.COM  *
4312198SEiji.Ota@Sun.COM  */
4412198SEiji.Ota@Sun.COM #include <sys/rds.h>
4512198SEiji.Ota@Sun.COM #include <netinet/in.h>
4612198SEiji.Ota@Sun.COM 
4712198SEiji.Ota@Sun.COM #include <sys/ib/clients/rdsv3/rdsv3.h>
4812198SEiji.Ota@Sun.COM #include <sys/ib/clients/rdsv3/rdma.h>
4912198SEiji.Ota@Sun.COM #include <sys/ib/clients/rdsv3/ib.h>
5012198SEiji.Ota@Sun.COM #include <sys/ib/clients/rdsv3/rdsv3_debug.h>
5112198SEiji.Ota@Sun.COM 
5212198SEiji.Ota@Sun.COM /*
5312198SEiji.Ota@Sun.COM  * This is stored as mr->r_trans_private.
5412198SEiji.Ota@Sun.COM  */
5512198SEiji.Ota@Sun.COM struct rdsv3_ib_mr {
5612676SEiji.Ota@Sun.COM 	list_node_t		m_obj; /* list obj of rdsv3_fmr_pool list */
5712676SEiji.Ota@Sun.COM 	struct rdsv3_ib_device	*m_device;
5812676SEiji.Ota@Sun.COM 	struct rdsv3_fmr_pool	*m_pool; /* hca fmr pool */
5912676SEiji.Ota@Sun.COM 	unsigned int		m_inval:1;
6012198SEiji.Ota@Sun.COM 
6112198SEiji.Ota@Sun.COM 	struct rdsv3_scatterlist	*sg;
6212198SEiji.Ota@Sun.COM 	unsigned int		sg_len;
6312198SEiji.Ota@Sun.COM 	uint64_t		*dma;
6412198SEiji.Ota@Sun.COM 	int			sg_dma_len;
6512198SEiji.Ota@Sun.COM 
6612198SEiji.Ota@Sun.COM 	/* DDI pinned memory */
6712198SEiji.Ota@Sun.COM 	ddi_umem_cookie_t	umem_cookie;
6812198SEiji.Ota@Sun.COM 	/* IBTF type definitions */
6912676SEiji.Ota@Sun.COM 	ibt_hca_hdl_t		rc_hca_hdl;
7012198SEiji.Ota@Sun.COM 	ibt_fmr_pool_hdl_t	fmr_pool_hdl;
7112198SEiji.Ota@Sun.COM 	ibt_ma_hdl_t		rc_ma_hdl;
7212198SEiji.Ota@Sun.COM 	ibt_mr_hdl_t		rc_fmr_hdl;
7312198SEiji.Ota@Sun.COM 	ibt_pmr_desc_t		rc_mem_desc;
7412198SEiji.Ota@Sun.COM };
7512198SEiji.Ota@Sun.COM 
7612198SEiji.Ota@Sun.COM /*
7712676SEiji.Ota@Sun.COM  * delayed freed fmr's
7812198SEiji.Ota@Sun.COM  */
7912676SEiji.Ota@Sun.COM struct rdsv3_fmr_pool {
8012676SEiji.Ota@Sun.COM 	list_t			f_list;	/* list of freed mr */
8112676SEiji.Ota@Sun.COM 	kmutex_t		f_lock; /* lock of fmr pool */
8212676SEiji.Ota@Sun.COM 	int32_t			f_listcnt;
8312198SEiji.Ota@Sun.COM };
8412198SEiji.Ota@Sun.COM 
8512198SEiji.Ota@Sun.COM static int rdsv3_ib_flush_mr_pool(struct rdsv3_ib_device *rds_ibdev,
8612198SEiji.Ota@Sun.COM 	ibt_fmr_pool_hdl_t pool_hdl, int free_all);
8712198SEiji.Ota@Sun.COM static void rdsv3_ib_teardown_mr(struct rdsv3_ib_mr *ibmr);
8812198SEiji.Ota@Sun.COM static void rdsv3_ib_mr_pool_flush_worker(struct rdsv3_work_s *work);
8912198SEiji.Ota@Sun.COM static struct rdsv3_ib_mr *rdsv3_ib_alloc_fmr(struct rdsv3_ib_device
9012198SEiji.Ota@Sun.COM 	*rds_ibdev);
9112198SEiji.Ota@Sun.COM static int rdsv3_ib_map_fmr(struct rdsv3_ib_device *rds_ibdev,
9212198SEiji.Ota@Sun.COM 	struct rdsv3_ib_mr *ibmr, struct buf *bp, unsigned int nents);
9312198SEiji.Ota@Sun.COM 
9412198SEiji.Ota@Sun.COM static struct rdsv3_ib_device *
rdsv3_ib_get_device(uint32_be_t ipaddr)9512198SEiji.Ota@Sun.COM rdsv3_ib_get_device(uint32_be_t ipaddr)
9612198SEiji.Ota@Sun.COM {
9712198SEiji.Ota@Sun.COM 	struct rdsv3_ib_device *rds_ibdev;
9812198SEiji.Ota@Sun.COM 	struct rdsv3_ib_ipaddr *i_ipaddr;
9912198SEiji.Ota@Sun.COM 
10012198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_get_device", "Enter: ipaddr: 0x%x", ipaddr);
10112198SEiji.Ota@Sun.COM 
10212198SEiji.Ota@Sun.COM 	RDSV3_FOR_EACH_LIST_NODE(rds_ibdev, &rdsv3_ib_devices, list) {
10312676SEiji.Ota@Sun.COM 		rw_enter(&rds_ibdev->rwlock, RW_READER);
10412198SEiji.Ota@Sun.COM 		RDSV3_FOR_EACH_LIST_NODE(i_ipaddr, &rds_ibdev->ipaddr_list,
10512198SEiji.Ota@Sun.COM 		    list) {
10612198SEiji.Ota@Sun.COM 			if (i_ipaddr->ipaddr == ipaddr) {
10712676SEiji.Ota@Sun.COM 				rw_exit(&rds_ibdev->rwlock);
10812198SEiji.Ota@Sun.COM 				return (rds_ibdev);
10912198SEiji.Ota@Sun.COM 			}
11012198SEiji.Ota@Sun.COM 		}
11112676SEiji.Ota@Sun.COM 		rw_exit(&rds_ibdev->rwlock);
11212198SEiji.Ota@Sun.COM 	}
11312198SEiji.Ota@Sun.COM 
11412198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_get_device", "Return: ipaddr: 0x%x", ipaddr);
11512198SEiji.Ota@Sun.COM 
11612198SEiji.Ota@Sun.COM 	return (NULL);
11712198SEiji.Ota@Sun.COM }
11812198SEiji.Ota@Sun.COM 
11912198SEiji.Ota@Sun.COM static int
rdsv3_ib_add_ipaddr(struct rdsv3_ib_device * rds_ibdev,uint32_be_t ipaddr)12012198SEiji.Ota@Sun.COM rdsv3_ib_add_ipaddr(struct rdsv3_ib_device *rds_ibdev, uint32_be_t ipaddr)
12112198SEiji.Ota@Sun.COM {
12212198SEiji.Ota@Sun.COM 	struct rdsv3_ib_ipaddr *i_ipaddr;
12312198SEiji.Ota@Sun.COM 
12412198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_add_ipaddr", "rds_ibdev: %p ipaddr: %x",
12512198SEiji.Ota@Sun.COM 	    rds_ibdev, ipaddr);
12612198SEiji.Ota@Sun.COM 
12712198SEiji.Ota@Sun.COM 	i_ipaddr = kmem_alloc(sizeof (*i_ipaddr), KM_NOSLEEP);
12812198SEiji.Ota@Sun.COM 	if (!i_ipaddr)
12912198SEiji.Ota@Sun.COM 		return (-ENOMEM);
13012198SEiji.Ota@Sun.COM 
13112198SEiji.Ota@Sun.COM 	i_ipaddr->ipaddr = ipaddr;
13212198SEiji.Ota@Sun.COM 
13312676SEiji.Ota@Sun.COM 	rw_enter(&rds_ibdev->rwlock, RW_WRITER);
13412198SEiji.Ota@Sun.COM 	list_insert_tail(&rds_ibdev->ipaddr_list, i_ipaddr);
13512676SEiji.Ota@Sun.COM 	rw_exit(&rds_ibdev->rwlock);
13612198SEiji.Ota@Sun.COM 
13712198SEiji.Ota@Sun.COM 	return (0);
13812198SEiji.Ota@Sun.COM }
13912198SEiji.Ota@Sun.COM 
14012198SEiji.Ota@Sun.COM static void
rdsv3_ib_remove_ipaddr(struct rdsv3_ib_device * rds_ibdev,uint32_be_t ipaddr)14112198SEiji.Ota@Sun.COM rdsv3_ib_remove_ipaddr(struct rdsv3_ib_device *rds_ibdev, uint32_be_t ipaddr)
14212198SEiji.Ota@Sun.COM {
14312198SEiji.Ota@Sun.COM 	struct rdsv3_ib_ipaddr *i_ipaddr, *next;
14412676SEiji.Ota@Sun.COM 	struct rdsv3_ib_ipaddr *to_free = NULL;
14512198SEiji.Ota@Sun.COM 
14612198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_remove_ipaddr", "rds_ibdev: %p, ipaddr: %x",
14712198SEiji.Ota@Sun.COM 	    rds_ibdev, ipaddr);
14812198SEiji.Ota@Sun.COM 
14912676SEiji.Ota@Sun.COM 	rw_enter(&rds_ibdev->rwlock, RW_WRITER);
15012198SEiji.Ota@Sun.COM 	RDSV3_FOR_EACH_LIST_NODE_SAFE(i_ipaddr, next, &rds_ibdev->ipaddr_list,
15112198SEiji.Ota@Sun.COM 	    list) {
15212198SEiji.Ota@Sun.COM 		if (i_ipaddr->ipaddr == ipaddr) {
15312198SEiji.Ota@Sun.COM 			list_remove_node(&i_ipaddr->list);
15412676SEiji.Ota@Sun.COM 			to_free = i_ipaddr;
15512198SEiji.Ota@Sun.COM 			break;
15612198SEiji.Ota@Sun.COM 		}
15712198SEiji.Ota@Sun.COM 	}
15812676SEiji.Ota@Sun.COM 	rw_exit(&rds_ibdev->rwlock);
15912676SEiji.Ota@Sun.COM 
16012676SEiji.Ota@Sun.COM 	if (to_free) {
16112676SEiji.Ota@Sun.COM 		kmem_free(i_ipaddr, sizeof (*i_ipaddr));
16212676SEiji.Ota@Sun.COM 	}
16312198SEiji.Ota@Sun.COM 
16412198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_remove_ipaddr",
16512198SEiji.Ota@Sun.COM 	    "Return: rds_ibdev: %p, ipaddr: %x", rds_ibdev, ipaddr);
16612198SEiji.Ota@Sun.COM }
16712198SEiji.Ota@Sun.COM 
16812198SEiji.Ota@Sun.COM int
rdsv3_ib_update_ipaddr(struct rdsv3_ib_device * rds_ibdev,uint32_be_t ipaddr)16912198SEiji.Ota@Sun.COM rdsv3_ib_update_ipaddr(struct rdsv3_ib_device *rds_ibdev, uint32_be_t ipaddr)
17012198SEiji.Ota@Sun.COM {
17112198SEiji.Ota@Sun.COM 	struct rdsv3_ib_device *rds_ibdev_old;
17212198SEiji.Ota@Sun.COM 
17312198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_update_ipaddr", "rds_ibdev: %p, ipaddr: %x",
17412198SEiji.Ota@Sun.COM 	    rds_ibdev, ipaddr);
17512198SEiji.Ota@Sun.COM 
17612198SEiji.Ota@Sun.COM 	rds_ibdev_old = rdsv3_ib_get_device(ipaddr);
17712198SEiji.Ota@Sun.COM 	if (rds_ibdev_old)
17812198SEiji.Ota@Sun.COM 		rdsv3_ib_remove_ipaddr(rds_ibdev_old, ipaddr);
17912198SEiji.Ota@Sun.COM 
18012198SEiji.Ota@Sun.COM 	return (rdsv3_ib_add_ipaddr(rds_ibdev, ipaddr));
18112198SEiji.Ota@Sun.COM }
18212198SEiji.Ota@Sun.COM 
18312198SEiji.Ota@Sun.COM void
rdsv3_ib_add_conn(struct rdsv3_ib_device * rds_ibdev,struct rdsv3_connection * conn)18412198SEiji.Ota@Sun.COM rdsv3_ib_add_conn(struct rdsv3_ib_device *rds_ibdev,
18512198SEiji.Ota@Sun.COM     struct rdsv3_connection *conn)
18612198SEiji.Ota@Sun.COM {
18712198SEiji.Ota@Sun.COM 	struct rdsv3_ib_connection *ic = conn->c_transport_data;
18812198SEiji.Ota@Sun.COM 
18912198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_add_conn", "rds_ibdev: %p, conn: %p",
19012198SEiji.Ota@Sun.COM 	    rds_ibdev, conn);
19112198SEiji.Ota@Sun.COM 
19212198SEiji.Ota@Sun.COM 	/* conn was previously on the nodev_conns_list */
19312198SEiji.Ota@Sun.COM 	mutex_enter(&ib_nodev_conns_lock);
19412198SEiji.Ota@Sun.COM 	ASSERT(!list_is_empty(&ib_nodev_conns));
19512198SEiji.Ota@Sun.COM 	ASSERT(list_link_active(&ic->ib_node));
19612198SEiji.Ota@Sun.COM 	list_remove_node(&ic->ib_node);
19712198SEiji.Ota@Sun.COM 
19812198SEiji.Ota@Sun.COM 	mutex_enter(&rds_ibdev->spinlock);
19912198SEiji.Ota@Sun.COM 	list_insert_tail(&rds_ibdev->conn_list, ic);
20012444SGiri.Adari@Sun.COM 	ic->i_on_dev_list = B_TRUE;
20112198SEiji.Ota@Sun.COM 	mutex_exit(&rds_ibdev->spinlock);
20212198SEiji.Ota@Sun.COM 	mutex_exit(&ib_nodev_conns_lock);
20312198SEiji.Ota@Sun.COM }
20412198SEiji.Ota@Sun.COM 
20512198SEiji.Ota@Sun.COM void
rdsv3_ib_remove_conn(struct rdsv3_ib_device * rds_ibdev,struct rdsv3_connection * conn)20612198SEiji.Ota@Sun.COM rdsv3_ib_remove_conn(struct rdsv3_ib_device *rds_ibdev,
20712198SEiji.Ota@Sun.COM     struct rdsv3_connection *conn)
20812198SEiji.Ota@Sun.COM {
20912198SEiji.Ota@Sun.COM 	struct rdsv3_ib_connection *ic = conn->c_transport_data;
21012198SEiji.Ota@Sun.COM 
21112198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_remove_conn", "rds_ibdev: %p, conn: %p",
21212198SEiji.Ota@Sun.COM 	    rds_ibdev, conn);
21312198SEiji.Ota@Sun.COM 
21412198SEiji.Ota@Sun.COM 	/* place conn on nodev_conns_list */
21512198SEiji.Ota@Sun.COM 	mutex_enter(&ib_nodev_conns_lock);
21612198SEiji.Ota@Sun.COM 
21712198SEiji.Ota@Sun.COM 	mutex_enter(&rds_ibdev->spinlock);
21812198SEiji.Ota@Sun.COM 	ASSERT(list_link_active(&ic->ib_node));
21912198SEiji.Ota@Sun.COM 	list_remove_node(&ic->ib_node);
22012444SGiri.Adari@Sun.COM 	ic->i_on_dev_list = B_FALSE;
22112198SEiji.Ota@Sun.COM 	mutex_exit(&rds_ibdev->spinlock);
22212198SEiji.Ota@Sun.COM 
22312198SEiji.Ota@Sun.COM 	list_insert_tail(&ib_nodev_conns, ic);
22412198SEiji.Ota@Sun.COM 
22512198SEiji.Ota@Sun.COM 	mutex_exit(&ib_nodev_conns_lock);
22612198SEiji.Ota@Sun.COM 
22712198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_remove_conn",
22812198SEiji.Ota@Sun.COM 	    "Return: rds_ibdev: %p, conn: %p", rds_ibdev, conn);
22912198SEiji.Ota@Sun.COM }
23012198SEiji.Ota@Sun.COM 
23112198SEiji.Ota@Sun.COM void
__rdsv3_ib_destroy_conns(struct list * list,kmutex_t * list_lock)23212198SEiji.Ota@Sun.COM __rdsv3_ib_destroy_conns(struct list *list, kmutex_t *list_lock)
23312198SEiji.Ota@Sun.COM {
23412198SEiji.Ota@Sun.COM 	struct rdsv3_ib_connection *ic, *_ic;
23512198SEiji.Ota@Sun.COM 	list_t tmp_list;
23612198SEiji.Ota@Sun.COM 
23712198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("__rdsv3_ib_destroy_conns", "Enter: list: %p", list);
23812198SEiji.Ota@Sun.COM 
23912198SEiji.Ota@Sun.COM 	/* avoid calling conn_destroy with irqs off */
24012198SEiji.Ota@Sun.COM 	mutex_enter(list_lock);
24112198SEiji.Ota@Sun.COM 	list_splice(list, &tmp_list);
24212198SEiji.Ota@Sun.COM 	mutex_exit(list_lock);
24312198SEiji.Ota@Sun.COM 
24412198SEiji.Ota@Sun.COM 	RDSV3_FOR_EACH_LIST_NODE_SAFE(ic, _ic, &tmp_list, ib_node) {
24512198SEiji.Ota@Sun.COM 		rdsv3_conn_destroy(ic->conn);
24612198SEiji.Ota@Sun.COM 	}
24712198SEiji.Ota@Sun.COM 
24812198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("__rdsv3_ib_destroy_conns", "Return: list: %p", list);
24912198SEiji.Ota@Sun.COM }
25012198SEiji.Ota@Sun.COM 
25112198SEiji.Ota@Sun.COM void
rdsv3_ib_destroy_mr_pool(struct rdsv3_ib_device * rds_ibdev)25212198SEiji.Ota@Sun.COM rdsv3_ib_destroy_mr_pool(struct rdsv3_ib_device *rds_ibdev)
25312198SEiji.Ota@Sun.COM {
25412676SEiji.Ota@Sun.COM 	struct rdsv3_fmr_pool *pool = rds_ibdev->fmr_pool;
25512676SEiji.Ota@Sun.COM 
25612198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_destroy_mr_pool", "Enter: ibdev: %p",
25712198SEiji.Ota@Sun.COM 	    rds_ibdev);
25812198SEiji.Ota@Sun.COM 
25912198SEiji.Ota@Sun.COM 	if (rds_ibdev->fmr_pool_hdl == NULL)
26012198SEiji.Ota@Sun.COM 		return;
26112198SEiji.Ota@Sun.COM 
26212676SEiji.Ota@Sun.COM 	if (pool) {
26312676SEiji.Ota@Sun.COM 		list_destroy(&pool->f_list);
26412676SEiji.Ota@Sun.COM 		kmem_free((void *) pool, sizeof (*pool));
26512676SEiji.Ota@Sun.COM 	}
26612676SEiji.Ota@Sun.COM 
26712198SEiji.Ota@Sun.COM 	(void) rdsv3_ib_flush_mr_pool(rds_ibdev, rds_ibdev->fmr_pool_hdl, 1);
26812198SEiji.Ota@Sun.COM 	(void) ibt_destroy_fmr_pool(ib_get_ibt_hca_hdl(rds_ibdev->dev),
26912198SEiji.Ota@Sun.COM 	    rds_ibdev->fmr_pool_hdl);
27012198SEiji.Ota@Sun.COM }
27112198SEiji.Ota@Sun.COM 
27212198SEiji.Ota@Sun.COM #define	IB_FMR_MAX_BUF_SIZE	0x1000000	/* 16MB max buf */
27312198SEiji.Ota@Sun.COM int
rdsv3_ib_create_mr_pool(struct rdsv3_ib_device * rds_ibdev)27412198SEiji.Ota@Sun.COM rdsv3_ib_create_mr_pool(struct rdsv3_ib_device *rds_ibdev)
27512198SEiji.Ota@Sun.COM {
27612198SEiji.Ota@Sun.COM 	uint_t h_page_sz;
27712198SEiji.Ota@Sun.COM 	ibt_fmr_pool_attr_t fmr_attr;
27812198SEiji.Ota@Sun.COM 	ibt_status_t ibt_status;
27912676SEiji.Ota@Sun.COM 	struct rdsv3_fmr_pool *pool;
28012198SEiji.Ota@Sun.COM 
28112198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_create_mr_pool",
28212198SEiji.Ota@Sun.COM 	    "Enter: ibdev: %p", rds_ibdev);
28312198SEiji.Ota@Sun.COM 
28412676SEiji.Ota@Sun.COM 	pool = (struct rdsv3_fmr_pool *)kmem_zalloc(sizeof (*pool), KM_NOSLEEP);
28512676SEiji.Ota@Sun.COM 	if (pool == NULL) {
28612198SEiji.Ota@Sun.COM 		return (-ENOMEM);
28712198SEiji.Ota@Sun.COM 	}
28812198SEiji.Ota@Sun.COM 
28912198SEiji.Ota@Sun.COM 	/* setup FMR pool attributes */
29012198SEiji.Ota@Sun.COM 	h_page_sz = rds_ibdev->hca_attr.hca_page_sz * 1024;
29112198SEiji.Ota@Sun.COM 
29212198SEiji.Ota@Sun.COM 	fmr_attr.fmr_max_pages_per_fmr = (IB_FMR_MAX_BUF_SIZE / h_page_sz) + 2;
29312198SEiji.Ota@Sun.COM 	fmr_attr.fmr_pool_size = RDSV3_FMR_POOL_SIZE;
29412198SEiji.Ota@Sun.COM 	fmr_attr.fmr_dirty_watermark = 128;
29512198SEiji.Ota@Sun.COM 	fmr_attr.fmr_cache = B_FALSE;
29612198SEiji.Ota@Sun.COM 	fmr_attr.fmr_flags = IBT_MR_NOSLEEP  | IBT_MR_ENABLE_LOCAL_WRITE |
29712198SEiji.Ota@Sun.COM 	    IBT_MR_ENABLE_REMOTE_WRITE | IBT_MR_ENABLE_REMOTE_READ;
29812198SEiji.Ota@Sun.COM 	fmr_attr.fmr_page_sz = h_page_sz;
29912198SEiji.Ota@Sun.COM 	fmr_attr.fmr_func_hdlr = NULL;
30012198SEiji.Ota@Sun.COM 	fmr_attr.fmr_func_arg = (void *) NULL;
30112198SEiji.Ota@Sun.COM 
30212198SEiji.Ota@Sun.COM 	/* create the FMR pool */
30312676SEiji.Ota@Sun.COM 	ibt_status = ibt_create_fmr_pool(rds_ibdev->ibt_hca_hdl,
30412676SEiji.Ota@Sun.COM 	    rds_ibdev->pd->ibt_pd, &fmr_attr, &rds_ibdev->fmr_pool_hdl);
30512198SEiji.Ota@Sun.COM 	if (ibt_status != IBT_SUCCESS) {
30612676SEiji.Ota@Sun.COM 		kmem_free((void *) pool, sizeof (*pool));
30712676SEiji.Ota@Sun.COM 		rds_ibdev->fmr_pool = NULL;
30812198SEiji.Ota@Sun.COM 		return (-ENOMEM);
30912198SEiji.Ota@Sun.COM 	}
31012676SEiji.Ota@Sun.COM 
31112676SEiji.Ota@Sun.COM 	list_create(&pool->f_list, sizeof (struct rdsv3_ib_mr),
31212676SEiji.Ota@Sun.COM 	    offsetof(struct rdsv3_ib_mr, m_obj));
31312676SEiji.Ota@Sun.COM 	mutex_init(&pool->f_lock, NULL, MUTEX_DRIVER, NULL);
31412676SEiji.Ota@Sun.COM 	rds_ibdev->fmr_pool = pool;
31512198SEiji.Ota@Sun.COM 	rds_ibdev->max_fmrs = fmr_attr.fmr_pool_size;
31612198SEiji.Ota@Sun.COM 	rds_ibdev->fmr_message_size = fmr_attr.fmr_max_pages_per_fmr;
31712676SEiji.Ota@Sun.COM 
31812676SEiji.Ota@Sun.COM 	RDSV3_DPRINTF2("rdsv3_ib_create_mr_pool",
31912676SEiji.Ota@Sun.COM 	    "Exit: ibdev: %p fmr_pool: %p", rds_ibdev, pool);
32012198SEiji.Ota@Sun.COM 	return (0);
32112198SEiji.Ota@Sun.COM }
32212198SEiji.Ota@Sun.COM 
32312198SEiji.Ota@Sun.COM void
rdsv3_ib_get_mr_info(struct rdsv3_ib_device * rds_ibdev,struct rds_info_rdma_connection * iinfo)32412198SEiji.Ota@Sun.COM rdsv3_ib_get_mr_info(struct rdsv3_ib_device *rds_ibdev,
325*12863SEiji.Ota@Sun.COM 	struct rds_info_rdma_connection *iinfo)
32612198SEiji.Ota@Sun.COM {
32712198SEiji.Ota@Sun.COM 	iinfo->rdma_mr_max = rds_ibdev->max_fmrs;
32812198SEiji.Ota@Sun.COM 	iinfo->rdma_mr_size = rds_ibdev->fmr_message_size;
32912198SEiji.Ota@Sun.COM }
33012198SEiji.Ota@Sun.COM 
33112198SEiji.Ota@Sun.COM void *
rdsv3_ib_get_mr(struct rds_iovec * args,unsigned long nents,struct rdsv3_sock * rs,uint32_t * key_ret)332*12863SEiji.Ota@Sun.COM rdsv3_ib_get_mr(struct rds_iovec *args, unsigned long nents,
33312198SEiji.Ota@Sun.COM 	struct rdsv3_sock *rs, uint32_t *key_ret)
33412198SEiji.Ota@Sun.COM {
33512198SEiji.Ota@Sun.COM 	struct rdsv3_ib_device *rds_ibdev;
33612198SEiji.Ota@Sun.COM 	struct rdsv3_ib_mr *ibmr = NULL;
33712198SEiji.Ota@Sun.COM 	ddi_umem_cookie_t umem_cookie;
33812198SEiji.Ota@Sun.COM 	size_t umem_len;
33912198SEiji.Ota@Sun.COM 	caddr_t umem_addr;
34012198SEiji.Ota@Sun.COM 	int ret;
34112198SEiji.Ota@Sun.COM 	struct buf *bp;
34212198SEiji.Ota@Sun.COM 
34312198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_get_mr", "Enter: args.addr: %p", args->addr);
34412198SEiji.Ota@Sun.COM 
34512198SEiji.Ota@Sun.COM 	rds_ibdev = rdsv3_ib_get_device(rs->rs_bound_addr);
34612198SEiji.Ota@Sun.COM 
34712198SEiji.Ota@Sun.COM 	if (rds_ibdev == NULL)
34812198SEiji.Ota@Sun.COM 		return (void *)(PTR_ERR(-EFAULT));
34912198SEiji.Ota@Sun.COM 
35012198SEiji.Ota@Sun.COM 	ibmr = rdsv3_ib_alloc_fmr(rds_ibdev);
35112198SEiji.Ota@Sun.COM 	if (IS_ERR(ibmr))
35212198SEiji.Ota@Sun.COM 		return (ibmr);
35312198SEiji.Ota@Sun.COM 
35412198SEiji.Ota@Sun.COM 	/* pin user memory pages */
35512198SEiji.Ota@Sun.COM 	umem_len   = ptob(btopr(args->bytes +
35612198SEiji.Ota@Sun.COM 	    ((uintptr_t)args->addr & PAGEOFFSET)));
35712198SEiji.Ota@Sun.COM 	umem_addr  = (caddr_t)((uintptr_t)args->addr & ~PAGEOFFSET);
35812444SGiri.Adari@Sun.COM 	ret = umem_lockmemory(umem_addr, umem_len,
35912444SGiri.Adari@Sun.COM 	    DDI_UMEMLOCK_WRITE | DDI_UMEMLOCK_READ,
36012444SGiri.Adari@Sun.COM 	    &umem_cookie, NULL, NULL);
36112198SEiji.Ota@Sun.COM 	if (ret != 0) {
36212198SEiji.Ota@Sun.COM 		kmem_free((void *) ibmr, sizeof (*ibmr));
36312580SGiri.Adari@Sun.COM 		ibmr = ERR_PTR(-ret);
36412198SEiji.Ota@Sun.COM 		return (ibmr);
36512198SEiji.Ota@Sun.COM 	}
36612198SEiji.Ota@Sun.COM 
36712198SEiji.Ota@Sun.COM 	/* transpose umem_cookie to buf structure for rdsv3_ib_map_fmr() */
36812198SEiji.Ota@Sun.COM 	bp = ddi_umem_iosetup(umem_cookie, 0, umem_len,
36912198SEiji.Ota@Sun.COM 	    B_WRITE, 0, 0, NULL, DDI_UMEM_SLEEP);
37012198SEiji.Ota@Sun.COM 
37112198SEiji.Ota@Sun.COM 	ret = rdsv3_ib_map_fmr(rds_ibdev, ibmr, bp, nents);
37212198SEiji.Ota@Sun.COM 	freerbuf(bp);	/* free bp */
37312198SEiji.Ota@Sun.COM 	if (ret == 0) {
37412198SEiji.Ota@Sun.COM 		ibmr->umem_cookie = umem_cookie;
37512198SEiji.Ota@Sun.COM 		*key_ret = (uint32_t)ibmr->rc_mem_desc.pmd_rkey;
37612676SEiji.Ota@Sun.COM 		ibmr->m_device = rds_ibdev;
37712676SEiji.Ota@Sun.COM 		ibmr->m_pool = rds_ibdev->fmr_pool;
37812198SEiji.Ota@Sun.COM 		RDSV3_DPRINTF4("rdsv3_ib_get_mr",
37912198SEiji.Ota@Sun.COM 		    "Return: ibmr: %p umem_cookie %p", ibmr, ibmr->umem_cookie);
38012198SEiji.Ota@Sun.COM 		return (ibmr);
38112198SEiji.Ota@Sun.COM 	} else { /* error return */
38212320SGiri.Adari@Sun.COM 		RDSV3_DPRINTF2("rdsv3_ib_get_mr", "map_fmr failed (errno=%d)\n",
38312198SEiji.Ota@Sun.COM 		    ret);
38412198SEiji.Ota@Sun.COM 		ddi_umem_unlock(umem_cookie);
38512198SEiji.Ota@Sun.COM 		kmem_free((void *)ibmr, sizeof (*ibmr));
38612198SEiji.Ota@Sun.COM 		return (ERR_PTR(ret));
38712198SEiji.Ota@Sun.COM 	}
38812198SEiji.Ota@Sun.COM }
38912198SEiji.Ota@Sun.COM 
39012198SEiji.Ota@Sun.COM static struct rdsv3_ib_mr *
rdsv3_ib_alloc_fmr(struct rdsv3_ib_device * rds_ibdev)39112198SEiji.Ota@Sun.COM rdsv3_ib_alloc_fmr(struct rdsv3_ib_device *rds_ibdev)
39212198SEiji.Ota@Sun.COM {
39312198SEiji.Ota@Sun.COM 	struct rdsv3_ib_mr *ibmr;
39412198SEiji.Ota@Sun.COM 
39512198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_alloc_fmr", "Enter: ibdev: %p", rds_ibdev);
39612198SEiji.Ota@Sun.COM 
39712198SEiji.Ota@Sun.COM 	if (rds_ibdev->fmr_pool_hdl) {
39812198SEiji.Ota@Sun.COM 		ibmr = (struct rdsv3_ib_mr *)kmem_zalloc(sizeof (*ibmr),
39912198SEiji.Ota@Sun.COM 		    KM_SLEEP);
40012676SEiji.Ota@Sun.COM 		ibmr->rc_hca_hdl = ib_get_ibt_hca_hdl(rds_ibdev->dev);
40112198SEiji.Ota@Sun.COM 		ibmr->fmr_pool_hdl = rds_ibdev->fmr_pool_hdl;
40212198SEiji.Ota@Sun.COM 		return (ibmr);
40312198SEiji.Ota@Sun.COM 	}
40412198SEiji.Ota@Sun.COM 	return (struct rdsv3_ib_mr *)(PTR_ERR(-ENOMEM));
40512198SEiji.Ota@Sun.COM }
40612198SEiji.Ota@Sun.COM 
40712198SEiji.Ota@Sun.COM static int
rdsv3_ib_map_fmr(struct rdsv3_ib_device * rds_ibdev,struct rdsv3_ib_mr * ibmr,struct buf * bp,unsigned int nents)40812198SEiji.Ota@Sun.COM rdsv3_ib_map_fmr(struct rdsv3_ib_device *rds_ibdev, struct rdsv3_ib_mr *ibmr,
40912198SEiji.Ota@Sun.COM 	struct buf *bp, unsigned int nents)
41012198SEiji.Ota@Sun.COM {
41112198SEiji.Ota@Sun.COM 	ibt_va_attr_t va_attr;
41212198SEiji.Ota@Sun.COM 	ibt_reg_req_t reg_req;
41312198SEiji.Ota@Sun.COM 	uint_t paddr_list_len;
41412198SEiji.Ota@Sun.COM 	uint_t page_sz;
41512198SEiji.Ota@Sun.COM 	ibt_status_t ibt_status;
41612198SEiji.Ota@Sun.COM 	/* LINTED E_FUNC_SET_NOT_USED */
41712198SEiji.Ota@Sun.COM 	unsigned int l_nents = nents;
41812198SEiji.Ota@Sun.COM 
41912198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_map_fmr", "Enter: ibmr: %p", ibmr);
42012198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_map_fmr", "buf addr: %p", bp->b_un.b_addr);
42112198SEiji.Ota@Sun.COM 
42212198SEiji.Ota@Sun.COM 	/* setup ibt_map_mem_area attributes */
42312198SEiji.Ota@Sun.COM 	bzero(&va_attr, sizeof (ibt_va_attr_t));
42412198SEiji.Ota@Sun.COM 	va_attr.va_buf   = bp;
42512198SEiji.Ota@Sun.COM 	va_attr.va_flags = IBT_VA_FMR | IBT_VA_BUF;
42612198SEiji.Ota@Sun.COM 
42712198SEiji.Ota@Sun.COM 	page_sz = rds_ibdev->hca_attr.hca_page_sz * 1024; /* in kbytes */
42812198SEiji.Ota@Sun.COM 	paddr_list_len = (bp->b_bcount / page_sz) + 2; /* start + end pg */
42912198SEiji.Ota@Sun.COM 
43012198SEiji.Ota@Sun.COM 	/* map user buffer to HCA address */
43112676SEiji.Ota@Sun.COM 	ibt_status = ibt_map_mem_area(ibmr->rc_hca_hdl,
43212198SEiji.Ota@Sun.COM 	    &va_attr, paddr_list_len, &reg_req, &ibmr->rc_ma_hdl);
43312198SEiji.Ota@Sun.COM 	if (ibt_status != IBT_SUCCESS) {
43412198SEiji.Ota@Sun.COM 		return (-ENOMEM);
43512198SEiji.Ota@Sun.COM 	}
43612198SEiji.Ota@Sun.COM 
43712198SEiji.Ota@Sun.COM 	/*  use a free entry from FMR pool to register the specified memory */
43812676SEiji.Ota@Sun.COM 	ibt_status = ibt_register_physical_fmr(ibmr->rc_hca_hdl,
43912676SEiji.Ota@Sun.COM 	    ibmr->fmr_pool_hdl,
44012198SEiji.Ota@Sun.COM 	    &reg_req.fn_arg, &ibmr->rc_fmr_hdl, &ibmr->rc_mem_desc);
44112198SEiji.Ota@Sun.COM 	if (ibt_status != IBT_SUCCESS) {
44212676SEiji.Ota@Sun.COM 		RDSV3_DPRINTF2("rdsv3_ib_map_fmr", "reg_phy_fmr failed %d",
44312676SEiji.Ota@Sun.COM 		    ibt_status);
44412676SEiji.Ota@Sun.COM 		(void) ibt_unmap_mem_area(ibmr->rc_hca_hdl,
44512198SEiji.Ota@Sun.COM 		    ibmr->rc_ma_hdl);
44612198SEiji.Ota@Sun.COM 		if (ibt_status == IBT_INSUFF_RESOURCE) {
44712198SEiji.Ota@Sun.COM 			return (-ENOBUFS);
44812198SEiji.Ota@Sun.COM 		}
44912198SEiji.Ota@Sun.COM 		return (-EINVAL);
45012198SEiji.Ota@Sun.COM 	}
45112198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_map_fmr", "Return: ibmr: %p rkey: 0x%x",
45212198SEiji.Ota@Sun.COM 	    ibmr, (uint32_t)ibmr->rc_mem_desc.pmd_rkey);
45312198SEiji.Ota@Sun.COM 	return (0);
45412198SEiji.Ota@Sun.COM }
45512198SEiji.Ota@Sun.COM 
45612198SEiji.Ota@Sun.COM void
rdsv3_ib_sync_mr(void * trans_private,int direction)45712198SEiji.Ota@Sun.COM rdsv3_ib_sync_mr(void *trans_private, int direction)
45812198SEiji.Ota@Sun.COM {
45912198SEiji.Ota@Sun.COM 	/* LINTED E_FUNC_SET_NOT_USED */
46012198SEiji.Ota@Sun.COM 	void *l_trans_private = trans_private;
46112198SEiji.Ota@Sun.COM 	/* LINTED E_FUNC_SET_NOT_USED */
46212198SEiji.Ota@Sun.COM 	int l_direction = direction;
46312198SEiji.Ota@Sun.COM 
46412198SEiji.Ota@Sun.COM 	/* FMR Sync not needed in Solaris on PCI-ex systems */
46512198SEiji.Ota@Sun.COM 
46612198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_sync_mr", "Enter:");
46712198SEiji.Ota@Sun.COM }
46812198SEiji.Ota@Sun.COM 
46912198SEiji.Ota@Sun.COM void
rdsv3_ib_flush_mrs(void)47012198SEiji.Ota@Sun.COM rdsv3_ib_flush_mrs(void)
47112198SEiji.Ota@Sun.COM {
47212198SEiji.Ota@Sun.COM 	struct rdsv3_ib_device *rds_ibdev;
47312198SEiji.Ota@Sun.COM 
47412198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_flush_mrs", "Enter:");
47512198SEiji.Ota@Sun.COM 
47612198SEiji.Ota@Sun.COM 	RDSV3_FOR_EACH_LIST_NODE(rds_ibdev, &rdsv3_ib_devices, list) {
47712198SEiji.Ota@Sun.COM 		if (rds_ibdev->fmr_pool_hdl) {
47812198SEiji.Ota@Sun.COM 			(void) rdsv3_ib_flush_mr_pool(rds_ibdev,
47912198SEiji.Ota@Sun.COM 			    rds_ibdev->fmr_pool_hdl, 0);
48012198SEiji.Ota@Sun.COM 		}
48112198SEiji.Ota@Sun.COM 	}
48212198SEiji.Ota@Sun.COM }
48312198SEiji.Ota@Sun.COM 
48412198SEiji.Ota@Sun.COM static void
rdsv3_ib_drop_mr(struct rdsv3_ib_mr * ibmr)48512676SEiji.Ota@Sun.COM rdsv3_ib_drop_mr(struct rdsv3_ib_mr *ibmr)
48612676SEiji.Ota@Sun.COM {
48712676SEiji.Ota@Sun.COM 	/* return the fmr to the IBTF pool */
48812676SEiji.Ota@Sun.COM 	(void) ibt_deregister_fmr(ibmr->rc_hca_hdl, ibmr->rc_fmr_hdl);
48912676SEiji.Ota@Sun.COM 	(void) ibt_unmap_mem_area(ibmr->rc_hca_hdl, ibmr->rc_ma_hdl);
49012676SEiji.Ota@Sun.COM 	(void) ddi_umem_unlock(ibmr->umem_cookie);
49112676SEiji.Ota@Sun.COM 	kmem_free((void *) ibmr, sizeof (*ibmr));
49212676SEiji.Ota@Sun.COM }
49312676SEiji.Ota@Sun.COM 
49412676SEiji.Ota@Sun.COM void
rdsv3_ib_drain_mrlist_fn(void * data)49512676SEiji.Ota@Sun.COM rdsv3_ib_drain_mrlist_fn(void *data)
49612198SEiji.Ota@Sun.COM {
49712676SEiji.Ota@Sun.COM 	struct rdsv3_fmr_pool *pool = (struct rdsv3_fmr_pool *)data;
49812676SEiji.Ota@Sun.COM 	ibt_hca_hdl_t hca_hdl;
49912676SEiji.Ota@Sun.COM 	ibt_fmr_pool_hdl_t fmr_pool_hdl;
50012676SEiji.Ota@Sun.COM 	unsigned int inval;
50112676SEiji.Ota@Sun.COM 	struct rdsv3_ib_mr *ibmr;
50212676SEiji.Ota@Sun.COM 	list_t *listp = &pool->f_list;
50312676SEiji.Ota@Sun.COM 	kmutex_t *lockp = &pool->f_lock;
50412676SEiji.Ota@Sun.COM 	int i;
50512198SEiji.Ota@Sun.COM 
50612676SEiji.Ota@Sun.COM 	inval = 0;
50712676SEiji.Ota@Sun.COM 	i = 0;
50812676SEiji.Ota@Sun.COM 	for (;;) {
50912676SEiji.Ota@Sun.COM 		mutex_enter(lockp);
51012676SEiji.Ota@Sun.COM 		ibmr = (struct rdsv3_ib_mr *)list_remove_head(listp);
51112676SEiji.Ota@Sun.COM 		if (ibmr)
51212676SEiji.Ota@Sun.COM 			pool->f_listcnt--;
51312676SEiji.Ota@Sun.COM 		mutex_exit(lockp);
51412676SEiji.Ota@Sun.COM 		if (!ibmr)
51512676SEiji.Ota@Sun.COM 			break;
51612676SEiji.Ota@Sun.COM 		if ((inval == 0) && ibmr->m_inval) {
51712676SEiji.Ota@Sun.COM 			inval = 1;
51812676SEiji.Ota@Sun.COM 			hca_hdl = ibmr->rc_hca_hdl;
51912676SEiji.Ota@Sun.COM 			fmr_pool_hdl = ibmr->fmr_pool_hdl;
52012676SEiji.Ota@Sun.COM 		}
52112676SEiji.Ota@Sun.COM 		i++;
52212676SEiji.Ota@Sun.COM 		rdsv3_ib_drop_mr(ibmr);
52312676SEiji.Ota@Sun.COM 	}
52412676SEiji.Ota@Sun.COM 	if (inval)
52512676SEiji.Ota@Sun.COM 		(void) ibt_flush_fmr_pool(hca_hdl, fmr_pool_hdl);
52612198SEiji.Ota@Sun.COM }
52712198SEiji.Ota@Sun.COM 
52812198SEiji.Ota@Sun.COM void
rdsv3_ib_free_mr(void * trans_private,int invalidate)52912198SEiji.Ota@Sun.COM rdsv3_ib_free_mr(void *trans_private, int invalidate)
53012198SEiji.Ota@Sun.COM {
53112198SEiji.Ota@Sun.COM 	struct rdsv3_ib_mr *ibmr = trans_private;
53212676SEiji.Ota@Sun.COM 	rdsv3_af_thr_t *af_thr;
53312198SEiji.Ota@Sun.COM 
53412198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_free_mr", "Enter: ibmr: %p inv: %d",
53512198SEiji.Ota@Sun.COM 	    ibmr, invalidate);
53612198SEiji.Ota@Sun.COM 
53712676SEiji.Ota@Sun.COM 	/* save af_thr at local as ibmr might be freed at mutex_exit */
53812676SEiji.Ota@Sun.COM 	af_thr = ibmr->m_device->fmr_soft_cq;
53912676SEiji.Ota@Sun.COM 	ibmr->m_inval = (unsigned int) invalidate;
54012676SEiji.Ota@Sun.COM 	mutex_enter(&ibmr->m_pool->f_lock);
54112676SEiji.Ota@Sun.COM 	list_insert_tail(&ibmr->m_pool->f_list, ibmr);
54212676SEiji.Ota@Sun.COM 	ibmr->m_pool->f_listcnt++;
54312676SEiji.Ota@Sun.COM 	mutex_exit(&ibmr->m_pool->f_lock);
54412676SEiji.Ota@Sun.COM 
54512676SEiji.Ota@Sun.COM 	rdsv3_af_thr_fire(af_thr);
54612198SEiji.Ota@Sun.COM }
54712198SEiji.Ota@Sun.COM 
54812198SEiji.Ota@Sun.COM static int
rdsv3_ib_flush_mr_pool(struct rdsv3_ib_device * rds_ibdev,ibt_fmr_pool_hdl_t pool_hdl,int free_all)54912198SEiji.Ota@Sun.COM rdsv3_ib_flush_mr_pool(struct rdsv3_ib_device *rds_ibdev,
55012198SEiji.Ota@Sun.COM     ibt_fmr_pool_hdl_t pool_hdl, int free_all)
55112198SEiji.Ota@Sun.COM {
55212198SEiji.Ota@Sun.COM 	/* LINTED E_FUNC_SET_NOT_USED */
55312198SEiji.Ota@Sun.COM 	int l_free_all = free_all;
55412198SEiji.Ota@Sun.COM 
55512198SEiji.Ota@Sun.COM 	RDSV3_DPRINTF4("rdsv3_ib_flush_mr_pool", "Enter: pool: %p", pool_hdl);
55612198SEiji.Ota@Sun.COM 
55712198SEiji.Ota@Sun.COM 	rdsv3_ib_stats_inc(s_ib_rdma_mr_pool_flush);
55812198SEiji.Ota@Sun.COM 
55912198SEiji.Ota@Sun.COM 	(void) ibt_flush_fmr_pool(ib_get_ibt_hca_hdl(rds_ibdev->dev),
56012198SEiji.Ota@Sun.COM 	    pool_hdl);
56112198SEiji.Ota@Sun.COM 	return (0);
56212198SEiji.Ota@Sun.COM }
563