2019-05-19 15:08:55 +03:00
// SPDX-License-Identifier: GPL-2.0-only
2016-04-05 00:48:11 +03:00
# include <linux/module.h>
# include <linux/scatterlist.h>
# include <linux/mempool.h>
# include <linux/slab.h>
# define SG_MEMPOOL_NR ARRAY_SIZE(sg_pools)
# define SG_MEMPOOL_SIZE 2
struct sg_pool {
size_t size ;
char * name ;
struct kmem_cache * slab ;
mempool_t * pool ;
} ;
# define SP(x) { .size = x, "sgpool-" __stringify(x) }
# if (SG_CHUNK_SIZE < 32)
# error SG_CHUNK_SIZE is too small (must be 32 or greater)
# endif
static struct sg_pool sg_pools [ ] = {
SP ( 8 ) ,
SP ( 16 ) ,
# if (SG_CHUNK_SIZE > 32)
SP ( 32 ) ,
# if (SG_CHUNK_SIZE > 64)
SP ( 64 ) ,
# if (SG_CHUNK_SIZE > 128)
SP ( 128 ) ,
# if (SG_CHUNK_SIZE > 256)
# error SG_CHUNK_SIZE is too large (256 MAX)
# endif
# endif
# endif
# endif
SP ( SG_CHUNK_SIZE )
} ;
# undef SP
static inline unsigned int sg_pool_index ( unsigned short nents )
{
unsigned int index ;
BUG_ON ( nents > SG_CHUNK_SIZE ) ;
if ( nents < = 8 )
index = 0 ;
else
index = get_count_order ( nents ) - 3 ;
return index ;
}
static void sg_pool_free ( struct scatterlist * sgl , unsigned int nents )
{
struct sg_pool * sgp ;
sgp = sg_pools + sg_pool_index ( nents ) ;
mempool_free ( sgl , sgp - > pool ) ;
}
static struct scatterlist * sg_pool_alloc ( unsigned int nents , gfp_t gfp_mask )
{
struct sg_pool * sgp ;
sgp = sg_pools + sg_pool_index ( nents ) ;
return mempool_alloc ( sgp - > pool , gfp_mask ) ;
}
/**
* sg_free_table_chained - Free a previously mapped sg table
* @ table : The sg table header to use
2019-04-28 10:39:30 +03:00
* @ nents_first_chunk : size of the first_chunk SGL passed to
* sg_alloc_table_chained
2016-04-05 00:48:11 +03:00
*
* Description :
* Free an sg table previously allocated and setup with
* sg_alloc_table_chained ( ) .
*
2019-04-28 10:39:30 +03:00
* @ nents_first_chunk has to be same with that same parameter passed
* to sg_alloc_table_chained ( ) .
*
2016-04-05 00:48:11 +03:00
* */
2019-04-28 10:39:30 +03:00
void sg_free_table_chained ( struct sg_table * table ,
unsigned nents_first_chunk )
2016-04-05 00:48:11 +03:00
{
2019-04-28 10:39:30 +03:00
if ( table - > orig_nents < = nents_first_chunk )
2016-04-05 00:48:11 +03:00
return ;
2019-04-28 10:39:30 +03:00
if ( nents_first_chunk = = 1 )
nents_first_chunk = 0 ;
__sg_free_table ( table , SG_CHUNK_SIZE , nents_first_chunk , sg_pool_free ) ;
2016-04-05 00:48:11 +03:00
}
EXPORT_SYMBOL_GPL ( sg_free_table_chained ) ;
/**
* sg_alloc_table_chained - Allocate and chain SGLs in an sg table
* @ table : The sg table header to use
* @ nents : Number of entries in sg list
* @ first_chunk : first SGL
2019-04-28 10:39:30 +03:00
* @ nents_first_chunk : number of the SGL of @ first_chunk
2016-04-05 00:48:11 +03:00
*
* Description :
* Allocate and chain SGLs in an sg table . If @ nents @ is larger than
2019-06-06 11:34:08 +03:00
* @ nents_first_chunk a chained sg table will be setup . @ first_chunk is
* ignored if nents_first_chunk < = 1 because user expects the SGL points
* non - chain SGL .
2016-04-05 00:48:11 +03:00
*
* */
int sg_alloc_table_chained ( struct sg_table * table , int nents ,
2019-04-28 10:39:30 +03:00
struct scatterlist * first_chunk , unsigned nents_first_chunk )
2016-04-05 00:48:11 +03:00
{
int ret ;
BUG_ON ( ! nents ) ;
2019-04-28 10:39:30 +03:00
if ( first_chunk & & nents_first_chunk ) {
if ( nents < = nents_first_chunk ) {
2016-04-05 00:48:11 +03:00
table - > nents = table - > orig_nents = nents ;
sg_init_table ( table - > sgl , nents ) ;
return 0 ;
}
}
2019-04-28 10:39:30 +03:00
/* User supposes that the 1st SGL includes real entry */
2019-06-06 11:34:08 +03:00
if ( nents_first_chunk < = 1 ) {
2019-04-28 10:39:30 +03:00
first_chunk = NULL ;
nents_first_chunk = 0 ;
}
2016-04-05 00:48:11 +03:00
ret = __sg_alloc_table ( table , nents , SG_CHUNK_SIZE ,
2019-04-28 10:39:30 +03:00
first_chunk , nents_first_chunk ,
GFP_ATOMIC , sg_pool_alloc ) ;
2016-04-05 00:48:11 +03:00
if ( unlikely ( ret ) )
2019-04-28 10:39:30 +03:00
sg_free_table_chained ( table , nents_first_chunk ) ;
2016-04-05 00:48:11 +03:00
return ret ;
}
EXPORT_SYMBOL_GPL ( sg_alloc_table_chained ) ;
static __init int sg_pool_init ( void )
{
int i ;
for ( i = 0 ; i < SG_MEMPOOL_NR ; i + + ) {
struct sg_pool * sgp = sg_pools + i ;
int size = sgp - > size * sizeof ( struct scatterlist ) ;
sgp - > slab = kmem_cache_create ( sgp - > name , size , 0 ,
SLAB_HWCACHE_ALIGN , NULL ) ;
if ( ! sgp - > slab ) {
printk ( KERN_ERR " SG_POOL: can't init sg slab %s \n " ,
sgp - > name ) ;
goto cleanup_sdb ;
}
sgp - > pool = mempool_create_slab_pool ( SG_MEMPOOL_SIZE ,
sgp - > slab ) ;
if ( ! sgp - > pool ) {
printk ( KERN_ERR " SG_POOL: can't init sg mempool %s \n " ,
sgp - > name ) ;
goto cleanup_sdb ;
}
}
return 0 ;
cleanup_sdb :
for ( i = 0 ; i < SG_MEMPOOL_NR ; i + + ) {
struct sg_pool * sgp = sg_pools + i ;
2018-10-31 01:05:37 +03:00
mempool_destroy ( sgp - > pool ) ;
kmem_cache_destroy ( sgp - > slab ) ;
2016-04-05 00:48:11 +03:00
}
return - ENOMEM ;
}
static __exit void sg_pool_exit ( void )
{
int i ;
for ( i = 0 ; i < SG_MEMPOOL_NR ; i + + ) {
struct sg_pool * sgp = sg_pools + i ;
mempool_destroy ( sgp - > pool ) ;
kmem_cache_destroy ( sgp - > slab ) ;
}
}
module_init ( sg_pool_init ) ;
module_exit ( sg_pool_exit ) ;