2007-05-09 05:00:38 +04:00
/*
* Copyright ( c ) 2007 Cisco Systems , Inc . All rights reserved .
*
* This software is available to you under a choice of one of two
* licenses . You may choose to be licensed under the terms of the GNU
* General Public License ( GPL ) Version 2 , available from the file
* COPYING in the main directory of this source tree , or the
* OpenIB . org BSD license below :
*
* Redistribution and use in source and binary forms , with or
* without modification , are permitted provided that the following
* conditions are met :
*
* - Redistributions of source code must retain the above
* copyright notice , this list of conditions and the following
* disclaimer .
*
* - Redistributions in binary form must reproduce the above
* copyright notice , this list of conditions and the following
* disclaimer in the documentation and / or other materials
* provided with the distribution .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND ,
* EXPRESS OR IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY , FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT . IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
* BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER LIABILITY , WHETHER IN AN
* ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING FROM , OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE .
*/
# include <linux/slab.h>
2019-03-31 19:10:07 +03:00
# include <rdma/uverbs_ioctl.h>
2007-05-09 05:00:38 +04:00
# include "mlx4_ib.h"
struct mlx4_ib_user_db_page {
struct list_head list ;
struct ib_umem * umem ;
unsigned long user_virt ;
int refcnt ;
} ;
2019-03-31 19:10:07 +03:00
int mlx4_ib_db_map_user ( struct ib_udata * udata , unsigned long virt ,
2008-04-23 22:55:45 +04:00
struct mlx4_db * db )
2007-05-09 05:00:38 +04:00
{
struct mlx4_ib_user_db_page * page ;
int err = 0 ;
2019-03-31 19:10:07 +03:00
struct mlx4_ib_ucontext * context = rdma_udata_to_drv_context (
udata , struct mlx4_ib_ucontext , ibucontext ) ;
2007-05-09 05:00:38 +04:00
mutex_lock ( & context - > db_page_mutex ) ;
list_for_each_entry ( page , & context - > db_page_list , list )
if ( page - > user_virt = = ( virt & PAGE_MASK ) )
goto found ;
page = kmalloc ( sizeof * page , GFP_KERNEL ) ;
if ( ! page ) {
err = - ENOMEM ;
goto out ;
}
page - > user_virt = ( virt & PAGE_MASK ) ;
page - > refcnt = 0 ;
2020-01-15 15:43:31 +03:00
page - > umem = ib_umem_get ( context - > ibucontext . device , virt & PAGE_MASK ,
PAGE_SIZE , 0 ) ;
2007-05-09 05:00:38 +04:00
if ( IS_ERR ( page - > umem ) ) {
err = PTR_ERR ( page - > umem ) ;
kfree ( page ) ;
goto out ;
}
list_add ( & page - > list , & context - > db_page_list ) ;
found :
2014-01-28 15:40:15 +04:00
db - > dma = sg_dma_address ( page - > umem - > sg_head . sgl ) + ( virt & ~ PAGE_MASK ) ;
2007-05-09 05:00:38 +04:00
db - > u . user_page = page ;
+ + page - > refcnt ;
out :
mutex_unlock ( & context - > db_page_mutex ) ;
return err ;
}
2008-04-23 22:55:45 +04:00
void mlx4_ib_db_unmap_user ( struct mlx4_ib_ucontext * context , struct mlx4_db * db )
2007-05-09 05:00:38 +04:00
{
mutex_lock ( & context - > db_page_mutex ) ;
if ( ! - - db - > u . user_page - > refcnt ) {
list_del ( & db - > u . user_page - > list ) ;
ib_umem_release ( db - > u . user_page - > umem ) ;
kfree ( db - > u . user_page ) ;
}
mutex_unlock ( & context - > db_page_mutex ) ;
}